IO学习系列之使用多线程复制同一个文件内容

这篇具有很好参考价值的文章主要介绍了IO学习系列之使用多线程复制同一个文件内容。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

  • 实例要求:
  • 使用多线程复制同一个文件内容;
  • 实例分析:
  • 1.创建两个线程,即线程1、线程2设置光标在指定文件中的偏移量,实现对同一个文件的复制。
  • 2.比如:可以指定线程1复制文件内容的前一半,而线程2复制文件内容的后一半
  • 3.根据时间片轮询法则,最终线程1和线程2可以把同一个文件复制成功。
  • 相关的线程库接口函数如下:
  • pthread_create函数:
#include <pthread.h>

int pthread_create(pthread_t *thread, const pthread_attr_t *attr,

                   void *(*start_routine) (void *), void *arg);
/*
功能:

		创建一个线程

参数:

    	thread: 创建的线程的线程号

    	attr: 线程属性 传 NULL 表示使用默认属性

    	start_routine: 线程处理函数--线程体

    	arg: 线程处理函数的参数

返回值:

    	成功  	0

    	失败  	错误码 
*/
  • pthread_self函数:
#include <pthread.h>

pthread_t pthread_self(void);
/*

功能:

		返回当前线程的线程号
	
参数:
		无

返回值:

		总是成功 当前线程的线程号
*/
  • pthread_exit函数:
#include <pthread.h>

void pthread_exit(void *retval);
/*
功能:

		退出线程

    	线程中调用 exit 会导致整个进程退出

    	所以 如果只想退出线程 需要使用 pthread_exit
参数:

		退出线程时的返回值 给 pthread_join 使用的

返回值:

		无
*/
  • pthread_join函数:
#include <pthread.h>

int pthread_join(pthread_t thread, void **retval);
/*
功能:

		阻塞等待一个结合态的线程退出,回收资源

		如果没有线程退出,该函数已知阻塞

参数:

		thread:线程的id

		retval:接受pthread_exit返回的值的 如果不关心 传NULL即可

返回值:

    	成功  0

    	失败  错误码
*/
  • pthread_detach函数:
#include <pthread.h>

int pthread_detach(pthread_t thread);
/*
功能:	

		标记一个线程为分离态

参数:	

		thread:线程的id

返回值:

		成功 	 0

		失败  	错误码
*/
  • pthread_cancel函数:
#include <pthread.h>

int pthread_cancel(pthread_t thread);
/*
功能:

	给线程发一个取消信号

	线程能否被取消以及能否被立即取消取决于线程的 state 和 type

	也就是与下面两个函数有关

	7.pthread_setcancelstate函数

	8.pthread_setcanceltype函数

参数:
	thread:线程的id

*/
//返回值:
int pthread_setcancelstate(int state, int *oldstate);

//state:
	PTHREAD_CANCEL_ENABLE  	//可被取消  --默认状态


	PTHREAD_CANCEL_DISABLE  //不可被取消


int pthread_setcanceltype(int type, int *oldtype);

//type:
	PTHREAD_CANCEL_ASYNCHRONOUS  	//可以被立即取消


	PTHREAD_CANCEL_DEFERRED   	//不可以被立即取消--默认状态

  • 示例代码:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdbool.h>
#include <pthread.h>

typedef struct INFO
{
    const char *src_file;
    const char *dest_file;
    int offset;
    int size;

}info_t;

//获取源文件的大小(字节)
//创建目标文件

int get_src_file_size_and_create_dest_file(const char *src_file,const char *dest_file);

//复制源文件内容到目标文件内容中

void *cp_src_file_to_dest_file(void *arg);

