এই প্রশ্নের চারপাশে দুটি বিষয় মেমরি পরিচালনা এবং থ্রেড সুরক্ষা। আপনি অসংখ্য পোস্ট থেকে দেখতে পাচ্ছেন যে সি তে নির্বিঘ্নে কাজ করা সহজ কাজ নয়, আমি একটি সমাধান চেয়েছিলাম যা হল:
- থ্রেড নিরাপদ. (স্ট্রোকক থ্রেড নিরাপদ নয়)
- ম্যালোক বা এর কোনও ডেরাইভেটিভ নিয়োগ করে না (মেমরি পরিচালনার সমস্যাগুলি এড়াতে)
- স্বতন্ত্র ক্ষেত্রগুলিতে অ্যারে সীমা পরীক্ষা করে (অজানা ডেটাতে বিভাগের ত্রুটিগুলি এড়ানোর জন্য)
- মাল্টি-বাইট ফিল্ড বিভাজক (utf-8) এর সাথে কাজ করে
- ইনপুটটিতে অতিরিক্ত ক্ষেত্র উপেক্ষা করে
- অবৈধ ক্ষেত্রের দৈর্ঘ্যের জন্য নরম ত্রুটির রুটিন সরবরাহ করে
আমি যে সমাধানটি এসেছি সেগুলি এই সমস্ত মানদণ্ডকে মেটায়। এখানে পোস্ট করা কিছু অন্যান্য সমাধানের চেয়ে সেটআপ করা সম্ভবত সম্ভবত আরও কিছু কাজ, তবে আমি মনে করি যে অন্যান্য সমাধানগুলির সাধারণ সমস্যাগুলি এড়াতে বাস্তবে অতিরিক্ত কাজটি মূল্যবান।
#include <stdio.h>
#include <string.h>
struct splitFieldType {
char *field;
int maxLength;
};
typedef struct splitFieldType splitField;
int strsplit(splitField *fields, int expected, const char *input, const char *fieldSeparator, void (*softError)(int fieldNumber,int expected,int actual)) {
int i;
int fieldSeparatorLen=strlen(fieldSeparator);
const char *tNext, *tLast=input;
for (i=0; i<expected && (tNext=strstr(tLast, fieldSeparator))!=NULL; ++i) {
int len=tNext-tLast;
if (len>=fields[i].maxLength) {
softError(i,fields[i].maxLength-1,len);
len=fields[i].maxLength-1;
}
fields[i].field[len]=0;
strncpy(fields[i].field,tLast,len);
tLast=tNext+fieldSeparatorLen;
}
if (i<expected) {
if (strlen(tLast)>fields[i].maxLength) {
softError(i,fields[i].maxLength,strlen(tLast));
} else {
strcpy(fields[i].field,tLast);
}
return i+1;
} else {
return i;
}
}
void monthSplitSoftError(int fieldNumber, int expected, int actual) {
fprintf(stderr,"monthSplit: input field #%d is %d bytes, expected %d bytes\n",fieldNumber+1,actual,expected);
}
int main() {
const char *fieldSeparator=",";
const char *input="JAN,FEB,MAR,APRI,MAY,JUN,JUL,AUG,SEP,OCT,NOV,DEC,FOO,BAR";
struct monthFieldsType {
char field1[4];
char field2[4];
char field3[4];
char field4[4];
char field5[4];
char field6[4];
char field7[4];
char field8[4];
char field9[4];
char field10[4];
char field11[4];
char field12[4];
} monthFields;
splitField inputFields[12] = {
{monthFields.field1, sizeof(monthFields.field1)},
{monthFields.field2, sizeof(monthFields.field2)},
{monthFields.field3, sizeof(monthFields.field3)},
{monthFields.field4, sizeof(monthFields.field4)},
{monthFields.field5, sizeof(monthFields.field5)},
{monthFields.field6, sizeof(monthFields.field6)},
{monthFields.field7, sizeof(monthFields.field7)},
{monthFields.field8, sizeof(monthFields.field8)},
{monthFields.field9, sizeof(monthFields.field9)},
{monthFields.field10, sizeof(monthFields.field10)},
{monthFields.field11, sizeof(monthFields.field11)},
{monthFields.field12, sizeof(monthFields.field12)}
};
int expected=sizeof(inputFields)/sizeof(splitField);
printf("input data: %s\n", input);
printf("expecting %d fields\n",expected);
int ct=strsplit(inputFields, expected, input, fieldSeparator, monthSplitSoftError);
if (ct!=expected) {
printf("string split %d fields, expected %d\n", ct,expected);
}
for (int i=0;i<expected;++i) {
printf("field %d: %s\n",i+1,inputFields[i].field);
}
printf("\n");
printf("Direct structure access, field 10: %s", monthFields.field10);
}
নীচে সংকলন এবং আউটপুট উদাহরণ দেওয়া আছে। মনে রাখবেন যে আমার উদাহরণে, আমি উদ্দেশ্যমূলকভাবে "এপ্রিল" বানান যাতে আপনি দেখতে পারেন যে নরম ত্রুটি কীভাবে কাজ করে।
$ gcc strsplitExample.c && ./a.out
input data: JAN,FEB,MAR,APRIL,MAY,JUN,JUL,AUG,SEP,OCT,NOV,DEC,FOO,BAR
expecting 12 fields
monthSplit: input field #4 is 5 bytes, expected 3 bytes
field 1: JAN
field 2: FEB
field 3: MAR
field 4: APR
field 5: MAY
field 6: JUN
field 7: JUL
field 8: AUG
field 9: SEP
field 10: OCT
field 11: NOV
field 12: DEC
Direct structure access, field 10: OCT
উপভোগ করুন!
strtok
একই জিনিস অর্জনের জন্য আপনি স্ট্যান্ডার্ড লাইব্রেরি থেকে ফাংশনটি ব্যবহার করতে পারেন ।