標(biāo)題: 關(guān)于C語言的指針,與教材不一樣的解釋 [打印本頁]
作者: 慢慢思考 時間: 2023-4-20 17:27
標(biāo)題: 關(guān)于C語言的指針,與教材不一樣的解釋
C語言中為什么要設(shè)計指針這個東西,這個東西的設(shè)計原理是什么,本來就是個很簡單的東西,小學(xué)生思維就夠了。還有,這些原理之類的,對于使用C來編程的人來說,也是必須完全了解的,了解的深度與你編程時的輕松度密切相關(guān),所以,弄清這個對你很重要。
總有人在問指針是干什么用的,那我先說說我的理解:獲取某個數(shù)據(jù)所在單元的地址值,由此推算出其它數(shù)據(jù)所在單元的地址值,以用于找到這個數(shù)據(jù)所在的存儲位置。
計算機(jī)編程,可以說,基本就是對存于存儲器中的數(shù)據(jù)進(jìn)行操作,這就牽扯到一個最重要的問題:你現(xiàn)在要操作的數(shù)據(jù),存在哪個地方?
想要弄清這個問題,最根本的方法,是了解芯片的設(shè)計和工作基本原理與基本結(jié)構(gòu)組成。當(dāng)然,如果知道存儲器的大概結(jié)構(gòu),也算是有了一個基礎(chǔ)。這里我不講這么多,因為學(xué)單片機(jī)的人,基本的知識應(yīng)該還是有的,不是那種不與硬件打交道的程序員,所以不需我多講。
C與匯編有兩個根本的區(qū)別,造成了C中指針的產(chǎn)生:一個是數(shù)據(jù)所存儲的地址是由C編輯器自動分配的,程序員沒自主權(quán),所以一個數(shù)據(jù)存儲的地址,程序員是不知道的;二是C編程中所用到的數(shù)據(jù)的長度與存儲單元容量的不一致導(dǎo)致一個數(shù)據(jù)可能會占據(jù)數(shù)個存儲單元,由此造成數(shù)據(jù)存儲地址值的不連續(xù),這個現(xiàn)象造成的原因在于,不管你是8位機(jī)、16位機(jī)還是其它位機(jī),一個存儲單元的位數(shù)都是8位,這個是芯片設(shè)計就如此,原因自然主要是兼容問題而不得不做出的選擇。本段所述的現(xiàn)象,在下面的比喻之中會進(jìn)行解釋。
用C語言編寫的程序,在程序運行中,你所要處理的數(shù)據(jù)存在哪,你是必須知道的,否則你找不到這個數(shù)據(jù)在哪,你的程序就沒法編下去了;如果你把數(shù)據(jù)所在的位置給弄錯了,那這個程序在運行中就得不到你想要的結(jié)果。
有關(guān)數(shù)據(jù)存在哪,及如何找到這個位置,我打如下一個比方:
我們知道,賓館的房間都是有編號的,而且是用數(shù)字來編寫的,很統(tǒng)一,一個房間一個編號;每個房間的大小都是一樣的,也就是說其容量是一樣的。賓館來了一家一家的,住哪間?賓館前臺來安排,一家人數(shù)少,可能就是一間,人數(shù)多了,可能就得幾間了,也就是說,同樣是一家,可能他只占據(jù)一個房間編號,也有可能占據(jù)數(shù)個房間編號(這就造成了地址編號的不連續(xù)),這個數(shù)據(jù),只有前臺知道。如果你想去找到哪一家,你怎么找?你一個房間一個房間地去看單子也好看現(xiàn)場也好,都不是個辦法,一般情況下,每一家都是有個名字的,如張家、李家、王家、陳家等等的,你告訴前臺,前臺根據(jù)這個名字查一下,就會把這家的第一個房間的編號(首地址)給了你,而你在事先知道他們家人數(shù)及賓館每間房能住的人數(shù)的情況下,也就知道了他們家占幾間房。這個比方里,與C語言中的情況極其相似,就是存儲內(nèi)容物的位置都是用統(tǒng)一數(shù)字編號來標(biāo)識的,存儲內(nèi)容的每一個單元(房間)的容量都是固定一致的,每一個(組)數(shù)據(jù)(一個家庭)都是有一個名字的,單個數(shù)據(jù)的長度(某個家庭人數(shù))你事先是知道的,內(nèi)容物的存儲位置分配是由他人分配而你事先是不知道的,其占據(jù)幾個存儲單元數(shù)量你是事先知道的。
同樣的原理,在C語言中,要找到一個數(shù)據(jù)的首地址,我們先要把這個數(shù)據(jù)的名字給編輯器(前臺),編輯器就會依據(jù)這個名字把這個數(shù)據(jù)存儲單元的首地址給你,這個動作,就是在這個數(shù)據(jù)的名字前面加一個“&”,其首地址值就出來了。
然后這里就又有了一個問題,這個查出的地址值,放在什么地方?賓館是可以將地址值寫在紙上,那計算機(jī)呢?它能放這個數(shù)據(jù)的地方,還是只有存儲器,當(dāng)然,你也可以為存儲這些個地址而單獨在芯片中設(shè)計一個專門的區(qū)域,那你可以算是創(chuàng)新,但是,現(xiàn)有的芯片怎么辦?沒辦法,還是只能存于現(xiàn)有的存儲器之中,于是,C編輯器還得給一個房間(存儲單元)專門用于存儲這個地址值,然后,這個單元編號是多少?程序員又不知道了,因為又是編輯器自動分配的,所以,沒法,又得給這個取個名字以便查找利用,所以,在將要存入的內(nèi)容存入之前,得先對要存入的內(nèi)容進(jìn)行定義(取名)一個變量,這樣我們以后就可以用名字直接找到這個內(nèi)容了。
下面就談?wù)?/font>C中有關(guān)獲取地址與使用地址的一些規(guī)則,有關(guān)這個概念的名詞,現(xiàn)有教材都稱之為指針,本人對這個名詞比較不感冒,這個名詞顯然沒能與“地址”一詞緊密聯(lián)系,于是不易理解不易記。ìF(xiàn)有計算機(jī)教材中,類似槽點很多):
存放入單元的內(nèi)容要要先取個名字,也就是所謂的定義(前面已經(jīng)解釋過),這個因為指針變量的功能不同,所以定義變量名時,也要給出區(qū)別,讓編輯器知道這個變量的用途,也讓讀程序的人知道這個變量是用于存儲地址的,也就是所謂的指針變量。其進(jìn)行區(qū)別的規(guī)則是變量名前面加一個“*”以示區(qū)別,如果不加會怎么樣?你照樣可以將取得的地址值存入其中,但是,你用這個變量去取這個地址所指向數(shù)據(jù)時,程序在編譯時就報錯了,這個現(xiàn)象,只是編輯器的硬性規(guī)定,也是有道理的。其實編輯器在分配指針變量在存儲器中的位置時,也沒做什么特殊處理,它與其它變量都是統(tǒng)一分配地址的,基本也是順序分配,誰先使用誰先排前面(不是誰先定義誰先排前面),一點也不特殊。
在使用指針來取得存儲單元中的數(shù)據(jù)時,C的規(guī)定也是在指針變量名前加一個“*”以示這個是用地址值來取得數(shù)據(jù),如a=*p,它的含意是:P所表示的存儲單元中的數(shù)據(jù),是一個地址值,a=*p就是把這個地址值所指向的單元中的數(shù)據(jù)送給a。其實吧,個人覺得這個“*”符號用得不好,改為“@”可能更讓人容易理解,這個“@”有“在”的意思,然后我們就可以理解為“把在這個p值所指向的單元中的值,把它給a”。如果是這樣,很多運算公式就很容易理解了,如@p++就是操作“在”p值為地址所指向的單元中的數(shù)據(jù),這個具體操作就是將其加1,而p++就是p值加1個數(shù)據(jù)長度值以將p值變?yōu)橄乱粋數(shù)據(jù)的地址值(不是地址值本身直接增加1,這個在后面再解釋),也就是說,只要在p前面加了@,那要操作的對象就并不是p本身,而是它所指向的單元,這個一定要牢記不是操作p本身,這樣你在編程時就不容易出錯了。
指針變量是可以直接賦一個常量值的,但是,這個值必須是你所要用到的單元的地址值,如果不是,程序編譯時不會報錯,但程序運行時肯定是亂了,也就是說,前提是你知道這個數(shù)據(jù)的絕對地址值。
接下來的一個問題,就是指針變量的長度,既然是表示地址,那它就是一個沒有負(fù)號的整數(shù),其長度最長也就是存儲器地址值的長度,如51機(jī)的RAM,其地址就只有八位,所以,在定義時,它的指針長度也就一個單元,多了浪費。
還有一個規(guī)則,指針變量的加減運算,是與其所指向的數(shù)據(jù)的長度綁定的,這個是在編譯處理時用到的,意思是其盡管這個指針變量也是一個數(shù)據(jù),但它在做加減運算時,例如p=p+1,并不是p值直接加1,而是加一個其所指向的數(shù)據(jù)的長度址,以直接變?yōu)橄乱粋數(shù)據(jù)的首地址值,這個處理,并不是在程序運行中進(jìn)行的,而是在編譯時進(jìn)行的,所以指針變量在使用之前一定要先定義,定義之后,編輯器才會知道這個是個指針變量,然后這樣做處理。
還有一個教材中沒說清楚的問題,這個所謂的指針,到底是只針對RAM,還是包括ROM?這個有興趣的朋友可以研究一下,方法也不難。
以上所說,其實都可以用編輯器調(diào)試中的匯編窗口來證實。
以上純屬個人理解,如有錯誤與不當(dāng),萬請指出。
作者: 慢慢思考 時間: 2023-4-20 20:06
C的指針問題,對于匯編來說,其實是個非常簡單的問題,就是用地址值來尋址的方式。當(dāng)然,在現(xiàn)有的匯編教材中,也沒有具體的說什么用地址值來來尋址的這種說法,在匯編中,除了立即尋址方式,其它的方式,都是用到地址值的尋址方式。事實上,只要是用到RAM,就得知道其地址才能進(jìn)行存取操作。
說起來,在C中,C的指針的操作,都只是取操作,前面已經(jīng)說得很清楚了,其存操作,是由編輯器來進(jìn)行的,程序員無權(quán)直接操作。也當(dāng)然,如果程序員知道編輯器的分配規(guī)則,那么程序員想把某數(shù)在編輯器分配時分配在某單元中,也是可以做到的,這個就是對芯片和C通透了。
作者: 慢慢思考 時間: 2023-4-20 20:48
這個帖子內(nèi)容,如果配上匯編窗口的實驗,應(yīng)該也是一篇不錯的文檔吧?
作者: 慢慢思考 時間: 2023-4-20 21:27
知道了指針的原理,那我們完全可以在編程時很容易地確定任一數(shù)據(jù)的存儲地址,以及任一數(shù)據(jù)中某一字節(jié)的地址,哪怕各數(shù)據(jù)的字節(jié)長度并不一致。
作者: Hephaestus 時間: 2023-4-20 22:04
C與匯編有兩個根本的區(qū)別,造成了C中指針的產(chǎn)生:一個是數(shù)據(jù)所存儲的地址是由C編輯器自動分配的,程序員沒自主權(quán),所以一個數(shù)據(jù)存儲的地址,程序員是不知道的;
這話說明你的匯編沒學(xué)好。以Keil C51自帶的A51匯編器為例:
這是大部分人常用的把變量Datum定義到50h地址上,嚴(yán)格地說是錯誤的。正確的寫法是這樣:
但是這個寫法把Datum定義到50h地址上,如果是純匯編編程那么沒有問題,如果是匯編和c語言混合編程,這種寫法會和BL51.exe分配內(nèi)存的方法發(fā)生沖突,不會出現(xiàn)編譯錯誤,也不會出現(xiàn)運行錯誤,但是可能會浪費RAM空間,不能實現(xiàn)資源的高效利用。那么最合理的寫法是這樣的:
給Datum變量保留一個字節(jié)的空間,在編譯期間,編譯器也是不知道Datum地址是多少,只有最后連接完成,才能確定,可以查看連接器生成的map文件來知道Datum的地址到底是幾。
作者: Y_G_G 時間: 2023-4-20 22:10
純技術(shù)問題不要搞這種花里胡哨的東西,對于學(xué)習(xí)意義不大
什么"老王教你用示波器","七天學(xué)會單片機(jī)"......之類的,往往是沒什么用的,說它不行,它好像有點用,說它行,好像用處不大
老外的書就簡單,往往枯燥,生澀難懂,但往往能真正的學(xué)到東西,很多時候能當(dāng)工具書來查資料
這是對指針的描述,幾行代碼,加黑字,很快就能理解指針了
51hei截圖20230420220523.png (1.11 MB, 下載次數(shù): 98)
下載附件
2023-4-20 22:10 上傳
作者: 溫xyz 時間: 2023-4-20 22:29
這個更通俗易懂。用@更容易理解,但現(xiàn)在都在用*,干脆就把*當(dāng)@吧。

