系统编程笔记

一、系统编程概述

操作系统的职责:
	操作系统用来管理所有的资源,
	并将不同的设备和不同的程序关联起来。
什么是Linux系统编程:
	在有操作系统的环境下编程,
	并使用操作系统提供的系统调用及各种库,
	对系统资源进行访问。
系统编程主要就是为了让用户能够更好和更方便的操作硬件设备,
并且对硬件设备也起到保护作用,
我们所写的程序,
本质就是对硬件设备的操作,
所以操作系统提供接口可以对硬件进行操作,
这就是系统编程。

二、系统调用概述

	本质都是要对硬件设备进行操作,
但是linux操作系统在硬件之上设置了内核,
也就是只有内核才可以直接操作硬件设备,
如果想操作内核,
需要调用内核的系统调用,
如果要操作内核中的系统调用,
有三种方式:
第一种:
	shell,
	用户通过shell命令,
	有shell解释器操作内核的系统调用
第二种:
	库函数,
	用户通过应用层库函数的借口,
	比如fread对内核的系统调用进行操作
第三种:
	应用层系统调用,
	他可以直接对内核的系统调用进行操作


系统调用是操作系统提供给用户程序的一组“特殊”函数接口
Linux的不同版本提供了两三百个系统调用。
用户程序可以通过这组接口获得操作系统(内核)提供的服务。

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

三、系统调用I/O函数

3.1 文件描述符

	文件描述符是非负整数。
	打开现存文件或新建文件时,
	系统(内核)会返回一个文件描述符。
	文件描述符用来指定已打开的文件。

	在系统调用(文件IO)中,
	文件描述符对文件起到标识作用,
	如果要操作文件,
	就是对文件描述符的操作

当一个程序运行或者一个进程开启时,
系统会自动创建三个文件描述符
0 					标准输入
1 					标准输出
2					标准输出出错


系统中也可以使用一些宏
#define STDIN_FILENO			 0 			//标准输入的文件描述符
#define STDOUT_FILENO 			1		 //标准输出的文件描述符
#define STDERR_FILENO 			2 				//标准错误的文件描述符



文件IO的文件描述符和标准IO的文件指针的对应关系
文件IO 			标准IO
0 				stdin
1 				stdout
2				 stderr

如果自己打开文件,
会返回文件描述符,
而文件描述符一般按照从小到大依次创建的顺序

3.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:标志位
 O_RDONLY 只读
 O_WRONLY 只写
 O_RDWR 读写
 O_CREAT 文件不存在则创建,需要通过第三个参数设置文件权限
 O_EXCL 一般与O_CREAT一起使用,标识如果文件存在则报错
 O_TRUNC 如果文件存在则清空
 O_APPEND 如果文件存在则追加
 mode:文件权限,当指定flags中有O_CREAT时,必须有这个参数,
如果没有O_CREAT,则不需要这个参数
 返回值:
 成功:文件描述符
 失败:‐1

3.2.1 文件IO和标准IO权限对比

标准IO 		文件IO 									权限含义
r 			O_RDONLY 								以只读的方式打开文件,如果文件不存在则报错
r+ O_		RDWR									以读写的方式打开文件,如果文件不存在则报错
w			O_WRONLY | O_CREAT | O_TRUNC,0664		以只写的方式打开文件,如果文件不存在则创建,如果文件存在则清空
w+			O_RDWR | O_CREAT | O_TRUNC,0664		以读写的方式打开文件,如果文件不存在	则创建,如果文件存在则清空
a			O_WRONLY | O_CREAT |O_APPEND, 0664		以只写的方式打开文件,如果文件不存在则创建,如果文件存在则追加
a+			O_RDWR | O_CREAT | O_APPEND,0664		以读写的方式打开文件,如果文件不存在则创建,如果文件存在则追加

3.2.2 函数调用出错后输出错误信息

 通过全局变量 errno
 #include <errno.h>
 errno是一个全局变量,当函数调用失败后,可以通过errno获取错误码

 通过一个函数 perror
 #include <stdio.h>
 void perror(const char *s);
 功能:输出函数调用失败的错误信息
 参数:
 s:打印错误信息的提示消息
 返回值:无
用法
 #include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
