从零开始UNIX环境高级编程(1):Unix基础知识

1. 概述

《UNIX环境高级编程》介绍的是不同版本的Unix操作系统提供的服务。那具体指的是哪些服务呢?首先,需要了解整个Unix操作系统的体系结构。

1.1 Unix体系结构

Unix操作系统的体系结构,如下图所示:

Unix操作系统体系结构
  • 硬件: 芯片、电路板、磁盘、键盘等。

  • 内核:有对所有硬件的完全访问权,并且只有内核才能操作硬件

  • 系统调用:操作内核的接口

  • shell:是一种特殊的应用程序,分为command-line interface (CLI) shell和Graphical shell

  • 公用函数库:建立在系统调用基础之上

  • 应用程序:可以进行系统调用,也可以调用公用函数库

从Unix体系结构可以看出,如果需要使用Unix系统提供的服务,那么必须进行系统调用或者库函数调用。因此,本书介绍的内容就是Unix系统函数和公用库函数的使用方式。

1.2 系统调用和库函数

  • 库函数分为两类:一种是对系统调用的封装,如malloc是对库函数sbrk(2)的封装。另外一种库函数没有进行系统调用,如strcpy,格式转换这类函数。
系统函数和库函数

1.3 Unix程序员手册

我们经常都能看到这种表示: ls(1) write(2) group(5) ... ..

其中括号里面的数字代表手册的区段,Unix程序员手册被分为8个部分,每个部分描述不同的内容。

手册区段

可以通过man命令来查询手册:

man 5 group

2. 登录

2.1 口令文件(/etc/passwd)

用户在登录系统时,需要输入登录名和密码。这些信息都存放在口令文件(通常是/etc/passwd文件)中查看登录名,组成字段信息如下,以逗号隔开。

ckt@ubuntu:~$ cat /etc/passwd
ckt:x:1000:1000:ckt,,,:/home/ckt:/bin/bash
  • 登录名:ckt
  • 加密口令:x
  • 用户ID:1000
  • 组ID:1000
  • 注释段:ckt,,,
  • 起始目录:/home/ckt
  • shell程序:/bin/bash

2.2 用户标识

用户ID,组ID和附属组ID用作用户的唯一标识,以区分不同的用户。

用户标识
  • 查看组文件/etc/group
 ckt@ubuntu:~$ cat /etc/group | grep zhm
 zhm:x:1001:ckt

使用man命令查看/etc/group格式为: group_name:password:GID:user_list

  • 组名称:zhm
  • 密码:x
  • 组ID:1001
  • 用户列表:ckt

  • 示例程序
    获得当前用户的user ID和group ID
#include "apue.h"

int main(int argc, char const *argv[])
{
    int user_id = -1;
    int group_id = -1;
    user_id = getuid();
    group_id = getgid();
    printf("user_id = %d, group_id = %d \n", user_id, group_id );
    return 0;
}
  • 运行结果
    以ckt作为用户运行程序,得到user_id和group_id为1000,和通过cat /etc/passwd得到的结果一致。
    再切换到root用户下运行,得到user_id和group_id为0。
ckt@ubuntu:~/work/unix/code$ ./a.out 
user_id = 1000, group_id = 1000 

root@ubuntu:/home/ckt/work/unix/code# ./a.out 
user_id = 0, group_id = 0 

2.3 shell

是一种特殊的应用程序,分为command-line interface (CLI) shell和Graphical shell。由/etc/passwd中设置的shell path可以知道,我们使用的shell是Bourne-Again shell。

常见shell

3. 文件和I/O操作

3.1 文件和目录

文件和目录.png

文件描述符

内核用来标识一个特定进程正在访问的文件

  • 文件描述符示例代码
    使用open打开一个文件,并打印出文件描述符的值。
#include "apue.h"
#include <fcntl.h>

int main(int argc, char const *argv[])
{
    int file_descr = -1;
    file_descr = open("/home/ckt/work/unix/code/MySignal.c", O_RDONLY);
    printf("file descriptor = %d\n", file_descr);
    return 0;
}
  • 运行结果

成功打开一个文件时,返回一个非负整数作为文件描述符。打开一个不存在的文件,返回-1。

ckt@ubuntu:~/work/unix/code$ ./a.out 
file descriptor = 3

ckt@ubuntu:~/work/unix/code$ ./a.out 
file descriptor = -1

3.2 输入和输出

