Linux環(huán)境編程對(duì)于初學(xué)者來說,必須深刻理解重點(diǎn)概念才能更好地編寫代碼,實(shí)現(xiàn)業(yè)務(wù)功能,下面就幾個(gè)重要的及常用的知識(shí)點(diǎn)進(jìn)行說明。搞懂這幾個(gè)概念后以免在將來的編碼出現(xiàn)混淆。
系統(tǒng)調(diào)用
所有的操作系統(tǒng)在其內(nèi)核里都有一些內(nèi)建的函數(shù),這些函數(shù)可以用來完成一些系統(tǒng)級(jí)別的功能。在Linux系統(tǒng)使用的這樣的函數(shù)叫做“系統(tǒng)調(diào)用”,英文是systemcall。這些函數(shù)代表了從用戶空間到內(nèi)核空間的一種轉(zhuǎn)換。
系統(tǒng)調(diào)用是Linux操作系統(tǒng)提供的服務(wù),是編寫應(yīng)用程序與內(nèi)核之間通信的接口,也就是我們所說的函數(shù)。相對(duì)于普通的函數(shù)調(diào)用來說,系統(tǒng)調(diào)用的性能消耗相對(duì)來說是大的。所以在程序追求性能的同時(shí),盡量避免系統(tǒng)調(diào)用。
用戶態(tài)的程序默認(rèn)是通過棧來傳遞參數(shù)的。而對(duì)于系統(tǒng)調(diào)用來說,內(nèi)核態(tài)和用戶態(tài)使用的是不同的棧,這使得系統(tǒng)調(diào)用的參數(shù)只能通過寄存器的方式進(jìn)行傳遞。
IO操作
什么是IO,通俗來講就是輸入輸出
IO分為標(biāo)準(zhǔn)IO和文件IO,我們常用的scanf、printf、getchar、putchar、gets、puts這些都是標(biāo)準(zhǔn)輸入輸出。Linux系統(tǒng)下一切皆文件的概念,所以在linux下的編程中對(duì)文件的IO操作有標(biāo)準(zhǔn)IO和文件IO兩種操作類型。標(biāo)準(zhǔn)IO是帶緩沖的IO屬于庫函數(shù),文件IO是不帶緩沖的屬于系統(tǒng)調(diào)用。
標(biāo)準(zhǔn)IO:
1.標(biāo)準(zhǔn)IO是由ANSIC標(biāo)準(zhǔn)定義 2.跨平臺(tái),可以在windows下運(yùn)行,也可以在Linux下運(yùn)行 3.通過緩沖機(jī)制來減少系統(tǒng)調(diào)用,實(shí)現(xiàn)更高的效率 4.文件流 標(biāo)準(zhǔn)IO用結(jié)構(gòu)體類型來存放文件的相關(guān)信息,標(biāo)準(zhǔn)IO所有操作圍繞著FILE來操作。
文件IO:
1.文件IO是POSIX提供的一組函數(shù) 2.只能運(yùn)行在可移植操作系統(tǒng)中,不能跨平臺(tái) 3.沒有緩沖機(jī)制 4.文件描述符是一個(gè)非負(fù)整數(shù),每打開一個(gè)文件,系統(tǒng)會(huì)自動(dòng)分配一個(gè)文件描述符(即從系統(tǒng)最小的且沒有被用的描述符來分配)
原子操作
原子在化學(xué)課程中是不可再分的顆粒。而對(duì)于Linux系統(tǒng)來說所謂原子操作是為了確保對(duì)一個(gè)整型數(shù)據(jù)的更改具有排他性。原子操作就是要么不執(zhí)行,一旦執(zhí)行就會(huì)執(zhí)行完成,是不可被打斷的一個(gè),或一系列的動(dòng)作,即在完成任務(wù)前不會(huì)被其他事件所打斷,就像原子不可被分割成顆粒一樣。單處理中,可以用單條指令完成的指令可以被看成是一個(gè)原子操作。軟件中的原子操作依賴于硬件原子操作的支持。當(dāng)然原子操作,也可以當(dāng)引用計(jì)數(shù)使用。
原子操作其實(shí)本質(zhì)上和鎖實(shí)現(xiàn)同樣的功能,都是為了保護(hù)共享對(duì)象,它具有原子性,和順序性。原子性確保指令執(zhí)行期間不被打斷,要么全部執(zhí)行,要么根本不執(zhí)行。而順序性確保即使兩條或多條指令出現(xiàn)在獨(dú)立的執(zhí)行線程中,甚至獨(dú)立的處理器上,它們本該執(zhí)行的順序依然要保持。
線程安全
所謂線程安全,就是指代碼可以在多線程環(huán)境下安全地執(zhí)行,輸出我們想要的結(jié)果。即符合正確的邏輯,是程序員期望的正常執(zhí)行結(jié)果。為了實(shí)現(xiàn)線程安全,Linux系統(tǒng)提供一些列的方法,或者只能使用局部變量或資源,或者就是利用鎖等同步機(jī)制,來實(shí)現(xiàn)全局變量或資源的訪問。
線程安全在Linux環(huán)境編程中極其重要,我們不僅要了解概念,更重要的是要在實(shí)際的編程中學(xué)會(huì)實(shí)現(xiàn)線程安全方式。下面來看一個(gè)簡單的例子:
#include#include #include staticintnCnt=0; void*Thread(void*arg) { for(inti=0;i10000;?++i)? ????{ ????????++nCnt; ????} ????return?NULL; } int?main() { ????pthread_t?t1; ????pthread_t?t2; ????/*?創(chuàng)建兩個(gè)線程?*/ ????pthread_create(&t1,?NULL,?thread,?NULL); ????pthread_create(&t2,?NULL,?thread,?NULL); ????pthread_join(t1,?NULL); ????pthread_join(t2,?NULL); ????printf("nCnt?is?%d?by?threads ",?nCnt); ????return?0; }
大家看出上面例子的問題了嗎?
對(duì),沒錯(cuò),在此例子中我們創(chuàng)建了兩個(gè)線程,線程函數(shù)是同一個(gè)函數(shù),在線程函數(shù)中是對(duì)全局變量nCnt的自增操作。這個(gè)例子中輸出結(jié)果和我們想要的是不一樣,就是因?yàn)閚Cnt執(zhí)行指令并不是原子的,兩個(gè)個(gè)線程對(duì)nCnt的并發(fā)訪問出現(xiàn)了問題。我們利用鎖就可以解決此問題。
阻塞與非阻塞
Linux環(huán)境編程中的阻塞與非阻塞,都是指I/O操作。而所有的I/O系統(tǒng)調(diào)用默認(rèn)都是阻塞的。那什么是阻塞?阻塞的系統(tǒng)調(diào)用是指當(dāng)進(jìn)行系統(tǒng)調(diào)用時(shí)除非出錯(cuò)或被信號(hào)打斷,那么系統(tǒng)調(diào)用將會(huì)一直陷入內(nèi)核態(tài)直到調(diào)用完成。非阻塞的系統(tǒng)調(diào)用是指無論I/O操作成功與否,調(diào)用都會(huì)立刻返回。阻塞和非阻塞IO是訪問設(shè)備的兩種模式,驅(qū)動(dòng)程序可以靈活的支持這兩種用戶空間對(duì)設(shè)備的訪問方式。
阻塞操作是指在執(zhí)行操作時(shí),若不能獲得資源,則阻塞進(jìn)程直到滿足條件再進(jìn)行操作。被阻塞的進(jìn)程進(jìn)入睡眠狀態(tài),被調(diào)度器的運(yùn)行隊(duì)列移走,直到等待的條件滿足
非阻塞是指在進(jìn)行操作時(shí),若不能獲得資源,他要么放棄,要么返回后重新查詢,直到可以進(jìn)行操作為止。
當(dāng)數(shù)據(jù)準(zhǔn)備好時(shí)二者的模式相同,即IO操作都是將進(jìn)程阻塞,直到IO操作完成
阻塞、非阻塞是設(shè)備文件、網(wǎng)絡(luò)文件的屬性
同步與異步
同步與異步,也是指I/O操作。POSIX定義如下:A synchronous I/O operation causes the requesting process to beblocked until that I/O operation completes An asynchronous I/O operation does not cause the requesting processto be blocked
兩者的區(qū)別就在于同步IO做IO操作時(shí)會(huì)將進(jìn)程阻塞,而異步IO做IO操作時(shí)不會(huì)阻塞進(jìn)程
當(dāng)把阻塞、非阻塞、同步和異步放在一起時(shí),難免會(huì)出現(xiàn)混淆。同步是否就是阻塞,異步是否就是非阻塞?實(shí)際上在I/O操作中,它們是不同的概念。同步既可以是阻塞的,也可以是非阻塞的,而常用的Linux的I/O調(diào)用實(shí)際上都是同步的。這里的同步和異步,是指I/O數(shù)據(jù)的復(fù)制工作是否同步執(zhí)行。
以系統(tǒng)調(diào)用read為例。阻塞的read會(huì)一直陷入內(nèi)核態(tài)直到read返回;而非阻塞的read在數(shù)據(jù)未準(zhǔn)備就緒時(shí),會(huì)直接返回,而當(dāng)有數(shù)據(jù)時(shí),非阻塞的read同樣會(huì)一直陷入內(nèi)核態(tài),直到read完成。這個(gè)read就是同步的操作, 即I/O的完成是在當(dāng)前執(zhí)行流程下同步完成的。如果是異步,則I/O操作不是隨系統(tǒng)調(diào)用同步完成的。調(diào)用返回后,I/O操作并沒有完成,而是由操作系統(tǒng)或者某個(gè)線程負(fù)責(zé)真正的I/O操作,等完成后通知原來的線程
審核編輯:湯梓紅
-
接口
+關(guān)注
關(guān)注
33文章
8598瀏覽量
151163 -
Linux
+關(guān)注
關(guān)注
87文章
11304瀏覽量
209521 -
操作系統(tǒng)
+關(guān)注
關(guān)注
37文章
6825瀏覽量
123333 -
編程
+關(guān)注
關(guān)注
88文章
3616瀏覽量
93738 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4331瀏覽量
62622
原文標(biāo)題:Linux環(huán)境編程必須搞懂的幾個(gè)概念
文章出處:【微信號(hào):良許Linux,微信公眾號(hào):良許Linux】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論