IO学习---->线程

news2025/4/22 8:40:20

1.创建两个线程,分支线程1拷贝文件的前一部分,分支线程2拷贝文件的后一部分

#include <head.h>
sem_t sem;
long half_size = 0;  // 全局变量,供所有线程共享
 
void* product(void *arg) 
{
    FILE *src = fopen("IO.text", "rb");
    FILE *dest = fopen("IO1.text", "rb+");
    if (src == NULL || dest == NULL) 
    {
        perror("文件打开失败");
        pthread_exit(NULL);
    }
 
    char buf[128] = {0};
    size_t n;
    while ((n = fread(buf, 1, sizeof(buf), src)) > 0) 
    {
        if (ftell(src) > half_size) 
        {
            n -= (ftell(src) - half_size); // 防止越界写入
        }
        if (fwrite(buf, 1, n, dest) != n) 
        {
            perror("线程1写入失败");
            fclose(src);
            fclose(dest);
            pthread_exit(NULL);
        }
 
        if (ftell(src) >= half_size) 
        {
            break;  // 退出循环,完成前半部分写入
        }
    }
 
    printf("线程1完成文件前半部分的拷贝。\n");
    fclose(src);
    fclose(dest);
    sem_post(&sem);  // 通知消费者线程
    pthread_exit(NULL);
}
 
void* consumer(void *arg) 
{
    sem_wait(&sem);  // 等待生产者完成前半部分
 
    FILE *src_child = fopen("IO.text", "rb");
    FILE *dest_child = fopen("IO1.text", "rb+");
    if (src_child == NULL || dest_child == NULL) 
    {
        perror("文件打开失败");
        pthread_exit(NULL);
    }
 
    fseek(src_child, half_size, SEEK_SET);
    fseek(dest_child, half_size, SEEK_SET);
 
    char buf[128] = {0};
    size_t n;
    while ((n = fread(buf, 1, sizeof(buf), src_child)) > 0) 
    {
        if (fwrite(buf, 1, n, dest_child) != n) 
        {
            perror("线程2写入失败");
            fclose(src_child);
            fclose(dest_child);
            pthread_exit(NULL);
        }
    }
 
    printf("线程2完成文件后半部分的拷贝。\n");
    fclose(src_child);
    fclose(dest_child);
    pthread_exit(NULL);
}
 
int main(int argc, const char *argv[]) 
{
    FILE *src = fopen("IO.text", "rb");
    if (src == NULL) 
    {
        perror("打开源文件失败");
        return -1;
    }
    FILE *dest = fopen("IO1.text", "wb");
    if (dest == NULL) 
    {
        perror("打开目标文件失败");
        fclose(src);
        return -1;
    }
 
    // 计算文件大小和一半位置
    fseek(src, 0, SEEK_END);
    long file_size = ftell(src);
    rewind(src);
    half_size = file_size / 2;
 
    sem_init(&sem, 0, 0);
 
    pthread_t tid1, tid2;
    if ((errno = pthread_create(&tid1, NULL, product, NULL)) != 0) 
    {
        perror("pthread_create error");
    }
    if ((errno = pthread_create(&tid2, NULL, consumer, NULL)) != 0) 
    {
        perror("pthread_create error");
    }
 
    pthread_join(tid1, NULL);
    pthread_join(tid2, NULL);
 
    sem_destroy(&sem);
    fclose(src);
    fclose(dest);
    return 0;
}

2.创建3个线程,线程A打印A,线程B打印B,线程C打印C,要求重复打印顺序ABC  (分别使用信号量和条件变量实现)

#include <head.h>
//创建信号量
sem_t sema;
sem_t semb;
sem_t semc;
void* pta(void* arg)
{
	while(1)
	{
		//sleep(1);
		sem_wait(&sema);
		printf("A\n");
		sem_post(&semb);
	}
	pthread_exit(NULL);
}

void* ptb(void* arg)
{
	while(1)
	{
		//sleep(1);
		sem_wait(&semb);
		printf("B\n");
		sem_post(&semc);
	}
	pthread_exit(NULL);
}

