人妖在线一区,国产日韩欧美一区二区综合在线,国产啪精品视频网站免费,欧美内射深插日本少妇

新聞動(dòng)態(tài)

Linux中文件描述符fd與文件指針FILE*互相轉(zhuǎn)換實(shí)例解析

發(fā)布日期:2022-04-14 16:01 | 文章來源:源碼之家

本文研究的主要是Linux中文件描述符fd與文件指針FILE*互相轉(zhuǎn)換的相關(guān)內(nèi)容,具體介紹如下。

1.文件描述符fd的定義:文件描述符在形式上是一個(gè)非負(fù)整數(shù)。實(shí)際上,它是一個(gè)索引值,指向內(nèi)核為每一個(gè)進(jìn)程所維護(hù)的該進(jìn)程打開文件的記錄表。當(dāng)程序打開一個(gè)現(xiàn)有文件或者創(chuàng)建一個(gè)新文件時(shí),內(nèi)核向進(jìn)程返回一個(gè)文件描述符。在程序設(shè)計(jì)中,一些涉及底層的程序編寫往往會(huì)圍繞著文件描述符展開。但是文件描述符這一概念往往只適用于UNIX、Linux這樣的操作系統(tǒng)。

2.文件指針FILE定義說明文件指針的一般形式為:

FILE *指針變量標(biāo)識(shí)符;

其中FILE應(yīng)為大寫,它實(shí)際上是由系統(tǒng)定義的一個(gè)結(jié)構(gòu),該結(jié)構(gòu)中含有文件名、文件狀態(tài)和文件當(dāng)前位置等信息。在編寫源程序時(shí)不必關(guān)心FILE結(jié)構(gòu)的細(xì)節(jié)。

使用系統(tǒng)調(diào)用的時(shí)候用文件描述符的時(shí)候比較多,但是操作比較原始。C庫(kù)函數(shù)在I/O上提供了一些方便的包裝(比如格式化I/O、重定向),但是對(duì)細(xì)節(jié)的控制不夠。

如果過度依賴其中的一種只會(huì)徒增麻煩,所以知道兩者的轉(zhuǎn)換是很有必要的。FILE*是對(duì)fd的封裝

當(dāng)然,有人會(huì)說知道文件路徑的話重新打開就是了,但是這會(huì)產(chǎn)生競(jìng)爭(zhēng)條件(Race Conditions),首先重新打開文件,相當(dāng)于是2個(gè)fd指向同一文件,然后如果在打開的期間文件被刪除了又被新建了一個(gè)同名文件,2個(gè)fd指向的便是不同的文件。

glibc庫(kù)提供了兩個(gè)轉(zhuǎn)換函數(shù)fdopen(3)和fileno(3),都是<stdio.h>中的

FILE *fdopen(int fd, const char *mode);
int fileno(FILE *stream);

PS:為了節(jié)省篇幅,還是繼續(xù)忽略返回值的檢查。

來看看測(cè)試吧,是不是我們想的那樣。

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
int main()
{
  const char* filename = "new.txt";
  int fd = open(filename, O_RDWR | O_CREAT, S_IRUSR | S_IWUSR);
  FILE* fp = fdopen(fd, "w+");
  int fd2 = fileno(fp);
  printf("fd=%d | fd2=%d\n", fd, fd2);
  fclose(fp);
  close(fd);
  return 0;
}
$ gcc test.c 
$ ./a.out 
fd=3 | fd2=3

參考fileno手冊(cè):

The function fileno() examines the argument stream and returns its integer descriptor.

FILE是對(duì)fd的封裝,fileno()是直接取得被封裝的fd,因此并未創(chuàng)建新的fd指向該文件。

參考fdopen手冊(cè):

The fdopen() function associates a stream with the existing file descriptor, fd. The mode of
the stream (one of the values "r", "r+", "w", "w+", "a", "a+") must be compatible with the
mode of the file descriptor.

fdopen()是講流(FILE對(duì)象)與已存在的文件描述符fd進(jìn)行關(guān)聯(lián),因此也是未創(chuàng)建新的fd。值得注意的是,F(xiàn)ILE指針的模式(mode)必須與文件描述符的模式兼容。

關(guān)于mode參數(shù)先擱置會(huì)兒,目前我們知道的是,使用fileno和fdopen進(jìn)行轉(zhuǎn)換,都是在原有的fd上進(jìn)行操作,并未產(chǎn)生新的fd。那么,再次審視剛才的代碼,是否發(fā)現(xiàn)了問題?

