os实训课程模拟考试(大题复习)

目录

一、Linux操作系统

(1)第1关:Linux初体验 

(2)第2关:Linux常用命令

(3)第3关:Linux 查询命令帮助语句

二、Linux之进程管理—(重点)

(1)第1关:获取进程常见属性

(2)第2关:进程创建操作-fork

(3)第3关:进程创建操作-vfork

(4)第4关:进程终止

三、生产者消费者问题实践

(1)第1关:生产者消费者问题实践

(2)第2关:进程互斥和同步

四、基于信号量的进程间通信

(1)第1关:信号量IPC操作考查

五、基于命名管道与信号的进程间通信

(1)第1关:命名管道与信号IPC操作考查

六、 Linux vi/vim编辑器(难度:简单)

(1)第1关:vi/vim基本用法

(2)第2关:vi/vim工作模式切换

(3)第3关:vi/vim命令模式

(4)第4关:vi/vim底线命令模式(重点)

七、读文件系统函数(难度:简单)

(1)第1关:读文件系统函数

八、写文件系统函数(难度:中等)

(1)第1关:写文件系统函数

 九、进程基础操作(难度:简单)

(1)第1关:进程基础操作考察


本篇博客的主要内容是在做大题时遇到的问题和一些总结(具体答案看我之前的博客)

一、Linux操作系统

(1)第1关:Linux初体验 

(2)第2关:Linux常用命令
  • newfile文件复制一份到newdir目录下并命名为newfileCpy
  • cp newfile newdir/newfileCpy

(3)第3关:Linux 查询命令帮助语句
  • linux中使用man命令来查询命令的帮助文件