void* ptc(void* arg)
{
	while(1)
	{
		sleep(1);
		sem_wait(&semc);
		printf("C\n");
		sem_post(&sema);
	}
	pthread_exit(NULL);
}

int main(int argc, const char *argv[])
{
	sem_init(&sema,0,1);
	sem_init(&semb,0,0);
	sem_init(&semc,0,0);

	pthread_t std1,std2,std3;
	if((errno=pthread_create(&std1,NULL,pta,NULL))!=0)
		PRINT_ERROR("pthread_create error");
	if((errno=pthread_create(&std2,NULL,ptb,NULL))!=0)
		PRINT_ERROR("pthread_create error");
	if((errno=pthread_create(&std3,NULL,ptc,NULL))!=0)
		PRINT_ERROR("pthread_create error");

	pthread_join(std1,NULL);
	pthread_join(std2,NULL);
	pthread_join(std3,NULL);

	sem_destroy(&sema);
	sem_destroy(&semb);
	sem_destroy(&semc);
	return 0;
}
#include <head.h>
//定义并初始化条件变量
pthread_cond_t cond3 =PTHREAD_COND_INITIALIZER;
pthread_cond_t cond1 =PTHREAD_COND_INITIALIZER;
pthread_cond_t cond2 =PTHREAD_COND_INITIALIZER;
//定义并初始化互斥锁
pthread_mutex_t mutex=PTHREAD_MUTEX_INITIALIZER;

//设置标志位取消sleep

int flag =0;
//pthread_cond_t cond;
//pthread_cond_init(&cond,NULL);
void* pta(void *arg)
{
    while(1)
    {
    //  sleep(1);//放弃CPU资源,使cpu一定先使用消费者线程

        pthread_mutex_lock(&mutex);
        if(flag!=0)

        pthread_cond_wait(&cond3,&mutex);
            printf("A\n");

            flag=1;
        pthread_cond_signal(&cond1);
        pthread_mutex_unlock(&mutex);

    }
    pthread_exit(NULL);
}
void* ptb(void *arg)
{
    while(1)
    {
        //上锁
        pthread_mutex_lock(&mutex);
        if(flag!=1)
        //阻塞休眠并解锁
        pthread_cond_wait(&cond1,&mutex);

            printf("B\n");

            flag=2;
        //解锁
        pthread_mutex_unlock(&mutex);
        pthread_cond_signal(&cond2);


    }
    pthread_exit(NULL);
}
void* ptc(void *arg)
{
    while(1)
    {
        //上锁
        pthread_mutex_lock(&mutex);
        if(flag!=2)
        //阻塞休眠并解锁
        pthread_cond_wait(&cond2,&mutex);

            printf("C\n");

            flag=0;
        //解锁
        pthread_mutex_unlock(&mutex);
        pthread_cond_signal(&cond3);


    }
    pthread_exit(NULL);
}
int main(int argc, const char *argv[])
{

    pthread_t std1,std2,std3,std4,std5;
    if( (errno=pthread_create(&std1,NULL,pta,NULL))!=0)
    {
        PRINT_ERROR("pthread_create error");
    }
    if((errno=pthread_create(&std2,NULL,ptb,NULL))!=0)
    {                                                                              
        PRINT_ERROR("pthread_create error");
    }
    if((errno=pthread_create(&std3,NULL,ptc,NULL))!=0)
    {
        PRINT_ERROR("pthread_create error");
    }

    pthread_join(std1,NULL);
    pthread_join(std2,NULL);
    pthread_join(std3,NULL);

    pthread_mutex_destroy(&mutex);
    pthread_cond_destroy(&cond3);
    pthread_cond_destroy(&cond1);
    pthread_cond_destroy(&cond2);

    return 0;
}
                                                                                   

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

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

相关文章

个人记录,Unity资源解压和管理插件

