Linux下的简单解释器

一、程序概述

1. 基本概述

这次大作业是在centos 7系统下完成的,使用的编程语言为:c++。


打开方式

存放路径为:/home/null/workspace

2. 实现功能

完成了要求实现的基本命令及部分可选做的扩展命令

1、pwd  //显示当前所在目录的路径名

2、list  <目录名>          //列出指定目录名中的所有目录及文件

3、cd  <目录名或路径>   //改变当前工作目录

4、mkdir<目录名>     //新建目录

5、rmdir  <目录名>      //删除目录

6、exit                 //退出命令解释程序

7、rename <旧文件名> <新文件名>  //重命名一个文件或目录

8、copy <已存在的文件名及其路径> <副本文件名及其路径>  //复制一个已存在的文件

二、设计思路

本次大作业要求使用c++,而且必须在linux的环境下操作,于是我使用了很多在linux下的函数库。

1、函数说明

1、getcwd()函数

char * getcwd(char * buf,size_t size);

getcwd()会将当前的工作目录绝对路径复制到参数buf所指的内存空间,参数size为buf的空间大小。

用于实现pwd的命令解释,显示当前目录所在的文件名。

2、opdir()函数&readdir()函数&closedir()函数

DIR *opendir(const char *name);

它返回一个DIR*类型,这是一个句柄,并且这个句柄要传给reddir函数。

struct dirent *readdir(DIR *dir);

该参数是opendir函数返回的句柄,而该函数的返回值是struct dirent* 类型。

该结构体:

struct dirent {

ino_t          d_ino;       /* inode number */

off_t          d_off;       /* offset to the next dirent */

unsigned short d_reclen;    /* length of this record */

unsigned char  d_type;      /* type of file */

char           d_name[256]; /* filename */

};

这个结构体的d_name存放的就是文件的名字,这里的文件包括普通文件,目录文件等等,在linux的思想中,所有的东西都是文件。

int closedir(DIR *dir);

在结束时使用,用于关闭dir。

用于实现list的命令解释,列出指定目录名中的所有目录及文件。

3、chdir()函数

int chdir (const char *pathname);

进程调用该函数可以改变当前工作目录

用于实现cd的命令解释,改变当前工作目录。

4、mkdir()函数

int mkdir(const char *pathname, mode_t mode);

用mode的方式创建以pathname命名的目录。如/home/null/helloworld 创建的文件夹名字则为helloworld。mode用于定义创建目录的权限 如 :

//00700权限,代表该文件所有者拥有读,写和执行操作的权限

//00070权限,代表该文件用户组拥有读,写和执行操作的权限

而我使用的mode_t 0777为八进制 转换成二进制为 000 111 111 111 对应为可读可写可执行,

每三个为一组。 分别为文件所有者,用户者,其他用户。即所有用户拥有可读可写可执行的权限。

用于实现mkdir的命令解释,新建目录。

5、rmdir()函数

int _rmdir(const char *dirname);

调用该函数删除dirname的目录

用于实现rmdir的命令解释,删除目录。

6、rename()函数

int rename(const char * oldpath,const char * newpath);

调用该函数将参数为oldpath的文件重命名为参数为newpath的文件,若newpath所指定的文件已存在,则会被删除。

用于实现rename的命令解释,删除目录。

7、open()函数&read()函数&write()函数

int open(constchar* pathname,int flags,mode_tmode);其中最后一项为可选项。

调用该函数可以打开参数为pathname的文件,参数flag可以决定用什么方式打开文件。eg:

O_RDONLY 以只读方式打开文件。

O_WRONLY 以只写方式打开文件。

O_RDWR 以可读写方式打开文件。上述三种旗标是互斥的,也就是不可同时使用,但可与下列的旗标利用OR(|)运算符组合。

O_CREAT 若欲打开的文件不存在则自动建立该文件。

ssize_t  read(int fd,void * buf ,size_t count);