int main(int argc, char const *argv[])
{
//使用open函数打开或者创建一个文件
 int fd;
 fd = open("file.txt", O_RDONLY);

 if(fd == -1)
 {
//通过全局变量errno打印错误码
 //注意需要添加头文件errno.h
 //printf("errno = %d\n", errno);

 //通过perror函数输出函数调用失败的错误信息
 perror("fail to open");
 return -1;
 }

 printf("fd = %d\n", fd);

 return 0;
 }

3.3 close函数

 #include <unistd.h>
 int close(int fd);
 功能:关闭一个文件描述符
参数:
fd:指定文件的文件描述符,open函数的返回值
 返回值:
 成功:0
 失败:‐1

3.3.1 案例

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
int main(int argc, char const *argv[])
{
int fd;
 fd = open("file.txt", O_RDONLY);

 if(fd == ‐1)
 {
 perror("fail to open");
 return ‐1;
 }

 printf("fd = %d\n", fd);

 //当不对文件进行任何操作时,就会关闭文件描述符
 //使用close函数关闭文件描述符
 //一旦关闭了文件描述符,就不能再通过原有的文件描述符对文件进行操作
 close(fd);

 return 0;
 }

 #include <stdio.h>
 #include <sys/types.h>
 #include <sys/stat.h>
 #include <fcntl.h>
 #include <unistd.h>

 int main(int argc, char const *argv[])
 {

 //测试1:一个进程(一个程序的运行)创建的文件描述符的个数
 //一个程序运行的时候最多可以创建2014个文件描述符,0~1023
 #if 0
 int fd;
 while (1)
 {
 fd = open("file.txt", O_RDONLY | O_CREAT, 0664);

 if(fd == ‐1)
 {
 perror("fail to open");
 return ‐1;
 }

 printf("fd = %d\n", fd);
 }
 #endif

 //测试2:文件描述符值的规律
 //文件描述符按照从小到大的顺序依次创建
 //如果中途有文件描述符被关闭了,则再创建的
 //文件描述符会先补齐之前的,然后依次递增创建
 //注意:不要认为最后创建的文件描述符一定是最大的
 #if 1
 int fd1, fd2, fd3, fd4;
 fd1 = open("test.txt", O_RDONLY | O_CREAT, 0664);
 fd2 = open("test.txt", O_RDONLY | O_CREAT, 0664);
 fd3 = open("test.txt", O_RDONLY | O_CREAT, 0664);
 fd4 = open("test.txt", O_RDONLY | O_CREAT, 0664);

 printf("fd1 = %d\n", fd1);
 printf("fd2 = %d\n", fd2);
 printf("fd3 = %d\n", fd3);
 printf("fd4 = %d\n", fd4);

 close(fd2);

 int fd5, fd6;
 fd5 = open("test.txt", O_RDONLY | O_CREAT, 0664);
 fd6 = open("test.txt", O_RDONLY | O_CREAT, 0664);
 printf("fd5 = %d\n", fd5);
 printf("fd6 = %d\n", fd6);

 #endif
 return 0;
 }

3.4 write函数

#include <unistd.h>
ssize_t write(int fd, const void *buf, size_t count);
功能:向文件写入数据
参数:
fd:指定的文件描述符
buf:要写入的数据
count:要写入的数据的长度
返回值:
成功:实际写入的字节数
 失败:‐1

3.4.1 向终端写入数据

终端stdin的文件描述符为1
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
int main(int argc, char const *argv[])
{
//向终端写入数据
 //对1这个文件描述符进行操作
 if(write(1, "hello world\n", 12) == -1)
 {
 perror("fail to write");
 return -1;
 }

 //向文件中写入数据

 return 0;
 }

3.4.2 向文件写入数据

 #include <stdio.h>
 #include <sys/types.h>
 #include <sys/stat.h>
 #include <fcntl.h>
 #include <unistd.h>

 int main(int argc, char const *argv[])
 {
 //向文件写入数据
int fd;
//以只写的方式打开文件,如果文件不存在则创建,如果文件存在则清空
fd = open("file.txt", O_WRONLY | O_CREAT | O_TRUNC, 0664);
if(fd == -1)
{
perror("fail to open");
return -1;
}
//使用write函数向文件写入数据
ssize_t bytes;
if((bytes = write(fd, "hello world\n", 12)) == -1)
{
perror("fail to write");
return -1;
}
printf("bytes = %ld\n", bytes);
write(fd, "nihao beijing", 5);
//关闭文件描述符
close(fd);
return 0;
}


