【MIT6.S081】Lab1: Xv6 and Unix utilities(详细解答版)

实验内容网址:https://xv6.dgs.zone/labs/requirements/lab1.html

Sleep

关键点:函数参数判断、系统函数调用

思路:

通过argc来判断函数参数是否正确,通过atoi函数来讲字符串转化为整型,调用sleep函数后退出程序。

代码:

#include "kernel/types.h"
#include "kernel/stat.h"
#include "user/user.h"

int
main(int argc, char *argv[])
{
    int time;
    if(argc != 2){
        printf("usage:%s timexx\n",argv[0]);
        exit(-1);
    }
    time = atoi(argv[1]);
    sleep(time);
    exit(0);
}

Pingpong

关键点:

pipe()、fork()、read()、write()、getpid()

思路:

管道是作为一对文件描述符公开给进程的小型内核缓冲区,一个用于读取,一个用于写入。将数据写入管道的一端使得这些数据可以从管道的另一端读取。
其初始化方式为:

int p[2];
pipe(p);
// p[0]是管道的读取端
// P[1]是管道的写入端

根据题目要求,父进程和子进程都需要向对方发送字节,那么可以定义2个管道,代码如下

#define READ 0
#define WRITE 1
int father2child[2];
int child2father[2];
pipe(father2child);
pipe(child2father);

创建进程使用fork 函数,在父进程中,fork返回子类的PID;在子进程中,fork返回零。

步骤:

  1. 新建 pingpong.c文件,并在MakeFile文件的UPROGS中加入程序。
  2. 编写如下的代码,运行make qemu进行编译并运行pingpong.
#include "kernel/types.h"
#include "kernel/stat.h"
#include "user/user.h"
#define READ 0
#define WRITE 1

int
main(int argc, char *argv[])
{
    int father2child[2];
    int child2father[2];
    char sendByte = 'A';
    pipe(father2child);

    pipe(child2father);
    if(fork() == 0){
        // 子进程
        // 管道关闭,这两个方向的管道不需要使用(感觉不用也可以,可以代码写完再看看哪里需要关闭)
        close(father2child[WRITE]);
        close(child2father[READ]);
        // 子进程接收父进程的字节并打印
        char tmp;
        if(read(father2child[READ],&tmp , 1) != 1 ){
            printf("read error\n");
        }
        printf("%d: received ping\n",getpid());

        // 通过管道写入字节发送给父进程
        write(child2father[WRITE],&sendByte,1);
        // 关闭管道退出程序
        close(father2child[READ]);
        close(child2father[WRITE]);
        exit(0);

    }else{
        // 父进程
        // 管道关闭
        close(father2child[READ]);
        close(child2father[WRITE]);
        // 父进程向子进程发送一个字节
        char tmp;
        write(father2child[WRITE], &sendByte, 1);
        // 父进程读取从子进程而来的字节
        if(read(child2father[READ], &tmp, 1) != 1){
            printf("read error\n");
        }
        printf("%d: received pong\n",getpid());
        // 关闭管道退出程序
        close(father2child[WRITE]);
        close(child2father[READ]);
        exit(0);

    }

    
}

这里的close掉某一些管道的做法个人认为可以不用,当然加上去更好,避免错误使用到这些管道端口。对于2对管道4个管道端口,父子进程都有这4个管道端口的文件描述符,对于父进程,可以把father2child[READ]和child2father[WRITE]关闭掉,这不会影响子进程对这两个端口的使用。对于子进程则把相反的管道端口关闭。

Primes

关键点:

pipe() 、fork()、递归思想、素数判断

思路:

一开始看的时候会看不懂题目,题目的要求就是找出2-35中的素数(质数)。本题寻找素数的方法可以看下图
在这里插入图片描述
对于一个进程,从左管道中读入第一个数,然后将第一个数打印出来(这个数是素数)。然后从左管道读入其他数,将其他数与第一个数相除,若不整除,则将该数写入右管道中。递归后可以得到2-35中的每一个素数。
本题涉及到递归思想,需要构造递归函数;然后是对于每个进程都需要有2个管道,一个管道是父进程与子进程的,一个管道是子进程与孙进程的;最后还需要及时关闭不使用的管道端口,避免系统资源不够用。

