线程相关作业
1.创建两个线程,分支线程1拷贝文件的前一部分,分支线程2拷贝文件的后一部分
#include "head.h"#define BUFFER_SIZE 1024// 线程参数结构体,包含文件名和文件偏移量
typedef struct {FILE *src_file;FILE *dest_file;long start_offset;long length;
} ThreadArgs;// 线程1:拷贝文件的前一部分
void *copyFirstPart(void *args) {ThreadArgs *thread_args = (ThreadArgs *)args;FILE *src = thread_args->src_file;FILE *dest = thread_args->dest_file;long start_offset = thread_args->start_offset;long length = thread_args->length;// 分配缓冲区char buffer[BUFFER_SIZE];fseek(src, start_offset, SEEK_SET);long bytes_read = 0;while (bytes_read < length) {size_t read_size = (length - bytes_read < BUFFER_SIZE) ? (length - bytes_read) : BUFFER_SIZE;size_t read_count = fread(buffer, 1, read_size, src);fwrite(buffer, 1, read_count, dest);bytes_read += read_count;}printf("Thread 1 finished copying the first part.\n");return NULL;
}// 线程2:拷贝文件的后一部分
void *copySecondPart(void *args) {ThreadArgs *thread_args = (ThreadArgs *)args;FILE *src = thread_args->src_file;FILE *dest = thread_args->dest_file;long start_offset = thread_args->start_offset;long length = thread_args->length;// 分配缓冲区char buffer[BUFFER_SIZE];fseek(src, start_offset, SEEK_SET);long bytes_read = 0;while (bytes_read < length) {size_t read_size = (length - bytes_read < BUFFER_SIZE) ? (length - bytes_read) : BUFFER_SIZE;size_t read_count = fread(buffer, 1, read_size, src);fwrite(buffer, 1, read_count, dest);bytes_read += read_count;}printf("Thread 2 finished copying the second part.\n");return NULL;
}int main() {FILE *src_file = fopen("head.h", "rb");if (src_file == NULL) {perror("Error opening source file");return 1;}// 获取源文件的大小fseek(src_file, 0, SEEK_END);long file_size = ftell(src_file);fseek(src_file, 0, SEEK_SET);// 创建两个目标文件FILE *dest_file1 = fopen("part1.h", "wb");FILE *dest_file2 = fopen("part2.h", "wb");if (dest_file1 == NULL || dest_file2 == NULL) {perror("Error opening destination file");fclose(src_file);return 1;}// 创建线程pthread_t thread1, thread2;// 设置线程参数ThreadArgs args1 = {src_file, dest_file1, 0, file_size / 2};ThreadArgs args2 = {src_file, dest_file2, file_size / 2, file_size - file_size / 2};// 创建两个线程,分别拷贝前后部分pthread_create(&thread1, NULL, copyFirstPart, &args1);pthread_create(&thread2, NULL, copySecondPart, &args2);// 等待线程完成pthread_join(thread1, NULL);pthread_join(thread2, NULL);// 关闭文件fclose(src_file);fclose(dest_file1);fclose(dest_file2);printf("File copy completed.\n");return 0;
}


