第4章 文件和目录(5)_贯穿案例2:mini shell(1)
6. 贯穿案例2:mini shell(1)
【阶段性任务】实现cd、pwd和quit命令

//job.h
#ifndef __JOB_H__
#define __JOB_H__ //接收命令行参数
typedef struct
{
char** args; //对应于主函数中的char* argv[]参数(含进程名本身)
}Program; //命令行中可以包含多个程序(命令),如
//#date;ls -l,单个或多个命令通过cmd传入Job结构体中
typedef struct
{
char* cmd; //单条命令或多条命令(用分号隔开)
int progs_num; //作业中包含程序的数量
Program* progs; //各个程序的命令行参数
}Job; //创建作业
extern Job* create_job(char* cmd);
//销毁作业
extern void destroy_job(Job* job);
//创建进程(命令)
extern Program* create_program(char** arg);
//销毁进程(命令)
extern void destroy_program(Program* prog);
//将命令加入作业中
extern int add_program(Job* job, Program* prog); #endif
//job.c
#include "job.h"
#include <malloc.h>
#include <assert.h>
#include <string.h> //创建作业
Job* create_job(char* cmd)
{
Job* job = (Job*)malloc(sizeof(Job));
assert( job != NULL); job->cmd = (char*)malloc(sizeof(char) * strlen(cmd));
assert(job->cmd != NULL);
strcpy(job->cmd, cmd); job->progs_num = ;
job->progs = NULL; return job;
} //销毁作业
void destroy_job(Job* job)
{
assert(job != NULL);
free(job->progs);
free(job->cmd);
free(job);
} //arg格式:command arg0 arg1 ==> 返回3
static int arg_num(char** arg)
{
int ret = ;
char* start = arg[]; while(start != NULL){
start = arg[++ret];
} return ret;
} //创建进程(命令)
Program* create_program(char** arg)
{
Program* prog = (Program*)malloc(sizeof(Program));
assert(prog != NULL); int counter = arg_num(arg);
prog->args = (char**)calloc(counter + , sizeof(char*)); //以NULL结尾 int i = ;
for(i=; i< counter; i++){
int len = strlen(arg[i]);
prog->args[i] = (char*)malloc(len);
assert(prog->args[i] != NULL); strcpy(prog->args[i], arg[i]);
} prog->args[i] = NULL; //指针数组,以NULL结尾 return prog;
} //销毁进程(命令)
void destroy_program(Program* prog)
{
assert(prog != NULL); int i = ;
while(prog->args[i] != NULL)
{
free(prog->args[i++]);
} free(prog->args);
free(prog);
} //将命令加入作业中
int add_program(Job* job, Program* prog)
{
//重新申请一片空间以增加一条命令进来,放入job->progs中
Program* ps = (Program*)malloc(sizeof(Program) * (job->progs_num + ));
memcpy(ps, job->progs, job->progs_num * sizeof(Program)); ps[job->progs_num++] = *prog;//将新的进程(命令)加入进来 free(job->progs); //释放旧的程序(命令)组
job->progs = ps; return job->progs_num - ; //返回新命令的索引号
}
//mshell.c
#include "job.h"
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <malloc.h>
#include <string.h>
#include <assert.h> char* prompt = "mshell> "; //命令行的提示符
#define MAX_COMMAND_LEN 256 //命令行最多的字符数 //cd命令
void cd_fun(Program* prog)
{
if(chdir(prog->args[]) < ){
perror("cd error");
}
} //pwd命令
void pwd_fun(Program* prog)
{
char buffer[];
memset(buffer, , sizeof(buffer)); if(getcwd(buffer, sizeof(buffer)) == NULL){
perror("pwd error");
} printf("%s\n", buffer);
} //分析命令所带的参数(含进程名本身)
void split_cmd(Job* job, char* arguments)
{
char** args = (char**)malloc(MAX_COMMAND_LEN * sizeof(char*));
assert( args != NULL); char* cmd = strtok(arguments, " "); args[] = (char*)malloc(strlen(cmd) * sizeof(char)); //命令本身
strcpy(args[], cmd); int i = ;
char* s = NULL;
while((s = strtok(NULL, " ")) != NULL){ //将参数分隔出来
args[i] = (char*)malloc(strlen(s) * sizeof(char));
strcpy(args[i++], s);
} //根据args创建一个Program
Program* prog = create_program(args);
add_program(job, prog); int j = ;
for(j=; j < i; j++){
free(args[j]);
} free(args);
} //多条命令的解析
void parse_cmd(Job* job, char* line)
{
char buff [MAX_COMMAND_LEN]; //以“;”号分隔多条命令
char* pos = line;
char* start = line;
int count = ;
while( start < (line + strlen(line)) ){
memset(buff, , sizeof(buff));
if((pos = strchr(pos, ';')) == NULL)
{
pos = line + strlen(line);
} count = pos-start;
if(count > ){
memcpy(buff, start, count);
split_cmd(job, buff);
}
start = ++pos;
}
} //执行命令
void execute_cmd(Job* job)
{
int i = ;
for(i=; i<job->progs_num; i++)
{
//cd命令
if(!strcmp(job->progs[i].args[], "cd")){
cd_fun(&job->progs[i]);
} //pwd命令
if(!strcmp(job->progs[i].args[], "pwd")){
pwd_fun(&job->progs[i]);
} //quit命令
if(!strcmp(job->progs[i].args[], "quit")){
exit();
return;
}
}
} int main(int argc, char* argv[])
{
char buffer[MAX_COMMAND_LEN];
memset(buffer, , MAX_COMMAND_LEN); ssize_t size = strlen(prompt) * sizeof(char);
write(STDOUT_FILENO, prompt, size); ssize_t len = ;
while(){
len = read(STDIN_FILENO, buffer, MAX_COMMAND_LEN);
buffer[len -] = ; //以NULL结尾 if(strlen(buffer) > ){
Job* job = create_job(buffer); //解析命令
parse_cmd(job, buffer);
//执行命令
execute_cmd(job); destroy_job(job);
} write(STDOUT_FILENO, prompt, size);
memset(buffer, , MAX_COMMAND_LEN);
} return ;
}
第4章 文件和目录(5)_贯穿案例2:mini shell(1)的更多相关文章
- 第8章 信号(6)_贯穿案例2:mini shell(3)
4. 贯穿案例2:mini shell(3) (1)之前存在问题 ①刚运行时,mshell作为前台进程.运行的其他命令会被加入新的进程组,并且调用tcsetpgrp将这个进程组设置为前台进程组,因此m ...
- 零基础学Python--------第10章 文件及目录操作
第10章 文件及目录操作 10.1 基本文件操作 在Python中,内置了文件(File)对象.在使用文件对象时,首先需要通过内置的open() 方法创建一个文件对象,然后通过对象提供的方法进行一些基 ...
- 第7章 进程关系(5)_贯穿案例2:mini shell(2)
5. 贯穿案例2:mini shell(2) (1)己经完成的功能:pwd.cd.exit命令 (2)阶段性目标: ①env.export.echo及其他命令 ②标准输入.输出重定向"> ...
- UNIX环境高级编程 第4章 文件和目录
第三章说明了关于文件I/O的基本函数,主要是针对普通regular类型文件.本章描述文件的属性,除了regular文件还有其他类型的文件. 函数stat.fstat.fstatat和lstat sta ...
- apue学习笔记(第四章 文件和目录)
本章将描述文件系统的其他特性和文件的性质. 函数stat.fstat.fstatat和lstat #include <sys/stat.h> int stat(const char *re ...
- Linux命令应用大词典-第 15章 文件、目录权限和属性
15.1 chmod:更改文件和目录的模式 15.2 chown:更改文件和目录的用户所有者和组群所有者 15.3 chgrp:更改文件或目录的所属组 15.4 umask:显示和设置文件及目录创建默 ...
- UNIX系统高级编程——第四章-文件和目录-总结
文件系统: 以UNIX系统V文件系统为例: 磁盘分为区,每个分区都有自己的文件系统: i节点是固定长度的记录项,包含了文件的相关信息.目录项包含文件名和i节点号.stat结构中除文件名和i节点编号 ...
- 《Unix环境高级编程》读书笔记 第4章-文件和目录
1. stat结构的基本形式: on error 24. 设备特殊文件 每个文件系统所在的存储设备都由其主.次设备号表示. 设备号所用的数据类型是基本系统数据类型dev_t. 主设备号标识设备驱动程序 ...
- apue 第4章 文件和目录
获取文件属性 #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> int stat(c ...
- 《UNIX环境高级编程》(APUE) 笔记第四章 - 文件和目录
4 - 文件和目录 1. 函数 stat.fstat.fstatat 和 lstat #inlcude <sys/stat.h> int stat(const char *restrict ...
随机推荐
- 大数开根号java模板
利用逼近的思路直接二分开方找出值 package lanqiao; import java.math.BigInteger; import java.util.Scanner; public clas ...
- spark 与 Hadoop 融合后 Neither spark.yarn.jars nor spark.yarn.archive is set
参考文献: http://blog.csdn.net/lxhandlbb/article/details/54410644 每次提交Spark任务到yarn的时候,总会出现uploading reso ...
- Dlib与OpenCV图片转换
re: 1. https://zhuanlan.zhihu.com/p/36489663 2. https://stackoverflow.com/questions/38180410/convert ...
- 查找单链表的倒数第k个值
刚开始,我想到的是一种笨方法,先遍历单链表,计算出单链表的长度len,然后再从头遍历单链表到第len-k个节点,那么 这个节点既是单链表的倒数第k个节点. 不过这种算法时间复杂度挺高的,还有一种更简单 ...
- Largest Submatrix of All 1’s
Given a m-by-n (0,1)-matrix, of all its submatrices of all 1’s which is the largest? By largest we m ...
- 【Quartz】Quartz将Job保存在数据库中所需表的说明
QRTZ_CALENDARS 以 Blob 类型存储 Quartz 的 Calendar 信息 QRTZ_CRON_TRIGGERS 存储 Cron Trigger,包括 Cron表达式和时区信息 ...
- ZH奶酪:【数据结构与算法】基础排序算法总结与Python实现
1.冒泡排序(BubbleSort) 介绍:重复的遍历数列,一次比较两个元素,如果他们顺序错误就进行交换. 2016年1月22日总结: 冒泡排序就是比较相邻的两个元素,保证每次遍历最后的元素最大. 排 ...
- redash docker 运行
redash .superset .metabase 都是很不错的数据分析工具,支持多种数据源,同时可以方便的生成报表 基本上都支持定制化报表界面.通知(定时),metabase 有点偏产品,supe ...
- dgraph 基本查询语法 一
dgraph 的查询语法是在graphql 上的扩展,添加了新的支持,同时官方提供了一个 学习的网站 https://tour.dgraph.io/ 基本环境(cluster 模式的) 参考 gith ...
- Oracle集合类型
Oracle集合类型介绍 集合类型 1. 使用条件: a. 单行单列的数据,使用标量变量 . b. 单行多列数据,使用记录 c. 单列多行数据,使用集合 *集 ...