PHP中文件读写操作

时间:2009-7-8     作者:smarteng     分类: PHP相关


PHP中提供了一系列的I/O函数,能简捷地实现我们所需要的功能,包括文件系统操作和目录操作(如“复制[copy]”)。下面给大家介绍的是基本的文件读写操作:(1)读文件;(2)写文件;(3)追加到文件。

  以下是一篇关于文件基本读写操作的文章,我曾经就是看了这篇文章后学会文件基本操作的,在这里发出来与大家共享:
读文件:[break]
PHP代码:
1.    <?php
2.
3.    $file_name = "data.dat";
4.    // 要读取的文件的绝对路径: homedata.dat
5.
6.    $file_pointer = fopen($file_name, "r");
7.    // 打开文件,8.    "r" 是一种模式,9.    或者说我们要进行的操作方法,10.    详见本文后面的介绍
11.
12.    $file_read = fread($file_pointer, filesize($file_name));
13.    // 通过文件指14.    针读取文件内容
15.
16.    fclose($file_pointer);
17.    // 关闭文件
18.
19.    print "读取到的文件内容是: $file_read";
20.    // 显示文件内容
21.    ?>
22.

写文件:
PHP代码:
1.    <?php
2.
3.    $file_name = "data.dat";
4.    // 绝对路径: homedata.dat
5.
6.    $file_pointer = fopen($file_name, "w");
7.    // "w"是一种模式,8.    详见后面
9.
10.    fwrite($file_pointer, "what you wanna write");
11.    // 先把文件剪切12.    为0字节大小,13.     然后写入
14.
15.    fclose($file_pointer);
16.    // 结束
17.
18.    print "数据成功写入文件";
19.
20.    ?>
21.

追加到文件后面:
PHP代码:
1.    <?php
2.
3.    $file_name = "data.dat";
4.    // 绝对路径: homedata.dat
5.
6.    $file_pointer = fopen($file_name, "a");
7.    // "w"模式
8.
9.    fwrite($file_pointer, "what you wanna append");
10.    // 不11.    把文件剪切12.    成0字节,13.     把数据追加到文件最后
14.
15.    fclose($file_pointer);
16.    // 结束
17.
18.    print "数据成功追加到文件";
19.
20.    ?>
21.

以上只是简单介绍,下面我们要讨论一些更深层的。

有时候会发生多人写入的情况(最常见是在流量较大的网站),会产生无用的数据写入文件, 例如:

info.file文件内容如下 ->

|1|Mukul|15|Male|India (n)
|2|Linus|31|Male|Finland (n)

现在两个人同时注册,引起文件破坏->

info.file ->

|1|Mukul|15|Male|India
|2|Linus|31|Male|Finland
|3|Rob|27|Male|USA|
Bill|29|Male|USA

上例中当PHP写入Rob的信息到文件的时候,Bill正好也开始写入,这时候正好需要写入Rob纪录的'n',引起文件破坏。

我们当然不希望发生这样的情况, 所以让我们看看文件锁定:
PHP代码:
1.    <?php
2.
3.    $file_name = "data.dat";
4.
5.    $file_pointer = fopen($file_name, "r");
6.
7.    $lock = flock($file_pointer, LOCK_SH);
8.    // 我使用4.0.2,9.    所以用LOCK_SH,10.    你可能需要直接写成 1.
11.
12.    if ($lock) {
13.
14.    $file_read = fread($file_pointer, filesize($file_name));
15.    $lock = flock($file_pointer, LOCK_UN);
16.    // 如果版本小于PHP4.0.2,17.     用 3 代替 LOCK_UN
18.
19.    }
20.
21.    fclose($file_pointer);
22.
23.    print "文件内容为 $file_read";
24.
25.    ?>
26.

