C语言缓冲区与重定向

目录

什么是缓冲区?

刷新策略 

模拟实现重定向

标准输出和标准错误有什么区别?


上文提到关闭1号文件(标准输出文件),根据文件描述符分配规则,再打开的文件的描述符就是1,看以下代码:

C语言缓冲区与重定向_第1张图片

运行代码后,前10s中file.txt文件被创建出来了,但是没有内容:

过了10秒钟,程序退出了,内容才被刷新到了磁盘文件上

 

 为什么数据没有被马上打印到文件上呢?

因为存在缓冲区,这里的缓冲区是C语言封装的缓冲区。

什么是缓冲区?

缓冲区的本质其实就是一段内存,用来暂时存放数据的地方。

为什么要有缓冲区呢?

联系生活,比如快递的出现,让货物的配送交给物流,这样就可以解放你的时间 ,而且集中配送货物可以提高总体的效率。对于缓冲区而言,缓冲区对于进程而言效果也是类似的,首先缓冲区解放使用缓冲区的进程的时间,比如进程往磁盘写入的时候,可以直接往缓冲区去写入,进程不用等待写入磁盘完成,可以继续执行后续的代码,其次缓冲区集中处理缓冲区中数据的刷新,减少了IO次数,因为外设上IO的效率是很低的(相比其他硬件),故而提高整机的效率。

缓冲区在哪里?

将上面的代码中的printf函数换成系统调用write:

发现,在sleep期间,进程已经把数据打印到了file.txt上,可见write内部并没有缓冲区,printf封装了write,这个缓冲区是C语言的缓冲区。

在C语言的文件描述结构体FILE当中封装了该FILE对应的语言级别的缓冲区,每次打开文件,每个FILE都有对应的fd和属于自己的语言级别缓冲区,写入时先写到缓冲区,当缓冲区需要刷新的时候,内部再调用write刷新到对应文件中,那么何时刷新呢?这取决于C语言的刷新策略。

刷新策略 

1.无缓冲:也就是立即刷新

进程退出时:当进程正常退出前,C语言内部就会一系列工作,包括调用fclose函数刷新缓冲区

用户强制刷新:用户调用fflush函数直接刷新缓冲区

2.行缓冲:逐行刷新,当输入的字符中有\n就直接刷新之前缓冲区中的内容,显示器文件就是这个刷新策略。

3.全缓冲:也即当缓冲区写满了才刷新,块设备对应的磁盘文件是采用全缓冲刷新策略的。

现在就可以解释原本的代码为什么不立即刷新到磁盘上了,首先1号文件标准输出被关闭了,而打开了磁盘文件,printf会往1号文件打印,也就是说标准输出被重定向到了磁盘文件,而标准输出的刷新策略本来是行刷新的,现在改变了刷新策略变成了全缓冲刷新策略,因为缓冲区没有被写满所以在sleep期间也没有看见数据输出到文件中,当进程退出或调用fflush时,缓冲区就被刷新到了磁盘文件上,于是进程退出后数据就才刷新到了文件当中。

模拟实现重定向

在上上篇博客中,实现了一个简易版的shell,现在根据现有知识再模拟实现一下重定向符号输出重定向>,追加重定向>>和输入重定向<添加入minishell吧。

进程程序替换_且随疾风前行->的博客-CSDN博客

实现方法:

先用一些宏标识几种重定向方法,用全局变量记录一下要重定向到的文件,和如何进行重定向,比如下面代码:

C语言缓冲区与重定向_第2张图片

 在命令分割之前检查一下命令中是否有重定向符号,并记录之。

然后再在创建子进程之后程序替换之前,将要重定向的文件打开,并用dup2函数重定向之即可。

代码如下:

#include
#include
#include
#include
#include
#include
#include
#include

#define NONE_REDIR -1
#define INPUT_REDIR 0
#define OUTPUT_REDIR 1
#define APPEND_REDIR 2
int g_redir_flag=NONE_REDIR;
char* g_redir_filename =NULL;

