C_and_CPP 板


LINE

※ 引述《TyrionLannis (小恶魔)》之铭言: : 由於不是问程式码相关的题目,故前面叙述恕删,最近刚开始看C++ Primer, : 读到Ch3多维阵列的部分(P128),它里面给了另一种用auto来跑for loop的方式,举个 : 例子来说: : int ia[2][2]={1,2,3,4}; : //印出阵列的每个元素值 : for(auto &row : ia) : for(auto col :row){ : cout<< col << endl; : } : 书中注明,auto &row中的&不能省略,否则编译器会把row转成一个pointer(指 : 向每列的第一个元素),跑到第二个loop的时候就变成违法的指令了(原文:That : loop attempts to iterate over an int*),所以说一定要要有&才会把row转成 : 一个一维阵列,然後我就有点不懂为什麽编译器会这样做了,毕竟前面讲auto的 : 内容好像没有有提到auto声明的时候加上reference会造成这种最後type的不同, : 想请问这是C++的规定还是背後有什麽特别的哲学(或者机制)吗? 以下提到的资料都可以在 N3242 里面查询到. range-based for 顾名思义就是寻访 range 里所有元素用的 for, range 范围是由成 对的迭代器所定义, 最简单的迭代器是指标: int array[5] = { 0, 1, 2, 3, 4 }; for (int* p = array; p != (array + 5); ++p) { cout << *p << " " << endl; } 上面的例子里, array (转型成指标)开始迭代器, (array + 5) 则是结束迭代器, 你可以用这两个迭代器来寻访阵列. 这边可以不 指定 p 的型别由编译器帮我们推导: // auto is deduced to int* for (auto p = array; p != (array + 5); ++p) { cout << *p << " " << endl; } 然後再来看看 N3242 里面的 6.5.4.1, range-based for 语句: for ( for-range-declaration : expression ) statement 等同於下面的写法: { auto && __range = expression; for ( auto __begin = begin-expr, __end = end-expr; __begin != __end; ++__begin ) { for-range-declaration = *__begin; statement } } 有看到开始结束迭代器了吗? 它们就是被用来控制回圈该跑几次. 另外虽然 for-range-declaration 只有写一次, 但它会在迭代中被 用来接 *__begin 的结果. 在 6.5.4.1 最後一段有写到这些迭代器 从哪里来: "... and _RangeT is the type of the expression, and begin-expr and end-expr are determined as follows: - if _RangeT is an array type, begin-expr and end-expr are __range and __range + __bound, respectively, where __bound is the array bound." 既然有迭代器的求法, 刚刚的回圈我们也可以试着用 range-based for 来作代换: // for-range-declaration: auto i // expression: array // statement: cout << i << " " << endl; for (auto i : array) { cout << i << " " << endl; } 等同於: { auto && __range = array; for ( auto __begin = __range, __end = __range + 5; __begin != __end; ++__begin ) { auto i = *__begin; cout << i << " " << endl; } } 这边 auto&& 推导的规则比较特殊, 先不细讲. __range 物件被用 来接冒号(:)右边的叙述 (这个例子为 array), 并且视情况延长生 命周期; 再来因为 array 的型别为 int[5], __begin 的型别被推 导为 int*, __bound 也被决定为 5. auto i 能否放在前面宣告, 取决於它可不可以从 *__begin 初始化而来. 来说结论: range-based for 中如果无法藉由标准里提的方法得出 /结束迭代器, 或是宣告不合法, 就可能造成编译错误. 你的例子里, 外层回圈从 *__begin 拿到的东西是 int(&)[2] (阵 列参考), 原本用 auto& 可以参考到子阵列; 但因为宣告改成 auto 导致先转型成指标 int* 才传递给内层回圈使用. 内层回圈因为无 法透过指标来获取开始/结束迭代器 (资讯不足), 所以报错. 参考资料: Draft for C++11 (N3242) https://bit.ly/2PmiJ6J --



