p是一個(gè)變量坯台,那么想怎么給它賦值秆剪,就可以怎么給它賦值,但是C語言是強(qiáng)類型語言本砰,如果賦值類型不匹配就會(huì)報(bào)錯(cuò)碴裙,如果賦值類型雖然不匹配但是可以默認(rèn)轉(zhuǎn)換,那么編譯可以通過但是會(huì)給警告点额。
比如可以這樣做:
p = 0;
p = 1;
都沒問題舔株,因?yàn)閜是一個(gè)指針,數(shù)字類型可以賦值給指針还棱,但是給它賦值是什么含義载慈?含義不對(duì),即使編譯通過珍手,運(yùn)行也會(huì)報(bào)錯(cuò)的办铡。
在第一個(gè)文件里存在一個(gè)錯(cuò)誤是i沒有初始化,必須初始化為0或者1或者什么數(shù)字琳要,不然一定會(huì)出錯(cuò)料扰,下面的討論假設(shè)初始化為1:
在第一個(gè)文件里面,p是一個(gè)變量焙蹭,這個(gè)變量的類型是指針,這個(gè)指針?biāo)赶虻氖恰白址羔槨薄?br>
那么既然是變量嫂伞,就可以讀和寫孔厉,所以可以賦值拯钻;由于是強(qiáng)類型,所以只要是指針類型或者能轉(zhuǎn)換成指針類型的就可以賦值撰豺;由于它是一個(gè)指向”字符串指針“的指針粪般,所以給它賦值的時(shí)候,如果所賦值對(duì)象類型不是指向”字符串指針“污桦,一般編譯器都會(huì)提示一下的亩歹。
name也是一個(gè)變量,這個(gè)變量的類型也是指針凡橱,這個(gè)指針?biāo)赶虻氖恰弊址當(dāng)?shù)組“小作。在C語言里面,數(shù)組其實(shí)就是指針稼钩,但是是常量顾稀,不能寫,只能讀坝撑,所以name這個(gè)變量可以讀可以寫静秆,但是它指向的內(nèi)存因?yàn)槭浅A繑?shù)組所以只能讀不能寫,除此以外name和p沒有任何區(qū)別巡李。
因此抚笔,name + i是指針加法,name + i還是一個(gè)指針侨拦,指向字符串?dāng)?shù)組殊橙,只不過所指內(nèi)容和name不一樣,是name的”下一個(gè)“阳谍,所以p = name +i 屬于兩個(gè)同類型指針賦值蛀柴,沒有任何問題;
p=name[i]和p = name + i是不一樣的矫夯,name定義為一個(gè)數(shù)組鸽疾,name[i]是取數(shù)組的第i個(gè)元素,相當(dāng)于(name + i)训貌。(感謝 @單車戀人 指正)但是這個(gè)賦值還是會(huì)成功的制肮,因?yàn)閜是指向字符指針的指針,本質(zhì)還是一個(gè)指針递沪,name是一個(gè)字符指針的數(shù)組豺鼻,取出的第i個(gè)元素是一個(gè)字符指針,賦值可以成功款慨,但是類型不一樣(字符指針的指針和字符指針)儒飒。
p=name+i 性質(zhì)就變了,就是把name所指向的內(nèi)容加1檩奠,然后賦值給p桩了,name作為一個(gè)指針指向的是 字符串?dāng)?shù)組附帽,還好字符串?dāng)?shù)組也是指針(指針與數(shù)組在C語言里面都按照指針處理),所以賦值可以成功井誉,但是會(huì)給出編譯警告蕉扮,因?yàn)楫吘顾麄儾皇窍嗤愋停琾是指向指針的指針颗圣,賦值內(nèi)容是數(shù)組喳钟,相當(dāng)于指針。
p=name+i 在這里name + i和前面的分析一樣在岂,它和p是相同類型奔则,是指針的指針;而p是一個(gè)指針了洁段,這時(shí)候要把它賦給p還是指針之間的賦值应狱,編譯可以通過,但是指針類型不同祠丝,也會(huì)給出編譯警告疾呻。這里要注意,p是一個(gè)變量沒問題写半,但是p能不能賦值取決于p指向的空間岸蜗,如果p指向一塊可讀不可寫的內(nèi)存,程序運(yùn)行到這里會(huì)運(yùn)行時(shí)退出叠蝇;如果p指向一塊可寫的內(nèi)存而且有足夠的空間璃岳,那么就沒問題。
抱歉第二個(gè)文件就不這樣詳細(xì)分析了悔捶,沒想到這么長(zhǎng)太累了铃慷,不過思路是類似的。寫了段代碼供參考蜕该,用的是64位Linnux下Gcc編譯器犁柜,所以打印指針的時(shí)候轉(zhuǎn)換成long類型,如果是32位轉(zhuǎn)換成int類型堂淡,這里p指向的是一塊不可寫內(nèi)存馋缅,所以在*p = name + i那里會(huì)運(yùn)行時(shí)報(bào)錯(cuò):
include "stdio.h"
int main()
{
char *name[]={"abc","edf","ghi"};
char *p;
int i = 1;
p = name[i];
printf("%ld\n", (long)p);
p = name + i;
printf("%ld\n", (long)p);
p = name + i; /此處執(zhí)行會(huì)出錯(cuò)/
printf("%ld\n", (long)p);
/*
int a[4]={1,3,5,7};
int (*p)[4];
p = 0;
p = &a;
printf("%ld\n", (long)p);
p = a;
printf("%ld\n", (long)p);
printf("%ld\n", (long)*p);
printf("%ld\n", (long)**p);
*/
}