C++中char[]能修改char*卻不行
少扯淡沒(méi)用的,直接上代碼
int main(int argc, char *argv[])
{
char p[74] = "abcefghijkmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmm";
char *a = "abcefghijkmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmm";
printf("%s,%s",p,a);
}
這誰(shuí)都能看明白,最終輸出兩次abcefghijkmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmm,沒(méi)問(wèn)題
把代碼再改改
int main(int argc, char *argv[])
{
char p[74] = "abcefghijkmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmm";
char *a = "abcefghijkmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmm";
p[8]= 'd';
a[8] = 'd';
//printf("%s,%s",p,a);
}
運(yùn)行,報(bào)錯(cuò)

錯(cuò)誤指向了a[8] = ‘d'

錯(cuò)誤指向第12行,為嘛尼?
看匯編

可以看到變量p,和a 都是采用了同樣的方式
d使用了
mov esi,offset string "abcefghijkmmmmmmmmmmmmmmmmmmmmmm"...
將字符串的偏移地址,賦值到esi寄存器
a使用了
mov dword ptr [a],offset string "abcefghijkmmmmmmmmmmmmmmmmmmmmmm"...
將字符串的偏移地址,賦值到a變量所在地址
仔細(xì)看圖 這兩句對(duì)字符串的取址操作,來(lái)源都是一個(gè)地方,01007500h,也就是程序數(shù)據(jù)段在內(nèi)存中的位置

既然兩個(gè)操作都是對(duì)同一個(gè)字符串的操作,為什么有的可以修改,有的不行?
這里面有個(gè)關(guān)于編譯文件后程序的存儲(chǔ)問(wèn)題,如上例
變量a,p他們都是操作相同的字符串,兩個(gè)字符串完全相同,所以,程序編譯后,生成的文件內(nèi),完全沒(méi)有必要保存兩個(gè)相同的內(nèi)容,只保留一個(gè)便可以,所以,你的程序,不管多少次使用這個(gè)字符串,實(shí)際上都是從一個(gè)地方引用的,這就是,這兩句代碼
mov esi,offset string "abcefghijkmmmmmmmmmmmmmmmmmmmmmm"... mov dword ptr [a],offset string "abcefghijkmmmmmmmmmmmmmmmmmmmmmm"...
的意思。
但這就出現(xiàn)了一個(gè)問(wèn)題,如果兩個(gè)或者多個(gè)變量都用了同一個(gè)字符串,然后最終要的,還進(jìn)行了修改,結(jié)果就是,所有引用這個(gè)字符串的變量,都變了,所以
a[8] = 'd';
要直接修改數(shù)據(jù)段,就報(bào)錯(cuò)了
但是p可以,為什么,應(yīng)為數(shù)組的處理是不一樣的,看代碼
mov esi,offset string "abcefghijkmmmmmmmmmmmmmmmmmmmmmm"... (0977588h) lea edi,[p] rep movs dword ptr es:[edi],dword ptr [esi]
首先,把字符串的地址給了esi,然后把p地址給了edi,
然后,通過(guò)rep movs 循環(huán)執(zhí)行,吧[esi]處的字符,賦值給[edi],也就是把字符串復(fù)制一份到p
所以,你操作的
p[8] ='d';
實(shí)事上是操作了一個(gè)新的字符串,不是數(shù)據(jù)段中的那個(gè)字符串,
再看關(guān)于a的操作
mov dword ptr [a],offset string "abcefghijkmmmmmmmmmmmmmmmmmmmmmm"... mov eax,1 shl eax,3 mov ecx,dword ptr [a] mov byte ptr [ecx+eax],64h
首先把1給了eax,然后執(zhí)行位移3,把EAX,變成8,把a(bǔ)的地址,也就是字符串的地址給了ecx,然后吧64h也就是d,賦值給[ecx+eax] 那個(gè)位置,也就是j的位置,因而你操作的是數(shù)據(jù)段中的那個(gè)字符串,就是上面的原因,系統(tǒng)會(huì)阻止你修改數(shù)據(jù)段,因而報(bào)錯(cuò)
到此這篇關(guān)于C++中char[]能修改char*卻不行的文章就介紹到這了,更多相關(guān)C++ char*不能修改內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++對(duì)象的淺復(fù)制和深復(fù)制詳解及簡(jiǎn)單實(shí)例
這篇文章主要介紹了C++對(duì)象的淺復(fù)制和深復(fù)制詳解及簡(jiǎn)單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-04-04
C++中產(chǎn)生臨時(shí)對(duì)象的情況及其解決方案
這篇文章主要介紹了C++中產(chǎn)生臨時(shí)對(duì)象的情況及其解決方案,以值傳遞的方式給函數(shù)傳參,類型轉(zhuǎn)換以及函數(shù)需要返回對(duì)象時(shí),并給對(duì)應(yīng)給出了詳細(xì)的解決方案,通過(guò)圖文結(jié)合的方式講解的非常詳細(xì),需要的朋友可以參考下2024-05-05
詳解C++成員函數(shù)的override和final說(shuō)明符的用法
這篇文章主要介紹了詳解C++成員函數(shù)的override和final說(shuō)明符的用法,分別用于重寫(xiě)和禁止繼承類,要的朋友可以參考下2016-01-01
c++ minicsv庫(kù)的編譯錯(cuò)誤與解決方案
有一個(gè)項(xiàng)目需要寫(xiě)csv文件以呈現(xiàn)數(shù)據(jù)。Github上有一個(gè)關(guān)于csv的輕量級(jí)讀寫(xiě)庫(kù)minicsv,于是下載之。但是編譯example時(shí)出現(xiàn)了以下問(wèn)題2016-11-11
C語(yǔ)言實(shí)現(xiàn)食堂就餐管理系統(tǒng)(帶鏈表)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)食堂就餐管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11

