C 函数参数 char **s与char *s[]
本文同时发表在https://github.com/zhangyachen/zhangyachen.github.io/issues/126
先来看一个小例子 : 编写函数遍历一个整型数组的元素,数组最后一个元素为-1标志数组的结束。
#include <stdio.h>
void test(int *a){
int x;
while((x = *a++) != -1) {
printf("%d\n",x);
}
}
int main(void) {
int arr[] = {1,2,3,4,5,6,-1};
test(arr);
return 0;
}
原理是在函数参数为数组名时,实际传递的是该数组第一个元素的地址,通过*a++
即可遍历该数组。实际上*a++
分三步:
- 产生一个a的拷贝
- 对a地址进行++操作,编译器发现a地址存储的是整型数据,所以a++指向a之后的sizeof(int)个字节,即数组下一个元素
- 对a的拷贝进行间接访问,即取出a的拷贝指向的数据。
下面对题目进行一下升级 : 编写函数遍历一个指针数组的元素,数组最后一个元素为NULL标志数组的结束。
#include <stdio.h>
void test_char(char *s){
char *string = NULL;
while((string = *s++) != NULL){
printf("%s\n",string);
}
}
int main(void) {
char *s[] = {"hello","world","ni","hao",NULL};
test_char(s);
return 0;
}
一运行代码,卧槽崩溃了。什么鬼。表面看起来很对啊,将数组首地址传递到函数内部,然后取出每个元素存储的字符串常量地址,再输出字符串。
盯着代码看了10分钟左右我才发现错误 : 虽然传递给函数的是数组第一个元素的地址,即0x000(为方便起见做的假设)。但是由于函数形参是char *
类型,所以*s
操作取得是0x000-0x001之间的数据赋值给char * string,而不是数组元素里的内容,由于这不是字符串常量的首地址,所以printf("%s\n",string);
就报错了。而且还有一点是s++也不是指向数组的第二个元素。因为数组是指针数组,存放的是字符串常量的首地址,每个数组元素是8字节(32位机器4字节)。而s是char *类型,s++的结果也只是0x001。
有什么办法让s++指向数组下一个元素,并且*s取出来的是数组元素的内容?其实将函数参数s声明为char **即可。因为编译器发现指针s存储的地址指向的是char *
类型,8字节,所以*s
取出来的是s存储的地址 - s存储的地址+8字节之间的内容。(后来想想第一个例子函数形参为int *a,需要一次取址取到数据。第二个例子为指针数组,需要两次取址取到数据,函数形参自然就是char **,当然这都是马后炮了)。
#include <stdio.h>
void test_char(char **s){
char *string = NULL;
while((string = *s++) != NULL){
printf("%s\n",string);
}
}
int main(void) {
char *s[] = {"hello","world","ni","hao",NULL};
test_char(s);
return 0;
}
函数形参也可以声明为char *s[]
,相似的还有函数形参为int a[]
与int *a
是等价的。
工作一年了还犯这种低级错误,真是有点羞愧啊。
C 函数参数 char **s与char *s[]的更多相关文章
- char *与const char **函数参数传参问题
传参方法 ## 函数 extern void f2 ( const char ** ccc ); const char ch = 'X'; char * ch_ptr; const char ** c ...
- C++ Error C2664:无法将参数 1 从“const char [9]”转换为“LPCWSTR”解决方案
问题出现 编译平台:VS2013 Windows 出现地方:在使用LoadLibrary( )函数动态链接DLL文件时出现的一个问题 Eg. 在使用 UNICODE字符的工程中, HIN ...
- 不能将参数1从“const char []”转换为“LPCTSTR
今天在使用vs2008+MFC时候,使用editControl的replacesel(“”)发生报错.如下::不能将参数1从“const char []”转换为“LPCTSTR” 其解决方案就是, 在 ...
- error C2664: “CWnd::MessageBoxW”: 不能将参数 1 从“const char [17]”转换为“LPCTSTR”
vs2008提示 error C2664: “CWnd::MessageBoxW”: 不能将参数 1 从“const char [17]”转换为“LPCTSTR” 在外面用vs2005编写mfc程序的 ...
- [转载] 已知strcpy的函数原型:char *strcpy(char *strDest, const char *strSrc),编写函数 strcpy(C++版)
已知strcpy的函数原型:char *strcpy(char *strDest, const char *strSrc)其中strDest 是目的字符串,strSrc 是源字符串.不调用C++/C ...
- VS2010 error C2664: “CWnd::MessageBoxW”: 不能将参数 1 从“const char [3]”转换为“LPCTSTR”
VS2010 (VC2010)建立工程时默认的字符集是Unicode,所以在代码中使用MessageBox时需要输入Unicode,用TEXT()这个宏输入参数,否则会报错: 代码:Message ...
- list排序成员函数对string对象与char*对象排序的差别
对list容器中的对象排序,不能使用sort()算法,只能采用其自身的排序函数sort().因为,算法sort()只支持随机存取的容器的排序,如vector等. 对基本数据对象list排序:成员函数s ...
- error: C2664: “zajiao::zajiao(const zajiao &)”: 无法将参数 1 从“const char [12]”转换为“char *”
原本打算在QT用一个字符串"ABCDEF12345"作为类zajiao的构造函数的参数,用来创建类zajiao的对象zajiao1. zajiao zajiao1("AB ...
- 给定一个字符串,把字符串内的字母转换成该字母的下一个字母,a换成b,z换成a,Z换成A,如aBf转换成bCg, 字符串内的其他字符不改变,给定函数,编写函数 void Stringchang(const char*input,char*output)其中input是输入字符串,output是输出字符串
import java.util.Scanner; /*** * 1. 给定一个字符串,把字符串内的字母转换成该字母的下一个字母,a换成b,z换成a,Z换成A,如aBf转换成bCg, 字符串内的其他字 ...
随机推荐
- C++ UTF8和GB2312相互转换
#include <Windows.h> #include <string> using std::string; void UTF8_to_GB2312(const char ...
- PHP时间戳和日期互转换
在php中我们要把时间戳转换日期可以直接使用date函数来实现,如果要把日期转换成时间戳可以使用strtotime()函数实现,下面我来给大家举例说明. 1.php中时间转换函数 strtotime ...
- 解决微信公众平台IP白名单
微信公众平台,作为自媒体的旗舰级产品,越来越多的人已经投入它的怀抱.正如它的广告词所说:再小的个体,也有品牌 好吧,闲话不多说,今天要说的是它的IP白名单机制. 我们现在安装的大部分的电信的家庭级别的 ...
- 【微服务】之二:从零开始,轻松搞定SpringCloud微服务系列--注册中心(一)
微服务体系,有效解决项目庞大.互相依赖的问题.目前SpringCloud体系有强大的一整套针对微服务的解决方案.本文中,重点对微服务体系中的服务发现注册中心进行详细说明.本篇中的注册中心,采用Netf ...
- IntelliJ IDEA(三) :常用快捷键
说IDEA对新手来说难,可能其中一个原因就是快捷键组合多而且复杂但是它也很全,基本所有功能都可以通过快捷键来完成,可以这么说,如果你掌握了所有IDEA的快捷键使用,那么你完全可以丢掉鼠标,而且不影响开 ...
- HDU 1317XYZZY spfa+判断正环+链式前向星(感觉不对,但能A)
XYZZY Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Subm ...
- nginx使用ssl模块配置支持HTTPS访问【解决ssl错误】
默认情况下ssl模块并未被安装,如果要使用该模块则需要在编译nginx时指定–with-http_ssl_module参数. 需求:做一个网站域名为 www.localhost.cn 要求通过http ...
- [P4基础]p4app的Docker镜像仓使用详解
How to use the p4lang/p4app Docker image 本文参考 孙勇峰博客 和 p4app README 以及 p4app Dockerfile 结合自己的理解做一些记录, ...
- springBoot数据库连接池常用配置
在配置文件中添加配置如下(我使用的是多数据源): spring.datasource.primary.url=jdbc\:mysql\://localhost\:3306/test?useUnicod ...
- 基于Abp的WebApi容器
简述 对 Abp的动态web api的改造过程 注册 1. 首先通过反射将<服务类型>通过ApiControllerBuilder 构建成成 DynamicApiControllerInf ...