步骤:

  1. 从第一个进程(戏称:始祖进程hhh)开始,定义管道,遍历2-35,将每个整型写入管道中,调用fork生成子进程,父进程则关闭管道的文件描述符并退出。
  2. 从第二个进程开始,需要使用递归了,则编写递归函数 primes(int* left_pipe) ,该函数的参数是父进程给子进程的左管道。在primes函数中,读取左管道第一个数作为素数并打印出来。接着创建新的管道(可以认为是右管道)。然后从左管道中读取每个整型,经过是否素数判断,若是则写入右管道中;接着进行fork创建孙进程,孙进程中继续调用primes(right_pipe),将右管道传给孙进程。子进程则关闭不使用的文件描述符,退出进程。
  3. 递归…直到最后一个进程连第一个数值也读不到的时候直接退出进程,程序结束。

代码:

#include "kernel/types.h"
#include "kernel/stat.h"
#include "user/user.h"
#define READ 0
#define WRITE 1

void primes(int* left_pipe){
    // 关闭左管道的write端口
    close(left_pipe[WRITE]);
    // 读取第一个值
    int first = 0;
    if(read(left_pipe[READ],&first,sizeof(int)) != sizeof(int)){
        exit(0);
    }
    // 按照要求打印
    printf("primes %d\n",first);
    // 创建管道
    int right_pipe[2];
    pipe(right_pipe);
    // 从左管道读入并写到右管道中
    int num;
    while(read(left_pipe[READ], &num,sizeof(int) == sizeof(int))){
        if(num%first != 0){
            // 将素数写入右管道中
            write(right_pipe[WRITE],&num, sizeof(int));
        }
    }
    // 创建子进程
    if(fork() == 0){
        primes(right_pipe);
    }
    else{
        close(left_pipe[READ]);
        close(right_pipe[READ]);
        close(right_pipe[WRITE]);
        // 等待子进程退出?
        wait(0);
    }
    exit(0);
}

int
main(int argc, char *argv[])
{
    // 定义一个管道
    int p[2];
    pipe(p);
    //从第一个进程开始,将2-35输入到子进程中
    for(int i = 2; i <= 35; i++){
        write(p[WRITE],&i,sizeof(int));
    }
    if(fork() == 0){
        // 子进程
        primes(p);
    }
    else{
        close(p[WRITE]);
        close(p[READ]);
        // 等待子进程退出?
        wait(0);
    }
    

    exit(0);
}

未解:

父进程使用wait(0)会等待子进程全部退出后才退出吗?

Find

关键点:user/ls.c实现、目录遍历、递归思想

思路:

首先要弄懂 user/ls.c中功能的实现。
fmtname()的作用是去除文件路径中的路径,只保留住文件名,如果文件名程度超过DIRSIZ则直接返回,若不超过则要补全空格。
如果path 是一个文件的话,则直接输出文件名。若path是一个目录的话。循环读取目录中的每个条目,并检查其索引节点号 (inum)。如果索引节点号为0,我们跳过该条目。否则,我们将条目的名称复制到 buf 的末尾,并添加一个终止字符。接着,我们获取该条目的状态信息,并打印其信息。如果获取状态信息失败,我们打印错误消息并继续下一个条目。
搞懂user/ls.c后,就可以在这上面做修改。

步骤:

  1. 新建find.c文件,首先进行参数检查
  2. 定义递归函数,传入本轮递归的路径和目标文件名。
  3. 递归函数中,如果path是文件名的话则进行文件名判断,再打印。如果是目录,则循环遍历每一个条目,过滤掉. …后进行递归。具体还是看代码吧,思路比较清晰。

代码:

#include "kernel/types.h"
#include "kernel/stat.h"
#include "user/user.h"
#include "kernel/fs.h"

// 用于分割出带有/路径的文件名
// input:a/b/filename
// output:filename
char*
my_fmtname(char *path)
{
  char *p;
  // Find first character after last slash.
  for(p=path+strlen(path); p >= path && *p != '/'; p--)
    ;
  p++;

  return p;
}

void find(char* path, char* target_filename){
    // 模仿ls函数
    char buf[512], *p;  // 用于存储文件或目录路径的缓冲区。
    int fd;             // 文件描述符,用于后续的文件或目录操作。
    struct dirent de;   // 一个 dirent 结构体,用于读取目录中的条目。
    struct stat st;     // 一个 stat 结构体,用于存储文件或目录的状态信息。
    // 尝试打开目录
    if((fd = open(path, 0)) < 0){
        fprintf(2, "find: cannot open %s\n", path);
        return;
    }
    // 尝试获取状态信息
    if(fstat(fd, &st) < 0){
        fprintf(2, "find: cannot stat %s\n", path);
        close(fd);
        return;
    }

    switch(st.type){
    // 如果path指向的是一个文件,判断该文件与目标文件名是否一样
    case T_FILE:
       
       if(strcmp(my_fmtname(path),target_filename) == 0){
        // 表示文件名一样
        printf("%s\n",path);
       }
       break;
    case T_DIR:
        if(strlen(path) + 1 + DIRSIZ + 1 > sizeof buf){
            printf("find: path too long\n");
            break;
        }
        strcpy(buf, path);
        p = buf+strlen(buf);
        *p++ = '/';
        // 循环读取目录中的每个条目,并检查其索引节点号
        while(read(fd, &de, sizeof(de)) == sizeof(de)){
            if(de.inum == 0)
                continue;
            // 过滤 . ..
            if(!strcmp(de.name,".") || !strcmp(de.name,".."))
                continue;
            memmove(p, de.name, DIRSIZ);
            p[DIRSIZ] = 0;
            if(stat(buf, &st) < 0){
                printf("find: cannot stat %s\n", buf);
                continue;
            }
            //递归!!!
            find(buf, target_filename);
        }
        break;
    }
    close(fd);
}