调用该函数可以由已打开的文件读取数据。read()会把参数fd 所指的文件传送count个字节到buf指针所指的内存中。若参数count为0,则read()不会有作用并返回0。返回值为实际读取到的字节数,如果返回0,表示已到达文件尾或是无可读取的数据,此外文件读写位置会随读取到的字节移动。简单来说就是从fd中调count个字节到缓冲字符区buf中。

ssize_t write (int fd,const void * buf,size_t count);

调用该函数参数buf所指的内存(read中的缓冲区)写入count个字节到参数fd所指的文件内。

用于实现copy的命令解释,复制一个已存在的文件。

三、详细设计


程序设计流程图

1、int main()

while(1) {

      //shell prompt

      cout<<"[null@]$";

      cin>>str;   

if(str == "pwd"){

        pwd();}

      else if(str == "list"){

        list();}

.....}

如流程图所示,在输出提示后则进入一个死循环给str赋值,当输入的str为可执行命令时则进入函数,执行完毕后等待下一个命令。

2、void pwd();

显示当前绝对工作路径。

char ptr[40];

 getcwd(ptr,sizeof(ptr)); 

 cout<<ptr<<endl; 

调用getcwd()函数,直接获取当前路径至ptr中,直接输出即可。

3、int list();   

列出指定目录名中的所有目录及文件。

    char path[20];

    cin>>path;   

    // char path[]="/home/null/workspace";

    const char *ptr=path;  //进入函数后,输入路径保存在path中,并转换为opdir支持的const char格式。

    DIR *dir; //声明一个句柄'dir'

    struct dirent *file; //readdir函数的返回值就存放在这个结构体中,所要查找的文件名也放在这里,用指针的方式就可以查找。

    int len = 0;

    int i;

    char filename[256][256];

    if(!(dir = opendir(ptr))) //如果需要打开的目录不存在

    {

      // printf("error path : %s!\n",path);

        cout<<"error path:"<<path<<endl; //输出错误的路径

        return -1;}

    while((file = readdir(dir)) != NULL)

    {

        if(strncmp(file->d_name, ".", 1) == 0) //把当前目录'.',上一级目录'..'去除

            continue;

        strcpy(filename[len++], file->d_name); //保存遍历到的文件名}

    closedir(dir); //关闭dir流

    for(i = 0; i < len; i++)

    {cout<<filename[i]<<" ";}

    cout<<endl;

    return 0;

进入函数后,需要输入查找得目录的绝对路径path,若opendir返回值为0,即打开目录错误,则输出错误路径,跳出函数。若打开成功则跳过if语句。随后执行 while((file = readdir(dir)) != NULL)语句,当读出来的file不为空则一直循环。第一个if语句是把所有'.'开头文件及文件夹去掉,因为在Linux中‘.’ ‘..’ ‘.xxx’为当前目录,上一级目录及隐藏文件,不符合我们要查找的内容。执行完循环后则输出查找到的文件夹并退出循环。

4、void cd();

改变当前工作目录

  char dirname[20];

  cin>>dirname;

  if(chdir(dirname) == -1)

  {  cout<<"the directory isn't exit!"<<endl; }

输入要进入的目录,调用linux c函数库的chdir()函数执行chadir命令。若返回值为1则cd命令输入错误(文件夹不存在)。

5、void _mkdir();

新建目录

  char filename[20]; cin >> filename;

  if(mkdir(filename, 0777) == 0)  {

    cout<<filename<<" created successful!"<<endl;}

  else {  cout<<filename<<" created unsuccessful!"<<endl;}

输入要创建的文件夹名称,调用linux c函数库的mkdir()函数执行makedir命令。若返回值为0则mkdir命令执行成功。

6、void _rmdir();

删除目录

char filename[20];

  cin >> filename;

  if(rmdir(filename) == 0)   {

    cout<<filename<<" deleted successful!"<<endl;}

  else  cout<<filename<<" deleted unsuccessful!"<<endl;

输入要删除的文件/文件夹名称,调用linux c函数库的rmdir()函数执行deldir命令。若返回值为0则deldir命令执行成功。

7、void _rename();

重命名一个文件或目录

  char filename[20], filename1[20];

  cin>>filename>>filename1;

  if(rename(filename,filename1) == 0)

  {  cout<<"changed successful!"<<endl;}

  else cout<<"changed unsuccessful!"<<endl;

输入要重命名的旧文件/文件夹名称以及新文件/文件夹名称,调用linux c函数库的rename()函数执行rename命令。若返回值为0则rename命令执行成功。

8、void copy();

复制一个已存在的文件

char src[20],des[20];

char buff[1024];

int temp; //读出了多少个字节

cin>>src>>des;

int fd,fd2;

fd = open(src,O_RDWR|O_CREAT);

fd2 = open(des,O_RDWR|O_CREAT);

while(temp = read(fd,buff,1024))

{write(fd2,buff,temp);}

 cout<<"copy successful!"<<endl;

输入需要复制的文件路径赋值至src,输入需要将复制的文件到的绝对路径至des。用open函数以O_RDWR|O_CREAT的旗标打开这两个路径 。O_RDWR|的含义是以可读写方式打开文件。O_CREAT 的含义是若欲打开的文件不存在则自动建立该文件。将fd的文件每次以1024个字节读入缓冲数组buff中,并用temp读出返回值(读了多少个字节),再将fd的文件以每次temp个字节将buff中的数组写入fd2中。

缺点:检测不到复制失败的情况。

四、使用情况


打开解释器


pwd命令


list命令


mkdir命令


rename命令


remdir命令


copy命令

五、小结

1、遇到的困难

写了一段时间的python对于这种接近底层的语言感到陌生,c++不像别的语言一样可以有很多打包好的方便的方法,也不像python一样可以不用定义数据类型的使用数据。就算有别人打包好的函数,还是有很多算法都是要自己去写的,而在linux系统下写这份大作业更是需要花时间去理解Linux c函数,不仅要理解函数的调用还要理解参数的意义和不同情况下的函数返回值。虽然一开始有一头雾水的感觉,但在查阅了资料和文档后还是能够实现要求的指令的。

2、总结和感想

一开始的时候,在装虚拟机和系统都费了不少的心思,比如虚拟机要破解、系统有各式各样的Linux系统,(如Linux Ubuntu 、Centos等等)、装系统的时候装了minimal 只有一个黑框框完全不知道该怎么办,即使查阅了网上的Linux的命令大全也不知道要怎么样才能编译c程序。后来通过看教学视频(腾讯课堂)和上机实验明白了要如何编译使用C程序和熟悉了一些基本的Linux的命令,迷上了vi编辑器。虽然很难用,但用得好只用键盘就能完成以前要鼠标点来点去的操作,是一个能实实在在的提高写代码效率的编辑器。也能理解为什么有人那么喜欢Linux系统了哈哈。虽然我为了明天能交作业用的是gedit

六、参考链接

https://blog.csdn.net/wwj_748/article/details/8309440

https://www.cnblogs.com/ace-wu/p/6640186.html (open write read )

https://blog.csdn.net/u014453898/article/details/54864666(linux下的c语言复制方法)

https://blog.csdn.net/qq_30941699/article/details/55517763(linux下c语言递归法查找文件,并打印绝对地址)

https://blog.csdn.net/silentob/article/details/76994618(const char * 、char const *、 char * const 三者的区别)

https://www.cnblogs.com/avril/archive/2010/03/22/1691477.html(int  main(int  argc,  char*  argv[])详解)

https://blog.csdn.net/jichl/article/details/7948724(ftw遍历目录树)

七、源代码



#include<iostream>

#include<cstring>

#include<sys/types.h>

#include<sys/stat.h>

#include<dirent.h>

#include<stdlib.h>

#include<fcntl.h>

#include<time.h>

#include<queue>

#include<ftw.h>

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#include <unistd.h>

using namespace std;

//declare method

void pwd();   

int list();    

void cd();     

void _mkdir();

void _rmdir();

void _rename();

void copy();

int main()

{

cout<<"可执行的命令:"<<endl;

cout<<"pwd                                         //显示当前所在目录的路径名"<<endl;

cout<<"list     <目录名>                            //列出指定目录名中的所有目录及文件"<<endl;

cout<<"cd       <目录名或路径>                       //改变当前工作目录"<<endl;

cout<<"mkdir    <目录名>                            //新建目录"<<endl;

cout<<"rmdir    <目录名>                            //删除目录"<<endl;

cout<<"rename   <旧文件名> <新文件名>                 //重命名一个文件或目录"<<endl;

cout<<"copy      <已存在的文件名及其路径> <副本文件名及其路径>    //复制一个已存在的文件"<<endl;

cout<<"rmdir    <目录名>                            //删除目录"<<endl;

// cout<<"find     <目录> -name <待查找的文件名>         //在指定的目录及其子目录中查找指定的文件,并输出查找到的文件的绝对路径。"<<endl;

cout<<"exit                                        //退出命令解释程序"<<endl;

   string str;

   while(1) {

      //shell prompt

      cout<<"[null@]$";

      cin>>str;     

/*

      switch(str){

    case "pwd":

          pwd();

          break;

    default:

    cout<<"no command not found"<<endl;

}

*/

      if(str == "pwd"){

         pwd();}

      else

      if(str == "list"){

         list();}

      else

      if(str == "cd"){

         cd();}

      else

      if(str == "mkdir"){

         _mkdir();}

      else

      if(str == "rmdir"){

         _rmdir();}

      else

      if(str == "rename"){

         _rename();}

      else

      if(str =="copy"){

         copy();}

      else

      if(str=="exit"){

         break;}

      else

      cout<<"no command not found"<<endl;

                        }

    return 0;

}

void pwd()

{

   char ptr[40];  

   getcwd(ptr,sizeof(ptr));   

   cout<<ptr<<endl;  

}


int list()

{

    char path[20];

    cin>>path;


//    char path[]="/home/null/workspace";

    const char *ptr=path;

    DIR *dir; //声明一个句柄'dir'

struct dirent *file; //readdir函数的返回值就存放在这个结构体中

    int len = 0;

    int i;

    char filename[256][256];

if(!(dir = opendir(ptr))) //如果需要打开的目录不存在

    {

       // printf("error path : %s!\n",path);

        cout<<"error path:"<<path<<endl;

        return -1;

    }

    while((file = readdir(dir)) != NULL)

    {

        if(strncmp(file->d_name, ".", 1) == 0) //把当前目录'.',上一级目录'..'去除

            continue;

strcpy(filename[len++], file->d_name); //保存遍历到的文件名

    }

    closedir(dir);

    for(i = 0; i < len; i++)

    {

        cout<<filename[i]<<" ";

    }

    cout<<endl;

    return 0;

}

void cd()

{

   char dirname[20];

   cin>>dirname;

   if(chdir(dirname) == -1)

   {

      cout<<"the directory isn't exit!"<<endl;


    }


}


void _mkdir()

{

  char filename[20];

  cin >> filename;

  if(mkdir(filename, 0777) == 0)

  {

     cout<<filename<<" created successful!"<<endl;

   }

   else

   {  cout<<filename<<" created unsuccessful!"<<endl;

   }

}


void _rmdir()

{

   char filename[20];

   cin >> filename;

   if(rmdir(filename) == 0)

   {

    cout<<filename<<" deleted successful!"<<endl;

   }

   else

    cout<<filename<<" deleted unsuccessful!"<<endl;

}

void _rename()

{


  char filename[20], filename1[20];

  cin>>filename>>filename1;

  if(rename(filename,filename1) == 0)

  {

    cout<<"changed successful!"<<endl;

   }

   else

     cout<<"changed unsuccessful!"<<endl;

}

void copy(){

char src[20],des[20];

char buff[1024];

int temp; //读出了多少个字节

cin>>src>>des;

// char const *src_path = argv[1];

// char const *des_path = argv[2];

int fd,fd2;

fd = open(src,O_RDWR|O_CREAT);

fd2 = open(des,O_RDWR|O_CREAT);

while(temp = read(fd,buff,1024))

{

write(fd2,buff,temp);

}

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容