华清day4 24-7-31

1> 使用父子进程完成两个文件的拷贝

父进程拷贝前一半内容,子进程拷贝后一半内容

子进程结束后退出,父进程回收子进程的资源

/* 
使用父子进程完成两个文件的拷贝
 父进程拷贝前一半内容,
 子进程拷贝后一半内容
子进程结束后退出,
父进程回收子进程的资源
*/
#include<myhead.h>
int main(int argc, const char *argv[])
{
	char buf[128] = " ";//定义搬运工
		int count = 0,n =0,num = 0;
	pid_t pid =-1;
	pid = fork();//创建子进程
	if(pid>0)//父进程执行的代码
	{
		int src = -1;
		if((src = open("./01dir.c",O_RDONLY))== -1)//被复制的文件以只读的形式打开
		{
			perror("src open error");
			return -1;
		}
		int dest = -1;
		if((dest = open("./zy.txt",O_RDWR|O_CREAT|O_TRUNC,0665))== -1)//被复制的文件以w+的形式打开
		{
			perror("dest open error");
			return -1;
		}
		
		n = (lseek(src, 0, SEEK_END)+1)/2;//遇到奇数时候确保能复制一半所以+1
		lseek(src, 0, SEEK_SET);//光标
	while(1)
		{
			count = read(src,buf,sizeof(buf));
			num +=count;
			if(num >n)
			{
				write(dest,buf,n%sizeof(buf));
				break;
			}
			write(dest,buf,count);
		}	
		printf("父进程\n");
	}
	else if(pid == 0)//子进程执行的代码
	{
		sleep(0.1);//确保子进程在父进程之后
		int src = -1;
		if((src = open("./01dir.c",O_RDONLY))== -1)//被复制的文件以只读的形式打开
		{
			perror("src open error");
			return -1;
		}
		int dest = -1;
		if((dest = open("./zy.txt",O_RDWR|O_CREAT|O_APPEND,0665))== -1)//目标文件以追加的方式打开
		{
			perror("dest open error");
			return -1;
		}
		lseek(dest,20,SEEK_SET);

		n = (lseek(src, 0, SEEK_END)+1)/2;//遇到奇数时候确保能复制一半所以+1
		lseek(src, n, SEEK_SET);//光标偏移到下半开始的地方
		while(1)
		{
			count = read(src,buf,sizeof(buf));
			if(count ==0)
			{
				break;
			}
			write(dest,buf,count);
		}		
		printf("子进程\n");
		
	}
	exit(0);
	wait(NULL);
	return 0;
}
//正确代码

#include<myhead.h>

//定义求源文件大小的函数
int get_file_len(const char *srcfile, const char *destfile)
{
    //以只读的形式打开源文件
    int sfd = open(srcfile, O_RDONLY);
    if(sfd == -1)
    {
        perror("open srcfile error");
        return -1;
    }

    //以创建的形式打开目标文件
    int dfd = open(destfile, O_WRONLY|O_CREAT|O_TRUNC, 0664);
    if(dfd == -1)
    {
        perror("open destfile error");
        return -1;
    }

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

    //关闭文件
    close(sfd);
    close(dfd);

    return size;     //将文件大小返回
}

//定义文件拷贝函数
void copy_file(const char *srcfile, const char *destfile, int start, int len)
{
    //以只读的形式打开源文件
    int sfd = open(srcfile, O_RDONLY);
    if(sfd == -1)
    {
        perror("open srcfile error");
        return ;
    }

    //以创建的形式打开目标文件
    int dfd = open(destfile, O_WRONLY);
    if(dfd == -1)
    {
        perror("open destfile error");
        return ;
    }

    //移动光标位置
    lseek(sfd, start, SEEK_SET);
    lseek(dfd, start, SEEK_SET);

    //定义搬运工
    char buf[128] = "";
    int sum = 0;           //记录拷贝的总个数
    while(1)
    {
        //从源文件中读取数据
        int res = read(sfd, buf, sizeof(buf));
        sum += res;               //将读取的个数累加

        if(res == 0 || sum>len)             //表示文件读取结束
        {
            write(dfd, buf, res-(sum-len));    //父进程将最后一次拷贝结束
            break;
        }
        //写入到目标文件中
        write(dfd, buf, res);
    }

    //关闭文件
    close(sfd);
    close(dfd);

    printf("拷贝成功\n");
}



/******************************主程序*************************/
int main(int argc, const char *argv[])
{
    //判断是否传入两个文件
    if(argc != 3)
    {
        printf("input file error\n");
        return -1;
    }

    //求出源文件长度
    int len = get_file_len(argv[1], argv[2]);

    //创建两个进程
    pid_t pid = fork();

    if(pid > 0)
    {
        //父进程,拷贝前一半内容
        copy_file(argv[1], argv[2], 0, len/2);

    }else if(pid == 0)
    {
        //子进程拷贝后一半内容
        copy_file(argv[1], argv[2], len/2, len-len/2);


        //退出进程
        exit(EXIT_SUCCESS);
    }else 
    {
        perror("fork error");
        return -1;
    }

    //回收子进程资源
    wait(NULL);
    return 0;
}

2> 详细绘制思维导图

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值