c語言strdup
㈠ c語言問題。大家幫忙!
bcmp(比較內存內容)
相關函數 bcmp,strcasecmp,strcmp,strcoll,strncmp,strncasecmp
表頭文件 #include<string.h>
定義函數 int bcmp ( const void *s1,const void * s2,int n);
函數說明 bcmp()用來比較s1和s2所指的內存區間前n個位元組,若參數n為0,則返回0。
返回值 若參數s1 和s2 所指的內存內容都完全相同則返回0 值,否則返回非零值。
附加說明 建議使用memcmp()取代。
範例 參考memcmp()。
b(拷貝內存內容)
相關函數 memccpy,memcpy,memmove,strcpy,ctrncpy
表頭文件 #include <string.h>
定義函數 void b ( const void *src,void *dest ,int n);
函數說明 b()與memcpy()一樣都是用來拷貝src所指的內存內容前n個位元組到dest所指的地址,不過參數src與dest在傳給函數時是相反的位置。
返回值
附加說明 建議使用memcpy()取代
範例 #include<string.h>
main()
{
char dest[30]=」string(a)」;
char src[30]=」string\0string」;
int i;
b(src,dest,30);/* src指針放在前*/
printf(b(): 「)
for(i=0;i<30;i++)
printf(「%c」,dest[i]);
memcpy(dest src,30); /*dest指針放在錢*/
printf(『\nmemcpy() : 「);
for(i=0;i<30;i++)
printf(「%c」,dest[i]);
執行 b() : string string
memcpy() :string sring
bzero(將一段內存內容全清為零)
相關函數 memset,swab
表頭文件 #include<string.h>
定義函數 void bzero(void *s,int n);
函數說明 bzero()會將參數s所指的內存區域前n個位元組,全部設為零值。相當於調用memset((void*)s,0,size_tn);
返回值
附加說明 建議使用memset取代
範例 參考memset()。
index(查找字元串中第一個出現的指定字元)
相關函數 rindex,srechr,strrchr
表頭文件 #include<string.h>
定義函數 char * index( const char *s, int c);
函數說明 index()用來找出參數s字元串中第一個出現的參數c地址,然後將該字元出現的地址返回。字元串結束字元(NULL)也視為字元串一部分。
返回值 如果找到指定的字元則返回該字元所在地址,否則返回0。
範例 #include<string.h>
main()
{
char *s =」」;
char *p;
p =index(s,』5』);
printf(%s\n」,p);
}
執行 5.68E+25
memccpy(拷貝內存內容)
相關函數 b,memcpy,memmove,strcpy,strncpy
表頭文件 #include<string.h>
定義函數 void * memccpy(void *dest, const void * src, int c,size_t n);
函數說明 memccpy()用來拷貝src所指的內存內容前n個位元組到dest所指的地址上。與memcpy()不同的是,memccpy()會在復制時檢查參數c是否出現,若是則返回dest中值為c的下一個位元組地址。
返回值 返回指向dest中值為c的下一個位元組指針。返回值為0表示在src所指內存前n個位元組中沒有值為c的位元組。
範例 #include<string.h>
main()
{
char a[]="string[a]";
char b[]="string(b)";
memccpy(a,b,'B',sizeof(b));
printf("memccpy():%s\n",a);
}
執行 memccpy():string(b)
memchr(在某一內存范圍中查找一特定字元)
相關函數 index,rindex,strchr,strpbrk,strrchr,strsep,strspn,strstr
表頭文件 #include<string.h>
定義函數 void * memchr(const void *s,int c,size_t n);
函數說明 memchr()從頭開始搜尋s所指的內存內容前n個位元組,直到發現第一個值為c的位元組,則返回指向該位元組的指針。
返回值 如果找到指定的位元組則返回該位元組的指針,否則返回0。
範例 #include <string.h>
main()
{
char *s="";
char *p;
p=memchr(s,'5',10);
printf("%s\n",p);
}
執行 5.68E+25
memcmp(比較內存內容)
相關函數 bcmp,strcasecmp,strcmp,strcoll,strncmp,strncasecmp
表頭文件 #include<string.h>
定義函數 int memcmp (const void *s1,const void *s2,size_t n);
函數說明 memcmp()用來比較s1和s2所指的內存區間前n個字元。字元串大小的比較是以ASCII碼表上的順序來決定,次順序亦為字元的值。memcmp()首先將s1第一個字元值減去s2第一個字元的值,若差為0則再繼續比較下個字元,若差值不為0則將差值返回。例如,字元串"Ac"和"ba"比較則會返回字元'A'(65)和'b'(98)的差值(-33)。
返回值 若參數s1和s2所指的內存內容都完全相同則返回0值。s1若大於s2則返回大於0的值。s1若小於s2則返回小於0的值。
範例 #include<string.h>
main()
{
char *a ="aBcDeF";
char *b="AbCdEf";
char *c="aacdef";
char *d="aBcDeF";
printf("memcmp(a,b):%d\n",memcmp((void*)a,(void*) b,6));
printf("memcmp(a,c):%d\n",memcmp((void*)a,(void*) c,6));
printf("memcmp(a,d):%d\n",memcmp((void*)a,(void*) d,6));
執行 memcmp(a,b):1 /*字元串a>字元串b,返回1*/
memcmp(a,c):-1 /* 字元串a<字元串c,返回-1*/
memcmp(a,d):0 /*字元串a=字元串d,返回0*/
memcpy(拷貝內存內容)
相關函數 b,memccpy,memcpy,memmove,strcpy,strncpy
表頭文件 #include<string.h>
定義函數 void * memcpy (void * dest ,const void *src, size_t n);
函數說明 memcpy()用來拷貝src所指的內存內容前n個位元組到dest所指的內存地址上。與strcpy()不同的是,memcpy()會完整的復制n個位元組,不會因為遇到字元串結束'\0'而結束。
返回值 返回指向dest的指針。
附加說明 指針src和dest所指的內存區域不可重疊。
範例 #include<string.h>
main()
{
char a[30]="string (a)";
char b[30]="string\0string";
int i;
strcpy(a,b);
printf("strcpy():");
for(i=0;i<30;i++)
printf("%c",a[i]);
memcpy(a,b,30);
printf("\nmemcpy() :");
for(i=0;i<30;i++)
printf("%c",a[i]);
}
執行 strcpy() : string a )
memcpy() : string string
memmove(拷貝內存內容)
相關函數 b,memccpy,memcpy,strcpy,strncpy
表頭文件 #include<string.h>
定義函數 void * memmove(void *dest,const void *src,size_t n);
函數說明 memmove()與memcpy()一樣都是用來拷貝src所指的內存內容前n個位元組到dest所指的地址上。不同的是,當src和dest所指的內存區域重疊時,memmove()仍然可以正確的處理,不過執行效率上會比使用memcpy()略慢些。
返回值 返回指向dest的指針。
附加說明 指針src和dest所指的內存區域可以重疊。
範例 參考memcpy()。
memset(將一段內存空間填入某值)
相關函數 bzero,swab
表頭文件 #include<string.h>
定義函數 void * memset (void *s ,int c, size_t n);
函數說明 memset()會將參數s所指的內存區域前n個位元組以參數c填入,然後返回指向s的指針。在編寫程序時,若需要將某一數組作初始化,memset()會相當方便。
返回值 返回指向s的指針。
附加說明 參數c雖聲明為int, 但必須是unsigned char ,所以范圍在0到255之間。
範例 #include <string.h>
main()
{
char s[30];
memset (s,'A',sizeof(s));
s[30]='\0';
printf("%s\n",s);
}
執行
rindex(查找字元串中最後一個出現的指定字元)
相關函數 index,memchr,strchr,strrchr
表頭文件 #include<string.h>
定義函數 char * rindex( const char *s,int c);
函數說明 rindex()用來找出參數s字元串中最後一個出現的參數c地址,然後將該字元出現的地址返回。字元串結束字元(NULL)也視為字元串一部分。
返回值 如果找到指定的字元則返回該字元所在的地址,否則返回0。
範例 #include <string.h>
mian()
{
char *s ="";
char *p;
p=rindex(s,'5');
printf("%s\n",p);
}
執行 567890
strcasecmp(忽略大小寫比較字元串)
相關函數 bcmp,memcmp,strcmp,strcoll,strncmp
表頭文件 #include<string.h>
定義函數 int strcasecmp (const char *s1, const char *s2);
函數說明 strcasecmp()用來比較參數s1和s2字元串,比較時會自動忽略大小寫的差異。
返回值 若參數s1和s2字元串相同則返回0。s1長度大於s2長度則返回大於0 的值,s1 長度若小於s2 長度則返回小於0的值。
範例 #include <string.h>
main()
{
char *a="aBcDeF";
char *b="AbCdEf";
if(!strcasecmp(a,b))
printf("%s=%s\n",a,b);
}
執行 aBcDeF=AbCdEf
strcat(連接兩字元串)
相關函數 b,memccpy,memcpy,strcpy,strncpy
表頭文件 #include <string.h>
定義函數 char *strcat (char *dest,const char *src);
函數說明 strcat()會將參數src字元串拷貝到參數dest所指的字元串尾。第一個參數dest要有足夠的空間來容納要拷貝的字元串。
返回值 返回參數dest的字元串起始地址
範例 #include <string.h.>
main()
{
char a[30]="string(1)";
char b[]="string(2)";
printf("before strcat() : %s\n",a);
printf("after strcat() : %s\n",strcat(a,b));
}
執行 before strcat () : string(1)
after strcat () : string(1)string(2)
strchr(查找字元串中第一個出現的指定字元)
相關函數 index,memchr,rinex,strbrk,strsep,strspn,strstr,strtok
表頭文件 #include<string.h>
定義函數 char * strchr (const char *s,int c);
函數說明 strchr()用來找出參數s字元串中第一個出現的參數c地址,然後將該字元出現的地址返回。
返回值 如果找到指定的字元則返回該字元所在地址,否則返回0。
範例 #include<string.h>
main()
{
char *s=」;
char *p;
p=strchr(s,'5');
printf("%s\n",p);
}
執行 5.68E+25
strcmp(比較字元串)
相關函數 bcmp,memcmp,strcasecmp,strncasecmp,strcoll
表頭文件 #include<string.h>
定義函數 int strcmp(const char *s1,const char *s2);
函數說明 strcmp()用來比較參數s1和s2字元串。字元串大小的比較是以ASCII 碼表上的順序來決定,此順序亦為字元的值。strcmp()首先將s1第一個字元值減去s2第一個字元值,若差值為0則再繼續比較下個字元,若差值不為0則將差值返回。例如字元串"Ac"和"ba"比較則會返回字元"A"(65)和'b'(98)的差值(-33)。
返回值 若參數s1和s2字元串相同則返回0。s1若大於s2則返回大於0的值。s1若小於s2則返回小於0 的值。
範例 #include<string.h>
main()
{
char *a="aBcDeF";
char *b="AbCdEf";
char *c="aacdef";
char *d="aBcDeF";
printf("strcmp(a,b) : %d\n",strcmp(a,b));
printf("strcmp(a,c) : %d\n",strcmp(a,c));
printf("strcmp(a,d) : %d\n",strcmp(a,d));
}
執行 strcmp(a,b) : 32
strcmp(a,c) :-31
strcmp(a,d) : 0
strcoll(採用目前區域的字元排列次序來比較字元串)
相關函數 strcmp,bcmp,memcmp,strcasecmp,strncasecmp
表頭文件 #include<string.h>
定義函數 int strcoll( const char *s1, const char *s2);
函數說明 strcoll()會依環境變數LC_COLLATE所指定的文字排列次序來比較s1和s2 字元串。
返回值 若參數s1和s2字元串相同則返回0。s1若大於s2則返回大於0的值。s1若小於s2則返回小於0 的值。
附加說明 若LC_COLLATE為"POSIX"或"C",則strcoll()與strcmp()作用完全相同。
範例 參考strcmp()。
strcpy(拷貝字元串)
相關函數 b,memcpy,memccpy,memmove
表頭文件 #include<string.h>
定義函數 char *strcpy(char *dest,const char *src);
函數說明 strcpy()會將參數src字元串拷貝至參數dest所指的地址。
返回值 返回參數dest的字元串起始地址。
附加說明 如果參數dest所指的內存空間不夠大,可能會造成緩沖溢出(buffer Overflow)的錯誤情況,在編寫程序時請特別留意,或者用strncpy()來取代。
範例 #include<string.h>
main()
{
char a[30]="string(1)";
char b[]="string(2)";
printf("before strcpy() :%s\n",a);
printf("after strcpy() :%s\n",strcpy(a,b));
}
執行 before strcpy() :string(1)
after strcpy() :string(2)
strcspn(返回字元串中連續不含指定字元串內容的字元數)
相關函數 strspn
表頭文件 #inclued<string.h>
定義函數 size_t strcspn ( const char *s,const char * reject);
函數說明 strcspn()從參數s字元串的開頭計算連續的字元,而這些字元都完全不在參數reject 所指的字元串中。簡單地說,若strcspn()返回的數值為n,則代表字元串s開頭連續有n個字元都不含字元串reject內的字元。
返回值 返回字元串s開頭連續不含字元串reject內的字元數目。
範例 #include <string.h>
main()
{
char *str="Linux was first developed for 386/486-based pcs.";
printf("%d\n",strcspn(str," "));
printf("%d\n",strcspn(str,"/-"));
printf("%d\n",strcspn(str,"1234567890"));
}
執行 5 /*只計算到「 」的出現,所以返回「Linux」的長度*/
33 /*計算到出現「/」或「-」,所以返回到「6」的長度*/
30 /* 計算到出現數字字元為止,所以返回「3」出現前的長度*/
strp(復制字元串)
相關函數 calloc,malloc,realloc,free
表頭文件 #include<string.h>
定義函數 char * strp( const char *s);
函數說明 strp()會先用maolloc()配置與參數s字元串相同的空間大小,然後將參數s字元串的內容復制到該內存地址,然後把該地址返回。該地址最後可以利用free()來釋放。
返回值 返回一字元串指針,該指針指向復制後的新字元串地址。若返回NULL表示內存不足。
範例 #include<string.h>
main()
{
char a[]="strp";
char *b;
b=strp(a);
printf("b[ ]=\"%s\"\n",b);
}
執行 b[ ]="strp"
strlen(返回字元串長度)
相關函數
表頭文件 #include<string.h>
定義函數 size_t strlen (const char *s);
函數說明 strlen()用來計算指定的字元串s的長度,不包括結束字元"\0"。
返回值 返回字元串s的字元數。
範例 /*取得字元串str的長度*/
#include<string.h>
main()
{
char *str = "12345678";
printf("str length = %d\n", strlen(str));
}
執行 str length = 8
strncasecmp(忽略大小寫比較字元串)
相關函數 bcmp,memcmp,strcmp,strcoll,strncmp
表頭文件 #include<string.h>
定義函數 int strncasecmp(const char *s1,const char *s2,size_t n);
函數說明 strncasecmp()用來比較參數s1和s2字元串前n個字元,比較時會自動忽略大小寫的差異。
返回值 若參數s1和s2 字元串相同則返回0。s1 若大於s2則返回大於0的值,s1若小於s2則返回小於0 的值。
範例 #include<string.h>
main()
{
char *a="aBcDeF";
char *b="AbCdEf";
if(!strncasecmp(a,b))
printf("%s =%s\n",a,b);
}
執行 aBcDef=AbCdEf
strncat(連接兩字元串)
相關函數 b,memccpy,memecpy,strcpy,strncpy
表頭文件 #inclue <string.h>
定義函數 char * strncat(char *dest,const char *src,size_t n);
函數說明 strncat()會將參數src字元串拷貝n個字元到參數dest所指的字元串尾。第一個參數dest要有足夠的空間來容納要拷貝的字元串。
返回值 返回參數dest的字元串起始地址。
範例 #include <string.h>
main()
{
char a[30]="string(1)";
char b[]="string(2)";
printf("before strnact() :%s\n", a);
printf("after strncat() :%s\n", strncat(a,b,6));
}
執行 before strnact() : string(1)
after strncat() : string(1) string
strncpy(拷貝字元串)
相關函數 b,memccpy,memcpy,memmove
表頭文件 #include<string.h>
定義函數 char * strncpy(char *dest,const char *src,size_t n);
函數說明 strncpy()會將參數src字元串拷貝前n個字元至參數dest所指的地址。
返回值 返回參數dest的字元串起始地址。
範例 #inclue <string.h>
main()
{
char a[30]="string(1)";
char b[]="string(2)";
printf("before strncpy() : %s\n",a);
printf("after strncpy() : %s\n",strncpy(a,b,6));
}
執行 before strncpy() : string(1)
after strncpy() : string(1)
strpbrk(查找字元串中第一個出現的指定字元)
相關函數 index,memchr,rindex,strpbrk,strsep,strspn,strstr,strtok
表頭文件 #include <include.h>
定義函數 char *strpbrk(const char *s,const char *accept);
函數說明 strpbrk()用來找出參數s 字元串中最先出現存在參數accept 字元串中的任意字元。
返回值 如果找到指定的字元則返回該字元所在地址,否則返回0。
範例 #include <string.h>
main()
{
char *s="";
char *p;
p=strpbrk(s,"a1 839"); /*1會最先在s字元串中找到*/
printf("%s\n",p);
p=strprk(s,"4398");/*3 會最先在s 字元串中找到*/
printf("%s\n",p);
執行 1.23E+29
strrchr(查找字元串中最後出現的指定字元)
相關函數 index,memchr,rindex,strpbrk,strsep,strspn,strstr,strtok
表頭文件 #include<string.h>
定義函數 char * strrchr(const char *s, int c);
函數說明 strrchr()用來找出參數s字元串中最後一個出現的參數c地址,然後將該字元出現的地址返回。
返回值 如果找到指定的字元則返回該字元所在地址,否則返回0。
範例 #include<string.h>
main()
{
char *s="";
char *p;
p=strrchr(s,'5');
printf("%s\n",p);
}
執行 567890
strspn(返回字元串中連續不含指定字元串內容的字元數)
相關函數 strcspn,strchr,strpbrk,strsep,strstr
表頭文件 #include<string.h>
定義函數 size_t strspn (const char *s,const char * accept);
函數說明 strspn()從參數s 字元串的開頭計算連續的字元,而這些字元都完全是accept 所指字元串中的字元。簡單的說,若strspn()返回的數值為n,則代表字元串s 開頭連續有n 個字元都是屬於字元串accept內的字元。
返回值 返回字元串s開頭連續包含字元串accept內的字元數目。
範例 #include<string.h>
main()
{
char *str="Linux was first developed for 386/486-based PCs.";
char *t1="";
printf("%d\n",strspn(str,t1));
}
執行 5 /*計算大小寫字母。不包含「 」,所以返回Linux的長度。*/
strstr(在一字元串中查找指定的字元串)
相關函數 index,memchr,rindex,strchr,strpbrk,strsep,strspn,strtok
表頭文件 #include<string.h>
定義函數 char *strstr(const char *haystack,const char *needle);
函數說明 strstr()會從字元串haystack 中搜尋字元串needle,並將第一次出現的地址返回。
返回值 返回指定字元串第一次出現的地址,否則返回0。
範例 #include<string.h>
main()
{
char * s="";
char *p;
p= strstr(s,"901");
printf("%s\n",p);
}
執行 9.01E+21
strtok(分割字元串)
相關函數 index,memchr,rindex,strpbrk,strsep,strspn,strstr
表頭文件 #include<string.h>
定義函數 char * strtok(char *s,const char *delim);
函數說明 strtok()用來將字元串分割成一個個片段。參數s指向欲分割的字元串,參數delim則為分割字元串,當strtok()在參數s的字元串中發現到參數delim的分割字元時則會將該字元改為\0 字元。在第一次調用時,strtok()必需給予參數s字元串,往後的調用則將參數s設置成NULL。每次調用成功則返回下一個分割後的字元串指針。
返回值 返回下一個分割後的字元串指針,如果已無從分割則返回NULL。
範例 #include<string.h>
main()
{
char s[]="ab-cd : ef;gh :i-jkl;mnop;qrs-tu: vwx-y;z";
char *delim="-: ";
char *p;
printf("%s ";strtok(s,delim));
while((p=strtok(NULL,delim)))printf("%s ",p);
printf("\n");
}
執行 ab cd ef;gh i jkl;mnop;qrs tu vwx y;z /*-與:字元已經被\0 字元取代*/
㈡ c語言程序 從數據文件中讀入數據到數組
先打開文件。
自己寫過getline函數,取文本的一行保存到一個字元串中。
用malloc函數給line分配空間。
對於第一行,取特定的字元串line取代替它,如果第二行是空行也這么做
在建立一個字元串數組,用strp將他們放入字元串數組strline[i]中。
構造一個結構數組,將得到的字元串line用strtok(line,"
")函數分開依次放入結構數組中的字元串成員中,記得不能用字元串數組的字元串,strtok函數會改變字元串的,而line就無所謂了。
知道用getline函數返回一個負數(一般我用負數標記文檔讀寫完畢)。結束讀操作。
現將前面的名字以及空行寫入new的文本中,用putline函數(將字元串寫入文檔中)。
再從字元串數組逆序寫入new的文本中。
如果結構數組不用了,就釋放strp分配的空間,並且以及line的空間,前面兩行無所謂,反正占的空間不大。所有用數組,還是動態分配都沒什麼大的關系。
㈢ 用C語言截取字元串
如果允許使用庫函數,那麼實際這個非常簡單:
#include<string.h>
#include<stdio.h>
voidparse_str(char*str)
{
char*s=NULL,*p;
if(str[0]!='s'||str[1]!='!')
return;
s=strp(str+2);
p=strtok(s,":");
while(p)
{
printf("%s ",p);
p=strtok(NULL,":");
}
free(s);
}
㈣ C語言中string.h中用到的字元串處理函數有哪些
1.函數名: stpcpy
功 能: 拷貝一個字元串到另一個
2.函數名: strcat
功 能: 字元串拼接函數
3.函數名: strchr
功 能: 在一個串中查找給定字元的第一個匹配之處\
4.函數名: strcmp
功 能: 串比較
5.函數名: strncmpi
功 能: 將一個串中的一部分與另一個串比較, 不管大小寫
6.函數名: strcpy
功 能: 串拷貝
7.函數名: strcspn
功 能: 在串中查找第一個給定字元集內容的段
8.函數名: strp
功 能: 將串拷貝到新建的位置處
9.函數名: stricmp
功 能: 以大小寫不敏感方式比較兩個串
10.函數名: strerror
功 能: 返回指向錯誤信息字元串的指針
11函數名: strcmpi
功 能: 將一個串與另一個比較, 不管大小寫
12函數名: strncmp
功 能: 串比較
13函數名: strncmpi
功 能: 把串中的一部分與另一串中的一部分比較, 不管大小寫
14函數名: strncpy
功 能: 串拷貝
15函數名: strnicmp
功 能: 不注重大小寫地比較兩個串
16函數名: strnset
功 能: 將一個串中的所有字元都設為指定字元
17函數名: strpbrk
功 能: 在串中查找給定字元集中的字元
18函數名: strrchr
功 能: 在串中查找指定字元的最後一個出現
19函數名: strrev
功 能: 串倒轉
20函數名: strset
功 能: 將一個串中的所有字元都設為指定字元
21函數名: strspn
功 能: 在串中查找指定字元集的子集的第一次出現
22函數名: strstr
功 能: 在串中查找指定字元串的第一次出現
23函數名: strtod
功 能: 將字元串轉換為double型值
24函數名: strtok
功 能: 查找由在第二個串中指定的分界符分隔開的單詞
25函數名: strtol
功 能: 將串轉換為長整數
26函數名: strupr
功 能: 將串中的小寫字母轉換為大寫字母
27函數名: swab
功 能: 交換位元組
㈤ c語言題目:乒乓球比賽
太強大了,
#include "stdio.h"
void main()
{
char i,j,k;/*i是a的對手,j是b的對手,k是c的對手*/
for(i='x';i<='z';i++)//i有三種對陣情況x-a,y-a,z-a,用循環遍歷
for(j='x';j<='z';j++)//j有三種對陣情況x-b,y-b,z-b,用循環遍歷
{
if(i!=j)//表示i 、j 不能為x,y,z三人中的同一個人
for(k='x';k<='z';k++)//k有三種對陣情況x-c,y-c,z-c,用循環遍歷
{
if(i!=k&&j!=k)//表示k、i、j不能為x,y,z三人中的同一個人
{
if(i!='x'&&k!='x'&&k!='z')//此乃題目條件:A說他不和X比(a!=x),而i 又非跟a比,所以i!='x';C說他不和X、Z比。即:i!='x'且k!='x'且k!='z'(同理)
printf("order is a--%c\tb--%c\tc--%c\n",i,j,k);//經過以上排列,選擇,最終確定對手組合
}
}
}
}
㈥ 關於 CGO 的字元串函數的解釋
CGO為Go語言提供了與C語言交互的手段。它提供了四個用於轉換Go和C類型字元串的函數。這些函數通過復制數據實現,但是使用時需注意幾個關鍵點。
第一類函數包括C.CString()與C.GoString()。C.CString()相當於C語言中的strp()函數,將Go的字元串復制為可以傳遞給C函數的C語言字元指針。然而,在調用C.free()時,需要特別注意Go字元串可能包含的\0字元。由於Go字元串中可能嵌入\0字元,使用C.CString()時,C代碼可能會從\0字元處截斷字元串,這通常不會引起注意,但有時文本可能包含\0字元。
第二類函數是C.GoString()與C.GoStringN()。C.GoString()類似於C的strp(),但與C.CString()相反,它是將C字元串轉換為Go字元串。它可以用於定義結構體欄位、聲明為C的char *類型或作為其他變數。C.GoStringN()等同於C的memmove(),用於復制整個N長度的C緩沖區到Go字元串。它通過復制來實現,並且不會單獨處理null字元。當你使用C結構體的欄位如char field[64]時,調用C.GoStringN(&field, 64)後,你將得到一個長度為64的Go字元串,其末尾可能是一串\0字元。
C.GoBytes()是C.GoStringN()的一個變體,不返回string類型,而是返回[]byte。它同樣進行內存拷貝操作,但不宣稱接受C字元串作為參數。
當處理固定長度的C字元串時,如char field[N]的欄位,上述函數可能無法滿足需求。這些欄位在傳統語義中表示當且僅當字元串有足夠的空間時以null字元結尾,也就是說最多有N-1個字元。如果字元串恰好有N個字元,它不會以null字元結尾。這可能導致C代碼中的bug,盡管它不是一個理想的設計。面對這類欄位時,文檔通常不會明確指出欄位內容不總是以null字元結尾,需要用戶自行假設。
C.GoString()或C.GoStringN()無法正確處理這種情況。使用C.GoStringN()相對出錯較少,它會返回一個長度為N的Go字元串,末尾可能有額外的\0字元。使用C.GoString()則可能會導致更嚴重的錯誤,因為它會對輸入字元串調用strlen()。如果字元串末尾沒有null字元,strlen()可能會訪問越界的內存地址。結果可能是生成包含大量垃圾數據的Go字元串,或者更糟糕的是,程序因訪問未映射內存地址而導致段錯誤。
為了解決這個問題,可以使用與C的strnp()等價的Go函數來確保復制不超過N個字元且在null字元處終止。可以手動實現這樣的函數,以最小化額外內存的使用。
這篇文章討論了Go和CGO類型之間的字元串轉換函數的使用注意事項和常見問題。它強調了在處理C語言字元串時,需要特別注意null字元和字元串長度問題,以及在使用CGO提供的函數時可能出現的錯誤和陷阱。