进程锁 读写文件的小例子 C++代码

代码

#include <unistd.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <cassert>
#include <pthread.h>
#include <cstdio>
#include <cstdlib>
#include <fstream>
#include <iostream>
#include <sstream>
#include <cstring>
#include <sys/wait.h>/***  返回一片共享内存标识符,用于后续获取该共享内存,以及销毁该共享内存*/class mutex_function{public:mutex_function() = default;// 定义进程锁结构体typedef struct mutex_struct{pthread_rwlock_t lock;pthread_rwlockattr_t lock_attr;int memory_addr;}mutex_struct_t;int create_share_memory_id(const int index_of_key,size_t md5){const char* file_path = "./";key_t id = ftok(file_path,md5);const int share_memory_id = shmget(id,md5,IPC_CREAT | 0666);if (share_memory_id == -1){std::cout << "share_memory_id error!\n" << std::endl;exit(EXIT_FAILURE);}else{return share_memory_id;}}// 初始化进程锁结构体int init_mutex_struct(mutex_struct_t* mutex_ptr){pthread_rwlockattr_init(&(mutex_ptr->lock_attr));pthread_rwlockattr_setpshared(&(mutex_ptr->lock_attr),PTHREAD_PROCESS_SHARED);pthread_rwlock_init(&(mutex_ptr->lock),&(mutex_ptr->lock_attr));return 0;}// 在共享内存上定义进程锁结构体并且返回其位置mutex_struct_t* mutex_struct_addr(const int index){int memory_id = create_share_memory_id(index,sizeof (mutex_struct_t));mutex_struct_t * share_ptr = (mutex_struct_t* )shmat(memory_id, 0, 0);if (share_ptr == (void *)-1){std::cout << "share_ptr error!\n" << std::endl;exit(EXIT_FAILURE);}share_ptr->memory_addr = memory_id;assert(init_mutex_struct(share_ptr)== 0);return share_ptr;}// 销毁进程锁结构体,利用其memory_addr变量索引到其占用的共享内存并销毁const int destroy_mutex_memory(mutex_struct_t * mutex_struct){pthread_rwlock_destroy(&(mutex_struct->lock));pthread_rwlockattr_destroy(&(mutex_struct->lock_attr));assert(shmctl(mutex_struct->memory_addr,IPC_RMID, nullptr) == 0);return 0;}public:mutex_struct_t* mutex_struct_ptr;};//read file
void read_file(const std::string file_name,int id){std::ifstream fp(file_name,std::ios::binary);std::stringstream ss;ss << fp.rdbuf();std::cout << "线程"<< id << "抢占了资源,输出:" << ss.str() << std::endl;sleep(3);std::cout << "线程"<< id << "释放了资源...\n" << std::endl;fp.close();
}
//write file
void write_file(const char *file_name,char *str,int id){FILE *fd = fopen(file_name,"a+");if (fd == nullptr){printf("fd is nullptr and open file fail\n");} else{std::cout <<"线程"<< id << "抢占了资源" << std::endl;fwrite(str,strlen(str),1,fd);char *next = "\0";fwrite(next,strlen(next),1,fd);sleep(5);std::cout << "线程"<< id << "释放了资源...\n" << std::endl;}fclose(fd);
}
int main(){std::string file_name = "/home/gsc/Projects/1.txt";char * str1 = "progress 1\n";char * str2 = "progress 2\n";// 创建自定义进程锁hsm::hal::mutex_function mutex_function_object;mutex_function_object.mutex_struct_ptr = mutex_function_object.mutex_struct_addr(111);pid_t c_pid = fork();if (c_pid == -1) {perror("fork");exit(EXIT_FAILURE);} else if (c_pid > 0) {std::cout << "Parent process " << getpid() << std::endl;
//            pthread_rwlock_rdlock(&(mutex_function_object.mutex_struct_ptr->lock));
//            read_file(file_name,getpid());
//            pthread_rwlock_unlock(&(mutex_function_object.mutex_struct_ptr->lock));pthread_rwlock_wrlock(&(mutex_function_object.mutex_struct_ptr->lock));write_file(file_name.c_str(),str1,getpid());pthread_rwlock_unlock(&(mutex_function_object.mutex_struct_ptr->lock));wait(nullptr);} else {std::cout << "Child process " << getpid() << std::endl;pthread_rwlock_rdlock(&(mutex_function_object.mutex_struct_ptr->lock));read_file(file_name,getpid());pthread_rwlock_unlock(&(mutex_function_object.mutex_struct_ptr->lock));//        pthread_rwlock_wrlock(&(mutex_function_object.mutex_struct_ptr->lock));
//        write_file(file_name.c_str(),str2,getpid());
//        pthread_rwlock_unlock(&(mutex_function_object.mutex_struct_ptr->lock));exit(EXIT_SUCCESS);}return 0;
}

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/446784.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java 中sleep()与wait()的区别

目录一、原理不同二、锁的处理机制不同三、使用的区域不同四、异常捕获不同五、总结一、原理不同 sleep()是Thread类的静态方法&#xff0c;是线程用来控制自身流程的&#xff0c;它会使此线程暂停执行指定的时间&#xff0c;而把执行机会让给其他的线程&#xff0c;等到计时时…

android--地图定位打卡

获取位置信息 1)位置信息 GPS卫星定位,在室外适用 基站(3个基站交叉,锁定手机位置)–基站定位不平均,有些地方实现不了3点定位 网络定位–通过手机IP地址,去锁定位置(消耗流量,对网络有要求) 谷歌地图的大致实现思路(通用) 2)实现定位功能的重要类 在百度地图和高德地图中不…