2.创建3个线程,线程A打印A,线程B打印B,线程C打印C,要求重复打印顺序ABC (分别使用信号量和条件变量实现)
#include "head.h"sem_t semA, semB, semC; // 信号量A, B, C// 线程A打印A
void* printA(void* arg) {while (1) {sem_wait(&semA); // 等待信号量Aprintf("A\n");sleep(1);sem_post(&semB); // 释放信号量B,通知线程B执行}return NULL;
}// 线程B打印B
void* printB(void* arg) {while (1) {sem_wait(&semB); // 等待信号量Bprintf("B\n");sleep(1);sem_post(&semC); // 释放信号量C,通知线程C执行}return NULL;
}// 线程C打印C
void* printC(void* arg) {while (1) {sem_wait(&semC); // 等待信号量C printf("C\n");sleep(1);sem_post(&semA); // 释放信号量A,通知线程A执行}return NULL;
}int main() {// 初始化信号量sem_init(&semA, 0, 1); // 初始化信号量A为1,线程A先执行sem_init(&semB, 0, 0); // 初始化信号量B为0,线程B等待线程Asem_init(&semC, 0, 0); // 初始化信号量C为0,线程C等待线程B// 创建线程pthread_t threadA, threadB, threadC;pthread_create(&threadA, NULL, printA, NULL);pthread_create(&threadB, NULL, printB, NULL);pthread_create(&threadC, NULL, printC, NULL);// 等待线程结束(实际上永远不会结束,因为是无限循环)pthread_join(threadA, NULL);pthread_join(threadB, NULL);pthread_join(threadC, NULL);// 销毁信号量sem_destroy(&semA);sem_destroy(&semB);sem_destroy(&semC);return 0;
}

#include "head.h"// 定义条件变量和互斥锁
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;// 用于控制线程执行的变量
int current_thread = 0; // 0: A, 1: B, 2: C// 线程 A
void* print_A(void* arg) {while (1) {pthread_mutex_lock(&mutex);while (current_thread != 0) {pthread_cond_wait(&cond, &mutex); // 等待轮到线程 A}sleep(1);printf("A\n"); // 打印 Afflush(stdout); // 刷新输出缓冲区current_thread = 1; // 切换到线程 Bpthread_cond_broadcast(&cond); // 唤醒其他线程pthread_mutex_unlock(&mutex);}return NULL;
}// 线程 B
void* print_B(void* arg) {while (1) {pthread_mutex_lock(&mutex);while (current_thread != 1) {pthread_cond_wait(&cond, &mutex); // 等待轮到线程 B} sleep(1);printf("B\n"); // 打印 Bfflush(stdout); // 刷新输出缓冲区current_thread = 2; // 切换到线程 Cpthread_cond_broadcast(&cond); // 唤醒其他线程pthread_mutex_unlock(&mutex);}return NULL;
}// 线程 C
void* print_C(void* arg) {while (1) {pthread_mutex_lock(&mutex);while (current_thread != 2) {pthread_cond_wait(&cond, &mutex); // 等待轮到线程 C}sleep(1);printf("C\n"); // 打印 Cfflush(stdout); // 刷新输出缓冲区current_thread = 0; // 切换到线程 Apthread_cond_broadcast(&cond); // 唤醒其他线程pthread_mutex_unlock(&mutex);}return NULL;
}int main() {pthread_t thread_A, thread_B, thread_C;// 创建线程pthread_create(&thread_A, NULL, print_A, NULL);pthread_create(&thread_B, NULL, print_B, NULL);pthread_create(&thread_C, NULL, print_C, NULL);// 等待线程结束pthread_join(thread_A, NULL);pthread_join(thread_B, NULL);pthread_join(thread_C, NULL);return 0;
}

相关文章:
线程相关作业
1.创建两个线程,分支线程1拷贝文件的前一部分,分支线程2拷贝文件的后一部分 #include "head.h"#define BUFFER_SIZE 1024// 线程参数结构体,包含文件名和文件偏移量 typedef struct {FILE *src_file;FILE *dest_file;long start_o…...
通义万相2.1开源版本地化部署攻略,生成视频再填利器
2025 年 2 月 25 日晚上 11:00 通义万相 2.1 开源发布,前两周太忙没空搞它,这个周末,也来本地化部署一个,体验生成效果如何,总的来说,它在国内文生视频、图生视频的行列处于领先位置,…...
【模拟CMOS集成电路设计】带隙基准(Bandgap)设计与仿真(基于运放的电流模BGR)
【模拟CMOS集成电路设计】带隙基准(Bandgap)设计与仿真 前言工程文件&部分参数计算过程,私聊~ 一、 设计指标指标分析: 二、 电路分析三、 仿真3.1仿真电路图3.2仿真结果(1)运放增益(2)基准温度系数仿真(3)瞬态启动仿真(4)静态…...
如何选择国产串口屏?
目录 1、迪文 2、淘晶驰 3、广州大彩 4、金玺智控 5、欣瑞达 6、富莱新 7、冠显 8、有彩 串口屏,顾名思义,就是通过串口通信接口(如RS232、RS485、TTL UART等)与主控设备进行通信的显示屏。其核心功能是显示信息和接收输入…...
Solana中的程序派生地址(PDAs):是什么,为什么,以及如何?
程序派生地址 (PDA) 在 Solana 中的应用:什么、为什么和如何? 在学习 Solana 时,你会经常听到关于 程序派生地址 (PDAs) 的讨论。它们就像这样 —— 强大、多功能,而且最重要的是,稍微被误解。如果你是一个开发者&…...
利用FatJar彻底解决Jar包冲突(一)
利用FatJar彻底解决Jar包冲突 序FatJar的加载与隔离⼀、 FatJar概念⼆、FatJar的加载三、FatJar的隔离四、隔离机制验证五、 FatJar的定位六、 打包注意点 序 今天整理旧电脑里的资料,偶然翻到大概10年前实习时写的笔记,之前经常遇到Java依赖冲突的问题…...
Spring MVC笔记
01 什么是Spring MVC Spring MVC 是 Spring 框架中的一个核心模块,专门用于构建 Web 应用程序。它基于经典的 MVC 设计模式(Model-View-Controller),但通过 Spring 的特性(如依赖注入、注解驱动)大幅简化了…...
BurpSuite插件jsEncrypter使用教程
一、前言 在当今Web应用安全测试中,前端加密已成为开发者保护敏感数据的常用手段。然而,这也给安全测试人员带来了挑战,传统的抓包方式难以获取明文数据,测试效率大打折扣。BurpSuite作为一款强大的Web安全测试工具,其…...
【C#实现手写Ollama服务交互,实现本地模型对话】
前言 C#手写Ollama服务交互,实现本地模型对话 最近使用C#调用OllamaSharpe库实现Ollama本地对话,然后思考着能否自己实现这个功能。经过一番查找,和查看OllamaSharpe源码发现确实可以。其实就是开启Ollama服务后,发送HTTP请求&a…...
Android Glide 框架线程管理模块原理的源码级别深入分析
一、引言 在现代的 Android 应用开发中,图片加载是一个常见且重要的功能。Glide 作为一款广泛使用的图片加载框架,以其高效、灵活和易用的特点受到了开发者的青睐。其中,线程管理模块是 Glide 框架中至关重要的一部分,它负责协调…...
每天记录一道Java面试题---day32
MySQL索引的数据结构、各自优劣 回答重点 B树:是一个平衡的多叉树,从根节点到每个叶子节点的高度差不超过1,而且同层级的节点间有指针相互连接。在B树上的常规检索,从根节点到叶子节点的搜索效率基本相当,不会出现大…...
Vue3 Pinia 符合直觉的Vue.js状态管理库
Pinia 符合直觉的Vue.js状态管理库 什么时候使用Pinia 当两个关系非常远的组件,要传递参数时使用Pinia组件的公共参数使用Pinia...
深度学习与大模型基础-向量
大家好!今天我们来聊聊向量(Vector)。别被这个词吓到,其实向量在我们的生活中无处不在,只是我们没注意罢了。 1. 向量是什么? 简单来说,向量就是有大小和方向的量。比如你从家走到学校&#x…...
【网络编程】完成端口 IOCP
10.11 完成端口 10.11.1 基本概念 完成端口的全称是I/O 完成端口,英文为IOCP(I/O Completion Port) 。IOCP是一个异 步I/O 的 API, 可以高效地将I/O 事件通知给应用程序。与使用select() 或是其他异步方法不同 的是,一个套接字与一个完成端口关联了起来…...
《苍穹外卖》SpringBoot后端开发项目重点知识整理(DAY1 to DAY3)
目录 一、在本地部署并启动Nginx服务1. 解压Nginx压缩包2. 启动Nginx服务3. 验证Nginx是否启动成功: 二、导入接口文档1. 黑马程序员提供的YApi平台2. YApi Pro平台3. 推荐工具:Apifox 三、Swagger1. 常用注解1.1 Api与ApiModel1.2 ApiModelProperty与Ap…...
管理网络安全
防火墙在 Linux 系统安全中有哪些重要的作用? 防火墙作为网络安全的第一道防线,能够根据预设的规则,对进出系统的网络流量进行严格筛选。它可以阻止未经授权的外部访问,只允许符合规则的流量进入系统,从而保护系统免受…...
明日直播|Go IoT 开发平台,开启万物智联新征程
在物联网技术飞速发展的当下,物联网行业却深受协议碎片化、生态封闭、开发低效等难题的困扰。企业和开发者们渴望找到一个能突破这些困境,实现高效、便捷开发的有力工具。 3 月 11 日(星期二)19:00,GitCode 特别邀请独…...
系统架构设计师—系统架构设计篇—软件架构风格
文章目录 概述经典体系结构风格数据流风格批处理管道过滤器对比 调用/返回风格主程序/子程序面向对象架构风格层次架构风格 独立构件风格进程通信事件驱动的系统 虚拟机风格解释器基于规则的系统 仓库风格(数据共享风格)数据库系统黑板系统超文本系统 闭…...
【MySQL_04】数据库基本操作(用户管理--配置文件--远程连接--数据库信息查看、创建、删除)
文章目录 一、MySQL 用户管理1.1 用户管理1.11 mysql.user表详解1.12 添加用户1.13 修改用户权限1.14 删除用户1.15 密码问题 二、MySQL 配置文件2.1 配置文件位置2.2 配置文件结构2.3 常用配置参数 三、MySQL远程连接四、数据库的查看、创建、删除4.1 查看数据库4.2 创建、删除…...
【Zinx】Day5-Part4:Zinx 的连接属性设置
目录 Day5-Part4:Zinx 的连接属性设置给连接添加连接配置的接口连接属性方法的实现 测试 Zinx-v1.0总结 Day5-Part4:Zinx 的连接属性设置 在 Zinx 当中,我们使用 Server 来开启服务并监听指定的端口,当接收到来自客户端的连接请求…...
盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来
一、破局:PCB行业的时代之问 在数字经济蓬勃发展的浪潮中,PCB(印制电路板)作为 “电子产品之母”,其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透,PCB行业面临着前所未有的挑战与机遇。产品迭代…...
三维GIS开发cesium智慧地铁教程(5)Cesium相机控制
一、环境搭建 <script src"../cesium1.99/Build/Cesium/Cesium.js"></script> <link rel"stylesheet" href"../cesium1.99/Build/Cesium/Widgets/widgets.css"> 关键配置点: 路径验证:确保相对路径.…...
基于ASP.NET+ SQL Server实现(Web)医院信息管理系统
医院信息管理系统 1. 课程设计内容 在 visual studio 2017 平台上,开发一个“医院信息管理系统”Web 程序。 2. 课程设计目的 综合运用 c#.net 知识,在 vs 2017 平台上,进行 ASP.NET 应用程序和简易网站的开发;初步熟悉开发一…...
Day131 | 灵神 | 回溯算法 | 子集型 子集
Day131 | 灵神 | 回溯算法 | 子集型 子集 78.子集 78. 子集 - 力扣(LeetCode) 思路: 笔者写过很多次这道题了,不想写题解了,大家看灵神讲解吧 回溯算法套路①子集型回溯【基础算法精讲 14】_哔哩哔哩_bilibili 完…...
UE5 学习系列(三)创建和移动物体
这篇博客是该系列的第三篇,是在之前两篇博客的基础上展开,主要介绍如何在操作界面中创建和拖动物体,这篇博客跟随的视频链接如下: B 站视频:s03-创建和移动物体 如果你不打算开之前的博客并且对UE5 比较熟的话按照以…...
Go 语言接口详解
Go 语言接口详解 核心概念 接口定义 在 Go 语言中,接口是一种抽象类型,它定义了一组方法的集合: // 定义接口 type Shape interface {Area() float64Perimeter() float64 } 接口实现 Go 接口的实现是隐式的: // 矩形结构体…...
select、poll、epoll 与 Reactor 模式
在高并发网络编程领域,高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表,以及基于它们实现的 Reactor 模式,为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。 一、I…...
Mobile ALOHA全身模仿学习
一、题目 Mobile ALOHA:通过低成本全身远程操作学习双手移动操作 传统模仿学习(Imitation Learning)缺点:聚焦与桌面操作,缺乏通用任务所需的移动性和灵活性 本论文优点:(1)在ALOHA…...
React---day11
14.4 react-redux第三方库 提供connect、thunk之类的函数 以获取一个banner数据为例子 store: 我们在使用异步的时候理应是要使用中间件的,但是configureStore 已经自动集成了 redux-thunk,注意action里面要返回函数 import { configureS…...
动态 Web 开发技术入门篇
一、HTTP 协议核心 1.1 HTTP 基础 协议全称 :HyperText Transfer Protocol(超文本传输协议) 默认端口 :HTTP 使用 80 端口,HTTPS 使用 443 端口。 请求方法 : GET :用于获取资源,…...
