字符串拷贝和内存拷贝函数:
strcpy
strncpy
memcpy
memmove
memccpy
bcopy
字符串和内存数据比较函数:
strcmp
strcasecmp
strncasecmp
memcmp
strcoll
bcmp
连接字符串的函数:
strcat
strncat
查找字符 / 字符串的函数:
strstr
strchr
strrchr
memchr
其它相关的函数:
index
rindex
strlen
strdup
memset
bzero
strspn
strcspn
strpbrk
strtok
路径名分割函数:
basename
dirname
字符串拷贝和内存拷贝函数:
strcpy (拷贝字符串)
定义函数: char *strcpy( char *dest, const char *src );
strcpy() 函数只能拷贝字符串。strcpy() 函数将源字符串src 的每个字节拷贝到目的字符串dest 中,src 字符串末尾的'/0' 也被拷贝过去。strcpy() 函数返回参数dest 的起始地址。如果参数dest 所指的内存空间不够大,可能会造成缓冲溢出(buffer Overflow) 的错误情况,在编写程序时请特别留意,或者用strncpy() 来取代。
strncpy (拷贝字符串)
定义函数: char *strncpy( char *dest, const char *src, size_t n );
strncpy() 会将参数src 字符串拷贝前n 个字符至参数dest 所指的地址。函数返回参数dest 的字符串起始地址。
注意n 的取值范围,不要超过src 和dest 的长度。
#include<string.h> #include<stdio.h>
int main() { char a1[30]="string(1)"; char b1[]="STRING(2)"; printf("before strcpy() : %s/n", a1 ); printf("after strcpy() : %s/n", strcpy( a1, b1 ) );
char a2[30]="string(1)"; char b2[]="STRING(2)"; printf("before strncpy() : %s/n", a2 ); printf("after strncpy() : %s/n", strncpy( a2, b2, 6 ) ); } |
memcpy (拷贝内存内容)
定义函数: void * memcpy( void * dest, const void *src, size_t n );
memcpy() 用来拷贝src 所指的内存内容前n 个字节到dest 所指的内存地址上。与strcpy() 不同的是,memcpy() 会完整的复制n 个字节,不会因为遇到字符串结束'/0' 而结束。memcpy() 函数可以拷贝任意类型的数据。memcpy() 函数返回指向dest 的指针。指针src 和dest 所指的内存区域不可重叠。在拷贝字符串时,通常都使用strcpy() 函数;在拷贝其它数据( 例如结构) 时,通常都使用memcpy() 函数。
memmove (拷贝内存内容)
定义函数: void *memmove(void *dest, const void *src, size_t n );
memmove() 与memcpy() 一样都是用来拷贝src 所指的内存内容前n 个字节到dest 所指的地址上。不同的是,当src 和dest 所指的内存区域重叠时,memmove() 仍然可以正确的处理,不过执行效率上会比使用memcpy() 略慢些。该函数返回指向dest 的指针。
#include<string.h>
int main() { char a[30]="string(1)"; char b[]="string(2)"; printf("before strcpy() :%s/n", a ); printf("after strcpy() :%s/n", strcpy( a, b ) );
a[30]="string(1)"; b[]="string(2)"; printf("before strncpy() : %s/n", a ); printf("after strncpy() : %s/n", strncpy( a, b, 6 ) ); } |
memccpy (拷贝内存内容)
定义函数: void * memccpy( void *dest, const void *src, int c, size_t n );
memccpy() 用来拷贝src 所指的内存内容前n 个字节到dest 所指的地址上。与memcpy() 不同的是,memccpy() 会在复制时检查参数c 是否出现,若是则返回dest 中值为c 的下一个字节地址。该函数返回指向dest 中值为c 的下一个字节指针。返回值为NULL 表示在src 所指内存前n 个字节中没有值为c 的字节。
#include<string.h> #include<stdio.h>
int main() { char a[]="string(a)"; char b[]="string(b)"; char *p;
p = ( char * )memccpy( a, b, 'k', sizeof( b ) );
if( p == NULL ) { // 注意 p 为 NULL 的情况,这时不能读取 p 所指的地方的内容 printf("the return pointer of mymccpy is null !/n"); } else { printf("memccpy(): %s, *p = %c/n", a, *p ); } } |
bcopy (拷贝内存内容)
定义函数: void bcopy ( const void *src,void *dest ,int n);
bcopy() 与memcpy() 一样都是用来拷贝src 所指的内存内容前n 个字节到dest 所指的地址,不过参数src 与dest 在传给函数时是相反的位置。建议使用memcpy() 取代 。
字符串和内存数据比较函数:
strcmp (比较字符串)
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 的值。
strcoll (采用目前区域的字符排列次序来比较字符串)
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() 作用完全相同。
strcasecmp (忽略大小写比较字符串)
int strcasecmp ( const char *s1, const char *s2 );
strcasecmp() 用来比较参数 s1 和 s2 字符串,比较时会自动忽略大小写的差异。若参数 s1 和 s2 字符串相同则返回 0 , s1 长度大于 s2 长度则返回大于 0 的值, s1 长度若小于 s2 长度则返回小于 0 的值。
str n casecmp (忽略大小写比较字符串)
int str n casecmp( const char *s1, const char *s2, size_t n );
strncasecmp() 用来比较参数 s1 和 s2 字符串前 n 个字符,比较时会自动忽略大小写的差异。若参数 s1 和 s2 字符串相同则返回 0 。 s1 若大于 s2 则返回大于 0 的值, s1 若小于 s2 则返回小于 0 的值。
strcmpi() 或 stricmp() :对两个字符串进行大小写不敏感的比较。
strncmp() :对两个字符串的一部分进行大小写敏感的比较。
strnicmp() :对两个字符串的一部分进行大小写不敏感的比较。
#include <stdio.h> #include <string.h>
int main() { char *a="aBcDeF"; char *b="AbCdEf"; char *c="aacdef"; char *d="ABCDEF";
printf("strcmpi(a,b) : %d/n",strcmpi(a,b)); printf("stricmp(a,b) : %d/n",stricmp(a,b)); printf("strcmp(a,d) : %d/n",strcmp(a,d));
printf("strncmp(a,b,3) : %d/n",strncmp(a,b,3)); printf("strnicmp(a,b,3) : %d/n",strnicmp(a,b,3));
getchar(); return 0; } |
memcmp (比较内存内容)
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 的值。
bcmp (比较内存内容)
int bcmp ( const void *s1,const void * s2,int n);
bcmp() 用来比较 s1 和 s2 所指的内存区间前 n 个字节,若参数 n 为 0 ,则返回 0 。若参数 s1 和 s2 所指的内存内容都完全相同则返回 0 值,否则返回非零值。 建议使用 memcmp() 取代。
#include <stdio.h> #include <string.h> int 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)); } |
连接字符串的函数:
strcat (连接两字符串)
char *strcat (char *dest, const char *src );
strcat() 会将参数src 字符串拷贝到参数dest 所指的字符串尾。第一个参数dest 要有足够的空间来容纳要拷贝的字符串。该函数返回参数dest 的字符串起始地址。
strncat (连接两字符串)
char * strncat( char *dest, const char *src, size_t n );
strncat() 会将参数src 字符串拷贝n 个字符到参数dest 所指的字符串尾。第一个参数dest 要有足够的空间来容纳要拷贝的字符串。该函数返回参数dest 的字符串起始地址。
#include<stdio.h> #include<string.h> int main() { //strcat char a[30]="string1ABCDE"; char b[]="STRING2"; printf( "before strcat() : %s/n",a ); printf( "after strcat() : %s/n",strcat( a, b ) );
//strncat char a1[30]="string1"; char b1[]="STRING2"; printf( "before strnact(): %s/n", a1 ); printf( "after strncat() : %s/n", strncat( a1, b1, 6 ) );
getchar(); return 0; } |
查找字符/ 字符串的函数:
strstr (在一字符串中查找指定的字符串)
char *strstr( const char *haystack, const char *needle );
strstr() 会从字符串haystack 中搜寻字符串needle ,并将第一次出现的地址返回。返回指定字符串第一次出现的地址,否则返回NULL 。
strchr (查找字符串中第一个出现的指定字符)
char *strchr( const char *s, int c );
strchr() 用来找出参数s 字符串中第一个出现的参数c 地址,然后将该字符出现的地址返回。如果找到指定的字符则返回该字符所在地址,否则返回NULL 。
strrchr (查找字符串中最后出现的指定字符)
char * strrchr( const char *s, int c );
strrchr() 用来找出参数s 字符串中最后一个出现的参数c 地址,然后将该字符出现的地址返回。如果找到指定的字符则返回该字符所在地址,否则返回NULL 。
memchr (在某一内存范围中查找一特定字符)
void *memchr( const void *s, int c, size_t n );
memchr() 从头开始搜寻s 所指的内存内容前n 个字节,直到发现第一个值为c 的字节,则返回指向该字节的指针。如果找到指定的字节则返回该字节的指针,否则返回NULL 。
#include<stdio.h> #include<string.h> int main() { char *s = "0123456789012345678901234567890"; char *p;
p = strchr(s,'9'); printf(" strchr(): %s/n",p);
p = strrchr(s,'9'); printf("strrchr(): %s/n",p);
p = strstr(s,"901"); printf(" strstr(): %s/n",p);
p = (char *)memchr( s, '9', 10 ); printf(" memchr(): %s/n",p); } |
其它相关的函数:
index (查找字符串中第一个出现的指定字符)
char * index( const char *s, int c );
index() 用来找出参数s 字符串中第一个出现的参数c 的地址,然后将该字符出现的地址返回。字符串结束字符(NULL) 也视为字符串一部分。如果找到指定的字符则返回该字符所在地址,否则返回NULL 。 [ 用Dev-C++4.9.9.2 调试时,提示index 为定义,在linux 下确认一下]
rindex (查找字符串中最后一个出现的指定字符)
char * rindex( const char *s, int c );
rindex() 用来找出参数s 字符串中最后一个出现的参数c 的地址,然后将该字符出现的地址返回。字符串结束字符(NULL) 也视为字符串一部分。如果找到指定的字符则返回该字符所在的地址,否则返回NULL 。 [ 用Dev-C++4.9.9.2 调试时,提示rindex 为定义,在linux 下确认一下]
strlen (返回字符串长度)
size_t strlen( const char *s );
strlen() 用来计算指定的字符串s 的长度,不包括结束字符"/0" ,该函数返回字符串s 的字符数。
strdup (复制字符串)
char *strdup( const char *s );
strdup() 会先用maolloc() 配置与参数s 字符串相同的空间大小,然后将参数s 字符串的内容复制到该内存地址,然后把该地址返回。该地址最后可以利用free() 来释放。函数返回一字符串指针,该指针指向复制后的新字符串地址,若返回NULL 表示内存不足。
#include<stdio.h> #include<stdlib.h> #include<string.h>
int main() { char a[] = "strdup"; char *b; b = strdup( a ); printf("b[]=/"%s/"/n", b ); free(b); return 0; } |
bzero (将一段内存内容全清为零)
void bzero(void *s,int n) ;
bzero() 会将参数s 所指的内存区域前n 个字节,全部设为零值。相当于调用memset((void*)s,0,size_tn); 建议使用memset 取代该函数。
memset (将一段内存空间填入某值)
void * memset (void *s ,int c, size_t n );
memset() 会将参数s 所指的内存区域前n 个字节以参数c 填入,然后返回指向s 的指针。在编写程序时,若需要将某一数组作初始化,memset() 会相当方便。附加说明:参数c 虽声明为int ,但必须是unsigned char ,所以范围在0 到255 之间。
#include<string.h> #include<stdio.h> #include<stdlib.h>
int main() { char s[30]; memset (s,'A',sizeof(s)); s[29]='/0';
for( int i = 0; i < 30; i++ ) { printf("%d ", s[i] ); } printf("/n%s/n",s);
memset (s,'/0',sizeof(s)); for( int i = 0; i < 30; i++ ) { printf("%d ", s[i] ); } printf("/n/n%s/n",s); return 0; }
输出结果为: 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 65 0 AAAAAAAAAAAAAAAAAAAAAAAAAAAAA 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 |
strspn (返回字符串中连续含指定字符串内容的字符数)
size_t strspn( const char *s, const char *accept );
strspn() 从参数s 字符串的开头计算连续的字符,而这些字符都完全是accept 所指字符串中的字符。简单的说,若strspn() 返回的数值为n ,则代表字符串s 开头连续有n 个字符都是属于字符串accept 内的字符。该函数返回字符串s 开头连续包含字符串accept 内的字符数目。
strcspn (返回字符串中连续不含指定字符串内容的字符数)
size_t strcspn( const char *s, const char *reject );
strcspn() 从参数s 字符串的开头计算连续的字符,而这些字符都完全不在参数reject 所指的字符串中。简单地说,若strcspn() 返回的数值为n ,则代表字符串s 开头连续有n 个字符都不含字符串reject 内的字符。该函数返回字符串s 开头连续不含字符串reject 内的字符数目。
#include<stdio.h> #include<string.h>
int main() { char *str="Linux was first developed for 386/486-based PCs."; char *t1="abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; printf( "strspn:/n%d/n",strspn( str,t1 ) );
printf( "strcspn: %d/n",strcspn( str, " " ) ); printf( "strcspn: %d/n",strcspn( str, "/-" ) ); printf( "strcspn: %d/n",strcspn( str, "1234567890" ) ); } |
strpbrk (查找字符串中第一个出现的指定字符)
char *strpbrk( const char *s, const char *accept );
strpbrk() 用来找出参数s 字符串中最先出现存在参数accept 字符串中的任意字符。如果找到指定的字符则返回该字符所在地址,否则返回NULL 。 [ 用Dev-C++4.9.9.2 调试时,提示index 为定义,在linux 下确认一下]
strtok (分割字符串)
char * strtok( char *s, const char *delim );
strtok() 用来将字符串分割成一个个片段。参数s 指向欲分割的字符串,参数delim 则为分割字符串,当strtok() 在参数s 的字符串中发现到参数delim 的分割字符时则会将该字符改为/0 字符。在第一次调用时,strtok() 必需给予参数s 字符串,往后的调用则将参数s 设置成NULL 。每次调用成功则返回下一个分割后的字符串指针。如果已无从分割则返回NULL 。
函数strtok() 具有“破坏性”,它会在原字符串中插入NUL 字符( 如果原字符串还要做其它的改变,应该拷贝原字符串,并将这份拷贝传递给函数strtok()) 。函数strtok() 是不能“重新进入”的,你不能在一个信号处理函数中调用strtok() 函数,因为在下一次调用strtok() 函数时它总是会“记住”上一次被调用时的某些参数。strtok() 函数是一个古怪的函数,但它在分解以逗号或空白符分界的数据时是非常有用的。
#include<stdio.h>
#include<string.h>
static char s0[] = "Now is the time for all good men . . . " ;
static char s1[] = "abcde--ABCDE--fghij--FGHIJ";
int main()
{
char *p;
p = strtok( s0, " ");
while( p )
{
printf( "%s/n", p );
p = strtok( NULL, " " );
}
p = strtok( s1, "--");
while( p )
{
printf( "%s/n", p );
p = strtok( NULL, "--" );
}
getchar();
return 0;
}
输出结果为:
Now
is
the
time
for
all
good
men
.
.
.
abcde
ABCDE
fghij
FGHIJ