Android 将整形颜色值转换成String类型

转换方法&#xff1a; val hexColor String.format("#%06X", [0xFFFFFF or intColor]);转换结果&#xff1a; #F2EADA

MacOS 的 zsh 和 bash 切换

目录一、从 bash 切换到 zsh1、使用系统自带的 zsh2、使用第三方的 zsh2.1、Clone代码到本地2.2、备份你已存在的 ~/.zshrc 文件2.3、新建一份新的 zsh 配置文件2.4、改变默认的shell脚本二、从 zsh 切换回 bash三、zsh 和 bash 的环境变量zsh、bash 都是shell&#xff0c;zsh …

android--在命令行中生成Android的数字证书keystore文件

标题 生成 密钥口令为 13458977480 密钥库口令为 13458977480 存放位置 查看证书的相关资料

linux查看系统日志

cd /var/log/gscubuntu:/var/log$ tail -f syslog

IDEA 创建 SpringBoot 项目

目录一、新建Springboot项目第一步&#xff1a;新建一个Springboot项目第二步&#xff1a;选择项目模板第三步&#xff1a;设置项目配置第四步&#xff1a;设置项目依赖第五步&#xff1a;设置项目名称及路径第六步&#xff1a;创建完成二、测试及运行1、测试代码2、设置默认端…

VC++软件

一个main fatal error LNK1169: 找到一个或多个多重定义的符号–报错 一个项目即一个程序&#xff0c;多个文件只能有一个main函数 删除掉多余的main 控制台按enter键闪退 在代码中加上 #include<stdlib.h> getchar();//让控制台停留 system("pause");//让…

IDEA 将 SpringBoot 项目打包成jar

目录一、打包配置1、File -> Project Structure2、Project Structure3、设置启动类及META-INF4、设置打包输出目录二、打包1、Build -> Artifacts2、Build三、查看打包文件四、运行新建SpringBoot项目&#xff1a;IDEA 创建 SpringBoot 项目 一、打包配置 1、File -> …

2014年考研英语一完型填空知识点

单词 单词释意commitv犯罪sufficientlyadv足够gainfuladj有收益的socioeconomicadj社会经济的discontentn/v不满意householdn家庭supervisionn监督offensiveadj冒犯的conditionn状态casualadj随意的causaladj因果关系的establishedadj已确立,公认的interactionn相互作用或影响…

如何查看软连接,以及相关注意事项

使用命令 ls -il 图片显示 参考链接 Linux 命令之软连接详解Linux软连接 查看/创建/删除

Git SSH key配置

一、检查本地Git配置 用如下命令&#xff08;如未特别说明&#xff0c;所有命令均默认在Git Bash工具下执行&#xff09;检查一下用户名和邮箱是否配置&#xff08;github支持我们用用户名或邮箱登录&#xff09;&#xff1a; git config --global --list 显示信息如下&#…

2014年英语一阅读理解Text1

单词解释chancellorn总理upfrontadj坦率的eligibleadj有资格的,合格的fortnightlyadv两星期一次的sign on办理reformn改革subsidisev补助zealn热情taxpayern纳税人claimantn(因失业)领取救济金者skip down边跳边走prospectn前景psychologicallyadv心理上地excludev不包括crucia…

HTTPS 工作原理

一、简介 HTTPS对于客户端开发人员来说并没有什么需要特别注意的地方&#xff0c;因为代码和写HTTP请求时并没有什么两样。但也正是因为这个原因&#xff0c;导致许多客户端开发人员对HTTPS并不了解&#xff0c;只知道它是安全的加密网络传输&#xff0c;对其具体的工作原理却一…

解决VM虚拟机中ubuntu系统上不了网的问题

最简单的方式 关闭虚拟机在对应的虚拟机上右键&#xff0c;点击设置&#xff0c;找到网络适配器&#xff0c;点击移除&#xff0c;再次点击添加&#xff0c;将网络适配器再次添加回来&#xff0c;点击确定重启虚拟机如果第一种方式解决不了问题&#xff0c;请使用第二种方式 …

Android Glide图片加载框架(一)基本用法

文章目录一、前言二、简介三、基本用法第一步&#xff1a;调用 Glide.with() 方法创建加载图片的实例第二步&#xff1a;调用 load() 方法指定待加载的图片资源第三步&#xff1a;调用 into() 方法绑定显示控件总结四、扩展用法1、占位图2、指定图片格式3、指定图片大小Android…

codeforces71A-C语言解题报告

71A题目地址 题目解答 1.输入单词 超过10个字母的单词,输出第一个字母中间个数最后一个字母 没有超过的,直接输出 2.循环读取输入 在for循环内部,接收用户输入的单词 知识点 1.接收用户输入 int a; scanf("%d",&a); 2.字符串数组 C语言中没有专门的字符串数组…

操作系统 进程 学习以及思考

进程管理逻辑图 将多个程序拷贝到进程中&#xff0c;占用内存&#xff0c;如图扇形区域&#xff0c;当酷狗进程需要资源的时候&#xff0c;会通过I/O子系统取用资源的过程中&#xff0c;会放弃对cpu的占用&#xff0c;cpu就会处理别的进程&#xff0c;因此提高了cpu的利用率&am…

Android Glide图片加载框架(二)源码解析之with()

文章目录一、前言二、如何阅读源码三、源码解析1、with()Android Glide图片加载框架系列文章 Android Glide图片加载框架&#xff08;一&#xff09;基本用法 Android Glide图片加载框架&#xff08;二&#xff09;源码解析之with() Android Glide图片加载框架&#xff08;二…