【C语言】字符串函数探幽

目录

1、strcpy()

a)如果src长度大于dest会发生什么?

i.执行到strcpy函数之前,查看a和b的值和内存:

ii、执行strcpy,观察内存

iii、得出结论

b)如果src长度小于dset呢?

c)手写strcpy

d)总结

2、strlen()

3、strcat()

最近学习C++时遇到了字符串的问题,结合之前的面试题深入分析下。

字符串库函数有strcpy、strlen、strstr等,可以参见C语言字符串操作总结大全

1、strcpy()

strcpy():char *strcpy(char* dest, const char *src);返回指向dset的指针

strcpy()被认为是不安全的函数,因为它的函数实现非常简单,没有考虑异常情况。下面考虑两种情况:

a)如果src长度大于dest会发生什么?

linux man page里面提到了这个BUG:

If the destination string of a strcpy() is not large enough, then anything might happen. Overflowing fixed-length string buffers is a favorite cracker technique for taking complete control of the machine. Any time a program reads or copies data into a buffer, the program first needs to check that there's enough space. This may be unnecessary if you can show that overflow is impossible, but be careful: programs can get changed over time, in ways that may make the impossible possible.

大概意思是这样做固定长度的字符串就会发生溢出。如果程序没有事先检查内存空间,就会发生难以预料的事情。

写好代码:char a[2] = "a"; char b[10] = "bbbbbbbb";  auto p=strcpy(a, b);  在VS断点调试,步骤如下:

i.执行到strcpy函数之前,查看a和b的值和内存:

发现VS编译器以16进制数c来作为未被使用的内存,a后面还有6字节的空内存。

ii、执行strcpy,观察内存

发现a虽然只有2个字节的内存,但是后面的内存都被覆盖掉了(对比上下两图可以发现值为9f的这个内存被覆盖成00了,其他的CC被62填充)

iii、得出结论

会覆盖字符串数组之外的内存(数组越界)。假设数组b无限大,那岂不是把整个堆都覆盖掉了?果然这个函数非常危险,要改用strncpy()。

b)如果src长度小于dset呢?

src从头到\0都被复制到dset中,dest多余的部分没有变化(右图)。这种情况没什么问题。

c)手写strcpy

发现网上大家的代码都没能解决数组越界的问题,我加了一行代码 if (*dst == '\0') break; 目前看来没什么问题。

char* my_strcpy(char *dst, const char *src) {
	assert(dst != NULL && src != NULL);//断言输入不为空,空则调用abort()异常终止进程
	char *ret = dst;
	while ((*dst = *src) != '\0') {
		dst++;
		src++;
		if (*dst == '\0') break;//如果此时目标数组已经结束,就不用继续复制了
	}
	return ret;
}

调试发现运行结束时不会覆盖dst后面的内存了。

d)总结

在一本面试书上看到过,说这个函数是MS考虑到普适性,故意写的很简单。不过这个问题也很老了,只是凭兴趣研究下。总结起来就是程序员要注意内存溢出的问题,特别在使用指针的时候,这种小问题很可能会导致一些重大安全问题。

2、strlen()

size_t  __cdecl strlen(const char * s){
    int i = 0;
    while( *s ) {
        i++;
        s++;
    }
    return i;
}

3、strcat()

char * __cdecl strcat(char * dst, const char * src){
    char *p = dst;
    while( *p )
        p++;
    while( *p ++ = *src ++ )
        ;
    return dst;
}

4、strcmp()

int strcmp(const char *str1, const char *str2){
   int ret=0;
   while( !(ret = *(unsigned char*)str1 - *(unsigned char*)str2 ) && *str1 )
     {
       str1++;
       str2++;
     }
     if(ret < 0)
        return -1;
     else if(ret > 0) 
        return 1;
     return 0;    
 }

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
回答: 在C语言中,string函数是一个字符串处理函数库,它包含在<string.h>头文件中。其中常用的函数有strcpy、strlen和strnset。 strcpy函数用于将一个字符串复制到另一个字符串中。它的函数原型是:char *strcpy(char *dest, const char *src)。其中,dest是目标字符串,src是源字符串。这个函数会将src字符串的内容复制到dest字符串中,并返回dest字符串的指针。\[1\] strlen函数用于计算字符串的长度。它的函数原型是:size_t strlen(const char *str)。这个函数接收一个字符串的首地址,然后遍历字符串直到遇到'\0'字符,返回字符串的长度。\[2\] strnset函数用于将指定的字符替换字符串中的一部分字符。它的函数原型是:char *strnset(char *str, int c, size_t n)。其中,str是要操作的字符串,c是要替换的字符,n是要替换的字符个数。这个函数会将字符串中的指定部分字符替换为指定的字符。\[3\] 这些函数都是C语言中常用的字符串处理函数,可以帮助我们进行字符串的复制、长度计算和字符替换等操作。 #### 引用[.reference_title] - *1* *3* [C语言中string函数详解](https://blog.csdn.net/weixin_30902251/article/details/99781150)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [c语言String字符串函数探幽](https://blog.csdn.net/Duary/article/details/106163396)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值