当前位置:  技术问答>linux和unix

关于LINUX下进程和线程对文件的同步问题,请高手来看看!!!

    来源: 互联网  发布时间:2015-04-30

    本文导语:  不知道大家有没有碰到过这样的问题,即在LINUX下的程序有时候会碰到多个进程访问同一个文件, 以及一个进程的不同线程对一个文件的同步问题,我尝试用了系统提供的fcntl函数来给文件加锁, 但是这只是一个进程...

不知道大家有没有碰到过这样的问题,即在LINUX下的程序有时候会碰到多个进程访问同一个文件,
以及一个进程的不同线程对一个文件的同步问题,我尝试用了系统提供的fcntl函数来给文件加锁,
但是这只是一个进程互斥锁,对于多线程的访问,后面的操作会覆盖前面的,看了stevens大爷的
网络编程的第9章中的my_lock和my_unlock的例子,还是不能很好的解决多线程同步的问题,
而他在11章利用信号灯对两个函数的另一种实现在LINUX下不行,因为LINUX下不支持有名信号灯,
哪位大虾能给提供一个比较好的解决方法,能给提供源代码的(上面两个函数的)我再加300分!

|
如果使用进程间同步,使用IPC(信号量(SYS-V)、消息队列(SYS-V)、SOCKET)都行
如果使用线程同步,建议使用posix 信号量和消息队列
这些都没问题,如果不能实现文件互斥,那么最大可能是你设计的文件互斥逻辑可能存在问题

|

/* recordio */

#include 
#include 
#include 
#include 
#include "recordio.h"


int initrecord(const char* pathname, unsigned int reclen)
{
int fd;
int rs;
RECORDINFO info;

if ( (fd = open(pathname, O_RDWR|O_CREAT, 0660)) == -1 ) {

perror("Fail to open file for initializing record");
return -1;
}

if ( (rs = read(fd, &info, sizeof(RECORDINFO))) == -1 ) {

perror("Fail to read record info for initializing record");
return -1;
}

if ( rs == 0 ) {

info.reclen = reclen;
info.recnum = 0;

if ( write(fd, &info, sizeof(RECORDINFO)) == -1 ) {

perror("Fail to write record info for initializing record");
return -1;
}
}

close(fd);
return 0;
}


int addrecord(const char* pathname, void* buf, unsigned int reclen, unsigned int recnum)
{
int fd;
int rs;
struct flock lock;

lock.l_type = F_WRLCK;
lock.l_start = 0;
lock.l_whence = SEEK_SET;
lock.l_len = 0;

if ( (fd = open(pathname, O_WRONLY|O_APPEND)) == -1 ) {

perror("Fail to open file for adding record");
return -1;
}

if ( fcntl(fd, F_SETLKW, &lock) == -1 ) {

      perror("Fail to lock file for adding record");
return -1;
}

        lock.l_type = F_UNLCK;

rs = write(fd, buf, reclen * recnum);

if ( fcntl(fd, F_SETLKW, &lock) == -1 ) {

perror("Fail to unlock file for adding record");
return -1;
}

close(fd);
return rs;
}


int getrecord(const char* pathname, void* buf, unsigned int reclen, unsigned int recnum)
{
int fd;
int len;
RECORDINFO info;
struct flock lock;

lock.l_type = F_WRLCK;
lock.l_start = 0;
lock.l_whence = SEEK_SET;
lock.l_len = 0;

if ( (fd = open(pathname, O_RDWR)) == -1 ) {

perror("Fail to open file for geting record");
return -1;
}

if ( fcntl(fd, F_SETLKW, &lock) == -1 ) {

perror("Fail to lock file for geting record");
return -1;
}

lock.l_type = F_UNLCK;

if ( read(fd, &info, sizeof(RECORDINFO)) != sizeof(RECORDINFO) ) {

perror("Fail to read record info");
return -1;
}

if ( info.reclen != reclen ) {

fprintf(stderr, "Invalid reclen parameter n");
return -1;
}

if ( lseek(fd, sizeof(RECORDINFO) + info.reclen * info.recnum, SEEK_SET) == -1 ) {

perror("Fail to seek read position");
return -1;
}

if ( (len = read(fd, buf, reclen * recnum)) == -1 ) {

perror("Fail to get record");
return -1;
}

if ( len == 0 ) {

if ( ftruncate(fd, sizeof(RECORDINFO)) == -1 ) {

perror("Fail to truncate file");
return -1;
}

info.recnum = 0;
if ( lseek(fd, 0, SEEK_SET) == -1 ) {


perror("Fail to return beginning");           
return -1;                 
}

if ( write(fd, &info, sizeof(RECORDINFO)) == -1 ) {

perror("Fail to update record info");
return -1;
        }

 if ( fcntl(fd, F_SETLKW, &lock) == -1 ) {

 perror("Fail to unlock file for geting record");
 return -1;
 }
 
close(fd);
return 0;

} else { 

info.recnum +=  len / info.reclen;

if ( lseek(fd, 0, SEEK_SET) == -1 ) {

perror("Fail to return beginning");
return -1;
}

if ( write(fd, &info, sizeof(RECORDINFO)) == -1 ) {

perror("Fail to update record info");
return -1;
}

if ( fcntl(fd, F_SETLKW, &lock) == -1 ) {
               
     perror("Fail to unlock file for geting record");
return -1;

}

close(fd);
return len;
}

}

               

    
 
 