输入和输出
  • 不带缓存I/O和标准I/O对比
    以不带缓存I/O的read函数为例,函数原型:ssize_t read(int fd, void *buf, size_t count); 在调用read时需要设置缓存区长度即count,如下图所示,不同大小的缓存区对读写操作时间也会有影响。读取相同大小的数据,设置的buffsize过小,read的调用次数就会增加,程序的系统CPU时间变长。
Unix环境高级编程 3.9 I/O效率

调用标准I/O函数无需设置buffsize,先将数据读入缓存流,在填满缓存区后才执行对磁盘的I/O磁盘。
标准I/O库提供缓存的目的就是为了尽可能的减少read的调用次数。


不带缓存I/O和标准I/O区别
  • printf是按行进行缓冲 - 示例程序
    printf是按行进行缓冲,只有当一行结束以后,才会将数据输出。
#include "apue.h"

int main(int argc, char const *argv[])
{
    printf("This Line has been cached...");
    sleep(3);
    printf("\nEnd by line break...\n");
    return 0;
}
  • 运行结果
    由于第一个printf没有加换行符,它的缓冲区没有被填满,所以不会马上显示到标准输出,只有当3秒的休眠结束都,执行到第2个printf的\n,缓冲区被填满了以后,才会将第一句printf输出。
示例代码运行效果

修改代码为:printf("This Line has been cached...\n"); 后运行,会立即输出第一句printf。

printf后面加上换行符运行效果

4. 程序及处理

4.1 进程和程序

程序和进程
  • 进程示例程序
    使用fork创建一个新进程,并打印出fork的返回值和当前的进程id。
#include "apue.h"

int main(int argc, char const *argv[])
{
    int values = -1;
    printf("current pid = %d\n", getpid());
    values = fork();
    printf("values return by fork : %d, current pid : %d\n", values, getpid()); 
    return 0;
}
  • 运行结果
    fork调用一次,会返回两次。返回值不为0,表示当前为父进程;返回值
    为0,表示当前调用是在子进程里面。
ckt@ubuntu:~/work/unix/code$ ./process_test
current pid = 2622
values return by fork : 2623, current pid : 2622
values return by fork : 0, current pid : 2623

4.2 出错处理

出错处理
  • 打印错误函数 - 示例代码
    使用strerror和perror进行错误打印。
#include "apue.h"
#include <errno.h>

int main(int argc, char const *argv[])
{
    printf("The default value of errno : %d\n", errno);
    fprintf(stderr, "%s\n", strerror(EACCES));
    errno = ENOENT;
    perror("printf the last value of errno ");
    return 0;
}
  • 运行结果
    程序出错后,错误值是存放在errno里面的。先打印出errno的默认值为0。先使用strerror(EACCES),将枚举值EACCES转为对应错误的字符串描述。再重新赋值errno = ENOENT,perror也会将error转化为对应错误的字符串。它们的区别就在于,perror可以传入一个字符串作为参数输出。
ckt@ubuntu:~/work/unix/code$ ./a.out 
The default value of errno : 0
Permission denied
printf the last value of errno : No such file or directory

4.3 信号

信号
  • 函数

sighandler_t signal(int signum, sighandler_t handler);

  • 信号示例代码
    当捕捉到ctrl+c信号时,就调用自定义函数MySigHanlder。SIGINT就代表我们要捕捉的信号是ctrl+c。
#include "apue.h"

void MySigHanlder();

int main(int argc, char const *argv[])
{
    char buf[MAXLINE];
    printf("please input...\n");
    if (signal(SIGINT, MySigHanlder) == SIG_ERR) 
    {

    }
    while (fgets(buf, MAXLINE,stdin) != NULL) 
    {
        printf("please input...\n");
    }
    return 0;
}

void MySigHanlder() 
{
    printf("\nsignal catch by MySigHanlder\n");
}
  • 捕捉ctrl+c - 运行效果
捕捉ctrl+c
  • 不捕捉ctrl+c - 运行效果
    如果将signal(SIGINT, MySigHanlder)注释掉,按照系统默认方式处理,直接停止程序。
不捕捉ctrl+c

5. Unix时间

Unix时间
  • 使用time命令查看ls命令执行的时间
ckt@ubuntu:~$ time ls
Android  AndroidStudioProjects  android-x86  ARM_Compiler_5   

real    0m0.006s
user    0m0.000s
sys 0m0.006s

FreeMind

按照书中讲解顺序,附上完整的思维导图。

Unix基础

参考

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

推荐阅读更多精彩内容