自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

原來 Mmap 這么簡單

系統(tǒng) Linux
從傳統(tǒng)讀寫文件的過程中,我們可以發(fā)現(xiàn)有個地方可以優(yōu)化:如果可以直接在用戶空間讀寫 頁緩存,那么就可以免去將 頁緩存 的數(shù)據(jù)復(fù)制到用戶空間緩沖區(qū)的過程。

 [[393886]]

本文轉(zhuǎn)載自微信公眾號「Linux內(nèi)核那些事」,作者songsong001 。轉(zhuǎn)載本文請聯(lián)系Linux內(nèi)核那些事公眾號。  

一、傳統(tǒng)的讀寫文件

一般來說,修改一個文件的內(nèi)容需要如下3個步驟:

  • 把文件內(nèi)容讀入到內(nèi)存中。
  • 修改內(nèi)存中的內(nèi)容。
  • 把內(nèi)存的數(shù)據(jù)寫入到文件中。

過程如圖 1 所示:

如果使用代碼來實(shí)現(xiàn)上面的過程,代碼如下:

  1. read(fd, buf, 1024);  // 讀取文件的內(nèi)容到buf 
  2. ...                   // 修改buf的內(nèi)容 
  3. write(fd, buf, 1024); // 把buf的內(nèi)容寫入到文件 

從圖 1 中可以看出,頁緩存(page cache) 是讀寫文件時的中間層,內(nèi)核使用 頁緩存 與文件的數(shù)據(jù)塊關(guān)聯(lián)起來。所以應(yīng)用程序讀寫文件時,實(shí)際操作的是 頁緩存。

二、使用 mmap 讀寫文件

從傳統(tǒng)讀寫文件的過程中,我們可以發(fā)現(xiàn)有個地方可以優(yōu)化:如果可以直接在用戶空間讀寫 頁緩存,那么就可以免去將 頁緩存 的數(shù)據(jù)復(fù)制到用戶空間緩沖區(qū)的過程。

那么,有沒有這樣的技術(shù)能實(shí)現(xiàn)上面所說的方式呢?答案是肯定的,就是 mmap。

使用 mmap 系統(tǒng)調(diào)用可以將用戶空間的虛擬內(nèi)存地址與文件進(jìn)行映射(綁定),對映射后的虛擬內(nèi)存地址進(jìn)行讀寫操作就如同對文件進(jìn)行讀寫操作一樣。原理如圖 2 所示:

前面我們介紹過,讀寫文件都需要經(jīng)過 頁緩存,所以 mmap 映射的正是文件的 頁緩存,而非磁盤中的文件本身。由于 mmap 映射的是文件的 頁緩存,所以就涉及到同步的問題,即 頁緩存 會在什么時候把數(shù)據(jù)同步到磁盤。

Linux 內(nèi)核并不會主動把 mmap 映射的 頁緩存 同步到磁盤,而是需要用戶主動觸發(fā)。同步 mmap 映射的內(nèi)存到磁盤有 4 個時機(jī):

  • 調(diào)用 msync 函數(shù)主動進(jìn)行數(shù)據(jù)同步(主動)。
  • 調(diào)用 munmap 函數(shù)對文件進(jìn)行解除映射關(guān)系時(主動)。
  • 進(jìn)程退出時(被動)。
  • 系統(tǒng)關(guān)機(jī)時(被動)。

三、mmap的使用方式

下面我們介紹一下怎么使用 mmap,mmap 函數(shù)的原型如下:

  1. void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset); 

下面介紹一下 mmap 函數(shù)的各個參數(shù)作用:

  • addr:指定映射的虛擬內(nèi)存地址,可以設(shè)置為 NULL,讓 Linux 內(nèi)核自動選擇合適的虛擬內(nèi)存地址。
  • length:映射的長度。
  • prot:映射內(nèi)存的保護(hù)模式,可選值如下:
    • PROT_EXEC:可以被執(zhí)行。
    • PROT_READ:可以被讀取。
    • PROT_WRITE:可以被寫入。
    • PROT_NONE:不可訪問。
  • flags:指定映射的類型,常用的可選值如下:
    • MAP_FIXED:使用指定的起始虛擬內(nèi)存地址進(jìn)行映射。
    • MAP_SHARED:與其它所有映射到這個文件的進(jìn)程共享映射空間(可實(shí)現(xiàn)共享內(nèi)存)。
    • MAP_PRIVATE:建立一個寫時復(fù)制(Copy on Write)的私有映射空間。
    • MAP_LOCKED:鎖定映射區(qū)的頁面,從而防止頁面被交換出內(nèi)存。
    • ...
  • fd:進(jìn)行映射的文件句柄。
  • offset:文件偏移量(從文件的何處開始映射)。