作者: pdwdzz 時間: 2023-4-20 22:36
指針的根源是接尋址寄存器,沒有這個寄存器的MCU無法用指針。
作者: Hephaestus 時間: 2023-4-20 22:40
有indirect/index register也不好用,比如51強(qiáng)行上指針代碼密度和執(zhí)行效率都是直線下降,得不償失。
作者: 東莞朋友 時間: 2023-4-21 09:52
其實就是間接尋址方式,給這么多教材解釋得繞來繞去
作者: 慢慢思考 時間: 2023-4-21 11:59
我寫的是有關(guān)單片機(jī)底層的工作原理和運行過程的東西,是比較接近其本質(zhì)的東西。如果不懂這些,在C中那有些動作可能是不知道如何去做的,比如說,結(jié)合指針操作把一個四字節(jié)變量中的第三個字節(jié)的內(nèi)容快速單獨地讀出來。
在單片機(jī)運行程序中,i=*p,這個語句執(zhí)行時操作的數(shù)據(jù)并不是p本身,而是另一個數(shù)據(jù):這個數(shù)據(jù)的首地址是p所指向的單元。
指針的本質(zhì)是操作地址,所以,了解單片機(jī)芯片存儲結(jié)構(gòu)及地址之類的概念,才能從底層上深入了解指針。
作者: Hephaestus 時間: 2023-4-21 12:41
真要研究指針,怎么也要找個正經(jīng)的c語言編譯器吧,如STM32用的三大編譯器IAR、GCC和MDK,找個天生殘疾的%的不正規(guī)c語言編譯器搞指針是%嗎?
作者: Y_G_G 時間: 2023-4-21 13:35
1: 匯編不需要C指針的功能,用法不一樣
2: 匯編的指針在不同單片機(jī)中,用法是不一樣的,你說的只能是8051的匯編,別的匯編用的不是這方法,像PIC的匯編壓根就沒有像8051這種指針尋址方式,它是由固定文件寄存器來進(jìn)行操作的,你的說法很容易誤導(dǎo)別人
3:C語言很少有人去動內(nèi)存,都是由編譯器自動分配,閑著沒事干才會去動內(nèi)存
作者: 15878397332 時間: 2023-4-21 14:14
指針的根源是接尋址寄存器
作者: lyonkon 時間: 2023-4-21 18:49
c51地址指針xdat區(qū)是兩個字節(jié)。變量的值存儲位置是編譯器決定的,但程序員要存儲的值是可以存到指定位置的。開機(jī)時候需要清理64k內(nèi)存,如果沒有地址指針,程序員就需要寫64k行代碼。0000h=0,,,,ffffh=0 .有地址指針只需要地址加一。
作者: wydev 時間: 2023-4-22 15:04
一句話,指針就是內(nèi)存地址,然后通過地址取這個地址的數(shù)據(jù),說一大堆沒用的云山霧繞的
作者: 慢慢思考 時間: 2023-4-24 09:42
關(guān)于EQU是什么意思有什么用法,看下面一段程序:
tryy EQU 0A0H
abyte EQU 0A0H
abit EQU 0A0H
ORG tryy
START:
MOV tryy,#tryy
MOV abyte,#tryy
MOV C,tryy
MOV C,abit
DB 1,2,3,4,tryy
ORG 0B0H
ddat: ds tryy
ACALL tryy
END
偽指令EQU如果用于分配存儲單元,那它是用于RAM;偽指令data用于RAM;偽指令ds則是用于ROM。
我的主貼最后有一段話,指針究竟是只針對RAM,還是包括ROM?
作者: Hephaestus 時間: 2023-4-24 12:42
老老實實看c51的幫助文件吧,你的回復(fù)就沒有任何正確的地方。
EQU就是簡單的替換,相當(dāng)于c語言的#define,如果純匯編那么可以用,因為內(nèi)存分配完全是你的責(zé)任,發(fā)生沖突也跟編譯器無關(guān)。如果匯編和c混合編程那么絕對不可以使用EQU定義變量,因為這么做連接器BL51.exe不知道這些地址已經(jīng)被占用,會發(fā)成沖突。
ds就是用于RAM,可讀可寫,用于ROM是很不合理的,讓編譯器分配一個字節(jié)內(nèi)容未知也不能寫的ROM你想干什么?
作者: 慢慢思考 時間: 2023-4-24 15:05
北航2012年出版的《單片機(jī)系統(tǒng)設(shè)計基礎(chǔ)》中強(qiáng)調(diào)DS及DW只能用于程序存儲器,不能用于數(shù)據(jù)存儲器,這是目前我找到的唯一說明了DS應(yīng)用存儲目標(biāo)的書。
我的其它的錯誤,還是希望一條條地說明,有討論就會有進(jìn)步。DS在編程中有什么實際用途,的確如你所說,本人也困惑。
作者: 李冬 時間: 2023-4-24 16:12
編程這么久了,也沒感覺到指針有多么厲害
作者: Hephaestus 時間: 2023-4-24 18:28
你直接看Keil自帶的幫助文件好了,中文的資料都是%為了晉升%灌水出來的,毫無價值的%。
作者: trthgyv 時間: 2023-4-26 23:16
很少有人把這個講的這么好了
作者: zyf20210305 時間: 2023-4-27 08:34
題主講的不錯,對于初學(xué)者來說是很有用的
作者: 濁世孤心 時間: 2023-4-27 20:15
講得很明細(xì)。
作者: li64331218 時間: 2023-4-28 13:26
爭論的焦點就是指針的用途。我理解指針就像上學(xué)時的樓管阿姨,手里有一大串鑰匙,對應(yīng)每個宿舍門,至于里邊住幾個人,從哪里來的。她不關(guān)心。只要找到能打開這扇門的鑰匙就行了。
作者: nlming 時間: 2023-5-4 09:26
很形象
作者: elviskuang 時間: 2023-5-10 11:53
比較難懂,不好理解
作者: 2408935979 時間: 2023-5-24 21:59
講的很好
作者: xb_00111 時間: 2023-6-13 14:12
講的很清楚,新手可以看一看。
作者: 隔壁老五 時間: 2023-7-12 11:57
c語言引入指針是為了程序設(shè)計簡單,因為cpu里可以間接尋址,這樣在類似查表之類的操作時,直接把間接尋址寄存器加減相應(yīng)的值就行了?梢钥焖偃〉脭(shù)據(jù)地址。
作者: dingmingzhou 時間: 2023-7-19 14:51
指針就是鐘表上的指針,指向哪里就可表示出哪里的數(shù)字。比如,指針指向3點鐘那里(存放3這個數(shù)據(jù)的地址),人們就知道是3點(3,就是存放3 de地址里 de 數(shù)據(jù))
作者: dingmingzhou 時間: 2023-7-19 14:52
言簡意賅
作者: KTTCO 時間: 2023-7-28 08:47
都快讓他們搞蒙了,剛?cè)雽W(xué)
作者: yzwzfyz 時間: 2023-8-25 15:49
【北航2012年出版的《單片機(jī)系統(tǒng)設(shè)計基礎(chǔ)》中強(qiáng)調(diào)DS及DW只能用于程序存儲器,不能用于數(shù)據(jù)存儲器】
個人不同意這個說法。
作者: aabbccddd 時間: 2023-10-18 10:20
解釋的挺好
作者: Decade768 時間: 2023-11-5 13:18
好資料,51黑有你更精彩!!!
作者: 52576525 時間: 2023-12-20 13:56
看懂啦!感謝樓主
作者: 1600277881 時間: 2024-1-6 17:19
也許從應(yīng)用倒過來理解"為什么要用指針"更加容易讓你了解指針。
比如就不讓你用指針寫C代碼, 可以嗎?
不用指針的話有什么功能是做不到的?
有什么功能不給你用指針做就效率大大的降低?
能正確的回答以上問題,你才算懂指針是個啥玩意兒。
作者: zkn866 時間: 2024-2-4 18:47
就是匯編語言的間接尋址
作者: cxxx180 時間: 2024-2-5 10:35
同意。一大堆去說間接/尋址/寄存器。
作者: shanghanlin 時間: 2024-2-17 16:18
你好老哥,我是測控技術(shù)與儀器專業(yè)點的大學(xué)生,我在學(xué)習(xí)中有很多難以理解的問題,想請教您,不知道是否能打擾一下您
作者: q202cs 時間: 2024-2-28 18:22
*就好比作了一個標(biāo)記并且釘了一顆釘子定位,釘了什么釘子呢?*abc,釘了一顆abc的釘子
作者: lyonkon 時間: 2024-3-6 21:12
指針包含兩個地址,1:指針值本身存放地址。2指針的值指向的地址。比如指針值是0x78,這個值存在哪里由編譯器分配。是讀指針方向所指的值,還是寫指針?biāo)阜较虻闹,自己決定。不會c語言,個人愚見。
作者: Tao濤 時間: 2024-4-30 18:43
簡單明了
作者: sunotea 時間: 2024-5-9 20:30
段位還不夠。還在學(xué)C語言。
作者: 鄉(xiāng)海峰 時間: 2024-7-17 10:54
幫我理解指針很有幫助
作者: cy009 時間: 2024-7-17 12:12
好文章,再附帶一個實例就更好了
作者: Fance9988 時間: 2024-8-15 00:45
一直沒理解:既然直接用一個變量能保存數(shù)據(jù),為什么還要用一個變量保存一個地址再通過這個地址去訪問到那個數(shù)據(jù)??是不是有點多此一舉了,
作者: xlhlydd 時間: 2024-11-28 11:03
在這里看這些能人討論,能學(xué)到知識,積累知識,大有益處。
作者: xhaity 時間: 2025-1-26 13:24
“.......一個是數(shù)據(jù)所存儲的地址是由C編輯器自動分配的,程序員沒自主權(quán),所以一個數(shù)據(jù)存儲的地址,程序員是不知道的;......” 我覺得這句有問題吧,一個是數(shù)據(jù)所存儲的地址可以由C編譯器(不是編輯器)自動分配,也可以程序員指定(特殊情況下還是需要程序員指定的),自動分配的地址一般程序員不需要知道,程序員也可以知到的
作者: xhaity 時間: 2025-1-26 23:40
數(shù)據(jù)變量比作某個人的話,存儲器就是房子,給每個人分配了房子是有地址的,存數(shù)據(jù)就是這個人回家(也必需根據(jù)地址),你要訪問道這個人也必須通過這個地址
作者: qinlu123 時間: 2025-3-10 11:27
指針厲害得很,主要是做接口用,很多人一輩子也不會程序分層所以不知道指針的妙處何在
作者: qinlu123 時間: 2025-3-10 11:31
不用看太高級的例程,看看我發(fā)的帖子就懂了
作者: zc960630 時間: 2025-5-20 16:42
這是什么書?推薦下
歡迎光臨 (http://www.torrancerestoration.com/bbs/) |
Powered by Discuz! X3.1 |