int main(int argc, char const *argv[])
{
    if(3 != argc)
    {
        printf("Usage : %s src_file dest_file\n",argv[0]);
        return -1;
    }

    int size = 0;

    size = get_src_file_size_and_create_dest_file(argv[1],argv[2]);

    //线程1复制文件内容的前一半
    //线程2复制文件内容的后一半
    info_t arg[2] = {

        {argv[1],argv[2],0,size/2},

        {argv[1],argv[2],size/2,size - size/2}

    };

    pthread_t thread_1_id = 0;

    pthread_t thread_2_id = 0;

    int ret = 0;

    //创建线程1
    if(0 != (ret = pthread_create(&thread_1_id,NULL,cp_src_file_to_dest_file,&arg[0])))
    {
        printf("pthread_create error : errno = [%d] errstr = [%s]\n",ret,strerror(ret));
        exit(EXIT_FAILURE);
    }
    //创建线程2
    if(0 != (ret = pthread_create(&thread_2_id,NULL,cp_src_file_to_dest_file,&arg[1])))
    {
        printf("pthread_create error : errno = [%d] errstr = [%s]\n",ret,strerror(ret));
        exit(EXIT_FAILURE);
    }

    //线程1阻塞等待一个结合态的线程退出并回收资源
    pthread_join(thread_1_id,NULL);

    //线程2阻塞等待一个结合态的线程退出并回收资源
    pthread_join(thread_2_id,NULL);
    return 0;
}

int get_src_file_size_and_create_dest_file(const char *src_file,const char *dest_file)
{
    //获取源文件的大小(字节)
    int src_fd = open(src_file,O_RDONLY);

    if(-1 == src_fd)
    {
        perror("open error");
        return -1;
    }

    int size = lseek(src_fd,0,SEEK_END);

    close(src_fd);

    //创建目标文件

    int dest_fd = open(dest_file,O_WRONLY | O_CREAT | O_TRUNC,0666);

    if(-1 == dest_fd)
    {
        perror("open error");
        return -1;
    }

    close(dest_fd);

    return size;
}

void *cp_src_file_to_dest_file(void *arg)
{
    //参数强制类型转换
    info_t info = *(info_t *)arg;

    int src_fd = 0;

    int dest_fd = 0;

    //打开源文件
    if(-1 == (src_fd = open(info.src_file,O_RDONLY)))
    {
        perror("open error");
        pthread_exit(NULL);
    }
    //打开目标文件
    if(-1 == (dest_fd = open(info.dest_file,O_WRONLY)))
    {
        perror("open error");
        pthread_exit(NULL);
    }

    //定位光标
    lseek(src_fd,info.offset,SEEK_SET);

    lseek(dest_fd,info.offset,SEEK_SET);

    int ret = 0;
    int num = 0;
    char buf[64] = {0};

    while (true)
    {
        ret = read(src_fd,buf,sizeof(buf));

        if(0 == ret)
        {
            break;
        }

        num += ret;

        if(num > info.size)
        {
            write(dest_fd,buf,info.size - (num - ret));
            break;
        }

        write(dest_fd,buf,ret);
       
    }
    //关闭文件
    close(src_fd);

    close(dest_fd);

}
  • 运行结果:
linux@ubuntu:~$ gcc th2.c -lpthread
linux@ubuntu:~$ ./a.out k1.c k2.c
linux@ubuntu:~$ diff k1.c k2.c 
linux@ubuntu:~$ 
  • 本示例代码,仅供参考

文章来源地址https://www.toymoban.com/news/detail-728771.html

