const char str1[] = "abc"const char str2[] = "abc"为什么str1,str2不相等?请问大牛
const char str1[] = "abc";const char str2[] = "abc";为什么str1,str2不相等???请教大牛
在vs下跑程序,分别添加监视,str1,str2,p1,p2. 为什么str1,str2不相同,而p1,p2相同?以下为监视结果:
+ str1 0x0017fd40 "abc" const char [4]
+ str2 0x0017fd34 "abc" const char [4]
+ p1 0x0101573c "abc" const char *
+ p2 0x0101573c "abc" const char *
------解决方案--------------------
str1和str2 是在堆栈中定义的一段内存 这是两个东西 所以是不同的地方
p1 和 p2 是编译器吧或者什么你不知道的东西在一个神秘的地方开辟了一个空间放了 abc\0
而p1 和p2 这样定义的东西是不能修改的 所以编译器或者什么东西看到 p2定义的时候发现 嘿!我见过它
然后就把p1 那个地方的abc 中a的地址给了 p2
------解决方案--------------------
char str1[] 和char *p 这两者分配的地址位置不一样
一个在栈上,一个在常量区
p1,p2相同----编译器的优化
------解决方案--------------------
const char str1[] = "abc";
const char str2[] = "abc";
这是在栈上分配的 从高地址到低地址分配 当然不相等
而const char *p1 = "abc";
const char *p2 = "abc";
这两个指针指向的是同一地址 编译器认为p1指向的“abc”和p2指向的“abc”是属于同一常量 很多书上都有这个提示
------解决方案--------------------
char* str1 = "abc";
char* str2 = "abc";
一般有 str1 == str2
------解决方案--------------------
VC调试(TC或BC用TD调试)时按Alt+8、Alt+6和Alt+5,打开汇编窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应内存和寄存器变化,这样过一遍不就啥都明白了吗。
对VC来说,所谓‘调试时’就是编译连接通过以后,按F10或F11键单步执行一步以后的时候,或者在某行按F9设了断点后按F5执行停在该断点处的时候。
(Linux或Unix下可以在用GDB调试时,看每句C对应的汇编并单步执行观察相应内存和寄存器变化。)
想要从本质上理解C指针,必须学习汇编以及C和汇编的对应关系。
从汇编的角度理解和学习C语言的指针,原本看似复杂的东西就会变得非常简单!
指针即地址。“地址又是啥?”“只能从汇编语言和计算机组成原理的角度去解释了。”
这辈子不看内存地址和内存值;只画链表、指针示意图,画堆栈示意图,画各种示意图,甚至自己没画过而只看过书上的图……能从本质上理解指针、理解函数参数传递吗?本人深表怀疑!
这辈子不种麦不收麦不将麦粒拿去磨面;只吃馒头、吃面条、吃面包、……甚至从没看过别人怎么蒸馒头,压面条,烤面包,……能从本质上理解面粉、理解面食吗?本人深表怀疑!!
提醒:
“学习用汇编语言写程序”
和
“VC调试(TC或BC用TD调试)时按Alt+8、Alt+6和Alt+5,打开汇编窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应内存和寄存器变化,这样过一遍不就啥都明白了吗。
(Linux或Unix下可以在用GDB调试时,看每句C对应的汇编并单步执行观察相应内存和寄存器变化。)
想要从本质上理解C指针,必须学习C和汇编的对应关系。”
不是一回事!
不要迷信书、考题、老师、回帖;
要迷信CPU、编译器、调试器、运行结果。
并请结合“盲人摸太阳”和“驾船出海时一定只带一个指南针。”加以理解。
任何理论、权威、传说、真理、标准、解释、想象、知识……都比不上摆在眼前的事实!
有人说一套做一套,你相信他说的还是相信他做的?
其实严格来说这个世界上古往今来所有人都是说一套做一套,不是吗?
不要写连自己也预测不了结果的代码!
------解决方案--------------------
3楼很好,你可以去看看面试试题!!里面很多这个。。
数组是按照地址排列的!!从低到高
指针只是指向的地址。。
------解决方案--------------------
str1和str2在栈里,p1和p2在资源里,一般是PE文件的.rdata段
- C/C++ code
#include<stdio.h> int main() { const char str1[] = "abc"; const char str2[] = "abc"; const char *p1 = "abc"; const char *p2 = "abc"; return 0; }
在vs下跑程序,分别添加监视,str1,str2,p1,p2. 为什么str1,str2不相同,而p1,p2相同?以下为监视结果:
+ str1 0x0017fd40 "abc" const char [4]
+ str2 0x0017fd34 "abc" const char [4]
+ p1 0x0101573c "abc" const char *
+ p2 0x0101573c "abc" const char *
------解决方案--------------------
str1和str2 是在堆栈中定义的一段内存 这是两个东西 所以是不同的地方
p1 和 p2 是编译器吧或者什么你不知道的东西在一个神秘的地方开辟了一个空间放了 abc\0
而p1 和p2 这样定义的东西是不能修改的 所以编译器或者什么东西看到 p2定义的时候发现 嘿!我见过它
然后就把p1 那个地方的abc 中a的地址给了 p2
------解决方案--------------------
char str1[] 和char *p 这两者分配的地址位置不一样
一个在栈上,一个在常量区
p1,p2相同----编译器的优化
------解决方案--------------------
const char str1[] = "abc";
const char str2[] = "abc";
这是在栈上分配的 从高地址到低地址分配 当然不相等
而const char *p1 = "abc";
const char *p2 = "abc";
这两个指针指向的是同一地址 编译器认为p1指向的“abc”和p2指向的“abc”是属于同一常量 很多书上都有这个提示
------解决方案--------------------
char* str1 = "abc";
char* str2 = "abc";
一般有 str1 == str2
------解决方案--------------------
VC调试(TC或BC用TD调试)时按Alt+8、Alt+6和Alt+5,打开汇编窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应内存和寄存器变化,这样过一遍不就啥都明白了吗。
对VC来说,所谓‘调试时’就是编译连接通过以后,按F10或F11键单步执行一步以后的时候,或者在某行按F9设了断点后按F5执行停在该断点处的时候。
(Linux或Unix下可以在用GDB调试时,看每句C对应的汇编并单步执行观察相应内存和寄存器变化。)
想要从本质上理解C指针,必须学习汇编以及C和汇编的对应关系。
从汇编的角度理解和学习C语言的指针,原本看似复杂的东西就会变得非常简单!
指针即地址。“地址又是啥?”“只能从汇编语言和计算机组成原理的角度去解释了。”
这辈子不看内存地址和内存值;只画链表、指针示意图,画堆栈示意图,画各种示意图,甚至自己没画过而只看过书上的图……能从本质上理解指针、理解函数参数传递吗?本人深表怀疑!
这辈子不种麦不收麦不将麦粒拿去磨面;只吃馒头、吃面条、吃面包、……甚至从没看过别人怎么蒸馒头,压面条,烤面包,……能从本质上理解面粉、理解面食吗?本人深表怀疑!!
提醒:
“学习用汇编语言写程序”
和
“VC调试(TC或BC用TD调试)时按Alt+8、Alt+6和Alt+5,打开汇编窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应内存和寄存器变化,这样过一遍不就啥都明白了吗。
(Linux或Unix下可以在用GDB调试时,看每句C对应的汇编并单步执行观察相应内存和寄存器变化。)
想要从本质上理解C指针,必须学习C和汇编的对应关系。”
不是一回事!
不要迷信书、考题、老师、回帖;
要迷信CPU、编译器、调试器、运行结果。
并请结合“盲人摸太阳”和“驾船出海时一定只带一个指南针。”加以理解。
任何理论、权威、传说、真理、标准、解释、想象、知识……都比不上摆在眼前的事实!
有人说一套做一套,你相信他说的还是相信他做的?
其实严格来说这个世界上古往今来所有人都是说一套做一套,不是吗?
不要写连自己也预测不了结果的代码!
------解决方案--------------------
3楼很好,你可以去看看面试试题!!里面很多这个。。
数组是按照地址排列的!!从低到高
指针只是指向的地址。。
------解决方案--------------------
str1和str2在栈里,p1和p2在资源里,一般是PE文件的.rdata段