本節(jié)是操作系統(tǒng)系列教程的第三篇文章,屬于操作系統(tǒng)第一章即基礎(chǔ)篇,在真正開始操作系統(tǒng)相關(guān)章節(jié)前在這一部分回顧一些重要的主題,以下是目錄,由于本文篇幅較多因此按上篇、中篇、下篇三次發(fā)布,目錄中黑體為本篇內(nèi)容,本文為該主題最后一篇。
什么是內(nèi)存
C/C++內(nèi)存模型
堆區(qū)與棧區(qū)的本質(zhì)
Java內(nèi)存模型
Jave中的堆區(qū)與棧區(qū)是如何實(shí)現(xiàn)的
Python內(nèi)存模型
指針與引用
**進(jìn)程的內(nèi)存模型
**
**幻想大師-操作系統(tǒng)
**
總結(jié)
指針與引用
在各種編程語(yǔ)言中我們應(yīng)該經(jīng)常聽到兩個(gè)詞,那就是引用或者指針。這兩個(gè)詞都是和內(nèi)存相關(guān)的,指針和引用的作用都是“如何找到存放在內(nèi)存上的數(shù)據(jù)”。
C/C++中有“指針”這樣一個(gè)概念,而其它語(yǔ)言比如Java、Python有的只是“引用”這樣一個(gè)概念。這兩者有什么區(qū)別呢?我們打個(gè)比方你就能理解了。
“引用”就好比一個(gè)人的外號(hào)一樣,就好有個(gè)程序員叫令狐沖,但是令狐沖同學(xué)在A公司的英文名可能是“Tom”,在B公司中可能又叫“Jerry”,那么在A公司中你只需要喊一聲“Tom”就能找到令狐沖同學(xué)。
而“指針”強(qiáng)調(diào)的是位置,比如令狐沖在A公司的工位是“10排第二個(gè)”,在B公司中的工位是“8排第六個(gè)”,下班后回的位置在“中關(guān)村”。
這個(gè)例子當(dāng)中的令狐沖同學(xué)就好比程序語(yǔ)言中的對(duì)象,令狐沖的各種外號(hào)就好比對(duì)象的引用,令狐沖當(dāng)前所在的位置就好比對(duì)象的指針。
雖然通過(guò)“引用”和“指針”都能找到令狐沖同學(xué),但是尋找的方式是不一樣的。
只有C/C++這樣的編譯型語(yǔ)言才會(huì)有“指針”這樣一個(gè)概念,指的是當(dāng)前的對(duì)象放在了內(nèi)存中的哪個(gè)位置上了。在比如Java、Python等語(yǔ)言中只有“引用”這樣一個(gè)概念。
在C/C++語(yǔ)言中,我們可以通過(guò)指針直接找到一個(gè)對(duì)象,因?yàn)槟阒肋@個(gè)對(duì)象就在內(nèi)存中指針?biāo)赶虻奈恢茫贘ava、Python等語(yǔ)言中,當(dāng)你利用引用找到對(duì)象時(shí)基本上是沖著解釋器喊一句“Hey,解釋器,幫我找到令狐沖這個(gè)對(duì)象”,解釋器通過(guò)記錄查找到這個(gè)對(duì)象,注意解釋器是知道對(duì)象在內(nèi)存中的真正位置的,由于直接管理內(nèi)存是一項(xiàng)非常繁瑣容易出錯(cuò)的事情(C/C++程序員一定對(duì)此有深刻體會(huì)),因此解釋器就接手了對(duì)內(nèi)存直接管理, Java、Python等程序員是沒(méi)有必要知道對(duì)象在內(nèi)存中的真正位置的 ,沒(méi)有指針也可以開心的寫程序而且程序更加健壯,何樂(lè)不為呢,因此這些語(yǔ)言中是沒(méi)有指針這樣一個(gè)概念。
Sun的一篇論文中提到了為什么Java里沒(méi)有指針。
Most studies agree that pointers are one of the primary features that enable programmers to inject bugs into their code. Given that structures are gone, and arrays and strings are objects, the need for pointers to these constructs goes away. Thus, Java has no pointer data types. ...
You no longer have dangling pointers and trashing of memory because of incorrect pointers, because there are no pointers in Java.
大意是Java設(shè)計(jì)者認(rèn)為指針太有技巧性以至于很容易出錯(cuò),因此Java中沒(méi)有指針。其實(shí)不只是Java,流行的語(yǔ)言當(dāng)中除了C/C++之外幾乎都沒(méi)有指針。
在這一節(jié)中,你只需要理解以下兩點(diǎn)就可以啦。
- 指針:直接在內(nèi)存中找到變量所在位置。所以指針是實(shí)實(shí)在在的內(nèi)存地址。
- 引用:告訴解釋器你想使用的變量,然后解釋器再去內(nèi)存中找到變量的位置。所以引用只是解釋器的一個(gè) 承諾 ,只要這個(gè)變量存在,解釋器就承諾能找到這個(gè)變量,程序員就可以使用這個(gè)變量,至于這個(gè)變量在內(nèi)存中的什么地方是不需要程序員關(guān)心的。
進(jìn)程的內(nèi)存模型
我們已經(jīng)在前面幾個(gè)小節(jié)中研究了C/C++以及Java、Python程序的內(nèi)存模型,接下來(lái)讓我們回到操作系統(tǒng)。
我們已經(jīng)知道了,不管什么語(yǔ)言,最后操作系統(tǒng)看到的都是C程序,C程序在內(nèi)存運(yùn)行起來(lái)就是進(jìn)程。而在前面的小節(jié)當(dāng)中我們已經(jīng)知道進(jìn)程在內(nèi)存中的樣子,但那里的描述其實(shí)是不完整的,也是不準(zhǔn)確的。接下里我們就來(lái)看一下,操作系統(tǒng)中的進(jìn)程在內(nèi)存中到底是什么樣子的,如下圖所示(注意這幅圖描述的是32位操作系統(tǒng)下進(jìn)程在內(nèi)存中是什么樣子的),我們需要注意以下幾點(diǎn):
- 在上圖中多出了一塊內(nèi)存,注意,這塊內(nèi)存就是操作系統(tǒng)在運(yùn)行的時(shí)候所占用的內(nèi)存。
- 每個(gè)進(jìn)程獨(dú)占一個(gè)連續(xù)的4G大小的內(nèi)存,從內(nèi)存地址0開始,一直到0xffffffff,其中最上方的1G留給了操作系統(tǒng)使用,下方的3G是留給進(jìn)程自己使用的,其中程序員可以操作的區(qū)域就是圖中的堆區(qū)和棧區(qū)。
- 你會(huì)發(fā)現(xiàn)代碼段下方也有一點(diǎn)空隙沒(méi)有使用,其實(shí)這是有特殊目的的,具體用途會(huì)在后面的章節(jié)中講解。
現(xiàn)在你已經(jīng)知道了進(jìn)程在內(nèi)存中的樣子,你一定會(huì)有疑問(wèn)吧,
為什么每個(gè)進(jìn)程認(rèn)為自己占用的是4G內(nèi)存呢? 如果我的PC上只有2G內(nèi)存,進(jìn)程還是認(rèn)為自己擁有4G內(nèi)存嗎 ?
操作系統(tǒng)上不是可以同時(shí)運(yùn)行很多進(jìn)程嗎,內(nèi)存是有限的,假如只有2G, 每個(gè)進(jìn)程都認(rèn)為自己擁有4G內(nèi)存,這不會(huì)有問(wèn)題嗎 ?
我們首先來(lái)回答第一個(gè)問(wèn)題:是的,每個(gè)進(jìn)程都認(rèn)為計(jì)算機(jī)上的真實(shí)內(nèi)存就是4G,而且是進(jìn)程自己獨(dú)占的,即使真正的物理內(nèi)存只有256MB。
第二個(gè)問(wèn)題:很顯然,不管你現(xiàn)在看這篇文章用的電腦,iPad,安卓手機(jī)還是iPhone,這些計(jì)算設(shè)備中的進(jìn)程都是這么認(rèn)為的,你能看到這篇文章說(shuō)明進(jìn)程認(rèn)為自己擁有4G內(nèi)存是不會(huì)出現(xiàn)問(wèn)題的。
在這里需要再次強(qiáng)調(diào)的是:
每個(gè)進(jìn)程都認(rèn)為真實(shí)的內(nèi)存就是4G,其中1G被操作系統(tǒng)使用,剩余部分被進(jìn)程使用,也就是可以被程序員使用。 注意這是不受真實(shí)物理內(nèi)存限制的 ,也就是說(shuō),即使真實(shí)的物理只有256MB,進(jìn)程同樣認(rèn)為在內(nèi)存是4G,其中1G是操作系統(tǒng)的,剩余3G是進(jìn)程自己獨(dú)占的,程序員依然可以按照內(nèi)存大小是3G來(lái)寫程序。所以在大小256MB的真實(shí)物理內(nèi)存上,程序員依然可以一次性申請(qǐng)超過(guò)256MB的內(nèi)存而且可以申請(qǐng)成功,后續(xù)內(nèi)存的使用也不受影響。
就像我第一次知道這種魔法時(shí)一樣,你肯定也會(huì)驚呼這怎么可能呢? 我們?cè)趺茨茉?56MB大小的內(nèi)存上申請(qǐng)超過(guò)256MB的內(nèi)存呢 ?但事實(shí)就是如此,你可以在物理內(nèi)存大小為256MB的內(nèi)存上面申請(qǐng)超過(guò)256MB的內(nèi)存,而且無(wú)論物理內(nèi)存大小,每個(gè)進(jìn)程都認(rèn)為自己擁有4G內(nèi)存,而且是獨(dú)占內(nèi)存。
這真的是太神奇了,這就是本課程的主角-操作系統(tǒng)帶來(lái)的神奇魔法。
幻象大師——操作系統(tǒng)
這種魔法確實(shí)是真實(shí)的,這個(gè)魔法就來(lái)自我們的幻象大師- 操作系統(tǒng) ,其實(shí)進(jìn)程看到的內(nèi)存是操作系統(tǒng)制造的幻覺(jué)。操作系統(tǒng)讓每個(gè)進(jìn)程都認(rèn)為內(nèi)存就只有兩部分,一部分是操作系統(tǒng)的一部分是自己的,這種魔法就稱之為虛擬內(nèi)存。后面的章節(jié)中會(huì)重點(diǎn)介紹操作系統(tǒng)是如何實(shí)現(xiàn)這種魔法的。
在虛擬內(nèi)存上程序員分配內(nèi)存不受真實(shí)物理內(nèi)存大小的限制 。
但這僅僅是進(jìn)程自己這么認(rèn)為,這是操作系統(tǒng)給進(jìn)程制造的幻覺(jué),所以被稱之為虛擬內(nèi)存。虛擬內(nèi)存是操作系統(tǒng)中極為重要的概念,和進(jìn)程一樣,對(duì)虛擬內(nèi)存的深刻理解也是編程高手的標(biāo)志之一。我會(huì)在后續(xù)文章中來(lái)為大家透徹講解操作系統(tǒng)是如何做到的。
總結(jié)
哈哈,這真是比較長(zhǎng)的一節(jié),希望你能堅(jiān)持學(xué)到這里,沒(méi)辦法,內(nèi)存真的是非常重要的, 要想學(xué)好操作系統(tǒng),對(duì)內(nèi)存的透徹理解是必不可少的 。
在這一節(jié)中我們認(rèn)識(shí)到了其實(shí)內(nèi)存僅僅就是一堆裝0或1的小盒子組成,是沒(méi)有什么神秘的。我們也了解了C/C++、Java、Python程序的內(nèi)存模型,也知道了操作系統(tǒng)中的進(jìn)程在內(nèi)存中是什么樣子的。同時(shí)操作系統(tǒng)中被被稱為虛擬內(nèi)存的神奇魔法也著實(shí)讓人驚嘆,想學(xué)習(xí)這么魔法請(qǐng)繼續(xù)關(guān)注操作系統(tǒng)系列文章。
-
內(nèi)存
+關(guān)注
關(guān)注
8文章
3052瀏覽量
74247 -
編程語(yǔ)言
+關(guān)注
關(guān)注
10文章
1950瀏覽量
34922 -
python
+關(guān)注
關(guān)注
56文章
4807瀏覽量
84976
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論