介紹完 mmap 函數(shù)的原型后,我們現(xiàn)在通過一個簡單的例子介紹怎么使用 mmap:

  1. int fd = open(filepath, O_RDWR, 0644);                           // 打開文件 
  2. void *addr = mmap(NULL, 8192, PROT_WRITE, MAP_SHARED, fd, 4096); // 對文件進(jìn)行映射 

在上面例子中,我們先通過 open 函數(shù)以可讀寫的方式打開文件,然后通過 mmap 函數(shù)對文件進(jìn)行映射,映射的方式如下:

  • addr 參數(shù)設(shè)置為 NULL,表示讓操作系統(tǒng)自動選擇合適的虛擬內(nèi)存地址進(jìn)行映射。
  • length 參數(shù)設(shè)置為 8192 表示映射的區(qū)域?yàn)?2 個內(nèi)存頁的大小(一個內(nèi)存頁的大小為 4 KB)。
  • prot 參數(shù)設(shè)置為 PROT_WRITE 表示映射的內(nèi)存區(qū)為可讀寫。
  • flags 參數(shù)設(shè)置為 MAP_SHARED 表示共享映射區(qū)。
  • fd 參數(shù)設(shè)置打開的文件句柄。
  • offset 參數(shù)設(shè)置為 4096 表示從文件的 4096 處開始映射。

mmap 函數(shù)會返回映射后的內(nèi)存地址,我們可以通過此內(nèi)存地址對文件進(jìn)行讀寫操作。我們通過圖 3 展示上面例子在內(nèi)核中的結(jié)構(gòu):

四、總結(jié)

本文主要介紹了 mmap 的原理和使用方式,通過本文我們可以知道,使用 mmap 對文件進(jìn)行讀寫操作時可以減少內(nèi)存拷貝的次數(shù),并且可以減少系統(tǒng)調(diào)用的次數(shù),從而提高對讀寫文件操作的效率。

由于內(nèi)核不會主動同步 mmap 所映射的內(nèi)存區(qū)中的數(shù)據(jù),所以在某些特殊的場景下可能會出現(xiàn)數(shù)據(jù)丟失的情況(如斷電)。為了避免數(shù)據(jù)丟失,在使用 mmap 的時候可以在適當(dāng)時主動調(diào)用 msync 函數(shù)來同步映射內(nèi)存區(qū)的數(shù)據(jù)。

 

責(zé)任編輯:武曉燕 來源: Linux內(nèi)核那些事
相關(guān)推薦

2023-11-01 14:49:07

2014-10-08 15:00:50

SUSE操作系統(tǒng)云計算

2019-03-15 10:55:12

通信系統(tǒng)手機(jī)

2023-09-22 08:00:00

分布式鎖Redis

2020-11-27 10:34:01

HTTPHTTPS模型

2020-09-24 06:44:54

HTTPS網(wǎng)站 HTTP

2020-10-22 08:01:52

XMLJSON轉(zhuǎn)換

2014-11-25 15:02:01

客服系統(tǒng)

2016-03-21 11:09:52

Tableau/大數(shù)據(jù)

2010-08-02 13:55:20

2025-03-20 12:33:36

2021-12-30 10:55:54

Python游戲腳本

2019-05-27 14:03:48

開發(fā)技能代碼

2022-01-27 14:12:49

Python游戲腳本

2021-06-10 06:57:39

Redis存儲數(shù)據(jù)庫

2020-11-02 14:38:56

Java 深度學(xué)習(xí)模型

2020-09-25 07:49:36

策略模式Spring

2019-05-14 10:50:11

HTTP協(xié)議HttpServlet

2021-02-01 12:18:55

策略模式Spring

2022-11-02 19:08:48

微服務(wù)輪詢消費(fèi)者
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號