※ 发信站: 批踢踢实业坊(ptt.cc), 来自: 123.193.76.85
※ 文章网址: https://webptt.com/cn.aspx?n=bbs/C_and_CPP/M.1544204166.A.F9A.html
1F:推 TyrionLannis: 感谢!排版跟解释都很清楚! 12/08 09:49
2F:→ TyrionLannis: 另外想请问,所以auto &跟auto 两个回传的结果不同 12/08 09:50
3F:→ TyrionLannis: (前者阵列,後者指标)背後的机制大概是什麽,是类似 12/08 09:52
int i = 0; auto vi = i; // auto deduced to int, vi's type is int auto& ri = i; // auto deduced to int, ri's type is int& auto pi = &i; // auto deduced to int*, pi's type is int* auto* pi2 = &i; // auto deduced to int, pi2's type is int* auto 是起到占位的作用, 占的部分由编译器帮你推导型别, 由於 auto& 占的部分变少, 所以不管 auto 的部分编译器怎麽填, 最後 物件的型别还是参考. 但是 array 并不像其他内建型别一样可以当 左值: int array[5]; int other_array[10]; array = other_array; // error auto pa = array; // auto deduced to int*, pa's type is int* auto& ra = array; // auto deduced to int[5], ra's type is int(&)[5] 所以当你单纯想用 auto 来接阵列时, 会发生 array-to-pointer 转换, 对型别为 T[N] 的阵列 A 而言, 得到的值会是 &A[0], 型别 则是 T* (T 可能包含 const)
4F:推 TyrionLannis: ?另外如果算是C++新手太钻这些点会不会没什麽帮助QQ 12/08 09:57
不会, 需要钻的东西太多了
5F:→ sarafciel: 原PO你先把C++的reference搞懂吧 12/08 10:08
可以用 C++ Insights 来看 range-based for 展开的模样: link: https://cppinsights.io ※ 编辑: poyenc (123.193.76.85), 12/10/2018 01:52:22







like.gif 您可能会有兴趣的文章
icon.png[问题/行为] 猫晚上进房间会不会有憋尿问题
icon.pngRe: [闲聊] 选了错误的女孩成为魔法少女 XDDDDDDDDDD
icon.png[正妹] 瑞典 一张
icon.png[心得] EMS高领长版毛衣.墨小楼MC1002
icon.png[分享] 丹龙隔热纸GE55+33+22
icon.png[问题] 清洗洗衣机
icon.png[寻物] 窗台下的空间
icon.png[闲聊] 双极の女神1 木魔爵
icon.png[售车] 新竹 1997 march 1297cc 白色 四门
icon.png[讨论] 能从照片感受到摄影者心情吗
icon.png[狂贺] 贺贺贺贺 贺!岛村卯月!总选举NO.1
icon.png[难过] 羡慕白皮肤的女生
icon.png阅读文章
icon.png[黑特]
icon.png[问题] SBK S1安装於安全帽位置
icon.png[分享] 旧woo100绝版开箱!!
icon.pngRe: [无言] 关於小包卫生纸
icon.png[开箱] E5-2683V3 RX480Strix 快睿C1 简单测试
icon.png[心得] 苍の海贼龙 地狱 执行者16PT
icon.png[售车] 1999年Virage iO 1.8EXi
icon.png[心得] 挑战33 LV10 狮子座pt solo
icon.png[闲聊] 手把手教你不被桶之新手主购教学
icon.png[分享] Civic Type R 量产版官方照无预警流出
icon.png[售车] Golf 4 2.0 银色 自排
icon.png[出售] Graco提篮汽座(有底座)2000元诚可议
icon.png[问题] 请问补牙材质掉了还能再补吗?(台中半年内
icon.png[问题] 44th 单曲 生写竟然都给重复的啊啊!
icon.png[心得] 华南红卡/icash 核卡
icon.png[问题] 拔牙矫正这样正常吗
icon.png[赠送] 老莫高业 初业 102年版
icon.png[情报] 三大行动支付 本季掀战火
icon.png[宝宝] 博客来Amos水蜡笔5/1特价五折
icon.pngRe: [心得] 新鲜人一些面试分享
icon.png[心得] 苍の海贼龙 地狱 麒麟25PT
icon.pngRe: [闲聊] (君の名は。雷慎入) 君名二创漫画翻译
icon.pngRe: [闲聊] OGN中场影片:失踪人口局 (英文字幕)
icon.png[问题] 台湾大哥大4G讯号差
icon.png[出售] [全国]全新千寻侘草LED灯, 水草

请输入看板名称,例如:Gossiping站内搜寻

TOP