文章目录
一、系统编程概述
在有操作系统的环境下编程,并使用操作系统提供的系统调用及各种库,对系统资源进行访问。系统编程主要就是为了让用户能够更好和更方便的操作硬件设备,并且对硬件设备也起到保护作用。我们所写的程序,本质就是对硬件设备的操作,所以操作系统提供接口可以对硬件进行操作,这就是系统编程。
系统调用
本质都是要对硬件设备进行操作,但是linux操作系统在硬件之上设置了内核,也就是只有内核才可以直接操作硬件设备。如果想操作内核,需要调用内核的系统调用。
内核中的系统调用有三种方式:
第一种:shell,用户通过shell命令,有shell解释器操作内核的系统调用。
第二种:库函数,用户通过应用层库函数的借口,比如fread对内核的系统调用进行操作。
第三种:应用层系统调用,他可以直接对内核的系统调用进行操作。
系统调用是操作系统提供给用户程序的一组“特殊”函数接口 Linux的不同版本提供了两三百个系统调用。 用户程序可以通过这组接口获得操作系统(内核)提供的服务。
系统调用按照功能逻辑大致可分为:
进程控制、进程间通信、文件系统控制、系统控制、内存管理、网络管理、socket控 制、用户管理。
系统调用的返回值:
通常,用一个负的返回值来表明错误。
返回一个0值表明成功。
错误信息存放在全局变量errno中,用户可用perror函数打印出错信息。
二、系统调用I/O函数
文件描述符
文件描述符是非负整数。打开现存文件或新建文件时,系统(内核)会返回一个文件描述 符。文件描述符用来指定已打开的文件。
在系统调用(文件IO)中,文件描述符对文件起到标识作用,如果要操作文件,就是对文件 描述符的操作。
当一个程序运行或者一个进程开启时,系统会自动创建三个文件描述符 0 标准输入 1 标准输出 2 标准输出出错。
文件IO的文件描述符和标准IO的文件指针的对应关系:
文件IO | 标准IO |
---|---|
0 | stdin |
1 | stdout |
2 | stderr |
打开文件open函数
返回值:
成功:文件描述符
失败:‐1
flags | 标志位 |
---|---|
O_RDONLY | 只读 |
O_WRONLY | 只写 |
O_RDWR | 读写 |
O_CREAT | 文件不存在则创建,需要通过第三个参数设置文件权限 |
O_EXCL | 一般与O_CREAT一起使用,标识如果文件存在则报错 |
O_TRUNC | 如果文件存在则清空 |
O_APPEND | 如果文件存在则追加 |
mode:文件权限,当指定flags中有O_CREAT时,必须有这个参数,如果没有O_CREAT,则不需要这个参数。
文件权限 | 字符 |
---|---|
可读 | r-4 |
可写 | w-2 |
可执行 | x-1 |
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<stdio.h>
#include <unistd.h>
#include <string.h>
#include<stdlib.h>
//判断文件打开成功 返回值-1
int main()
{
int fd;
char *buf="qian ru shi";
fd=open("./text",O_RDWR);
if(fd==-1)//返回-1打开失败
{
printf("open text failed\n");
fd=open("./text",O_RDWR|O_CREAT,0600);//然后以读写的方式创建,权限为6=4+2
if(fd>0)
{
printf("create text success\n");
}
}
printf("open susceess:=%d\n",fd);
return 0;
}
文件写入write函数
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<stdio.h>
#include <unistd.h>
#include <string.h>
#include<stdlib.h>
//O_APPEND 将文件里面原有内容加载写入内容的前面
int main()
{
int fd;
char *buf="wenjian daxiao";
//fd=open("./text",O_RDWR|O_APPEND); 不覆盖原来内容
fd=open("./text",O_RDWR|O_TRUNC); //覆盖文档原来内容写入
printf("file:%d\n",fd);
int n_write=write(fd,buf,strlen(buf)); //返回的是内容的字节数
if(n_write !=-1)
{
printf("write %d byte to text\n",n_write);
}
close(fd);
return 0;
}
读取文件read函数
因为写入操作后光标在文件末尾直接读取没有内容所以需要将光标移动到头。
移动光标函数需要的头文件
#include <sys/types.h>
#include <unistd.h>
off_t lseek(int fd, off_t offset, int whence);
1.文件描述符 2.偏移量 3.固定点 由尾向前偏移的话offset为负
可以用来计算文件大小
int whence | .固定点 |
---|---|
SEEK_SET | 指向头 |
SEEK_CUR | 当前位置 |
SEEK_END | 尾 |
关闭文件close函数
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<stdio.h>
#include <unistd.h>
#include <string.h>
#include<stdlib.h>
int main()
{
int fd;
char *buf="qian ru shi"; //./text代表当前文件夹下的text
fd=open("./text",O_RDWR); //int open(const char *pathname, int flags);第一个入口参数为指针字符串就是指针
/*if(fd==-1) //返回值返回-1表示打开失败
{
printf("open text failed\n"); //r=4 w=2 x=1
fd=open("./text",O_RDWR|O_CREAT,0600); //O_CREAT若不存在则创建该文件./text,添加O_CREAT必须给定第三个参数即6=4+2表示可读可写
if(fd>0)
{
printf("create text success\n");
}
}*/
printf("open susceess:=%d\n",fd);
//ssize_t write(int fd, const void *buf, size_t count); 1.路径 2.内容 3.大小 功能:写入
//write(fd,buf,sizeof(buf)); linux下一个字符指针8个字节 因此sizeof(buf)只会显示qian ru
int n_write=write(fd,buf,strlen(buf)); //返回的是内容的字节数
if(n_write !=-1)
{
printf("write %d byte to text\n",n_write);
}
//ssize_t read(int fd, void *buf, size_t count);
/*
close(fd); 关闭以免光标在最后读取读不出来
fd=open("./text",O_RDWR); 重新打开 光标在起始位
*/
//用移动光标的方法
//lseek(fd,0,SEEK_SET); 从头偏移
//lseek(fd,-11,SEEK_END); 从尾向前偏移用负号
lseek(fd,-11,SEEK_CUR); //因为写入操作后光标在文件末尾直接读取没有内容所以需要将光标移动到头
char *readBuf;
readBuf=(char *)malloc(sizeof(char)*n_write+1); //申请地址写入的字符数加上一个结束符\0
int n_read=read(fd,readBuf,n_write);
printf("read %d,context:%s\n",n_read,readBuf);
close(fd);
return 0;
}