二、Linux之进程管理—(重点

(1)第1关:获取进程常见属性
  • 获取进程本身的进程ID的系统调用函数是getpid()
  • 获取父进程的进程ID的系统调用函数是getppid()
(2)第2关:进程创建操作-fork
  • 这题需要注意的地方:pid_t pid =fork();
(3)第3关:进程创建操作-vfork
  • 这题需要注意的地方:pid_t pid = vfork();
  • 子进程中输出"Children"字符串(提示:需要换行)
(4)第4关:进程终止
  • 用这个注册函数:atexit()on_exit()调用成功返回0调用失败返回一个非零值
 void exit(){
        printf("%d",getpid());
    }
	if(atexit(exit)){
        
    }

三、生产者消费者问题实践

(1)第1关:生产者消费者问题实践
  • 观察题目代码给出的生产者函数。根据这个来写!!
  • sem_t empty; :类似等待一个信号(生产)

  • sem_t full; :类似等待一个信号(消费)

  • sem_wait(&empty);
  • sem_wait(&full);

    (在每一个操作时和释放时都要记得加锁解锁

  • 加锁 pthread_mutex_lock(&mutex);
  • 释放锁 pthread_mutex_unlock(&mutex);

锁对象:pthread_mutex_t mutex;


  • 设置一个参数(仿照生产者函数)int nextc = 0;
  • 先给一个等待消费条件:sem_wait(&full);
  • 再加锁:pthread_mutex_lock(&mutex);
  • 意思是:从之前生产了的东西拿东西出来消费:nextc=buffer[out];

  • 与生产者函数中用到的一样:int out = 0;

  • 这个跟着改:printf("Consume one message:%d\n", nextc);

  • fflush(stdout);  //printf后请一定调用这句刷新输出缓存out = (out + 1) % SIZE;

  • out = (out + 1) % SIZE; (这个跟生产者函数一样,对着改就行)
  • pthread_mutex_unlock(&mutex);   ——> 再解锁

最后记得给一个消费完的信号:sem_post(&empty); 。意思其实就是消费完,给一个叫"空"的信号。


具体的答案《消费:Consumer()函数》

void *Consumer()
{
    // ============================= begin ======================

	//请补充消费者线程函数代码
    int nextc = 0;
    int i = 0;
	for(; i < 10; ++i)
	{  
		int time = rand() % 10 + 1;
        usleep(time*100000); 
        sem_wait(&full); 
		pthread_mutex_lock(&mutex);
		nextc=buffer[out];
		printf("Consume one message:%d\n", nextc);
		fflush(stdout);//printf后请一定调用这句刷新输出缓存
		out = (out + 1) % SIZE;
		pthread_mutex_unlock(&mutex);         //互斥锁解锁
		sem_post(&empty);
	}


 // ============================= end ========================

}

对比题目提供的《生产函数:Producer()》

int in = 0;
int out = 0;
int buffer[SIZE];
sem_t empty;
sem_t full;
pthread_mutex_t mutex;

void *Producer()
{
    int nextp = 0;
    int i = 0;
	for(; i < 10; ++i)
	{  
		int time = rand() % 10 + 1;
        usleep(time*100000); 
        sem_wait(&empty); 
		pthread_mutex_lock(&mutex);
		buffer[in] = nextc;
		printf("Produce one message:%d\n", nextp);
		fflush(stdout);//printf后请一定调用这句刷新输出缓存
		in = (in + 1) % SIZE;
		pthread_mutex_unlock(&mutex);         //互斥锁解锁
		sem_post(&full);
	}
}
(2)第2关:进程互斥和同步

(这个就简单了!!结合上面学的东西,以及仿照这个题目女儿的"苹果"消费函数写就好

  • 把消费等待的信号换成"orange(桔子)"
  • 把printf(....)输出换一下就好了
  • 儿子消费函数答案
void *Son()
{
    // ============================== begin ===========================

	//请添加儿子线程的函数代码
    while(1)
	{
		int time = rand() % 10 + 1;          //随机使程序睡眠0点几秒
		usleep(time * 100000);        
		sem_wait(&orange); 
		pthread_mutex_lock(&mutex);
		printf("儿子取了一个桔子\n") ;
        fflush(stdout);
		pthread_mutex_unlock(&mutex);         //互斥锁解锁
		sem_post(&empty);
	}

    // ============================== end ==============================
}

四、基于信号量的进程间通信

(1)第1关:信号量IPC操作考查
  • 搞清楚函数:semctl(int semid,int semnum,int cmd,...);
  • semid要操作的信号量集合:(题目中给出变量:"int semid;")
  • semnum集合中信号量的编号(题目中就是:"i")
  • cmd执行的操作:(题目中有提示:#define GETVAL 获取信号量的值返回信号的值

具体答案如下:

(此代码不严谨!!因为没有判断是否函数调用成功!但是满足评测要求)

 for ( i = 0 ; i < MAX_SEMAPHORES ; i++ )
    {
        // =================================== begin ===============================

        /*请调用semctl函数,读取并输出与上述数组输出相同的输出*/
 
        // 注意:getval_arg.val 在这里不需要设置,因为 GETVAL 不使用它  
        int x = semctl(semid, i, GETVAL);  

        printf("Semaphore %d, value %d\n", i, x); 

            // =================================== end ===============================

        }

更完整、更严谨的答案:
(记得还要在顶部导入包:#include <stdlib.h>   //以使用exit函数)

for ( i = 0 ; i < MAX_SEMAPHORES ; i++ )
    {
        // =================================== begin ===============================

        /*请调用semctl函数,读取并输出与上述数组输出相同的输出*/
 
        // 注意:getval_arg.val 在这里不需要设置,因为 GETVAL 不使用它  
        int x = semctl(semid, i, GETVAL);  
        if (x == -1) {  
        printf("GETVAL for semaphore %d failed (%d)\n", i, errno);  
        exit(EXIT_FAILURE);  
    }
        printf("Semaphore %d, value %d\n", i, x); 

            // =================================== end ===============================

        }

五、基于命名管道与信号的进程间通信

(1)第1关:命名管道与信号IPC操作考查

没啥好说的,不会的操作系统考试就靠记呗!!

  • sleep(1); 休眠1秒
  • 打开读通道——> open("FIFO",O_RDONLY);
  • 打开写通道——> open("FIFO",O_WRONLY);
  • 开始写——>write(fd,"heool0penEuler",1);

答案如下:

 if(pid==0)
    {
        // ============================= begin =========================
        /*子进程打开读管道,随后关闭管道*/
        int fd;
        fd = open("FIFO",O_RDONLY);  /*rd->only 写*/
        /*关闭*/
        close(fd);
    }
    else
    {
        
        /*父进程打开写通道,休眠1秒,尝试写入*/
        int fd;
        fd = open("FIFO",O_WRONLY);  /*wr->only 写*/
        int ret;
        sleep(1); /*休眠1秒*/
        /*写入*/
        ret = write(fd,"heool0penEuler",1);

        // ============================= end =========================
    }

六、 Linux vi/vim编辑器(难度:简单

注意这类题目在操作之前需要初始化任务环境

(1)第1关:vi/vim基本用法
(2)第2关:vi/vim工作模式切换
  • wq :保存并退出
  • i:插入
(3)第3关:vi/vim命令模式
  • dd:删除
  • yy:复制
  • p:粘贴
(4)第4关:vi/vim底线命令模式(重点
  • 从第一行到最后一行寻找 word1 字符串,并将该字符串取代为 word2。 :1,$s/word1/word2/g
  • 将文件第2-5行内容另存为oldFileCpy.txt文件:2,5 w oldFileCpy.txt

七、读文件系统函数(难度:简单

(1)第1关:读文件系统函数

(此题注意:知道read()函数是啥就行了。其次知道其返回值和其函数的参数)

  • 函数原型:read(int fd, void *buf, size_t count);
  • fd:看上文(读文件的地址)
  • *buf:字符数组缓冲区的位置
  • 最后这个size_t count:就是读到的全部字符的个数——> sizeof(buffer)-1
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#define rwmode 0
int main()
{
    int fd;
    char buffer[1024];
    int n;
    fd = open("/data/workspace/myshixun/case1/testFIle", rwmode);
    if (fd < 0)
    {
        printf("Open file error!\n");
        exit(1);
    }
    else
        printf("open testFIle ok!\n");
    //-------------begin----------------------  
    //请使用read函数将其读入buffer中
    n = read(fd,buffer,sizeof(buffer)-1);
    //--------------end----------------------  
    buffer[n] = '\0';
    printf("%s\n", buffer);
    close(fd);
    return 0;
}

八、写文件系统函数(难度:中等)

(1)第1关:写文件系统函数

(这题其实就是一个综合。先用read()读函数,再用write()写函数

  • 函数原型:ssize_t write(int fd, const void *buf, size_t count);
  • 读的文件的地址题目也给出:int resource_fd

  • char buffer[FILESIZE]:字符数组

  • #define FILESIZE 1024 (该字符数组最大容量,也是题目要求读取的量)
  • 所以:(read(resource_fd,buffer,FILESIZE ))>0,因为读到了就要返回得到的个数。
  • char buffer[FILESIZE], *p; p = buffer; :要写入的字符数组信息地址在p

  • 写的目的文件地址题目也给出了:int  destination_fd

  • 具体要写的字符个数(前面被赋值):readbytes = read(resource_fd,buffer,FILESIZE)

  • 所以:(writebytes = write(destination_fd,p,readbytes))>0,因为写到了就要返回个数

综合解答:

#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#define resource_mode 0
#define destination_mode 0774
#define FILESIZE 1024

int main(int argc, char *argv[])
{
    int resource_fd, destination_fd;
    char buffer[FILESIZE], *p;
    int readbytes, writebytes;
    if (argc != 3)
    {
        printf("Usage:copy from resource file to destination file\n %s src_file dest_file\n", argv[0]);
        exit(0);
    }
    if ((resource_fd = open(argv[1], resource_mode)) == -1)
    {
        perror("Can't open source file");
        exit(0);
    }
    if ((destination_fd = creat(argv[2], destination_mode)) == -1)
    {
        perror("Can't create destination file");
        exit(0);
    }
    // ======================= begin =======================================

    // 请使用read函数读取前1024字节的内容读到缓冲区buffer中
    while ((readbytes = read(resource_fd,buffer,FILESIZE))>0)

    // ========================== end =======================================
    {
        p = buffer;
        if ((readbytes == -1) && (errno != EINTR))
            break;
        else if (readbytes > 0)
        {
            // ======================= begin =======================================

            // 请使用write函数读取到的前1024字节的内容写到目的文件中
            while ((writebytes = write(destination_fd,p,readbytes))>0)

            // ========================== end =======================================
            {
                if ((writebytes == -1) && (errno != EINTR))
                    break;
                else if (writebytes == readbytes)
                    break;
                else if (writebytes > 0)
                {
                    p += writebytes;
                    readbytes -= writebytes;
                }
            }
            if (writebytes == -1)
                break;
        }
    }
    close(resource_fd);
    close(destination_fd);
    return 0;
}

 九、进程基础操作(难度:简单

(1)第1关:进程基础操作考察

(这一关没啥好说的,难度很简单)

  • 了解fork()函数即可
  • 注意空格的输出(父、子进程都一样):printf("bye! "); 
#include<stdio.h>
#include<unistd.h>
#include<fcntl.h>
#include<stdlib.h>
int main(int argc,char *argv[])
{
  /*请开始填写*/
  pid_t pid =fork();
  if(pid==0){
    printf("bye! ");
  }
  else if(pid>0){
    printf("bye! ");
  }
      return 0;
}

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

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

相关文章

论文翻译 | PRCA:通过可插拔奖励驱动的上下文适配器拟合用于检索问答的黑盒大语言模型

摘要 检索问答(ReQA)任务采用检索增强框架&#xff0c;该框架由检索器和生成器组成。 生成器根据检索器检索到的文档制定答案。将大型语言模型(llm)作为生成器是有益的&#xff0c;因为它们具有先进的QA功能&#xff0c;但它们通常太大而无法根据预算限制进行微调&#xff0c;而…

RpcRrovider分发rpc服务(OnMessage和Closure回调)

目录 1.完善rpcprovider.cc的OnConnection 2.完善rpcprovider.cc的OnMessage 3.完整rpcprovider.h 4.完整rpcprovider.cc 这篇文章主要完成&#xff0c;protobuf实现的数据序列化和反序列化。 1.完善rpcprovider.cc的OnConnection rpc的请求是短连接的&#xff0c;请求一次…

Java--回顾方法的调用

1.静态方法与非静态方法 1.当二者皆为静态方式时&#xff0c;可直接类名.方法名调用其方法 2.当调用的方法是静态&#xff0c;被调用的方法为非静态时&#xff0c;调用将会报错 3.出现2情况可通过进行实例化这个类的方式进行调用&#xff0c;如图所示 4.当处于一个类下&#xf…

如何把项目文文件/文件夹)上传到Gitee(全网最细)

目录 1、首先必须要有一个Gitee官网的账号 2、点击右上角的号&#xff0c;点击新建仓库 3、按照下图步骤&#xff0c;自己起仓库名字&#xff0c;开发语言 4、点击初始化readme文件 5、在自己的电脑上选择姚上传的文件夹&#xff0c;或者文件&#xff0c;这里都是一样的&a…

新奥集团校招面试经验分享、测评笔试题型分析

一、走进新奥集团 新奥集团成立于1989年&#xff0c;总部位于河北廊坊&#xff0c;是中国领先的清洁能源企业集团。业务涵盖城市燃气、能源化工、环保科技等多个领域&#xff0c;致力于构建现代能源体系&#xff0c;提升生活品质。 二、新奥集团校招面试经验分享 新奥集团的…

哥斯拉短视频:成都柏煜文化传媒有限公司

哥斯拉短视频&#xff1a;巨兽传奇的视听盛宴 在短视频的海洋中&#xff0c;成都柏煜文化传媒有限公司 有一种特殊的存在总能吸引人们的目光&#xff0c;那就是以哥斯拉为主题的短视频。这些视频以震撼的视觉效果、扣人 ​心弦的剧情和独特的怪兽文化&#xff0c;为我们呈现了…

UE5基本操作(二)

文章目录 前言相机的移动速度修改默认地图使用初学者内容包文件夹结构 总结 前言 在我们的上一篇文章中&#xff0c;我们已经介绍了一些Unreal Engine 5&#xff08;UE5&#xff09;的基本操作。UE5是一款强大的游戏开发引擎&#xff0c;它提供了许多工具和功能&#xff0c;使…

C++进阶

C进阶 一、细节1.cout与输出缓冲区2.constexpr3.NULL和nullptr是不同的类型4.关于inline5.函数杂合用法6.const char*、char const*、char * const7.进程地址空间&#xff0c;所谓静态区常量区不准8.位运算9.多态9.1 内存切片9.2 转型9.3 构造函数和析构函数里是静态绑定9.4 dy…

如何将Hive表的分区字段插入PG表对应的时间戳字段?

文章目录 1、背景描述2、场景分析 1、背景描述 数据仓库的建设通常是为业务和决策服务的。在数仓开发的应用层阶段&#xff0c;BI可以直接从主题层/业务层取数&#xff0c;而前端需要根据具体的作图需求通过后端查询数据库 作图的指标需要根据主题层/业务层做查询计算&#xf…

C#——SortedList 排序列表详情

SortedList 排序列表 SortedList 类用来表示键/值对的集合&#xff0c;这些键/值对按照键值进行排序&#xff0c;并且可以通过键或索引访问集合中的各个项。 我们可以将排序列表看作是数组和哈希表的组合&#xff0c;其中包含了可以使用键或索引访问各项的列表。如果您使用索…

文章分享 | seq-DAP-seq和ChIP-seq联合检测揭示花发育器官转录因子调控机制

技术简介 MADS转录因子的同源蛋白SEPALLATA3 (SEP3)和AGAMOUS (AG)是调控拟南芥花发育分化的重要DNA结合蛋白。在雌蕊发育阶段&#xff0c;SEP3和AG形成异源四聚体&#xff0c;通过识别CArG-box序列来调控基因的表达。Seq-DAP-seq技术与ChIP-seq联合使用&#xff0c;可用于分析…

无需向量量化的自回归图像生成

摘要 https://arxiv.org/pdf/2406.11838 传统观点认为&#xff0c;用于图像生成的自回归模型通常伴随着向量量化的标记。我们观察到&#xff0c;尽管离散值空间可以方便地表示分类分布&#xff0c;但它对于自回归建模来说并不是必需的。在这项工作中&#xff0c;我们提出使用扩…

Kafka入门-分区及压缩

一、生产者消息分区 Kafka的消息组织方式实际上是三级结构&#xff1a;主题-分区-消息。主题下的每条消息只会保存在某一个分区中&#xff0c;而不会在多个分区中被保存多份。 分区的作用就是提供负载均衡的能力&#xff0c;或者说对数据进行分区的主要原因&#xff0c;就是为…

spring和springboot的关系是什么?

大家好&#xff0c;我是网创有方的站长&#xff0c;今天给大家分享下spring和springboot的关系是什么&#xff1f; Spring和Spring Boot之间的关系可以归纳为以下几个方面&#xff1a; 技术基础和核心特性&#xff1a; Spring&#xff1a;是一个广泛应用的开源Java框架&#…

我们后端程序员不是操作MyBatis的CRUD Boy

大家好&#xff0c;我是南哥。 一个对Java程序员进阶成长颇有研究的人&#xff0c;今天我们接着新的一篇Java进阶指南。 为啥都戏称后端是CRUD Boy&#xff1f;难道就因为天天怼着数据库CRUD吗&#xff1f;要我说&#xff0c;是这个岗位的位置要的就是你CRUD&#xff0c;你不…

JeecgBoot中如何对敏感信息进行脱敏处理?

数据脱敏即将一些敏感信息通过加密、格式化等方式处理&#xff0c;展示给用户一个新的或是格式化后的信息&#xff0c;避免了敏感信息的暴露。 一、接口脱敏注解 针对接口数据实现脱敏加密&#xff0c;只加密&#xff0c;一般此方案用于数据加密展示。 1.1 注解介绍 注解作用域…

Qt信号槽的坑

1、重载的信号&#xff08;以QSpinBox为例&#xff09; 像是点击按钮之类的信号槽很好连接&#xff0c;这是因为它的信号没有重载&#xff0c;如果像SpinBox那样有重载信号的话&#xff08;Qt5.12的见下图&#xff0c;不过Qt5.15LTS开始就不再重载而是换信号名了&#xff09;&…

Linux常用命令大全(超详细!!!)

文章目录 1.Linux是什么1.1 关于Linux我们主要学习什么1.1 学习Linux常见命令的前置知识 2. Linux常见命令2.1 ls命令2.2 cd命令2.3 pwd命令2.4 touch命令2.5 cat命令2.6 echo命令2.7 vim命令2.8 mkdir 命令2.9 rm命令2.10 cp命令2.11 mv命令2.12 grep命令2.13 ps命令2.14 nets…

<Python><ffmpeg>基于python使用PyQt5构建GUI实例:音频格式转换程序(MP3/aac/wma/flac)(优化版2)

前言 本文是基于python语言使用pyqt5来构建的GUI,功能是使用ffmpeg来对音频文件进行格式转换,如mp3、aac、wma、flac等音乐格式。 UI示例: 环境配置 系统:windows 平台:visual studio code 语言:python 库:pyqt5、ffmpeg 概述 本文是建立在之前的博文的基础上的优化版…

(笔记)Error: qemu-virgl: Failed to download resource “qemu-virgl--test-image“解决方法

错误&#xff1a; > Downloading https://www.ibiblio.org/pub/micro/pc-stuff/freedos/files/distributions/1.2/FD12FLOPPY.zip curl: (22) The requested URL returned error: 404Error: qemu-virgl: Failed to download resource "qemu-virgl--test-image" D…