3.5 read函数

 #include <unistd.h>
 ssize_t read(int fd, void *buf, size_t count);
 功能:从文件中读取数据
 参数:
 fd:指定的文件描述符
 buf:保存读取到的数据
 count:最大一次读取多少个字节
 返回值:
 成功:实际读取的字节数
 失败:‐1
 注意:如果读取到文件末尾,返回0

3.5.1 从终端中读取数据

终端输出stdout ,文件描述符为0
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
int main(int argc, char const *argv[])
{
//使用read函数从终端读取数据
 //使用0文件描述符从终端读取数据

 //如果终端输入的字节数大于第三个参数
 //则只会读取第三个参数对应的字节数,返回值也是与第三个参数一致

 //如果终端输入的字节数小于第三个参数‘
 //则只会读取输入的数据加上换行符,返回值就是实际输入的数据+1

 ssize_t bytes;
 char str[32] = "";
 if((bytes = read(0, str, 6)) == -1)
 {
 perror("fail to read");
 return -1;
 }

 printf("str = [%s]\n", str);
 printf("bytes = %ld\n", bytes);

 return 0;
 }

3.5.2 从文件中读取数据

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#define N 64
int main(int argc, char const *argv[])
 {
 //使用read从文件中读取数据
 int fd;
 if((fd = open("test.txt", O_RDONLY | O_CREAT, 0664)) == -1)
 {
 perror("fail to open");
 return -1;
 }

 //读取文件内容
 char buf[N] = "";
 ssize_t bytes;
 #if 1
 if((bytes = read(fd, buf, 32)) == -1)
 {
 perror("fail to read");
 return -1;
 }

 printf("buf = [%s]\n", buf);
 printf("bytes = %ld\n", bytes);

 char buf1[N] = "";
 bytes = read(fd, buf1, 32);
 printf("buf1 = [%s]\n", buf1);
 printf("bytes = %ld\n", bytes);

 //如果文件中的数据都读取完毕,则read会返回0
 char buf2[N] = "";
 bytes = read(fd, buf2, 32);
 printf("buf2 = [%s]\n", buf2);
 printf("bytes = %ld\n", bytes);
 #endif

 #if 0
 //读取文件中的所有内容
 while((bytes = read(fd, buf, 32)) != 0)
 {
 printf("buf = [%s]\n", buf);
 printf("bytes = %ld\n", bytes);
 }
 #endif
 //关闭文件描述符
 close(fd);

 return 0;
 }

3.6 remove函数

1 #include <stdio.h>
2 int remove(const char *pathname);
3 功能:删除指定文件
4 参数:
5 pathname:包含路径的文件名
6 返回值:
7 成功返回0。
8 失败返回‐1
#include <stdio.h>
int main(int argc, char const *argv[])
{
//使用remove函数删除文件
if(remove("./file.txt") == -1)
{
perror("fail to remove");
return -1;
 }

 printf("delete done\n");

 return 0;
 }

四、系统调用和库函数

库函数由两类函数组成
(1)不需要调用系统调用
不需要切换到内核空间即可完成函数全部功能,
并且将结果反馈给应用程序,
如strcpy、bzero等字符串操作函数。
(2)需要调用系统调用
需要切换到内核空间,
这类函数通过封装系统调用去实现相应功能,
如printf、fread等

库函数与系统调用的关系
系统提供的很多功能都必须通过系统调用才能实现。

系统调用是需要时间的,
程序中频繁的使用系统调用会降低程序的运行效率,
当运行内核代码时,
CPU工作在内核态,
在系统调用发生前需要保存用户态的栈和内存环境,
然后转入内核态工作。
系统调用结束后
又要切换回用户态。
这种环境的切换会消耗掉许多时间
库函数访问文件的时候根据需要,
设置不同类型的缓冲区,
从而减少了直接调用IO系统调用的次数,
提高了访问效率。
总结:
	大多数库函数的本质也是系统调用,
	只不过库函数有了缓冲区,
	用于减少系统调用的次数
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值