到了这里,关于IO学习系列之使用多线程复制同一个文件内容的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • UDP和TCP可以使用同一个端口吗?

    之前有读者在面试的时候,被问到: TCP 和 UDP 可以同时监听相同的端口吗? 关于端口的知识点,还是挺多可以讲的,比如还可以牵扯到这几个问题: 多个 TCP 服务进程可以同时绑定同一个端口吗? 客户端的端口可以重复使用吗? 客户端 TCP 连接 TIME_WAIT 状态过多,会导致端

    2023年04月08日
    浏览(90)
  • 使用ES同一个字段,不同条件or查询

    需求: type 字段是使用逗号分隔的字符串,要求多个 值只要与此字段 模糊匹配,即可成功 布尔查询(Bool Query) 布尔查询是一种联合查询,可以对多个查询条件进行组合,布尔查询有四个子查询: 有时我们在查询es时,希望能够一次返回符合多个查询条件的结果,如

    2024年02月11日
    浏览(95)
  • Unity 3D开发--SceneManager场景管理(异步使用同一个过渡场景)

    在U3D开发过程中经常使用到多场景的切换,有同步SceneManager.LoadScene()和异步SceneManager.LoadSceneAsync()两种方法,同步的话一般就会卡住界面直到加载完成,使用异步的话一般都做一个加载的进度条,每次切换的时候都需要一个加载动画,所以需要建一个专门的过渡加载场景来进

    2024年02月14日
    浏览(43)
  • 4.18 TCP 和 UDP 可以使用同一个端口吗?

    目录 TCP 和 UDP 可以同时绑定相同的端口吗? 多个 TCP 服务进程可以绑定同一个端口吗? 重启 TCP 服务进程时,为什么会有“Address in use”的报错信息? 重启 TCP 服务进程时,如何避免“Address in use”的报错信息? 客户端的端口可以重复使用吗? 多个客户端可以 bind 同一个端口

    2024年02月11日
    浏览(54)
  • Django的mysql数据库问题:同一个模型(同一张表)中的不同记录也是可以相互关联的【使用“自引用关系”】

    是的,确实可以在Django的模型中使用外键来建立同一模型中不同记录之间的关联关系。这样的关联关系被称为自引用关系(self-referential relationship)或者自关联关系。通过在模型中定义外键字段,你可以使模型的实例与同一模型中的其他实例产生关联。 在Django中,这通常通过

    2024年01月18日
    浏览(65)
  • 【解惑】当处理同一个字段的并发问题时,使用乐观锁来处理库存数量

    以下是一个使用乐观锁处理库存数量并发问题的c#示例代码: 上述示例代码使用乐观锁来处理并发问题,确保库存数量在更新过程中不受影响。通过使用数据库事务来保证更新的原子性,并且在更新数据之前检查版本号是否一致,以避免潜在的并发冲突。 乐观锁是一种乐观的

    2024年02月13日
    浏览(51)
  • Linux系统中判断多个设备使用了同一个IP地址,判断IP地址冲突的问题

    在Linux系统中,如果多个设备使用了同一个IP地址,就会发生IP地址冲突的问题。IP地址冲突会导致网络连接不稳定,甚至会导致网络中断。因此,在使用Linux系统时,需要及时发现和解决IP地址冲突问题。 下面介绍几种在Linux系统中判断IP地址冲突的方法。 1.使用ping命令 ping命

    2024年02月16日
    浏览(76)
  • 同一个请求同一个参数,postman发送成功,jmeter失败

    当使用jmeter 请求时吗,注意请求参数中是否包含中文。 问题来由: 1.先用postman发送,成功。  2.一模一样,header一样,请求一样,但是就是一直查询不出来。 3.经过fiddler抓包,再次对比请求头,请求体,还是没有发现有什么差别。 4.猜想: 此接口其他参数传送,都没有问题

    2024年02月13日
    浏览(55)
  • Vue - 你会在同一个元素上使用v-for和v-if吗

    难度级别:初级及以上                                 提问概率:50%  在初学者看来,v-for和v-if同时使用是非常方便的,二者共同使用的常见场景有两种。例如有两个列表,分别用于渲染学生数据和老师数据,然后有两个单选按钮,用于切换当前页面中需要

    2024年04月12日
    浏览(35)
  • IO学习系列之使用read和write复制文件内容

    read函数: 功能:从 文件fd 中读取 count个字节 ,存放进 指针buf ; 具体内容: write函数: 功能:把 指针buf 中的内容,写 count个字节 到 文件fd 中; 具体内容: 示例代码:

    2024年02月07日
    浏览(46)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包