在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函数形参类型的解析和指针作为输入型输出型参数的不同的更多相关文章

  1. 通过指定函数/方法形参类型提高PHP代码可靠性

    指定形参类型是PHP 5就支持的一项特性.形参支持array - 数组. object - 对象两种类型. class User{ public $name; public $password; fu ...

  2. C/C++——strcpy函数的实现

    题目:     已知strcpy函数的原型是:         char * strcpy(char * strDest,const char * strSrc);     1.不调用库函数,实现st ...

  3. 《征服 C 指针》摘录5:函数形参 和 空的下标运算符[]

    一.函数的形参的声明 C 语言可以像下面这样声明函数的形参: void func(int a[]) {     // ... } 对于这种写法,无论怎么看都好像要向函数的参数传递数组. 可是,在 C ...

  4. python学习道路(day4note)(函数,形参实参位置参数匿名参数,匿名函数,高阶函数,镶嵌函数)

    1.函数 2种编程方法 关键词面向对象:华山派 --->> 类----->class面向过程:少林派 -->> 过程--->def 函数式编程:逍遥派 --> ...

  5. memcpy、memmove、memset及strcpy函数实现和理解

    memcpy.memmove.memset及strcpy函数实现和理解 关于memcpy memcpy是C和C++ 中的内存拷贝函数,在C中所需的头文件是#include<string.h> ...

  6. strcpy函数的C/C++实现

    2013-07-05 14:07:49 本函数给出了几种strcpy与strncpy的实现,有ugly implementation,也有good implementation.并参考标准库中的imp ...

  7. strcmp函数和strcpy函数

    (一)strcmp函数 strcmp函数是比較两个字符串的大小,返回比較的结果.一般形式是: i=strcmp(字符串,字符串); 当中,字符串1.字符串2均可为字符串常量或变量:i   是用于存放比 ...

  8. C#中:函数访问级别对函数形参访问级别的约束

    Inconsistent accessibility: parameter type 'Program.CommandLineInfo' is less accessible than method ...

  9. strcpy函数用法

    字符串是数组类型,不能通过赋值运算进行,要通过strcpy进行拷贝,其中目的字符串必须是字符串变量,源字符串可以是常量,复制后源字符串保持不变. strcpy()是C中的一个复制字符串的库函数,在C+ ...

随机推荐

  1. ConcurrentHashMap 与 Hashtable

    粘贴复制于:https://blog.csdn.net/lzwglory/article/details/79978788 集合是编程中最常用的数据结构.而谈到并发,几乎总是离不开集合这类高级数据结构 ...

  2. jenkins集成python时出现"Non-ASCII character '\xe6' in file"错误解决方法

    我的问题: 使用python3.5,在Linux环境下手动执行python文件时不报错,但是用jenkins自动执行时就报"Non-ASCII character '\xe6' in fil ...

  3. Ubuntu16.04安装TensorFlow及Mnist训练

    版权声明:本文为博主原创文章,欢迎转载,并请注明出处.联系方式:460356155@qq.com TensorFlow是Google开发的开源的深度学习框架,也是当前使用最广泛的深度学习框架. 一.安 ...

  4. Python 执行 shellcode

    import urllib2 import ctypes import base64 # 从我们的web服务器上下载shellcode url = "http://rinige.com/sh ...

  5. 解决Docker中运行的MySQL中文乱码

    docker exec -it mysql bash 如果没有安装vim,请参考 解决Docker容器中不能用vim编辑文件 vim /etc/mysql/mysql.conf.d/mysql.cnf ...

  6. kubernetes-kubeadm自动生成的证书过期的解决方法

    拉取kubernetes的源码: git clone https://github.com/kubernetes/kubernetes.git 切换版本: cd kubernetes &&am ...

  7. webpack学习记录-认识loader(二)

    Loader 就像是一个翻译员,能把源文件经过转化后输出新的结果,并且一个文件还可以链式的经过多个翻译员翻译. loader参考文章:https://webpack.docschina.org/loa ...

  8. Super Mario HDU - 4417 (主席树)

    Mario is world-famous plumber. His “burly” figure and amazing jumping ability reminded in our memory ...

  9. 无需认证的mail,适用于ZABBIX等运维系统

    cat main.cf | grep "^\s[^# \t].$" queue_directory = /var/spool/postfix command_directory = ...

  10. 首次使用Oracle SQL Developer 提示: enter the full pathname for java.exe

    https://www.cnblogs.com/520future/p/7699095.html 首次使用Oracle SQL Developer 提示: enter the full pathnam ...