sendfile
現在流行的web 伺服器裡面都提供 sendfile 選項來提升伺服器效能,那到底 sendfile是什麼,又怎麼影響效能的呢? sendfile其實是 Linux2.0+以後的推出的一個系統調用,web伺服器可以透過調整自身的配置來決定是否要利用 sendfile這個系統調用。先來看看不用 sendfile的傳統網路傳輸過程:
read(file,tmp_buf, len);
write(socket,tmp_buf, len);
硬碟 >> kernel buffer >> user buffer>> kernel socket buffer >>協定棧
一般來說一個網路應用是透過讀取硬碟數據,然後寫數據到socket 來完成網路傳輸的。上面2行用程式碼解釋了這一點,不過上面2行簡單的程式碼掩蓋了底層的許多操作。來看看底層是怎麼執行上面2行程式碼的:
1.系統呼叫 read()產生一個上下文切換:從 user mode 切換到 kernel mode,然後 DMA 執行拷貝,把文件資料從硬碟讀到一個 kernel buffer 裡。
2、資料從 kernel buffer拷貝到 user buffer,然後系統呼叫 read() 返回,這時又產生一個上下文切換:從kernel mode 切換到 user mode。
3. 系統呼叫write()產生一個上下文切換:從user mode切換到kernel mode,然後把步驟2讀到user buffer的資料拷貝到kernel buffer(資料第2次拷貝到kernel buffer),不過這次是個不同的kernel buffer,這個buffer和socket相關聯。
4、系統呼叫 write()返回,產生一個上下文切換:從 kernel mode 切換到 user mode(第4次切換了),然後 DMA 從 kernel buffer拷貝資料到協定堆疊(第4次拷貝了)。
上面4個步驟有4次上下文切換,有4次拷貝,我們發現如果能減少切換次數和拷貝次數將會有效提升效能。在kernel2.0+ 版本中,系統呼叫 sendfile() 就是用來簡化上面步驟提升效能的。 sendfile() 不只減少切換次數而且還能減少拷貝次數。
再來看看用 sendfile()來進行網路傳輸的過程:
sendfile(socket,file, len);
硬碟 >> kernel buffer (快速拷貝到kernelsocket buffer) >>協定堆疊
1、 系統呼叫sendfile()透過 DMA把硬碟資料拷貝到 kernel buffer,然後資料被 kernel直接拷貝到另外一個與 socket相關的 kernel buffer。這裡沒有 user mode和 kernel mode之間的切換,在 kernel直接完成了從一個 buffer到另一個 buffer的拷貝。
2、DMA 把資料從 kernelbuffer 直接拷貝給協定棧,沒有切換,也不需要資料從 user mode 拷貝到 kernel mode,因為資料就在 kernel 裡。
文章參考:http://www.th7.cn/system/lin/201306/41314.shtml
').addClass('pre-numbering').hide(); $(this).addClass('has-numbering').parent().append($numbering); for (i = 1; i ').text(i)); }; $numbering.fadeIn(1700); }); });以上就介紹了nginx sendfile 參數解釋,包含了方面的內容,希望對PHP教學有興趣的朋友有幫助。