最近一直在忙毕设,出现了很多问题。
今天终于把物联网毕业设计应用层到底层的数据连接上了。
现在上一个让我最无奈的问题。
在ARM开发板系统里面,文件到底可不可以同时被不同的程序读和写?
之前一直都存在一个非常严重的误区。
一直认为,在C语言里物物皆文件,而write和read一类的读写文件的函数,应该是不能同时使用的,系统是会出错的。
实际上应该更加准确的去形容:write和read的函数不是不可以同时读写同一个文件,而是如果同时读写的话,当出现先后的问题的时候,读出的数据并不是我们想要的结果,应该尽量避免同时读写同一个文件,尤其是在使用同一个进程多个线程分别进行write和read操作的例子的时候。
我要做是对ARM开发板有一个内部自动运行的进程会一直不断的读取串口数据,而另一个不相关的用户进程,会在用户触发某些条件的时候写入串口文件,一开始有以上的误区,所以想用两个串口分别工作,一个用来读,一个用来写,结果根本收不到数据。(这里说的串口是Rx、Tx)
后来,我觉得不行还是得两个进程对同一个串口读取。我就现在Linux系统下尝试了一下。
write.c
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <string.h>
int main(void)
{
int fd;
char *c = "Hello, world!\n";
fd = open("./a.txt",O_WRONLY|O_APPEND|O_CREAT,0777);
if (fd <0)
{
perror("open file failed");
return 0;
}
while(1){
if(write(fd,c,strlen(c)) < 0)
{
perror("write failed");
return -1;
}
}
return 0;
}
这个write会创建一个文件,并且不断的往内部写入数据,我这里并没有给进程退出的出口,直接让其进入死循环中。
运行时直接加上&,脱离bash,进入后台运行。
@ubuntu:~/$ ./write &
read.c
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
int main(void)
{
int fd;
char buff[2];
fd = open("./a.txt",O_RDONLY);
if(fd <0 )
{
perror("open file failed ");
return -1;
}
while(1){
if(read(fd,buff,2)<0)
{
perror("read failed :");
return -2;
}
}
printf("%s",buff);
return 0;
}
用这个程序,在write函数写入的时候不断的读取。
我先运行write然后再运行read,结果并没有发生什么错误。
然后我又反着运行,先运行read然后write,最后也是正常的。只是两个程序都没有停止的时候。
实践证明,同一个文件是可以被不同的进程同时的读写,这里我也提出了一些可能,当wirte运行的时候,如果read开始读取文件,发现文件正在写入会进入阻塞状态,等到写入完成后,开始读取文件内容。反之一样,read的时候write会进入阻塞状态,等待read完成操作。这个论点还需要更进一步的证明。