void CheckRdir(char*command)
{
    char*begin=command;
    while(*begin)
    {
        if(*begin=='>')
        {
            if(*(begin+1)=='>')
            {
                g_redir_flag=APPEND_REDIR;
                *begin='\0';
                begin+=2;
                while(*begin==' ')++begin;
                g_redir_filename=begin;
                break;
            }
            else
            {
                g_redir_flag=OUTPUT_REDIR;
                *begin='\0';
                begin+=1;
                while(*begin==' ')++begin;
                g_redir_filename=begin;
                break;
            }
        }
        else if(*begin=='<')
        {
            g_redir_flag=INPUT_REDIR;
            *begin='\0';
            begin+=1;
            while(*begin==' ')++begin;
            g_redir_filename=begin;
            break;
        }
        else ++begin;
    }
}
int main()
{
    //保持完整的字符串
    char inbuff[1024];
    //保持打散之后的字符串
    char *argv[32];
    char g_envar[64];//保存环境变量
    while(1)
    {
        printf("Please Enter#");
        fflush(stdout);
        memset(inbuff,'\0',sizeof(inbuff));
        if(fgets(inbuff,sizeof inbuff,stdin)==NULL)
            continue;
        inbuff[strlen(inbuff)-1]='\0';

        //检查重定向符号,并记录重定向方法
        CheckRdir(inbuff);

        argv[0]=strtok(inbuff," ");
        int index=1;
        if(strcmp(argv[0],"ls")==0)argv[index++]="--color=auto";
        if(strcmp(argv[0],"ll")==0)argv[0]="ls",argv[2]="-l",argv[1]="--color=auto",index=3;
        while(argv[index++]=strtok(NULL," "));//第二次调用解析原字符串时,传入NULL
        if(strcmp(argv[0],"export")==0&&argv[1]!=NULL)
        {
            strcpy(g_envar,argv[1]);
            if(putenv(g_envar)==0)
                printf("%s export success!\n",argv[1]);
            continue;
        }
        //内建命令
        if(strcmp(argv[0],"cd")==0 && argv[1]!=NULL)
        {
            chdir(argv[1]);
            printf("cd success!\n");
            continue;
        }
        pid_t id=fork();
        if(id==0)//child
        {
            if(g_redir_flag!=NONE_REDIR)
            {
                int fd;
                switch(g_redir_flag)
                {
                    case INPUT_REDIR:
                        fd=open(g_redir_filename,O_RDONLY,0666);
                        dup2(fd,0);
                        break;
                    case OUTPUT_REDIR:
                        fd=open(g_redir_filename,O_WRONLY|O_CREAT|O_TRUNC,0666);
                        dup2(fd,1);
                        break;
                    case APPEND_REDIR:
                        fd=open(g_redir_filename,O_WRONLY|O_CREAT|O_APPEND,0666);
                        dup2(fd,1);
                        break;
                    default:
                        break;
                }
            }
            execvp(argv[0],argv);
            exit(0);
        }
        int status=0;
        pid_t ret=waitpid(id,&status,0);
        if(ret>0)printf("exit code:%d\n",WEXITSTATUS(status));

    }
    return 0;
}

编译运行程序,我们的minshell就可以进行重定向了:

C语言缓冲区与重定向_第3张图片

 

在打印的时候标准输出和标准错误都往屏幕上打印,二者有区别吗?

标准输出和标准错误有什么区别?

看如下代码:

C语言缓冲区与重定向_第4张图片

 运行程序,标准输出和标准错误都输出到屏幕中:

但是我们使用重定向符号将程序运行的重定向输出输出到文件时,只重定向了标准输出到file.txt文件中,标准错误还是打印到屏幕中:

这是因为重定向默认是使用输出重定向,在命令行中可以显式写出要如何重定向,

比如:

./a.out 1>stdout.txt 2>stderr.txt 让文件输出到标准输出文件的重定向输出到stdout.txt,标准错误输出的重定向输出到stderr.txt 文件:

C语言缓冲区与重定向_第5张图片

还可以用 ./a.out>all.txt 2>&1表示让输出到标准输入和标准错误都重定向输出到all.txt文件中:

C语言缓冲区与重定向_第6张图片 

之所以有标准输出和标准错误,主要是用于区分哪些是程序的日常输出,哪些是程序运行中产生的错误。

你可能感兴趣的:(Linux,linux,运维,服务器)