就是经典的两个AssetStudio 和 Ripper 没有什么干货&#xff0c;就是记录一下&#xff0c;内容没有很详细 AssetStudio 说错了&#xff0c;AssetStudio比较出名&#xff08;曾经&#xff09;&#xff0c;但好像堕落了 这个工具有个好处就是分类选择&#xff0c;&#xff08;…

day19-前端Web——Vue3+TS+ElementPlus

目录 1. Vue工程化1.1 介绍1.2 环境准备1.2.1 NodeJS安装双击安装包选择安装目录验证NodeJS环境变量配置npm的全局安装路径 1.3 Vue项目-创建1.4 Vue项目开发流程1.5 API风格1.6 案例 2. TS2.1 概述2.2 快速入门2.3 常用类型2.3.1 基础类型2.3.2 联合类型2.3.3 函数类型2.3.4 对…

隐私保护在 Facebook 用户身份验证中的应用

在这个数字化的时代&#xff0c;个人隐私保护成为了公众关注的焦点。社交媒体巨头 Facebook 作为全球最大的社交平台之一&#xff0c;拥有数十亿用户&#xff0c;其在用户身份验证过程中对隐私保护的重视程度直接影响着用户的安全感和信任度。本文将探讨 Facebook 在用户身份验…

【JavaWeb学习Day23】

Maven高级 分模块设计与开发 分模块设计&#xff1a;将一个大项目分成若干个子模块&#xff0c;方便项目的维护、扩展&#xff0c;也方便模块间的相互引用&#xff0c;资源共享。 策略&#xff1a; 1.策略一&#xff1a;按照功能模块拆分&#xff0c;比如&#xff1a;公共组…

个人记录的一个插件,Unity-RuntimeMonitor

