## 一、shell ### 1.1 shell 解释器 ``` /bin/bash [默认] /bin/sh ``` ### 1.2 shell 脚本调用 系统自动调用: ``` /etc/profile 系统环境变量, 每次系统启动时调用 ~/.bashrc 用户环境变量, 用户登录时调用 ``` 用户调用: ``` 1) chmod +x xxx.sh ./xxx.sh 2) bash xxx.sh 3) . xxx.sh 相当于 source xxx.sh ``` ### 1.3 shell 脚本的定义 第一行: ``` #!/bin/bash ``` ### 1.4 shell 变量 ``` 定义: 变量名=值 引用: $变量名, "$变量名 xxx" 重新赋值: 变量名=新值 算术计算: 变量名=$(( 变量名 算术运算符 值 )) 预定变量: $#, $*, $?, $1~9, $0 进程名, $$ 进程ID 环境变量: $环境变量名 export 变量名=值 设置环境变量 特殊的变量写法: (变量名=值; 其它语句) 不影响外部的同名变量 {变量名=值; 其它语句;} 影响外部的同名变量 ``` ### 1.5 条件测试语句 语法: ``` test 操作符 变量 test 变量1 操作符 变量2 [ 操作符 变量 ] [ 变量1 操作符 变量2 ] ``` 文件相关: ``` -e 是否存在 -d 是目录 -f 是文件 -r 可读 -w 可写 -x 可执行 -L 符号连接 -c 是否字符设备 -b 是否块设备 -s 文件非空 ``` 字符串相关: ``` = 两个字符串相等 != 两个字符串不相等 -z 空串 -n 非空串 ``` 数字相关: ``` -eq 数值相等 -ne 数值不相等 -gt 数 1 大于数 2 -ge 数 1 大于等于数 2 -lt 数 1 小于数 2 -le 数 1 小于等于数 2 ``` 复合测试相关: ``` command1 && command2 左边命令执行成功(即返回 0)shell 才执行右边的命令 command1 || command2 左边的命令未执行成功(即返回非 0)shell 才执行右边的命令 ``` 多重条件判定: ``` -a 多个条件必须都为真(true) 结果才为true -o 多个条件中只需要一个为真,结果为true ! 条件取反 ``` ### 1.6 分支语句 if 分支: ``` if [ 条件1 ]; then 执行第一段程序 elif [条件2 ]; then 执行第二段程序 else 执行第三段程序 fi ``` case 分支: ``` case $变量名称 in “第一个变量内容”) 程序段一 ;; “第二个变量内容”) 程序段二 ;; *) 其它程序段 exit 1 esac ``` for 语句: ``` for (( 初始值; 限制值; 执行步阶 )) do 程序段 done ``` while 语句: ``` while [ condition ] do 程序段 done ``` until 语句: ``` until [ condition ] do 程序段 done ``` 循环语句中可以使用 break 和 continue。 ### 1.7 函数 定义函数: ``` 函数名(){ 命令 ... } function 函数名(){ 命令 ... } ``` 调用函数: ``` 函数名 param1 param2…… ``` 函数体内,可以使用$1,$2 ...$9 读取传递到函数的参数。 ## 二、系统调用和进程(上) ### 2.1 系统调用 I/O 函数 ``` #include #include #include int open(const char *pathname, int flags); 文件存在时 int open(const char *pathname, int flags, mode_t mode); 文件不存在时 #include int close(int fd); ssize_t write(int fd, const void *addr,size_t count); ssize_t read(int fd, void *addr, size_t count); #include int remove(const char *pathname); 【库函数】 ``` ### 2.2 fcntl 函数 ```c // 修改标准的输入设备的标识为非阻塞 int flags = fcntl(STDIN_FILENO, F_GETFL); flags |= O_NONBLOCK; fcntl(STDIN_FILENO, F_SETFL, flags); ``` 取消非阻塞 ```c int flags = fcntl(STDIN_FILENO, F_GETFL); flags ^= O_NONBLOCK; fcntl(STDIN_FILENO, F_SETFL, flags); ``` 获取文件的类型(文件、目录) ``` struct stat info; stat(char *path, &info); info.st_mode & S_IFDIR 是否为目录的验证 info.st_mode & S_IFREG 是否为文件的验证 ``` 打开目录和读取目录: ``` dir *opendir(char *path) dirent *readdir(dir *) ``` ### 2.3 进程的定义 ``` 进程拥有自己独立的处理环境和系统资源(处理器、存储器、I/O 设备、数据、程序) ``` 进程的三种状态: ``` 就绪态: 进程已经具备执行的一切条件,正在等待分配 CPU 的处理时间。 执行态: 该进程正在占用 CPU 运行。 等待态: 进程因不具备某些执行条件而暂时无法继续执行的状态。 ``` 进程控制块(PCB): ``` OS 是根据 PCB 来对并发执行的进程进行控制和管理的。 系统在创建一个进程的时候会开辟一段内存空间存放与此进程相关的 PCB 数据结构。 PCB 是操作系统中最重要的记录型数据结构,包含进程号、返回状态、运行时间等。 ``` 获取进程号: ``` #include #include pid_t getpid(void) 获取本进程号(PID) pid_t getppid(void) 获取父进程号(PPID) pid_t getpgid(pid_t pid) 获取进程组号(PGID),参数为0时返回当前PGID,否则返回参数指定的进程的PGID ``` ### 2.4 进程控制函数 ``` #include #include pid_t fork(void); 创建独立空间的子进程 #include unsigned int sleep(unsigned int sec); 挂起或休眠sec秒 #include #include pid_t wait(int *status); pid_t waitpid(pid_t pid, int *status, int options) ``` 获取子进程的返回状态: ``` WIFEXITED(status) 如果子进程是正常终止的,取出的字段值非零 WEXITSTATUS(status) 返回子进程的退出状态,退出状态保存在 status 变量的 8~16 位(低位的第2个字节)。在用此宏前应先用宏 WIFEXITED 判断子进程是否正常退出,正常退出才可以使用此宏。 ```