鱼C论坛

 找回密码
 立即注册
查看: 2197|回复: 6

[已解决]关于学习C语言指针数组中的一些问题

[复制链接]
发表于 2018-6-2 17:21:09 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
#include <stdio.h>

int main()
{
        char *p1[5]={
                "让编程改变世界",
                "HELLO WORLD",
                "勇敢不止步",
                "china",
                "csu"
        };

        char *p2=p1[0];
                 
        int i;

        for(i=0;i<5;i++)
        {
                printf("%s\n",*(p2+i));
        }


        return 0;
}
学习指针数组过程中写了一个理解指针数组概念的代码如上,p1是一个指针数组存放5个字符串,定义p2为一个指针,我的理解是p1既然是一个指针数组,那么p1中的每一个元素都代表一个地址,于是将p1[0]赋值给p2,然后希望通过叠加的方式,利用p2将p1数组中的所有字符串打印出来,但是结果输出结果是空白,请问各位神通广大的鱼游问题出在哪里了?
最佳答案
2018-6-2 23:17:45
本帖最后由 simplerjiang 于 2018-6-2 23:24 编辑

首先你在代码中的操作,是用过for循环,将p2以字符串方式,逐个字符来输出。
但是你要知道,指针数组是存放的指针,这不代表里面的指针地址是连续的。

当你使用(p2+i)的时候,它其实是读取了p1[0] 这个地址的下i个字节的地址,并不是跳到下一个字符串p1[1],

字符不同于字节,字符是一个抽象意义,对于整型数组来说,一个字符是4个字节(不同计算机不同)
也就是当你 (p + 1) ,它是跳了4个字节,为了读取下一个整型,因为计算机知道p是一个整型数组。
同理对于字符串char[] 字符串类型来说, (p + 1) ,一次是跳了1个字节,对于它来说一个字符就是1个字节。

但是因为p2只是一个纯指针,他不知道你这个地址到底是储存了整型还是字符串,对于它来说
每跳一次都是一个字节,所以你(p2 + i) 只是得到了p2这个地址的下一字节地址,很不巧的是,这个地址储存的甚至不是一个char。

因为对于中文字来说,它们使用的是双字节编码,一个汉字等于两个字节,当p2指针跳到下一个地址(p2+1)时
它读取到的是错误的,它读取到的是第一个汉子的第二个字节,所以它应该是读不出来的。

同时字符串的变量本身也是一个指针,我们用printf("%s",str); 输出的时候,
计算机是以读取到\0结尾,所以当p2读到\0时,就会认为已经到结尾了。

更不巧的是,双字节编码的头两位二进制就是00 ,等于我们的'\0' (这里学艺不精,我也不太确定是不是,如果有错,请见谅并指正)
因此每当计算机读到(p2 + i) ,它都以为,这就是个空字符串,因为读到的第一个就是\0

我并没有拷贝你的代码编译运行,所以可能分析会有偏差。
但是你这个输出方法是错误的。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2018-6-2 22:01:09 From FishC Mobile | 显示全部楼层
指针数组不等于指针
请问p2如何知道数组的长度多少?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2018-6-2 23:04:06 | 显示全部楼层
BngThea 发表于 2018-6-2 22:01
指针数组不等于指针
请问p2如何知道数组的长度多少?

指针数组存放的不是指针数据吗?
char *p2=p1[0];
不就相当于把p1中存放的第一个指针赋值给了p2吗?
这个地方概念有点模糊了,求大神解释!T_T
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2018-6-2 23:17:45 | 显示全部楼层    本楼为最佳答案   
本帖最后由 simplerjiang 于 2018-6-2 23:24 编辑

首先你在代码中的操作,是用过for循环,将p2以字符串方式,逐个字符来输出。
但是你要知道,指针数组是存放的指针,这不代表里面的指针地址是连续的。

当你使用(p2+i)的时候,它其实是读取了p1[0] 这个地址的下i个字节的地址,并不是跳到下一个字符串p1[1],

字符不同于字节,字符是一个抽象意义,对于整型数组来说,一个字符是4个字节(不同计算机不同)
也就是当你 (p + 1) ,它是跳了4个字节,为了读取下一个整型,因为计算机知道p是一个整型数组。
同理对于字符串char[] 字符串类型来说, (p + 1) ,一次是跳了1个字节,对于它来说一个字符就是1个字节。

但是因为p2只是一个纯指针,他不知道你这个地址到底是储存了整型还是字符串,对于它来说
每跳一次都是一个字节,所以你(p2 + i) 只是得到了p2这个地址的下一字节地址,很不巧的是,这个地址储存的甚至不是一个char。

因为对于中文字来说,它们使用的是双字节编码,一个汉字等于两个字节,当p2指针跳到下一个地址(p2+1)时
它读取到的是错误的,它读取到的是第一个汉子的第二个字节,所以它应该是读不出来的。

同时字符串的变量本身也是一个指针,我们用printf("%s",str); 输出的时候,
计算机是以读取到\0结尾,所以当p2读到\0时,就会认为已经到结尾了。

更不巧的是,双字节编码的头两位二进制就是00 ,等于我们的'\0' (这里学艺不精,我也不太确定是不是,如果有错,请见谅并指正)
因此每当计算机读到(p2 + i) ,它都以为,这就是个空字符串,因为读到的第一个就是\0

我并没有拷贝你的代码编译运行,所以可能分析会有偏差。
但是你这个输出方法是错误的。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2018-6-2 23:21:43 | 显示全部楼层
simplerjiang 发表于 2018-6-2 23:17
首先你在代码中的操作,是用过for循环,将p2以字符串方式,逐个字符来输出。
但是你要知道,指针数组是存 ...

如果有什么不明确的,可以说出来,我写明白些
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2018-6-3 12:32:00 | 显示全部楼层
simplerjiang 发表于 2018-6-2 23:21
如果有什么不明确的,可以说出来,我写明白些

谢谢大神,明白了!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2018-6-3 13:22:39 | 显示全部楼层
瓜瓜咚 发表于 2018-6-3 12:32
谢谢大神,明白了!

明白请设置最佳答案
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-11-29 22:51

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表