我們來檢查下close(fd)的返回值,把close(fd)改成下列代碼

  if (-1 == close(fd)) {
    perror("close");
    exit(1);
  }
$ gcc test.c 
$ ./a.out 
close: Bad file descriptor

沒錯(cuò),fclose在關(guān)閉文件指針的時(shí)候,內(nèi)部其實(shí)也關(guān)閉了文件描述符(否則資源就泄露了),既然這里fp內(nèi)部的文件描述符和fd是同一個(gè),當(dāng)fp被關(guān)閉時(shí),fd也被關(guān)閉了,再次關(guān)閉fd就會(huì)出現(xiàn)“損壞的文件描述符”錯(cuò)誤。

OK,現(xiàn)在回顧下fopen的第2個(gè)參數(shù),又r/r+/w/w+/a/a+一共6種設(shè)置(windows平臺(tái)的rb/rb+/wb/wb+暫且不談),對(duì)比Linux手冊(cè)我將對(duì)應(yīng)的open設(shè)置列出來

依然是進(jìn)行測(cè)試,修改fd_mode和fp_mode,看看實(shí)驗(yàn)結(jié)果

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
const int security = S_IRUSR | S_IWUSR;
const int fd_mode = O_RDWR | O_CREAT | O_TRUNC;
const char* fp_mode = "r";
int main()
{
  int fd = open("new.txt", fd_mode, security);
  FILE* fp = fdopen(fd, fp_mode);
  if (fp == NULL) {
    perror("fdopen");
    exit(1);
  }
  close(fd);
  return 0;
}

在fd_mode等價(jià)于"w+"時(shí),fp_mode的6種設(shè)置(r/r+/w/w+/a/a+)均返回非空指針。

在fd_mode等價(jià)于"w"時(shí),fp_mode6種設(shè)置只有"a"和"w"返回非空指針。

繼續(xù)嘗試"r"/"r+"/"a"/"a+"的設(shè)置,可以發(fā)現(xiàn)所謂“兼容”只與讀寫權(quán)限有關(guān),O_RDWR兼容O_RDONLY和O_WRONLY,而后兩者則只與自身兼容。

有意思的是O_APPEND(在末尾添加)和O_TRUNC(截?cái)辔募念^添加)也兼容。

The file position indicator of the new stream is set to that
belonging to fd, and the error and end-of-file indicators are cleared. Modes "w" or "w+" do
not cause truncation of the file. The file descriptor is not dup'ed, and will be closed when
the stream created by fdopen() is closed.

繼續(xù)查看fdopen的手冊(cè)內(nèi)容,可以看到"w"和"w+"在這里不會(huì)導(dǎo)致文件截?cái)唷?/p>

后一句也印證了我們前面的實(shí)驗(yàn)結(jié)果:文件描述符不會(huì)被復(fù)制,文件指針被關(guān)閉時(shí)文件描述符也會(huì)被關(guān)閉。

PS:其實(shí)fdopen的手冊(cè)上還有最后一句:The result of applying fdopen() to a shared memory object is undefined.

將fdopen用于共享內(nèi)存對(duì)象的結(jié)果是未定義的。

總結(jié)

以上就是本文關(guān)于Linux中文件描述符fd與文件指針FILE*互相轉(zhuǎn)換實(shí)例解析的全部?jī)?nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專題,如有不足之處,歡迎留言指出。感謝朋友們對(duì)本站的支持!

版權(quán)聲明:本站文章來源標(biāo)注為YINGSOO的內(nèi)容版權(quán)均為本站所有,歡迎引用、轉(zhuǎn)載,請(qǐng)保持原文完整并注明來源及原文鏈接。禁止復(fù)制或仿造本網(wǎng)站,禁止在非www.sddonglingsh.com所屬的服務(wù)器上建立鏡像,否則將依法追究法律責(zé)任。本站部分內(nèi)容來源于網(wǎng)友推薦、互聯(lián)網(wǎng)收集整理而來,僅供學(xué)習(xí)參考,不代表本站立場(chǎng),如有內(nèi)容涉嫌侵權(quán),請(qǐng)聯(lián)系alex-e#qq.com處理。

相關(guān)文章

實(shí)時(shí)開通

自選配置、實(shí)時(shí)開通

免備案

全球線路精選!

全天候客戶服務(wù)

7x24全年不間斷在線

專屬顧問服務(wù)

1對(duì)1客戶咨詢顧問

在線
客服

在線客服:7*24小時(shí)在線

客服
熱線

400-630-3752
7*24小時(shí)客服服務(wù)熱線

關(guān)注
微信

關(guān)注官方微信
頂部