該函數(shù)可以改變已打開(kāi)的文件的性質(zhì)。
#include <fcntl.h>

int fcntl(int fields, int cmd,
/**//* int arg */); //若成功則依賴(lài)于cmd,若出錯(cuò)則返回-1
第三個(gè)參數(shù)總是一個(gè)整數(shù),與上面所示函數(shù)原型中的注釋部分相對(duì)應(yīng)。但是在作為記錄鎖用時(shí),第三個(gè)參數(shù)則是指向一個(gè)結(jié)構(gòu)的指針。
fcntl函數(shù)有5種功能:
1.復(fù)制一個(gè)現(xiàn)有的描述符(cmd = F_DUPFD).
2.獲得/設(shè)置文件描述符標(biāo)記(cmd = F_GETFD || F_SETFD).
3.獲得/設(shè)置文件狀態(tài)標(biāo)記(cmd = F_GETFL || F_SETFL).
4.獲得/設(shè)置異步I/O所有權(quán)(cmd = F_GETOWN || F_SETOWN).
5.獲得/設(shè)置記錄鎖(cmd = F_GETLK, F_SETLK || F_SETLKW).
1
#include <fcntl.h>
2
#include <unistd.h>
3
#include <iostream>
4
#include <errno.h>
5
using namespace std;
6
7
int main(int argc, char* argv[])
8

{
9
int fd, var;
10
//fd = open("new",O_RDWR);
11
12
if (argc != 2)
13
{
14
perror("--");
15
cout << "請(qǐng)輸入?yún)?shù),即文件名!" << endl;
16
}
17
18
if ((var = fcntl(atoi(argv[1]), F_GETFL, 0)) < 0)
19
{
20
strerror(errno);
21
cout << "fcntl file error." << endl;
22
}
23
24
switch (var & O_ACCMODE)
25
{
26
case O_RDONLY :
27
cout<<"Read only.."<<endl;
28
break;
29
30
case O_WRONLY :
31
cout<<"Write only.."<<endl;
32
break;
33
34
case O_RDWR :
35
cout<<"Read wirte.."<<endl;
36
break;
37
38
default :
39
break;
40
}
41
42
if (val & O_APPEND)
43
cout << ",append" << endl;
44
45
if (val & O_NONBLOCK)
46
cout << ",noblocking" << endl;
47
48
cout << "exit 0" << endl;
49
50
exit(0);
51
}
52
53
fcntl文件鎖有兩種類(lèi)型:建議性鎖和強(qiáng)制性鎖
建議性鎖是這樣規(guī)定的:每個(gè)使用上鎖文件的進(jìn)程都要檢查是否有鎖存在,當(dāng)然還得尊重已有的鎖。內(nèi)核和系統(tǒng)總體上都堅(jiān)持不使用建議性鎖,它們依靠程序員遵守這個(gè)規(guī)定。
強(qiáng)制性鎖是由內(nèi)核執(zhí)行的。當(dāng)文件被上鎖來(lái)進(jìn)行寫(xiě)入操作時(shí),在鎖定該文件的進(jìn)程釋放該鎖之前,內(nèi)核會(huì)阻止任何對(duì)該文件的讀或?qū)懺L(fǎng)問(wèn),每次讀或?qū)懺L(fǎng)問(wèn)都得檢查鎖是否存在。
系統(tǒng)默認(rèn)fcntl都是建議性鎖,強(qiáng)制性鎖是非POSIX標(biāo)準(zhǔn)的。如果要使用強(qiáng)制性鎖,要使整個(gè)系統(tǒng)可以使用強(qiáng)制性鎖,那么得需要重新掛載文件系統(tǒng), mount使用參數(shù) -0 mand打開(kāi)強(qiáng)制性鎖,或者關(guān)閉已加鎖文件的組執(zhí)行權(quán)限并且打開(kāi)該文件的set-GID權(quán)限位。
建議性鎖只在cooperating processes之間才有用,對(duì)cooperating process的理解是最重要的,它指的是會(huì)影響其它進(jìn)程的進(jìn)程或被別的進(jìn)程所影響的進(jìn)程,舉兩個(gè)例子:
(1)我們可以同時(shí)在兩個(gè)窗口中運(yùn)行同一個(gè)命令,對(duì)同一個(gè)文件進(jìn)行操作,那么這兩個(gè)進(jìn)程就是cooperating processes;
(2) cat file | sort, 那么cat和sort產(chǎn)生的進(jìn)程就是使用了pipe的cooperating processes。
使用fcntl文件鎖進(jìn)行I/O操作必須小心:進(jìn)程在開(kāi)始任何I/O操作前如何去處理鎖,在對(duì)文件解鎖前如何完成所有的操作,是必須考慮的。如果在設(shè)置鎖之前打開(kāi)文件,或者讀取該鎖之后關(guān)閉文件,另一個(gè)進(jìn)程就可能在上鎖/解鎖操作和打開(kāi)/關(guān)閉操作之間的幾分之一秒內(nèi)訪(fǎng)問(wèn)該文件。當(dāng)一個(gè)進(jìn)程對(duì)文件加鎖后,無(wú)論它是否釋放所加的鎖,只要文件關(guān)閉,內(nèi)核都會(huì)自動(dòng)釋放加在文件上的建議性鎖(這也是建議性鎖和強(qiáng)制性鎖的最大區(qū)別), 所以不要想設(shè)置建議性鎖來(lái)達(dá)到永久不讓別的進(jìn)程訪(fǎng)問(wèn)文件的目的(強(qiáng)制性鎖才可以);強(qiáng)制性鎖則對(duì)所有進(jìn)程起作用。
fcntl使用三個(gè)參數(shù) F_SETLK/F_SETLKW, F_UNLCK和F_GETLK, 來(lái)分別要求、釋放、測(cè)試record locks。 record locks是對(duì)文件一部分而不是整個(gè)文件的鎖,這種細(xì)致的控制使得進(jìn)程更好地協(xié)作以共享文件資源。fcntl能夠用于讀取鎖和寫(xiě)入鎖,read lock也叫shared lock(共享鎖), 因?yàn)槎鄠€(gè)cooperating process能夠在文件的同一部分建立讀取鎖;write lock被稱(chēng)為exclusive lock(排斥鎖), 因?yàn)槿魏螘r(shí)刻只能有一個(gè)cooperating process在文件的某部分上建立寫(xiě)入鎖。如果cooperating processes對(duì)文件進(jìn)行操作,那么它們可以同時(shí)對(duì)文件加read lock,在一個(gè)cooperating process加write lock之前,必須釋放別的cooperating process加在該文件的read lock和wrtie lock,也就是說(shuō),對(duì)于文件只能有一個(gè)write lock存在,read lock和wrtie lock不能共存。