我们发现,如果没有自定义环境变量,系统自带的环境变量就会被打印,但是如果自定义环境变量系统自带的环境变量就不会被打印。
那么如果我两个都想要怎么办呢?
这个函数传入你的自定义环境变量就可以了,作用就是将你定义的环境变量导入到系统当中。
这里穿插一个问题,一个程序运行之前,是先调用main还是先调用exec函数呢?
是先调用exec函数,因为它的作用上面说了,是将程序加载到内存中,Linux中,它就是加载器。
调用exec函数之后会将自己的参数等等传给main函数,这就是为什么之前说main函数有三个参数,谁传给他的。
int execvpe(const char *file, const char *argv[], …,char *const envp[]);
这个参数就不说了,都说过了。
注意:上面这些接口都是execve系统调用,其他的都是封装,为了让我们有更好的选择性。
模拟实现简易的shell
首先来利用main函数的参数来实现一个功能:
那么我们可有利用这个模拟实现一个简单的shell。
第一步先设置输入和输出,并且创建一个字符数组储存输入的参数。
我们输入一个字符串是abc,然后会按回车,也就是说实际上是abc\n,如果我要在打印信息%s后面加一个\n那么就会多出一行,不加容易出现缓冲区不刷新问题,所以我们要去除输入末尾的\n。
第二步要进行字符串分割,因为我们在屏幕输入的是ls -a -l这种,但是exec函数要用到的是字符指针数组类型的,所以我们创建一个字符指针数组,然后进行分割放进字符指针数组:
这里要说一下内建命令,我们在输入ls什么的时候不同文件会有颜色,但是如果调用exec里面就需要自己添加颜色选项,我们又不能在屏幕输入,所以只能在代码中添加,首先判断一定要是ls命令才行,然后添加颜色选项。
像这种不需要让子进程来执行,而是shell自己执行的就叫做内建命令。
第三步是打印,创建一个子进程帮我们工作,这是因为exec函数会替换掉原来程序中所有的代码和数据:
然后我们还可以设置一个条件编译来看看字符指针数组中的字符切割是否正确:
先来测试一下上面的程序是否正确
但是如果我们输入cd …就会发现根本没有任何变化,这是为什么呢?
先创建一个其他程序来看一下一个进程的状态:
用ls /proc/pid -al
cwd是当前进程的工作目录,也是我们平时说的当前路径,exe是当前程序执行的是磁盘路径的哪一个程序。
那么这个当前路径可以改变嘛?通过一个函数是可以的:
谁调用这个函数就更改谁的工作目录,参数是更改到哪个目录。
如果更改了工作目录,那么以后这个程序再进行创建文件等等操作,就会再新的工作目录创建,因为系统默认是跟可执行程序同一个目录下去创建新文件。
那么刚才我们的shell不能cd …是因为他只能让当前工作目录发生变化,因为shell是通过创建子进程去执行命令,我们让目录进行变化的时候是让子进程去帮助执行,也就是说改变的其实是子进程的目录,和父进程没有任何关系,所以说这里还需要创建一个内建命令:
之有前还有一个命令,是echo $?,返回的是最近一次退出码
首先创建两个全局变量保存退出码和信号,然后再用他们储存子进程返回的结果:
最后进行判断:
这里简单的完善一下就可以了,主要是综合了上面所说的大部分内容。