没有什么干货,仅仅是个人的记录 基于GUI做的一个工具:好处就是Monitor必须,Unity天然支持实时的Monitor;唯一不好处,就是默认字体太小了,layout居中,居右也是要自行设计的。 (下面文字是有一点点写错,但意思和功能就很牛逼了;并不是都按2 x shift,而是一个 shift 添…

【NexLM 开源系列】如何封装多个大模型 API 调用

&#x1f31f; 在这系列文章中&#xff0c;我们将一起探索如何搭建一个支持大模型集成项目 NexLM 的开发过程&#xff0c;从 架构设计 到 代码实战&#xff0c;逐步搭建一个支持 多种大模型&#xff08;GPT-4、DeepSeek 等&#xff09; 的 一站式大模型集成与管理平台&#xff…

Git和GitHub基础教学

文章目录 1. 前言2. 历史3. 下载安装Git3.1 下载Git3.2 安装Git3.3 验证安装是否成功 4. 配置Git5. Git基础使用5.1 通过Git Bash使用5.1.1 创建一个新的仓库。5.1.1.1 克隆别人的仓库5.1.1.2 自己创建一个本地仓库 5.1.2 管理存档 5.2 通过Visual Studio Code使用 6. Git完成远…

笔记六:单链表链表介绍与模拟实现

在他一生中&#xff0c;从来没有人能够像你们这样&#xff0c;以他的视角看待这个世界。 ---------《寻找天堂》 目录 文章目录 一、什么是链表&#xff1f; 二、为什么要使用链表&#xff1f; 三、 单链表介绍与使用 3.1 单链表 3.1.1 创建单链表节点 3.1.2 单链表的头插、…

坐落于杭州的电商代运营公司品融电商

坐落于杭州的电商代运营公司品融电商 在中国电商行业蓬勃发展的浪潮中&#xff0c;品融电商&#xff08;PINKROON&#xff09;作为一家扎根杭州的新锐品牌管理公司&#xff0c;凭借其独特的全域增长方法论和实战经验&#xff0c;迅速崛起为行业标杆。自2020年成立以来&#x…

【算法学习之路】8.栈和队列

栈和队列 前言一.简介二.题目12 前言 我会将一些常用的算法以及对应的题单给写完&#xff0c;形成一套完整的算法体系&#xff0c;以及大量的各个难度的题目&#xff0c;目前算法也写了几篇&#xff0c;题单正在更新&#xff0c;其他的也会陆陆续续的更新&#xff0c;希望大家点…

OpenMCU(三):STM32F103 FreeRTOS移植

概述 本文主要描述了STM32F103移植FreeRTOS的简要步骤。移植描述过程中&#xff0c;忽略了Keil软件的部分使用技巧。默认读者熟练使用Keil软件。本文的描述是基于OpenMCU_RTOS这个工程&#xff0c;该工程已经下载放好了移植STM32F103 FreeRTOS的所有文件 OpenMCU_RTOS工程的愿景…

大数据 spark hive 总结

Apache Spark 简介 是一个开源的统一分析引擎&#xff0c;专为大规模数据处理而设计。它提供了高级API&#xff0c;支持Java、Scala、Python和R语言&#xff0c;并且包含了一个优化过的执行引擎&#xff0c;该引擎支持循环计算&#xff08;如机器学习算法&#xff09;和交互式…

小程序开发总结

今年第一次帮别人做小程序。 从开始动手到完成上线&#xff0c;一共耗时两天。AI 让写代码变得简单、高效。 不过&#xff0c;小程序和 Flutter 等大厂开发框架差距实在太大&#xff0c;导致我一开始根本找不到感觉。 第一&#xff0c;IDE 不好用&#xff0c;各种功能杂糅在…

QLoggingCategory类使用

QLoggingCategory类使用 QLoggingCategory的概述 QLoggingCategory是Qt的日志策略类&#xff1b;可以通过声明不同的日志策略对象来输出不同的日志信息。打印信息类型如下&#xff1a;宏 Q_DECLARE_LOGGING_CATEGORY(name) 定义一个返回QLoggingCategory对象函数&#xff0c;…

GPU加速生信分析-宏基因组MAG去污染

Deepurify利用多模态深度语言模型来过滤污染的基因组&#xff0c;从而提高了宏基因组组装基因组&#xff08;MAGs&#xff09;的质量&#xff0c;并且可以利用GPU加速。 宏基因组组装的基因组 &#xff08;MAG&#xff09; 为使用宏基因组测序数据探索微生物暗物质提供了有价值…

数据结构(蓝桥杯常考点)

数据结构 前言&#xff1a;这个是针对于蓝桥杯竞赛常考的数据结构内容&#xff0c;基础算法比如高精度这些会在下期给大家总结 数据结构 竞赛中&#xff0c;时间复杂度不能超过10的7次方&#xff08;1秒&#xff09;到10的8次方&#xff08;2秒&#xff09; 空间限制&#x…

从0到1入门Linux

一、常用命令 ls 列出目录内容 cd切换目录mkdir创建新目录rm删除文件或目录cp复制文件或目录mv移动或重命名文件和目录cat查看文件内容grep在文件中查找指定字符串ps查看当前进程状态top查看内存kill终止进程df -h查看磁盘空间存储情况iotop -o直接查看比较高的磁盘读写程序up…

React:类组件(中)

dangerouslySetInnerHTML React写进{}内的东西&#xff0c;不允许被当作代码块解析&#xff0c;是为了防止xss攻击和代码注入 XSS&#xff08;跨站脚本攻击&#xff0c;Cross-Site Scripting&#xff09; 是一种常见的安全漏洞&#xff0c;攻击者通过注入恶意脚本到网页中&…

第六次CCF-CSP认证(含C++源码)

第六次CCF-CSP认证 数位之和&#xff08;easy&#xff09;思路及AC代码遇到的问题 开心消消乐&#xff08;easy&#xff09;思路及AC代码 画图&#xff08;mid&#xff09;思路及AC代码 数位之和&#xff08;easy&#xff09; 题目链接 思路及AC代码 既然题目要求我们输出各位…

SpringBoot 如何调用 WebService 接口

前言 调用WebService接口的方式有很多&#xff0c;今天记录一下&#xff0c;使用 Spring Web Services 调用 SOAP WebService接口 一.导入依赖 <!-- Spring Boot Web依赖 --><dependency><groupId>org.springframework.boot</groupId><artifactId…