您可能感兴趣的文章:

  • Windows和Linux下C++类成员方法作为线程函数方法介绍
  • 请问Linux核心支持多线程吗?开发库有线程库吗?线程好用吗?(稳定?)
  • Linux下GCC内置原子操作函数(多线程资源访问)介绍
  • 求个linux多线程编程的例子,要有线程池的 通用些更好
  • 请问Linux下线程开销为什么这么大?一个线程要占用近10M内存
  • Linux多线程时一些线程分不到时间片
  • 说说windows线程和linux线程的区别?
  • linux 下多线程 每个线程能否使用alarm来处理,信号是否会乱呢?
  • linux下,如何在多线程中每个线程设置一个定时器,在线等,急。。。谢谢
  • 在创建linux线程时为什么会多一个线程?
  • linux如何主动将线程放入到线程调度队列中重新排队?
  • 在linux RH73下为什么单进程只能开1021个线程线程.
  • Linux下如何让主线程挂起一个指定的时间段而子线程继续运行?谢谢
  • LINUX下有没有线程概念,和WIN下线程概念有什么不同,谢谢.
  • linux线程编程问题
  • linux线程最大数,奇怪,怎么只有300?而且线程是用完就没了,不是同时300啊!
  • 跪求:linux下pthread 双线程计算和单线程的运行时间完全相同是为什么?
  • linux线程与进程的问题
  • linux 下子线程不能执行的问题?
  • linux多线程编程的问题
  • linux多线程编程
  • linux僵尸(zombie)进程介绍及清除
  • 高手请教!linux怎样通过pid获取进程信息,如:进程名、进程状态等?
  • linux下进程占用内存空间详解
  • linux命令如何实现重启父进程而不会使其子进程退出
  • Linux进程的内核栈和用户栈概念,相互关系及切换过程
  • Linux中最多同时可以开多少个进程,一个进程可以开多少个线程?
  • linux下进程间通信:共享内存原理及具体用法举例(基于c/c++语言)
  • Linux守护进程 的子进程 终端处理
  • Linux内核进程与应用进程的通信方式有哪些?
  • 在linux,如何用共享内存来实现进程间的通讯?(这些进程没有父子关系)
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • Linux下时钟同步问题:Clock skew detected原因分析及解决方法
  • linux时钟为何与BIOS时钟不同步?如何使其同步?
  • Linux下用ntpdate同步时间及date显示设置时间
  • linux虚拟机时间与windows时间如何同步
  • linux服务器之间如何实现时间同步?
  • 送分:LINUX 下如何实现双机的备份和同步?
  • linux中同步与原子操作的相关知识辨析【请教高手】
  • linux下的同步问题
  • window系统的东东如何及时同步到linux系统上?
  • 老板要用linux服务器共享一些CAD文件,并且要同步更新,linux有这功能吗?
  • linux 多线程sys_read同步问题
  • samba性能配置问题:linux和window如何能保持同步
  • linux管道问题,管道建立后如何同步?
  • 关于Linux下线程同步的小疑问
  • linux 如何实现两台服务器文件的同步?
  • 谁能给一个linux/unix下多线程同步的例子
  • 多台linux服务器,如何同步用户名和密码?
  • 如何在LINUX实现同步录音和放音
  • 请教:如何把一台Linux机器设置为时间同步服务器
  • linux下进程同步有mutex吗?
  • linux 下,实时同步两台机器上的数据用什么技术好?
  • linux c/c++ IP字符串转换成可比较大小的数字
  • 在win分区上安装linux和独立分区安装linux有什么区别?可以同时安装吗?(两个linux系统)
  • linux哪个版本好?linux操作系统版本详细介绍及选择方案推荐
  • 在虚拟机上安装的linux上,能像真的linux系统一样开发linux程序么?
  • secureCRT下Linux终端汉字乱码解决方法
  • 我重装window后,把linux的引导区覆盖了,进不了linux怎么办?急啊,望热心的人帮助 (现在有linux的盘)
  • Linux c字符串中不可打印字符转换成16进制
  • 安装vmware软件,不用再安装linux系统,就可以模拟linux系统了,然后可以在其上学习一下LINUX下的基本操作 了?
  • Linux常用命令介绍:更改所属用户群组或档案属性
  • 红旗Linux主机可以通过127.0.0.1访问,但如何是连网的Win2000机器通过Linux的IP去访问Linux


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3