上例中,如果两个文件read.php和read2.php都要存取该文件,那么它们都可以读取,但是当一个程序需要写入的时候,它必须等待,直到读操作完成,文件所释放。
PHP代码:
1.    <?php
2.
3.    $file_name = "data.dat";
4.
5.    $file_pointer = fopen($file_name, "w");
6.
7.    $lock = flock($file_pointer, LOCK_EX);
8.    // 如果版本低于PHP4.0.2,9.     用 2 代替 LOCK_EX
10.
11.    if ($lock) {
12.
13.    fwrite($file_pointer, "what u wanna write");
14.    flock($file_pointer, LOCK_UN);
15.    // 如果版本低于PHP4.0.2,16.     用 3 代替 LOCK_UN
17.
18.    }
19.
20.    fclose($file_pointer);
21.
22.    print "数据成功写入文件";
23.
24.    ?>
25.

虽然"w"模式用来覆盖文件, 单我觉得不适用。
PHP代码:
1.    <?php
2.
3.    $file_name = "data.dat";
4.
5.    $file_pointer = fopen($file_name, "a");
6.
7.    $lock = flock($file_pointer, LOCK_EX);
8.    // 如果版本低于PHP4.0.2,9.     用 2 代替 LOCK_EX
10.
11.    if ($lock) {
12.
13.    fseek($file_pointer, 0, SEEK_END);
14.    // 如果版本小于PHP4.0RC1,15.     使用 fseek($file_pointer, filsize($file_name));
16.
17.    fwrite($file_pointer, "what u wanna write");
18.    flock($file_pointer, LOCK_UN);
19.    // 如果版本低于PHP4.0.2,20.     用 3 代替 LOCK_UN
21.
22.    }
23.
24.    fclose($file_pointer);
25.
26.    print "数据成功写入文件";
27.
28.    ?>
29.

Hmmm..., 对于追加数据与其他操作有点不同,就是FSEEK! 确认文件指针在文件尾部总是一个好习惯。

如果是在Windows系统下, 上面的文件中文件名前面需要加上''.

FLOCK杂谈:

Flock()只在文件打开后才锁定。 在上列中文件打开后才获得锁定,现在文件的内容只是在当时的内容, 而不反映出别的程序操作的结果,因此不只是在文件追加操作,就是对读取操作也应该使用fseek。
(此处翻译可能不是很确切, 但我想意思到了)。

关于模式:

'r' - 只读方式打开, 文件指针置于文件头

'r+' - 读写方式打开,文件指针置于文件头

'w' - 只写打开,文件指针置于文件头, 文件被剪切为0字节, 如果文件不存在, 尝试建立文件

'w+' - 读写打开,文件指针置于文件头, 文件大小被剪切为0字节,如果文件不存在, 尝试建立文件

'a' - 只写方式打开,文件指针置于文件尾,如果文件不存在,尝试建立文件

'a+' - 读写打开,文件指针置于文件尾,如果文件不存在, 尝试建立文件

顺便说一下创建文件目录的代码

//创建类似"../../../xxx/xxx.txt"的目录

function createdirs($path, $mode = 0777) //mode 077
{
$dirs = explode('/',$path);
$pos = strrpos($path, ".");
if ($pos === false) { // note: three equal signs
// not found, means path ends in a dir not file
$subamount=0;
}
else {
$subamount=1;
}

for ($c=0;$c < count($dirs) - $subamount; $c++) {
$thispath="";
for ($cc=0; $cc <= $c; $cc++) {
$thispath.=$dirs[$cc].'/';
}
if (!file_exists($thispath)) {
//print "$thispath
";
mkdir($thispath,$mode); //mkdir函数创建目录
}
}
}
//调用如createdirs("xxx/xxxx/xxxx",);

//原函数中使用$GLOBALS["dirseparator"]我改成了'/'

function recur_mkdirs($path, $mode = 0777) //mode 0777
{
//$GLOBALS["dirseparator"]
$dirs = explode($GLOBALS["dirseparator"],$path);
$pos = strrpos($path, ".");
if ($pos === false) { // note: three equal signs
// not found, means path ends in a dir not file
$subamount=0;
}
else {
$subamount=1;
}