从头文件、函数声明、函数功能和其他说明等角度对以下函数进行对比和学习。
1:strnlen
<string.h>
size_t strnlen(const char *str, size_t maxlen);
计算字符串str的(unsigned int型)长度,直到碰到第一个结束符'\0'或计数至maxlen为止,然后返回长度。
不包括结束符'\0',该长度最大为maxlen。
1.1 strlen
string.h(C)或cstring(C++)
unsigned int strlen(char *s);
计算字符串的长度,从开始到遇到第一个'\0'。
如果只定义没有给s赋初值,这个结果不定,它会从s首地址一直找下去,直到遇到'\0'停止。
2:strlcpy
#include <string.h>
size_t strlcpy(char *dest, const char *src, size_t size)
在已知dest缓冲区大小并不会造成缓冲区溢出前提下,将src地址开始的字符串复制到以dest开始的地址空间。复制完,会在dst后追加'\0'字符。
strlcpy函数创建的目的主要是针对strcpy函数缓冲区溢出的问题,因为一旦strcpy调用中src的字符串长度超过了dst的缓冲区大小,就会造成缓冲区安全漏洞问题
2.1 strncpy
和strlcpy的区别是:如果src的长度小于n个字节,则以'\0'填充dest直到复制完n个字节;复制完,若src前n个字节不含结束符,则dest不会以结束符结束,函数也不会在dst后追加'\0'字符,需手动加'\0'。
2.2 strcpy
#include <string.h> 和 #include <stdio.h>
char *strcpy(char* dest, const char *src);
将src地址开始且含有'\0'结束符的字符串复制到以dest开始的地址空间。
src和dest所指内存区域不可以重叠且dest必须有足够空间来容纳src的字符串。
3:strlcat
#include <string.h>
size_t strlcat(char *dest, const char *src, size_t destsz);
把src所指向的字符串的前destsz个字节复制到dest所指向的字符串后面(删除*dest原来末尾的“\0”)。追加完后,末尾加上结束符。
strlcat函数创建的目的主要是针对strcat函数缓冲区域出的问题,因为一旦strcat调用中src的字符串长度超过了dst的缓冲区大小,就会造成缓冲区安全漏洞问题。
3.1 strcat
#include <string.h>
extern char *strcat(char *dest, const char *src);
把src所指向的字符串(包括“\0”)复制到dest所指向的字符串后面(删除*dest原来末尾的“\0”)。
src和dest所指内存区域不可以重叠且dest必须有足够的空间来容纳src的字符串。
4:snprintf
#include <stdio.h>
int snprintf(char* dest_str,size_t size,const char* format,...);
将可变个参数(...)按照format格式化成字符串,然后将其复制到str中。
(1) 如果格式化后的字符串长度 < size,则将此字符串全部复制到str中,并给其后添加一个字符串结束符('\0');
(2) 如果格式化后的字符串长度 >= size,则只将其中的(size-1)个字符复制到str中,并给其后添加一个字符串结束符('\0'),返回值为欲写入的字符串长度。
若成功则返回预写入的字符串长度,若出错返回负值。
4.1 sprintf
#include <stdio.h>
int sprintf(char *string, char *format [,argument,...]);
如果成功,返回写入的字符总数,不包括字符串追加在字符串末尾的空字符。如果失败,返回一个负数。
与snprintf的返回值不同,sprintf的返回值是成功写入的字符串长度,此处需要谨慎处理。
memcpy相关函数见memset,memcpy,memmove和memchr