关于strcpy函数形参类型的解析和指针作为输入型输出型参数的不同
在C语言中,字符串一直都是热点,关于strcpy函数大家都很熟悉,但是真正了解的很少,一旦用到总会报一大堆莫名其妙错误,今天我就来给大家详细剖析一下strcpy函数。
虽然不能看到strcpy的内部实现,但是我们通过查阅<string.h>可以看到strcpy函数的声明。
char * __cdecl strcpy(char *, const char *);
那个_cdecl是一个函数调用约定,暂且不讨论,我们今天就来说一下strcpy指针形参加const与不加的区别,帮助大家更好使用这个函数
首先我们要理解这两种语句有何不同
char *p="abcd";
char str[]="abcd";
这两条语句都是存储abcd字符串,但是经过编译链接后,会产生不同的结果
语句1,常量字符串会保存在程序的常量区,编译时会将该字符串在常量区的起始地址拷贝过来存在指针p中,
语句2 常量字符串也会保存在程序的常量区,但是在字符数组str初始化时i,会将字符串拷贝到str中,即数组中存储字符串副本
即str[0]='a';str[1]='b';str[2]='c';str[3]='d';str[4]='\0';
如图所示
我们看到在地址0x0042201C 处存储的是字符串常量abcd,而语句1汇编指令mov dword ptr ds:[ebp-4], 0xoo42201C,作用就是把常量字符串地址存到[ebp-4]这个内存空间(即变量名为p的内存地址)
语句2则是将该常量值一个个拷贝到数组中,即字符串存储在数据段中
那么这样区分之后,会产生一个是否允许修改的差异,我们都知道常量区中的内容不允许修改,而数据区的内容是可读可写的,因此,如果我们这样写
p[]='w';
str[]='q';
会发现 虽然语句1编译链接都通过,但是运行时会报错,这是因为语句1试图非法修改常量区的值,而常量区是允许修改,只能读取
而语句2则可以使程序正常运行,因为str数组只是常量字符串的一份副本,这份副本存在数据区,可以修改,而且不会影响到常量区字符串的值
明白了关于指向字符串常量的指针和存储字符串常量的数组之间的差异后,我们接下来讨论指针形参输出型和输入型问题
对于形参是指针类型的,我们都知道是传址调用,也明白函数内形参的改变会影响到形参,这就涉及一个实参是否允许修改的问题
char * strcpy(char * strDest,const char * strSrc);
这个函数形参,一个是不加const修饰,一个是加上了const修饰,有何区别呢?这个函数是将strSrc指向的字符串复制到strDest中,(连同字符串结束符'\0'一起复制),
那么就是说,strDest指向的值是可以修改的,而strSRC指向的值在函数中是不允许修改的,我们把加上const修饰的参数称为输入型参数,即只允许读取,不允许写入,把不加const修饰的参数称为输出型参数,即可以在函数内部进行读写改变,从而在主调函数中看到改变。
通过刚才的探讨,我们可以很容易知道如下四条语句哪些会使程序运行时出错
char *p="";
char *q="abcd";
char str1[]="";
char str2[]="hijk";
strcpy(p,q);①
strcpy(p,str1);②
strcpy(str1,p);③
strcpy(str1,str2);④
很明显,标号①②的语句都会运行时出错,因为strcpy的第一个形参要求是可以写入的,而p,q都是指向了常量区字符串的首地址,不可写入
标号③④都是可以正常运行,但是推荐写法③,因为写法④设计一个隐式转换问题,将str2转换成了常指针了。
下面给出一个strcpy函数的实现
char * strcpy(char *strDest,const char *strSrc)
{
assert((strDest!=NULL)&&(strSrc!=NULL)); //断言两个指针都不是空指针
char *address=strDest; //函数要返回复制后的字符串首地址
while((*(strDest++)=*(strSrc++))!='\0');//连同结束符一起复制
return address; //返回复制后的字符串首地址
}
总结
1 char *p="1234";指针p指向常量区,不允许修改内容及p[0]='a'非法
2 char str[5]="abcd" 字符数组str复制了常量区字符串“abcd”的值,而字符数组是在数据段,可以进行修改及str[0]='1'合法
3 对于strcpy函数的第一个形参,是输出型形参,因此只能是数组名,而不能是字符指针变量
4 对strcpy的第二个形参,是输入型形参,可以是数组名或者是字符指针变量,但最好是字符指针变量
关于strcpy函数形参类型的解析和指针作为输入型输出型参数的不同的更多相关文章
- 通过指定函数/方法形参类型提高PHP代码可靠性
指定形参类型是PHP 5就支持的一项特性.形参支持array - 数组. object - 对象两种类型. class User{ public $name; public $password; fu ...
- C/C++——strcpy函数的实现
题目: 已知strcpy函数的原型是: char * strcpy(char * strDest,const char * strSrc); 1.不调用库函数,实现st ...
- 《征服 C 指针》摘录5:函数形参 和 空的下标运算符[]
一.函数的形参的声明 C 语言可以像下面这样声明函数的形参: void func(int a[]) { // ... } 对于这种写法,无论怎么看都好像要向函数的参数传递数组. 可是,在 C ...
- python学习道路(day4note)(函数,形参实参位置参数匿名参数,匿名函数,高阶函数,镶嵌函数)
1.函数 2种编程方法 关键词面向对象:华山派 --->> 类----->class面向过程:少林派 -->> 过程--->def 函数式编程:逍遥派 --> ...
- memcpy、memmove、memset及strcpy函数实现和理解
memcpy.memmove.memset及strcpy函数实现和理解 关于memcpy memcpy是C和C++ 中的内存拷贝函数,在C中所需的头文件是#include<string.h> ...
- strcpy函数的C/C++实现
2013-07-05 14:07:49 本函数给出了几种strcpy与strncpy的实现,有ugly implementation,也有good implementation.并参考标准库中的imp ...
- strcmp函数和strcpy函数
(一)strcmp函数 strcmp函数是比較两个字符串的大小,返回比較的结果.一般形式是: i=strcmp(字符串,字符串); 当中,字符串1.字符串2均可为字符串常量或变量:i 是用于存放比 ...
- C#中:函数访问级别对函数形参访问级别的约束
Inconsistent accessibility: parameter type 'Program.CommandLineInfo' is less accessible than method ...
- strcpy函数用法
字符串是数组类型,不能通过赋值运算进行,要通过strcpy进行拷贝,其中目的字符串必须是字符串变量,源字符串可以是常量,复制后源字符串保持不变. strcpy()是C中的一个复制字符串的库函数,在C+ ...
随机推荐
- (hdu)4858 项目管理 (vector)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4858 Problem Description 我们建造了一个大项目!这个项目有n个节点,用很多边连接起 ...
- Photoshop调出清晰的阴雨天气山水风景照
既然我们前期拍摄到了一张效果还不错的照片,那么下一步就是通过后期处理得到最终的影像. 在处理之前,我们一定要做到胸有成竹,而不是盲目调整. 也就是说在还没调整照片的时候,就要计划和预想到最终的照片应该 ...
- Web并发页面访问量统计实现
Web并发页面访问量统计实现 - huangshulang1234的博客 - CSDN博客https://blog.csdn.net/huangshulang1234/article/details/ ...
- 好久好久没写,,百度API逆地址解析以及删除指定marker
百度地图Api中 除覆盖物有两个方法:map.removeOverlay()或者 map.clearOverlays(),其中 clearOverlays()方法一次移除所有的覆盖物removeOve ...
- 常用的flex布局
演示地址:https://xibushijie.github.io/static/flex.html
- Ubuntu下基于Virtualenv构建Python开发环境
1.安装virtualenv并建立虚拟环境 1).更新pip版本 sudo pip install --upgrade pip 如果出现如下异常: File , in <module> f ...
- 配置ssh免密登录服务器
当前服务器环境为ubantu 14.04 一.本地 ①生成id_rsa id_rsa.pub ssh-keygen -t rsa -C "xxxxx@xxxxx.com" ss ...
- nowcoder16450 托米的简单表示法
题目链接 思路 仔细理解一下题意可以发现. 对于每个完整的括号序列都是独立的,然后就想到分治.高度是序列中所有括号序列的最大值,宽度是所有括号序列宽度和\(+1\). 然后仔细想了一下,这种分治应该是 ...
- 转:centos 7 安装音频视频解码器
(原文:https://blog.csdn.net/zhou1519/article/details/39035233/) 1.安装额外的软件源epel和nux-dextop rpm -Uvh htt ...
- Hadoop基础-镜像文件(fsimage)和编辑日志(edits)
Hadoop基础-镜像文件(fsimage)和编辑日志(edits) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.查看日志镜像文件(如:fsimage_00000000000 ...