作者poyenc (发箍)
看板C_and_CPP
标题Re: [问题] 关於不同资料却得到相同地址的问题
时间Wed Sep 11 23:21:55 2019
※ 引述《ac01965159 (leeleo)》之铭言:
: 最近在练习写程式的时候碰到一些问题,想来请教一下,以下是程式码:
: https://pastebin.com/pzgHN0bt
: 执行结果:
: https://i.imgur.com/9v2nirr.jpg
: 有两个问题:
: 1.我的想法是,a为一个储存a[0]的位置的指标,而a[0]又存放着指向a[0][0]资料的指
: 标,但是照理说,这两笔资料不是应该存在不同位置的吗?
: 2.那既然上面都已经输出了相同的地址,那我把一样的地址拿去取值,却得到不一样的
: 结果,不知道原因为何。
这个误会有点大, 不过只要补足几个观念就可以:
1. C/C++ 没有多维阵列, 但是有
阵列的阵列
2.
阵列可以转型成指标 (array to pointer conversion), 反
过来则不行. 例如: array of T 可以转型成 pointer to T.
3. 对
指标或阵列 x
使用 operator[] 作下标 (subscripting)
运算
得到的是 *((x) + (i)) (i 为整数型别)
考虑以下定义:
int ai[
10];
// array of 10 int
int aai[
20][
10];
// array of 20 array of 10 int
int aaai[
30][
20][
10];
// array of 30 array of 20 array of 10 int
用阵列的阵列去抽丝剥茧就可以厘清各种叙述的型别 (反而用
多维阵列去理解会卡关), 以你的程式码而言, 虽然 std::cout 印
出的値一样, 但是叙述型别却很不同:
┌───┬─────┐
│ 叙述 │ 型别 │
├───┼─────┤
│ a
│int[
2][
2]
│
├───┼─────┤
│ a[
0]
│ int[
2]
│
└───┴─────┘
对於
型别不同的两个叙述, 不管得到的指标値是否相同, 它的
意义
本质上就不一样. 可以用下面的程式码观察看看:
cout << (a +
1) << endl;
cout << (a[
0] +
1) << endl;
对阵列的操作和指标基本上类似, 不管是下标还是用 operator+ 作
运算, 都需要先知道每个元素所占的空间大小:
int x[
2];
int y[
3][
4];
assert(
reinterpret_cast<
char*>(&x[
1])
==
reinterpret_cast<
char*>(x)
+
1 *
sizeof(
int)
);
assert(
reinterpret_cast<
char*>(&y[
2][
0])
==
reinterpret_cast<
char*>(y)
+
2 *
sizeof(
int[
4])
+
0 *
sizeof(
int)
);
另外留个小问题给原PO思考:
给定两个 int 物件 a 和 b, 已知叙述: &a + 1 == &b 为真,
试问叙述: *(&a + 1) == *(&b) 是否也为真?
有时候我们探讨指标的时候, 不只要考虑它的型别, 同时也要考虑
我们是透过什麽途径来取得这个指标, 因为
指标代表的不仅仅只是
记忆体的位址.
--
P1389R0: Guidelines for Teaching C++ to Beginners
https://bit.ly/2GvDWKb
SG20 Education and Recommended Videos for Teaching C++
https://www.cjdb.com.au/sg20-and-videos
--
※ 发信站: 批踢踢实业坊(ptt.cc), 来自: 123.193.76.216 (台湾)
※ 文章网址: https://webptt.com/cn.aspx?n=bbs/C_and_CPP/M.1568215322.A.F75.html
1F:推 Gway: 推用心 09/11 23:24
※ 编辑: poyenc (123.193.76.216 台湾), 09/11/2019 23:30:41
2F:推 cutekid: 看 p 大发文,学习排版、上色标示重点的技巧,大推(Y) 09/12 04:09
3F:推 j5128709: 新手QQ 想知道小问题 答案是”真”嘛 09/12 11:54
4F:推 ac01965159: 感谢大大的用心,我也觉得答案是 “是” 09/12 14:26
5F:→ poyenc: 答案是: 不一定为真 09/12 16:33
6F:→ dces4212: why 09/12 18:52
7F:→ EricTCartman: = = 最好不要是virtual address 刚好一样这种答案喔 09/13 00:02
8F:推 ac01965159: 是因为如果b没被初始化,所以才是不一定吗? 09/13 10:17
9F:→ poyenc: 简单说语言的设计, 指标被允许储存除了位址以外的资讯, 像 09/13 12:35
10F:→ poyenc: 是型别等, 唯一被保证透过指标运算还可以正确取値的只有阵 09/13 12:36
11F:→ poyenc: 列元素, 也就是说同个阵列里, 元素 A 的位址可以用元素 B 09/13 12:37
12F:→ poyenc: 的位址算出来, 而且可以正确取値; 但是不同阵列间的元素就 09/13 12:38
13F:→ poyenc: 不能这样算, 因为单纯的指标运算也许会丢失必要的资讯; 另 09/13 12:39
14F:→ poyenc: 外, 物件在这个例子里被视为只有一个元素的阵列. 09/13 12:40
15F:→ poyenc: 这是语言里抽象机器 (abstract machine) 想要表达的概念, 09/13 12:42
16F:→ poyenc: 如果用实体机器的行为去解释就会发生冲突, 譬如在 64bit 09/13 12:44
17F:→ poyenc: 机器上用 sizeof(int*) 就以为指标大小是 8 个 bytes, 语 09/13 12:44
18F:→ poyenc: 言并没有规定指标内容就是虚拟记忆体的位址 09/13 12:45
19F:推 ac01965159: 抱歉有些地方不太能理解,如果单纯的指标运算会丢失 09/13 14:07
20F:→ ac01965159: 一些资讯,那电脑为什麽不会自己去判断那些物件的型 09/13 14:07
21F:→ ac01965159: 别等等资讯呢? 09/13 14:07
22F:→ poyenc: 这就要看编译器实作了, 我说的只是一种其中可能性, 而探讨 09/13 14:09
23F:→ poyenc: 语言特性的时候, 不要被实作限制想像空间, 例如用位址去查 09/13 14:11
24F:→ poyenc: 找物件型别是可行的, 不过遇到指标转型该怎麽办呢? 09/13 14:13
25F:推 ac01965159: 原来如此,感谢。 09/13 14:17