void
main(int argc, char *argv[])
{
  
    if(argc != 3){
        printf("usage: %s <dir> <filename>",argv[0]);
        exit(0);
    }
    find(argv[1], argv[2]);
    exit(0);
}

Xargs

待施工

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

守护人类健康:人工智能赋能医疗领域创新应用

编者按&#xff1a;每年的4月7日是世界卫生日&#xff0c;又称世界健康日&#xff0c;旨在引起世界各国人民对卫生、健康工作的关注&#xff0c;提高人们对卫生领域的素质和认识&#xff0c;强调健康对于劳动创造和幸福生活的重要性。那么&#xff0c;如果医疗技术能够更加智能…

vue vue3 日期选择的组件,封装组件

一、背景 基于element日期选择组件&#xff0c;自行封装了一个组件。 以下是达到的效果&#xff1a; 1.选择年&#xff0c;日期选择组件默认填充是&#xff1a;当时的年&#xff1b; 2.选择月&#xff0c;日期选择组件默认填充的是&#xff1a;当时的年月&#xff1b; 3.选择日…

Redis监控方案以及相关黄金指标提升稳定性和可靠性

Redis监控方案以及相关黄金指标提升稳定性和可靠性 1. 需要了解的词2. 「基准性能」相关指标2.1 Latency2.2 最大响应延迟2.3 平均响应延迟2.4 OPS(instantaneous_ops_per_sec)2.5 Hit Rate 3. 「内存」相关指标3.1 内存使用量(used_memory)3.2 内存碎片率(mem_fragmentation_r…

商业银行布局AI大模型的“三大路径”

随着人工智能技术的迅猛发展&#xff0c;AI创新应用模式持续涌现&#xff0c;2022年11月OpenAI推出的对话式通用人工智能工具ChatGPT正式上线&#xff0c;标志着人工智能技术的发展迈入了全新阶段。随着ChatGPT的蹿红&#xff0c;一时间&#xff0c;人工智能大模型技术迅速成为…

区块链相关概念

区块链是什么&#xff0c;就算是做计算机技术开发的程序员&#xff0c;100个当中都没有几个能把这个概念理解明白&#xff0c;更不要说讲清楚了。那对于普通人来说&#xff0c;就更扯了。 除了“挖矿”表面意思似乎比较好理解外&#xff0c;其他的基础概念真TMD绕。 去中心化、…

OpenHarmony应用启动流程分析——ApplicationAbility初始化

作者&#xff1a;汪语 一、引言 本文基于OpenAtom OpenHarmony&#xff08;以下简称“OpenHarmony”&#xff09; 4.0 Release版本的源码&#xff0c;对应用进程初始化后MainThread初始化及调用AttachApplication、LaunchApplication、LaunchAbility的过程做了分析和总结&…

HarmonyOS-数据请求(http / axios)

一、http数据请求 步骤&#xff1a; 1.在module.json5中申请ohos.permission.INTERNET权限 "module": {"requestPermissions": [{ "name": "ohos.permission.INTERNET" }],...} 2.在xxx.ets页面中导入&#xff1a;import http fro…

Jenkins结合gitlab自动化持续集成

最近在公司有负责搭建自动化测试环境&#xff0c;自动化脚本写好后&#xff0c;毋庸置疑是需要将自动化脚本进行持续集成测试&#xff0c;能够根据企业的定制化需求&#xff0c;通过Jenkins触发执行构建任务&#xff0c;定时执行自动化脚本等&#xff0c;今天就给大家介绍一下J…

Mybatis一级缓存

一级缓存简介 在常见的应用系统中&#xff0c;数据库是比较珍贵的资源&#xff0c;很容易成为整个系统的瓶颈。在设计和护系统时&#xff0c;会进行多方面的权衡&#xff0c;并且利用多种优化手段&#xff0c;减少对数据库的直接访问。使用缓存是一种比较有效的优化手段&#x…

干货分享 | 在TSMaster中加载基于DotNet平台的SeedKey

在UDS诊断过程中&#xff0c;会涉及到安全访问的问题&#xff0c;也就是所谓的Seed&Key。TSMaster 诊断模块支持通过.dll文件载入 Seed&Key 算法用于安全访问解锁。在最近发布的TSMaster 2024.03版本中不仅支持了C/C&#xff0c;Delphi等语言封装的DLL文件&#xff0c;…

【CVE复现计划】CVE-2024-0195

CVE-2024-0195 简介&#xff1a; SpiderFlow是新一代开源爬虫平台&#xff0c;以图形化方式定义爬虫流程&#xff0c;不写代码即可完成爬虫。基于springbootlayui开发的前后端不分离,也可以进行二次开发。该系统/function/save接口存在RCE漏洞&#xff0c;攻击者可以构造恶意命…

蓝帕控制阀门将莅临2024年第13届生物发酵展

参展企业介绍 感谢你正在或即将使用蓝帕控制阀门(江苏)有限公司系列产品&#xff0c;感谢你关注蓝帕控制阀门(江苏)有限公司&#xff01; 蓝帕控制阀门(江苏)有限公司&#xff0c;流体控制领域的国际品牌之一&#xff0c;总部位于意大利米兰&#xff0c;成立多年以来&#xf…

流式密集视频字幕

流式密集视频字幕 摘要1 IntroductionRelated Work3 Streaming Dense Video Captioning Streaming Dense Video Captioning 摘要 对于一个密集视频字幕生成模型&#xff0c;预测在视频中时间上定位的字幕&#xff0c;理想情况下应该能够处理长的输入视频&#xff0c;预测丰富、…

Rust 标准库 API 文件和文件夹操作 File,读取/创建/修改/追加/删除/重命名文件等

File::create 使用File的关联函数&#xff08;类似Java中的静态方法&#xff09;create&#xff0c;创建文件&#xff0c;如果存在&#xff0c;则覆盖。 use std::fs::{File, Metadata};fn main() -> std::io::Result<()> {let file: File File::create("foo.…

C++ 学习笔记

文章目录 【 字符串相关 】C 输入输出流strcpy_s() 字符串复制输出乱码 【 STL 】各个 STL 支持的常见方法 ? : 运算符switch case 运算符 switch(expression) {case constant-expression :statement(s);break; // 可选的case constant-expression :statement(s);break; //…

基于arcgis /envi PCA(主成分分析)实现过程

基于arcgis /envi PCA(主成分分析)实现过程 1 提取研究范围 2对研究范围进行重采样 &#xff08;根据数据情况进行选做&#xff0c;如数据较大建议进行该步骤操作&#xff09; 3 对研究范围内数据进行归一化处理 4 将空值替换为0 5 对同期不同要素数据进行波段合成 对波段…

如何本地部署JumpServer堡垒机并结合内网穿透实现远程访问

文章目录 前言1. 安装Jump server2. 本地访问jump server3. 安装 cpolar内网穿透软件4. 配置Jump server公网访问地址5. 公网远程访问Jump server6. 固定Jump server公网地址 前言 JumpServer 是广受欢迎的开源堡垒机&#xff0c;是符合 4A 规范的专业运维安全审计系统。JumpS…

云南省气象探空业务升级为北斗探空观测系统

云南省气象探空业务升级为北斗探空观测系统 近日&#xff0c;云南省首套北斗探空观测系统在普洱市思茅区高空气象观测站建成并调试成功&#xff0c;这意味着云南省气象探空业务将从L波段雷达探测升级到北斗探空观测系统。 &#xff08;图片来源于网络&#xff09; 北斗探空观…

距离度量方法——欧氏距离、曼哈顿距离、切比雪夫距离、闵可夫斯基距离

目录 一、 欧氏距离&#xff08;Euclidean Distance&#xff09; 1、简介 2、代码实现 二、曼哈顿距离&#xff08;Manhattan Distance&#xff09; 1、简介 2、代码实现 三、切比雪夫距离&#xff08;Chebyshev Distance&#xff09; 1、简介 2、代码实现 四、闵可夫…

CSS实现热门创作者排行榜(毛玻璃效果)

CSS实现热门创作者排行榜&#xff08;毛玻璃效果&#xff09; 效果展示 CSS 知识点 CSS 基础知识回顾filter 属性运用回顾 整体页面布局实现 <div class"container"><h3>Popular Creator Rank List</h3><!-- 用户列表容器 --><div cl…