linux系統編程之文件與IO(一):文件描述符、open,close
1. 什麼是IO?
輸入/輸出是主存和外部設備之間拷貝數據的過程
- 設備->內存(輸入操作)
- 內存->設備(輸出操作)
高級I/O
- ANSI C提供的標準I/O庫稱為高級I/O,通常也稱為帶緩衝的I/O
低級I/O
- 通常也稱為不帶緩衝的I/O
2. 文件描述符:fd
- 對於Linux而言,所有對設備或文件的操作都是通過文件描述符進行的。
- 當打開或者創建一個文件的時候,內核向進程返回一個文件描述符(非負整數)。後續對文件的操作只需通過該文件描述符,內核記錄有關這個打開文件的信息。
- 一個進程啟動時,默認打開了3個文件,標準輸入、標準輸出、標準錯誤,對應文件描述符是0(STDIN_FILENO)、1(STDOUT_FILENO)、2(STDERR_FILENO),這些常量定義在unistd.h頭文件中。C庫函數中與之對應的是:stdin,stdout,stderr,不過這三個是FILE指針類型。
3.文件描述符與文件指針相互轉換
可以通過以下兩個函數實現:
- fileno:將文件指針轉換為文件描述符
#include <stdio.h>
int fileno(FILE *stream)
測試程序:
#include <stdlib.h>
#include <stdio.h>
int main(void)
{
printf("fileno(stdin) = %d\n", fileno(stdin));
printf("fileno(stdout) = %d\n", fileno(stdout));
printf("fileno(stderr) = %d\n", fileno(stderr));
return 0;
}
測試結果:
- fdopen:將文件描述符轉換為文件指針
#include <stdio.h>
FILE *fdopen(int fd, const char *mode) //mode :r,w,r+,w+,a,a+
- open系統調用
有幾種方法可以獲得允許訪問文件的文件描述符。最常用的是使用open()(打開)系統調用
函數原型
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
int open(const char *pathname, int flags);
int open(const char *pathname, int flags, mode_t mode);
參數
path :文件的名稱,可以包含(絕對和相對)路徑
flags:文件打開模式
mode:用來規定對該文件的所有者,文件的用戶組及系 統中其他用戶的訪問權限
返回值
打開成功,返回文件描述符;
打開失敗,返回-1
文件打開方式:
O_EXCL表示:當O_EXCL|O_CREAT時,若文件存在,則打開失敗,不存在,則打開成功
訪問權限:
open系統調用的幾點說明:
可以利用按位邏輯加(bitwise-OR)(|)對打開方式的標誌值進行組合。
如打開一個新文件:
#define NEWFILE (O_WRONLY|O_CREAT|O_TRUNC)
對訪問權限位進行訪問所用到的標識符,均可以通過
#include <sys/stat.h>
訪問到,同樣可以通過|運算來對訪問權限進行組合也可以直接給出數字表示如0655
#define MODE755 (S_IRWXU|S_IRGRP|S_IXGRP|S_IROTH|S_IXOTH)
注:文件的訪問權限是根據:umask&~mode得出來的,例如umask=0022,mode = 0655 則訪問權限為:644
測試程序:
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#define ERR_EXIT(m) \
do \
{ \
perror(m); \
exit(EXIT_FAILURE); \
} while(0)
int main(void)
{
umask(0);
int fd;
fd = open("test.txt", O_WRONLY | O_CREAT, 0666);
if (fd == -1)
ERR_EXIT("open error");
printf("open succ\n");
return 0;
}
測試結果一:採用默認的umask值
測試結果二:重新設置umask值
- close系統調用
為了重新利用文件描述符,用close()系統調用釋放打開的文件描述符
函數原型:
#include <unistd.h>
int close(int fd);
函數參數:
- fd :要關閉的文件的文件描述符
返回值
如果出現錯誤,返回-1
調用成功返回0
注:若沒有顯示調用close(),當程序退出時也會關閉文件
- creat系統調用
為了維持與早期的UNIX系統的向後兼容性,Linux也提供可選的創建文件的系統調用,它稱為creat()。現代的linux內核很少採用creat創建文件,因為open可以完成創建功能
函數原型:
int creat(const char *path, mode_t mode);
參數
path :文件的名稱,可以包含(絕對和相對)路徑
mode: 用來規定對該文件的所有者,文件的用戶組及系 統中其他用戶的訪問權限
返回值
打開成功,返回文件描述符;
打開失敗,返回-1
在UNIX的早期版本中,open()系統調用僅僅存在兩個參數的形式。如文件不存在,它就不能打開這些文件。文件的創建則由單獨的系統調用creat()完成。在Linux及所有UNIX的近代版本中,creat()系統調用是多餘的。
creat()調用
fd = creat(file, mode);
完全等價於近代的open()調用
fd = open(file, O_WRONLY | O_CREAT | O_TRUNC, mode);