Linux中系统编程和系统调用

系统编程概述

》 操作系统的职责
操作系统用来管理所有的资源,并将不同的设备和不同的程序关联起来。
》 什么是Linux系统编程
在有操作系统的环境下编程,并使用操作系统提供的系统调用及各种库,
对系统资源进行访问。学会了C语言再知道一些使用系统调用的方法,
就可以进行Linux系统编程了

系统调用概述

类UNIX系统的软件层次image
应用程序与系统调用image

》系统调用是操作系统提供给用户程序的一组“特殊”函数接口。
》 Linux的不同版本提供了两三百个系统调用。
》 用户程序可以通过这组接口获得操作系统(内核)提供的服务。
》 例如:
用户可以通过文件系统相关的系统调用,请求系统打开文件、关闭文件或读写文件。

image

》 系统调用按照功能逻辑大致可分为:
进程控制、进程间通信、文件系统控制、系统控制、
内存管理、网络管理、socket控制、用户管理。
》 系统调用的返回值:
通常,用一个负的返回值来表明错误,返回一个0值表明成功。
错误信息存放在全局变量errno中,用户可用perror函数打印出错信息。

系统调用遵循的规范
》 在Linux中,应用程序编程接口(API)遵循POSIX标准。
POSIX标准基于当时现有的UNIX 实践和经验,描述了操作系统的系统调用编程接口(实际上就是API),用于保证应用程序可以在源代码一级上在多种操作系统上移植运行。
如:
Linux下写的open、write 、read可以直接移植到unix操作系统下。

系统调用I/O函数

》 系统调用中操作I/O的函数,都是针对文件描述符的。
通过文件描述符可以直接对相应的文件进行操作。如:open、close、write 、read、ioctl
》 文件描述符
文件描述符是非负整数。打开现存文件或新建文件时,
系统(内核)会返回一个文件描述符。文件描述符用来指定已打开的文件。
》 #define STDIN_FILENO 0 //标准输入的文件描述符
#define STDOUT_FILENO 1 //标准输出的文件描述符
#define STDERR_FILENO 2 //标准错误的文件描述符
程序运行起来后这三个文件描述符是默认打开的。
*open函数:打开一个文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
当文件存在时使用:
int open(const char *pathname, int flags);
当文件不存在时使用:
int open(const char *pathname,
int flags, mode_t mode);
参数:
	pathname:文件的路径及文件名。
	flags:open函数的行为标志。
	mode:文件权限(可读、可写、可执行)的设置。
返回值:
	成功返回打开的文件描述符。
	失败返回-1,可以利用perror去查看原因。
flags的取值及其含义
取值 含义
O_RDONLY 以只读的方式打开
O_WRONLY 以只写的方式打开
O_RDWR 以可读、可写的方式打开
flags除了取上述值外,还可与下列值位或
O_CREAT 文件不存在则创建文件,使用此选项时需使用mode说明文件的权限
O_EXCL 如果同时指定了O_CREAT,且文件已经存在,则出错
O_TRUNC 如果文件存在,则清空文件内容
O_APPEND 写文件时,数据添加到文件末尾
O_NONBLOCK 当打开的文件是FIFO、字符文件、块文件时,此选项为非阻塞标志位
* mode 的取值及其含义
取值 八进制数 含义
S_IRWXU 00700 文件所有者的读、写、可执行权限
S_IRUSR 00400 文件所有者的读权限
S_IWUSR 00200 文件所有者的写权限
S_IXUSR 00100 文件所有者的可执行权限
S_IRWXG 00070 文件所有者同组用户的读、写、可执行权限
S_IRGRP 00040 文件所有者同组用户的读权限
S_IWGRP 00020 文件所有者同组用户的写权限
S_IXGRP 00010 文件所有者同组用户的可执行权限
S_IRWXO 00007 其他组用户的读、写、可执行权限
S_IROTH 00004 其他组用户的读权限
S_IWOTH 00002 其他组用户的写权限
S_IXOTH 00001 其他组用户的可执行权限
*close函数:关闭一个文件
#include <unistd.h>
int close(int fd);
参数:
	fd是调用open打开文件返回的文件描述符。
返回值:
	成功返回0。
	失败返回-1,可以利用perror去查看原因。
*write函数:把指定数目的数据写到文件
#include <unistd.h>
ssize_t write(int fd, const void *addr,size_t count);
参数:
	fd:文件描述符。
	addr:数据首地址。
	count:写入数据的字节个数。
返回值:
	成功返回实际写入数据的字节个数。
	失败返回-1,可以利用perror去查看原因。
*read函数:把指定数目的数据读到内存
#include <unistd.h>
ssize_t read(int fd, void *addr, size_t count);
参数:
	fd:文件描述符。
	addr:内存首地址。
	count:读取的字节个数。
返回值:
	成功返回实际读取到的字节个数。
	失败返回-1,可以利用perror去查看原因。
remove库函数:删除文件
#include <stdio.h>
int remove(const char *pathname);
参数:
	pathname :文件的路名+文件名。
返回值:
	成功返回0。
	失败返回-1,可以利用perror去查看原因。

系统调用与库

库函数由两类函数组成
》不需要调用系统调用
不需要切换到内核空间即可完成函数全部功能,并且将结果反馈给应用程序,如strcpy、bzero等字符串操作函数。
》需要调用系统调用
需要切换到内核空间,这类函数通过封装系统调用去实现相应功能,如printf、fread等。
库函数与系统调用的关系:
并不是所有的系统调用都被封装成了库函数,系统提供的很多功能都必须通过系统调用才能实现。image
系统调用是需要时间的,程序中频繁的使用系统调用会降低程序的运行效率。
》当运行内核代码时,CPU工作在内核态,在系统调用发生前需要保存用户态的栈和内存环境,然后转入内核态工作。
》系统调用结束后,又要切换回用户态。这种环境的切换会消耗掉许多时间。
库函数访问文件的时候根据需要,设置不同类型的缓冲区,从而减少了直接调用IO系统调用的次数,提高了访问效率。image
image
应用程序调用printf函数时,函数执行的过程:
image

posted @ 2021-08-30 10:50  早晨9点  阅读(165)  评论(0)    收藏  举报