匿名管道和命名管道

news2024/11/29 20:33:47

目录

管道

pipe创建一个管道

让子进程写入,父进程读取

如何把消息发送/写入给父进程

父进程该怎么读取呢

管道本质

结论:管道的特征:

测试管道大小

写端退了,测试结果

测试子进程一直写,父进程读一会就退出

管道到的应用场景

写一个进程池(pipe_use)

思路步骤

管道创建

创建父子进程

父进程写,子进程读

子进程pid有了管道也有了,就差在父进程添加字段了

先更改一下,在class里构造一下

添加字段

把初始化改造成函数

debug测试函数,纯输入函数

第二步开始控制进程了(想让子进程做什么)

改变一下,直接从键盘(0号描述符)里读,不从管道(3)里读了,就没有管道的概念了,slaver就不用传参了,父进程通过管道写,子进程通过标准输入读

现在开始选择任务和进程

也可以轮询选择,定义一个计数器,++弄,再%等

清理收尾

bug的地方:

​编辑

以上是匿名管道 

总文件总代码

makefile中代码

Task.hpp中代码

ProcessPool.cc中代码

命名管道

mkfifo命名管道

创建一个共享文件

这里先用server控制管道文件

谁控制的先运行运行谁就好了

命名管道的删除​编辑

思路

用到了open

服务端读取数据

打开顺序一定

无法输入空格问题(getline)

会发现一个问题,客户端退出了,服务端还没退出

sever端

写成进程池的样子

日志

用到了可变参数(形参实例化是从右到左)

开始写日志,level日志等级

先定义时间,time,时间戳

ctime

vsnprint

测试,先./server,然后会阻塞,然后./client,就会打印出,logmessage里的信息

为啥./client之前不打印

因为等待写入方式打开后,自己才会打开文件,向后执行,open阻塞了!​编辑​编辑

往文件里打印(上面是往屏幕打印)

分类

日志放入一个文件测试结果:​编辑

​编辑

日志分类测试结果:

log.hpp里头文件

优化一下调用

​编辑

client.cc

common.hpp

log.hpp

makefile

server.cc


匿名管道

https://blog.csdn.net/2401_83427936/article/details/142603367

首先自己要用用户层缓冲区,还得把用户层缓冲区拷贝到管道里,(从键盘里输入数据到用户层缓冲区里面),然后用户层缓冲区通过系统调用(write)写到管道里,然后再通过read系统调用,被对方(读端)读取,就要从管道拷贝到读端,然后再显示到显示器上。

pipe创建一个管道

pipe的介绍

1完成这件事:

看图分析

运行结果

#include<iostream>
#include<unistd.h>
using namespace std;
int main()
{
    //创建管道
    //先创建一个pipefd数组
    int pipefd[2];
    //用n接受一下,判断是否成功
    int n = pipe(pipefd);
    if(n<0) return 1;//创建失败了

    //创建成功
    //测试一下文件描述符是3和4
    cout<<"pipefd[0]:"<<pipefd[0]<<"pipefd[1]:"<<pipefd[1]<<endl;
    

    return 0;
}

2完成这件事:

创建一个子进程

 pid_t id = fork();
    if(id < 0)return 2;//创建失败
    if(id == 0)//创建成功
    {
        //子进程

    }
    //父进程

让子进程写入,父进程读取

要想让子进程进程写,就需要在进程中关闭读端

if(id == 0)//创建成功
{
     //子进程
     close(pipefd[0]);
}

同理

//父进程
close(pipefd[1]);

都用完结束后,可以都关掉

    if(id == 0)//创建成功
    {
        //子进程
        close(pipefd[0]);
        //.....
        close(pipefd[1]);
    }
    //父进程
    close(pipefd[1]);
    //.....
    close(pipefd[0]);

IPC code,写通信代码

3这件事也完成了:

结构就有了

然后在pipefd[1]这个管道里写,定义一个Writer函数

    if(id == 0)//创建成功
    {
        //子进程
        close(pipefd[0]);
        //.....IPC code,写通信代码
        //在pipefd[1]这个管道里写
        Writer(pipefd[1]);


        close(pipefd[1]);

        exit(0);//正常退出
    }

同理父进程的        

    //父进程
    close(pipefd[1]);
    //.....IPC code,写通信代码
    //在pipefd[0]这个管道里写
    Reader(pipefd[0]);


    close(pipefd[0]);

//子进程
void Writer(int wfd)
{

}
//父进程
void Reader(int rfd)
{

}

Writer

//子进程
void Writer(int wfd)
{
    string s = "hello,I am child";
    pid_t self = getpid();
    int number = 0;

    char buffer[10];
    while(true)
    {
        buffer[0] = 0;//字符串清空,只是为了提醒阅读代码的人,我把这个数组当字符串了

    }
}

用到snprintf
介绍

将s和self和number放进buffer

  char buffer[100];
    while(true)
    {
        buffer[0] = 0;//字符串清空,只是为了提醒阅读代码的人,我把这个数组当字符串了
        snprintf(buffer,sizeof(buffer),"%s    pid:%d\n",s.c_str(),self);
        cout<< buffer <<endl;
        sleep(1);
    };

用cout打印测试一下,打印成功说明写入buffer成功了

等待进程少不了,子进程exit后需要回收

 //父进程
    close(pipefd[1]);
    //.....IPC code,写通信代码
    //在pipefd[0]这个管道里写
    Reader(pipefd[0]);

    //等待进程缺少不了
    pid_t rid = waitpid(id,nullptr,0);
    if(rid < 0) return 3;//等待失败了

    close(pipefd[0]);

如何把消息发送/写入给父进程

用到了write

用write写入管道(管道也是文件),用strlen,不用+1,不用管\0,因为C语言规定\0结尾,和文件没有关系,wfd写入管道

//子进程
void Writer(int wfd)
{
    string s = "hello,I am child";
    pid_t self = getpid();
    int number = 0;
    char buffer[100];
    while(true)
    {
        buffer[0] = 0;//字符串清空,只是为了提醒阅读代码的人,我把这个数组当字符串了
        snprintf(buffer,sizeof(buffer),"%s    pid:%d  %d\n",s.c_str(),self,number++);
        //用write写入管道(管道也是文件),用strlen,不用+1,不用管\0,因为C语言规定\0结尾,和文件没有关系,wfd写入管道
        write(wfd,buffer,strlen(buffer));
        //cout<< buffer <<endl;
        sleep(1);
    };
}

父进程该怎么读取呢

用到了read,fd是文件描述符,从特定的文件描述符里读取,放在这个buf里,buf的长度是count

这里就需要考虑到\0,因为buffer中需要\0

//父进程
void Reader(int rfd)
{
    char buffer[100];
    while(true)
    {
        buffer[0] = 0;
                                    //用sizeof是为了留个空间放\0
        ssize_t n = read(rfd, buffer, sizeof(buffer));//sizeof!=strlen
        if(n > 0)
        {
            //添加\0,因为要放在buffer数组中读取
            buffer[n]=0;
            cout << "father get a message[" << getpid() <<"]"<< buffer <<endl;
        }
    }
}

运行结果

也会发现:为什么子进程sleep,父进程不sleep,父进程还是会跟着子进程sleep,因为父子进程是要协同的

管道本质

通信是为了更好的发送变化的数据,管道本质上是文件

所以必须要用到系统调用接口来访问管道,其是由系统管理,read和write

,操作系统相当于中介 

结论:管道的特征:

1:具有血缘关系的进程进行进程间通信

2:管道只能单向通信

3:父子进程是会进程协同的,同步与互斥的--保护管道文件的数据安全

4:管道是面向字节流的

5:管道是基于文件的,而文件的生命周期是随进程的

再测试,把子进程sleep去掉,就是让子进程写快一点,父进程sleep几秒,就是让父进程读慢一点,看有什么现象

 管道的四种情况

测试管道大小

把c一直往管道里写,把父进程中休眠50秒

结果差不多64kb

写端退了,测试结果

结果是:

读端正常读,写端关闭,读端就会读到0,表明读到了文件(pipe)结尾,不会被阻塞

read读取成功会返回读到的字符个数,读到结尾返回0

读到结尾父进程也就可以停止读取了,break后去把僵尸的子进程回收

break到这里

最后子进程会被waitpid回收

测试子进程一直写,父进程读一会就退出

定义一个cnt控制退出的时间

这里也要修改一下,加个sleep(5),观察,close提前关闭

结果:通过13号信号杀死 

管道到的应用场景

都会变成一个进程

写一个进程池(pipe_use)

首先创建好文件

创建5个进程

channel通道的意思

cmdfd文件描述符

slaverid代表哪个子进程

把它放进vector容器里

思路步骤

管道创建

void(n),假装使用一下,要不然编译不过

创建父子进程

父进程写,子进程读

子进程要读取,就要关闭自己的写端,父进程同理

子进程中的任务

子进程pid有了管道也有了,就差在父进程添加字段了

先更改一下,在class里构造一下

添加字段

测试一下:结果:文件描述符0,1,2是默认打开,3是从管道里读,4是写入管道

把初始化改造成函数

debug测试函数,纯输入函数

第二步开始控制进程了(想让子进程做什么)

这里打印的rfd都是3,正常吗,文件描述符是可以被子进程继承的

父进程对应的写端拿到的是4-8,子进程拿到的读端fd是3

改变一下,直接从键盘(0号描述符)里读,不从管道(3)里读了,就没有管道的概念了,slaver就不用传参了,父进程通过管道写,子进程通过标准输入读

用到了dup2,将从pipefd[0]中读变成从0开始读

想让父进程固定的向管道里写入指定大小字节的内容,必须读取四个字节,四个字节四个字节的写和读,这里的管道64kb

必须读取四个字节

如果父进程不给子进程发送数据呢?阻塞等待!

开始控制子进程

生成一个随机数种子

可以随机选择任务和选择进程

cmd是任务码,测试一下,父进程控制子进程,父进程发送给子进程(通过cmdcode连续)

在Task.hpp里

要用到函数指针

main中的任务了就属于

再把任务装载进来

输出型参数用*

现在开始选择任务和进程

再把main中的任务弄成全局的

进行判断一下

测试 ,comcode和任创建的任务一致

这里的write是父进程进行写入,向子进程发送,子进程不得闲,先写到管道里,等得闲了再读

也可以轮询选择,定义一个计数器,++弄,再%等

整理一下控制代码,这里是输入型参数,只需要读

这样就可以轮询方式选择进程了,不用随机了

结果

清理收尾

思路:把所有文件的描述符都关掉

等待方式设置为0 

read返回0,就是失败了,然后slaver就会调完

结束完就会exit直接退出

打印下更好显示

关闭文件描述符后sleep(10)秒,

然后这10个子进程一瞬间都应该break,然后最后到exit直接就退了,10秒结束后,父进程再回收他       

测试时不弄死循环,用cnt,5秒后自动结束控制,正常退出流程

测试结果

手动控制一下

定义一个select,输入0就是退出了,判断完后,就走到了选择任务


然后直接把cmdcode改为选择的select,-1是因为是从下标0开始的,输入1就是0下标的

测试

bug的地方:

这样会有一些bug(一个子进程不是只有一个写端(每一次子进程的创建都是有继承))

 这样会有一些bug(一个子进程不是只有一个写端(每一次子进程的创建都是有继承))

按理说这样是对的,可是这样就错了

因为下面两个红线还没有关掉,它们进程了最开始的w

这样倒着回收是可以的

正确改法

修改一下

最后一个push_back的就都是父进程的写入fd,

然后加一句这个红线的,每创建子进程后都先把上一次父进程的读端fd关掉就可以了,这里很妙,因为vector一开始是空的

方便看

这里这样就可以了        

管道已经完成

以上是匿名管道 

总文件总代码

makefile中代码

ProcessPool:ProcessPool.cc
	g++ -o $@ $^ -std=c++11
.PHNOY:clean
clean:
	rm -f ProcessPool

Task.hpp中代码

#pragma once

#include<iostream>
#include<vector>

using namespace std;

typedef void (*task_t)();

void task1()
{
    cout<< "lol 刷新日志" <<endl;
}

void task2()
{
    cout<< "lol 更新野区" <<endl;
}

void task3()
{
    cout<< "lol 检测软件更新" <<endl;
}

void task4()
{
    cout<< "lol 释放技能" <<endl;
}

ProcessPool.cc中代码

注意:

这里为啥是取地址一个comcode,不是一个0吗,要发送的数据就是这个,所以要把地址传给函数,可以是个数组

换成数组的话,read这也接收数据的时候,就得用数组去接受,要是写入超过int大小的话,就可能会出错,这个就是通信的双方要遵守的约定,这个判断一下,就是派发的这个任务是不是合法的,假设你的tasks任务中,只有4个任务,所以任务编号就是0 ~ 3,如果你接受到的任务编号是10或者-20,那么这些就是非法的,你执行的话,程序就会崩溃,所以要做一个简单的判断。

write以后,cmdcode的值也会跟着传到read对吧,write就是为了把cmdcode的值传递给给另外一个进程,以前见到的都是用的char buffer[];,这样&cmdcode能更方便的传值过去是不,看要传的是什么数据,只是传递一个int数据的话,就这样传递,如果是文本数据,或者是其他的话,可能就需要数组了,具体问题,具体讨论

#include "Task.hpp"
#include<iostream>
#include<string>
#include<vector>
#include<unistd.h>
#include<assert.h>
#include <sys/types.h>
#include <sys/wait.h>
using namespace std;

//打算创建5个进程
const int processnum = 5;
//全局任务
vector<task_t> tasks;


//先描述
class channel//管道
{ 
public:
    channel(int cmdfd,pid_t slaverid,string& processname)
    :_cmdfd(cmdfd)
    ,_slaverid(slaverid)
    ,_processname(processname)
    {}

public:
    int _cmdfd;//文件描述符
    pid_t _slaverid;//代表哪个子进程
    string _processname;//子进程的名字,方便打印日志
};

//子进程中读的任务
// void slaver(int rfd)
// {
//     while(true)
//     {
//         cout<< getpid() <<" - "<< "read fd is->"<<rfd<<endl;
//         sleep(1000);
//     }
// }
//改变一下从fd为0的地方开始读
void slaver()
{
    //read(0);
    while(true)
    {
        int cmdcode = 0;
        int n = read(0, &cmdcode, sizeof(int));
        if(n == sizeof(int))
        {
            //执行cmdcode对应的任务列表
            cout<< "slaver say@ get a command:" << getpid() << ":cmdcode:" << cmdcode <<endl;

            //判断一下并执行
            if(cmdcode >= 0 && cmdcode < tasks.size())  tasks[cmdcode]();
        }
        if(n == 0) break;
    }
}

//初始化
void Init(vector<channel>& channels)
{
    for(int i =0;i < processnum;i++)
    {
        int pipefd[2];
        int n = pipe(pipefd);//创建管道
        //返回值小于0就创建失败了
        assert(!n);
        (void)n;

        pid_t id = fork();
        if(id == 0)
        {
            //子进程:读
            close(pipefd[1]);

            //改变一下从fd为0的地方读
            dup2(pipefd[0],0);
            close(pipefd[0]);
            //任务
            slaver();
            cout<< "process: " << getpid() << "quit" <<endl;
            //slaver(pipefd[0]);

            exit(0);
        }
        //父进程:写
        close(pipefd[0]);
        //channel添加字段
        string name = "processs-" + to_string(i);
        //插入的是自定义类型,要构造一下,第一个传的是文件描述符,要写入的fd
        channels.push_back(channel(pipefd[1], id, name));
    }
}
//测试函数,纯输入函数
//输入:const &
//输出:*
//输入输出:&
void debug(const vector<channel>& channels)
{
    for(auto&e : channels)
    {
        cout<< e._cmdfd <<"  "<<e._slaverid<<"   "<<e._processname<<endl;
    }

}

void Loadtask(vector<task_t> *tasks)
{   
    tasks->push_back(task1);
    tasks->push_back(task2);
    tasks->push_back(task3);
    tasks->push_back(task4);
}
void memu()
{
    cout<< "########################" <<endl;
    cout<< "1:lol 刷新日志     2:lol 更新野区" <<endl;
    cout<< "1:lol 检测软件更新   4:lol 释放技能" <<endl;
    cout<< "           0:退出             " <<endl;
    cout<< "########################" <<endl;
}
//2:开始控制子进程
void ctrlSlaver(vector<channel> &channels)
{
    int which = 0;
    int cnt = 5;
    while(true)
    {
        int select = 0;
        memu();
        cout<< "Please Enter@:";
        cin>> select;
        if(select == 0) break;
        //1:选择任务
        //int cmdcode = rand()%tasks.size();
        int cmdcode = select - 1;

        //2:随机选择进程
        //int processpos = rand()%channels.size();
        
        //2:轮询选择进程
        cout<< "father say:"<< "cmdcode:" << cmdcode << "   already sendto  " <<channels[which]._slaverid << "process name   " 
            <<channels[which]._processname << endl;

        //3:发送任务
        write(channels[which]._cmdfd, &cmdcode, sizeof(cmdcode));
        which++;
        which%=channels.size();//保证不大于其长度
        cnt--;
        if(cnt == 0) break;
        sleep(1);
    }
}
void QuitProcess(const vector<channel> &channels)
{
    for(const auto& e : channels) close(e._cmdfd);
    sleep(10);
    for(const auto& e : channels) waitpid(e._slaverid, nullptr, 0);//进程的pid=_slaverid,关上了以后记得回收
}

int main()
{
    Loadtask(&tasks);

    //srand(time(nullptr)^getpid()^1023);//种一个随机数种子
    //在组织
    vector<channel> channels;
    //1:初始化
    Init(channels);

    debug(channels);

    //2:开始控制子进程
    ctrlSlaver(channels);

    //3:清理收尾
    QuitProcess(channels);

    return 0;
}


 管道是

毫不相关的进程进程间通信::命名管道

命名管道

https://blog.csdn.net/2401_83427936/article/details/142603367​​​​​​​

首先自己要用用户层缓冲区,还得把用户层缓冲区拷贝到管道里,(从键盘里输入数据到用户层缓冲区里面),然后用户层缓冲区通过系统调用(write)写到管道里,然后再通过read系统调用,被对方(读端)读取,就要从管道拷贝到读端,然后再显示到显示器上。

mkfifo命名管道

1号手册是指令,。2号系统调用接口

创建一个管道,p开头就是命名管道,并不会直接刷新到磁盘中,实际是个符号

这样会阻塞

这样会显示出来(先输入左边的,再输入右边的就会显示),左右两边是两个进程

>>追加写入的方式,但空间一直是0

所以这就是文件里大小一直是0的原因         

你怎么知道打开的是同一个文件

正好符合前提

所以要创建两个可执行程序,各自跑各自的,创建一个common是为了方便使用头文件        

client是客户   server是服务者

makefile中一下运行两个程序

mkfifo,用程序的方式创建管道,第一个参数是要创建的这个管道在那个路径下叫什么名字,也就是要保持唯一性的那些点,第二个是创建一个管道

这里是3号手册是函数。

返回 -1创建失败 

创建一个共享文件

./myfifo

server.cc和client.cc想看到同一个文件,包含上头文件就可以了

这里先用server控制管道文件

创建管道失败了设置为1 ,如果失败了就exit(1)

谁控制的先运行运行谁就好了

make一下生成两个可执行程序,因为是server控制的,所以要先运行server

运行后就会多一个myfifo命名管道

命名管道的删除

想删除这个myfifo用unlink(成功返回0 ,失败返回-1) 

命令行删除

代码也可以删(成功返回0 ,失败返回-1),头文件是unistd.h

创建完文件,5秒后就删除了

思路

用到了open

打开管道文件,第二个参数是只进行读取

enum中

fd<0打开失败了

服务端读取数据

客户端,只要用就行

 第二个参数就是打开文件为了写入的

用户输入完成以后,就要发送输入的消息到另一端

打开顺序一定

然后打开的顺序就一定了,先打开server,然后再打开另一个cc

先打开服务端,会阻塞在这里,然后再打开客户端,进行输入

右边输入啥,左边就会有啥

无法输入空格问题(getline)

但有一个问题就是cin没法输入空格,,要用到getline


会发现一个问题,客户端退出了,服务端还没退出

客户端退出,会read到0,所以服务端(读端)也要退出

 改正

sever端

等待写入方式打开后,自己才会打开文件,向后执行,open阻塞了!

优化一下

写成进程池的样子

日志

创建一个新文件

用到了可变参数(形参实例化是从右到左)

可变参数必须右至少一个具体的参数

举个例子:步骤:s指向可变部分

这里的sum第一个参数是几个数求和的意思,传不同的类型不可以的,因为上面va_arg里已经写死了

开始写日志,level日志等级

先定义时间,time,时间戳

ctime

头文件

tm_year
tm_min
tm_huor小时

打印具体年月日 

年是从1900年开始的

 年月日时分秒

vsnprint

vsnprint,跟不带v的区别就是,去除了...换成了可变参数部分

把日记等级转换成字符串风格,所有有可能的地方都需要返回

改进

va_start(s,format),用format修饰s的指向,上面的sum是(s,n),类似

这里要用c_str,因为返回的是string

用完以后再用end

这里是往显示器打印的,这里要*3,以为%s和%s中间有几个空格,空间不够

把这里修改一下,打开失败的话

这样就形成日志了

打印最后一行就是正常打开

这里也改一下

测试,先./server,然后会阻塞,然后./client,就会打印出,logmessage里的信息

为啥./client之前不打印

因为等待写入方式打开后,自己才会打开文件,向后执行,open阻塞了!

往文件里打印(上面是往屏幕打印)

先把这些内容全放在Log,日志类

分类

1:向屏幕打印

2:向一个文件打印

3:分类打印

打印格式printMethod

这里构造默认往屏幕去印

析构

打印方式也改一下

打印单个

以0666的权限打开这个文件

打印多个文件(添加一下level)

实现一下

优化一下

以后再打印日志就不用这样打了

这样就可以了,要记住先创建一个Log对象

这样以后就写入一个文件了,写入log.txt

这样就把日志分类了

结果

但是日志文件这么多太混乱了

这样操作后就统一进入一个文件了

makefile也修改一下,先把path定义的log目录创建一下

日志放入一个文件测试结果:

日志分类测试结果:

log.hpp里头文件

优化一下调用

然后修改一下server.cc

 

client.cc

#include "common.hpp"
#include "log.hpp"

int main()
{
    int fd = open(FIFO_FILE,O_WRONLY);
    if(fd < 0)
    {
        perror("open");
        exit(FIFO_OPEN_ERR);
    }
    string line;
    while(true)
    {
        cout<< "Please Enter@ ";
        // cin>> line;
        getline(cin, line);

        write(fd, line.c_str(),line.size());
    }

    close(fd);
    return 0;
}


common.hpp

#pragma noce
#include<iostream>
#include<vector>
#include<string>
#include<unistd.h>
#include <sys/types.h>
#include <sys/wait.h> 
#include <sys/stat.h>
#include<fcntl.h>
#include<stdio.h>
using namespace std;

#define FIFO_FILE "./myfifo"
#define MODE 0664 //用于设置文件的权限,0664代表着8进制写法,4是其他用户可读不可写

enum
{
    FIFO_CREATE_ERR = 1,
    FIFO_DELETE_ERR,
    FIFO_OPEN_ERR
};

class Init
{
public:
    Init()
    {
        //创建管道
        int n = mkfifo(FIFO_FILE,MODE);
        if(n == -1)
        {
            perror("mkfofi");
            exit(FIFO_CREATE_ERR);
        }
    }
    ~Init()
    {
         //删除命名管道
        int m = unlink(FIFO_FILE);
        if(m == -1)
        {
            perror("unlink");
            exit(FIFO_DELETE_ERR);
        }      
    }
};

log.hpp

#pragma noce
#include <stdarg.h>
// #include "common.hpp"
#include <iostream>
#include <stdio.h>
#include<string.h>//strerror(errno)头文件
#include<stdlib.h>
using namespace std;

#define Info        0
#define Debug       1
#define Warning     2
#define Error       3
#define Fatal       4 // 致命的

//打印方式
#define Screen      1 //屏幕
#define Onefile     2 //一个文件
#define Classfile   3 //多个文件

#define LogFile    "log.txt" 

class Log
{
public:
    Log()
    {
        printMehod = Screen;
        path = "./log/";
    }
    void Enable(int method)
    {
        printMehod = method;
    }
    string levelToString(int level)
    {
        switch (level)
        {
        case Info:
            return "Info";
        case Debug:
            return "Debug";
        case Warning:
            return "Warning";
        case Error:
            return "Error";
        case Fatal:
            return "Fatal";
        default:
            return "";
        }
        return "";
    }
    // void logmessage(int level, const char *format, ...)
    // {
    //     time_t t = time(nullptr);
    //     struct tm *ctime = localtime(&t);
    //     char leftbuffer[1024];
    //     snprintf(leftbuffer, sizeof(leftbuffer), "[%s][%d-%d-%d  %d:%d:%d]", levelToString(level).c_str(),
    //             ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, ctime->tm_hour, ctime->tm_min, ctime->tm_sec);

    //     va_list s;
    //     va_start(s, format);
    //     char rightbuffer[1024];
    //     vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);
    //     va_end(s);

    //     // 格式:默认部分+自定义部分
    //     char logtxt[1024 * 3];
    //     snprintf(logtxt, sizeof(logtxt), "%s  %s\n", leftbuffer, rightbuffer);
    //     //cout << logtxt << endl; // 暂时打印
    //     printLog(level, logtxt);
    // }
    void operator()(int level, const char* format, ...)
    {
        time_t t = time(nullptr);
        struct tm *ctime = localtime(&t);
        char leftbuffer[1024];
        snprintf(leftbuffer, sizeof(leftbuffer), "[%s][%d-%d-%d  %d:%d:%d]", levelToString(level).c_str(),
                ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, ctime->tm_hour, ctime->tm_min, ctime->tm_sec);

        va_list s;
        va_start(s, format);
        char rightbuffer[1024];
        vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);
        va_end(s);

        // 格式:默认部分+自定义部分
        char logtxt[1024 * 3];
        snprintf(logtxt, sizeof(logtxt), "%s  %s\n", leftbuffer, rightbuffer);
        //cout << logtxt << endl; // 暂时打印
        printLog(level, logtxt);
    }

    void printLog(int level, const string &logtxt)
    {
        switch(printMehod)
        {
            case Screen:
                cout<< logtxt <<endl;
            break;
            case Onefile:
                printOneFile(LogFile, logtxt);//"log.txt" 
            break;
            case Classfile:
                printClassFile(level, logtxt);
            break;
            default:
            break;
        }

    }
    void printOneFile(const string &logname, const string &logtxt)
    {
        //          "./log/" "log.txt"
        string _logname =path + logname;
        int fd = open(_logname.c_str(), O_WRONLY|O_CREAT|O_APPEND, 0666);
        if(fd < 0) return;
        write(fd, logtxt.c_str(), logtxt.size());
        close(fd);
    }
    void printClassFile(int level, const string &logtxt)
    {
        string filename = LogFile;//"log.txt" 
        filename += ".";//"log.txt." 
        filename += levelToString(level); //log.txt.Debug/Waring/Fatal
        printOneFile(filename, logtxt);
    }
    ~Log()
    {}
private:
    int printMehod;
    string path;
};


makefile

.PHONY:all
all:client server
server:server.cc
	g++ -o $@ $^ -g -std=c++11
	mkdir log
client:client.cc
	g++ -o $@ $^ -g -std=c++11
.PHONY:clean
clean:
	rm -f server client

server.cc

#include "common.hpp"
#include "log.hpp"


int main()
{
    //logmessage(Info, "hello");
    //创建管道
    Init init;
    Log log;
    //log.Enable(Onefile);
    log.Enable(Classfile);

    // //创建管道
    // int n = mkfifo(FIFO_FILE,MODE);
    // if(n == -1)
    // {
    //     perror("mkfofi");
    //     exit(FIFO_CREATE_ERR);
    // }
    // sleep(5);
    //打开管道
    int fd = open(FIFO_FILE,O_RDONLY);
    if(fd < 0)
    {
        //log.logmessage(Fatal, "error string:%s,error code:%d",strerror(errno), errno);
        //优化后
        log(Fatal, "error string:%s,error code:%d",strerror(errno), errno);
        exit(FIFO_OPEN_ERR);
    }
    // log.logmessage(Info, "server open file done,error string:%s,error code:%d",strerror(errno), errno);
    // log.logmessage(Warning, "server open file done,error string:%s,error code:%d",strerror(errno), errno);
    //优化后
    log(Info, "server open file done,error string:%s,error code:%d",strerror(errno), errno);
    log(Warning, "server open file done,error string:%s,error code:%d",strerror(errno), errno);
    //......
    //开始通信
    while(true)
    {
        char buffer[1024] = {0};
        int x = read(fd, buffer, sizeof(buffer));
        if(x > 0)
        {
            buffer[x] = 0;
            cout<< "client say# " << buffer <<endl;
        }
        else if(x == 0)
        {
            //log.logmessage(Debug, "sclient quit too!,error string:%s,error code:%d",strerror(errno), errno);
            //优化后
            log(Debug, "sclient quit too!,error string:%s,error code:%d",strerror(errno), errno);

            //cout<< "client quit too!\n" <<endl;
            break;
        }
        else break;
    }
    close(fd);

    // //删除命名管道
    // int m = unlink(FIFO_FILE);
    // if(n == -1)
    // {
    //     perror("unlink");
    //     exit(FIFO_DELETE_ERR);
    // }


    return 0;
}


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2215688.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PAT甲级-1076 Forwards on Weibo

题目 题目大意 已知微博上粉丝都可能会转发自己所关注的人的动态&#xff0c;给定总人数n和层数l&#xff0c;和每个id关注的人数及所关注人的id。要求查询一组用户的潜在转发量&#xff0c;粉丝层级不能超过l。 思路 求潜在转发量&#xff0c;就是一层一层的找粉丝数&#…

【已知当前表字段名注入】

一、通配符注入 like 测试注入 usernameadmin&passwordadmin 提示username or password error输入单引号闭合测试 这里用or防止admin字段不存在数据库中&#xff0c;如果admin不存在用and连接admin不为真&#xff0c;username的逻辑还是没绕过&#xff0c;用or就算我们前面…

嵌入式Linux开发板配置静态IP

嵌入式Linux开发板配置静态IP Chapter1 嵌入式Linux开发板配置静态IPChapter2 Linux命令之hwclock - 查询和设置硬件时钟 Chapter1 嵌入式Linux开发板配置静态IP 修改interfaces配置文件&#xff0c;普通用户interfaces文件权限只可读&#xff0c;首先切换到root权限。 sudo …

vue单页面 与多页面的区别

引用&#xff08;【Vue相关】单页面应用(SPA)与多页面应用(MPA)的区别_vue mpa-CSDN博客&#xff09; 只要用框架基本上都是单页面工程

特斯拉全新发布会上,无人驾驶汽车亮相,机器人与用户近距离互动

在科技日新月异的今天&#xff0c;特斯拉再次以其前瞻性的技术和创新理念引领了行业的潮流。近日&#xff0c;特斯拉在美国加利福尼亚州伯班克华纳兄弟工作室召开了一场主题为“WE ROBOT”的新品发布会&#xff0c;会上不仅发布了无人驾驶汽车&#xff0c;还展示了特斯拉人形机…

4S店汽车行业专业线上小程序源码系统 功能强大 带完整的安装代码包以及搭建部署教程

系统概述 随着移动互联网的迅速发展&#xff0c;越来越多的传统行业开始拥抱数字化转型。在汽车销售领域&#xff0c;4S店&#xff08;即集整车销售、零配件供应、售后服务、信息反馈四位于一体的汽车服务企业&#xff09;也不例外。为了更好地服务于客户&#xff0c;并提升自…

C++11——异常

异常的介绍 异常是C1中的一种处理错误的方式&#xff0c;当某一函数发现自己无法处理的错误时就可以抛出异常&#xff0c;让函数的直接或间接的调用者处理这个错误&#xff0c;异常的使用可以避免一些难以发现的bug被更好的发现并被处理。 异常的使用 异常的使用包含是三个部…

从空口分析BLE AUDIO ASCS

ASCS&#xff1a;AUDIO STREAM CONTROL SERVICE&#xff0c;音频流控制服务&#xff0c;顾名思义会对音频流的一些参数去做控制的服务&#xff0c;下面我们还是以手机和耳机为例&#xff0c;结合空口来分析ASCS都有哪些内容&#xff1a; 1&#xff1a;ASE CODEC CONFIG 配置A…

保姆级教程下载finalshell以及连接云服务器基础的使用教程

废话不多说&#xff0c;我们直接进行安装 一、软件下载 下载地址&#xff1a; FinalShell SSH工具,服务器管理,远程桌面加速软件,支持Windows,macOS,Linux,版本4.5.10,更新日期2024.9.26 - FinalShell官网 (hostbuf.com)https://www.hostbuf.com/t/988.html 点击链接进行下载…

【Next.js 项目实战系列】01-创建项目

原文链接 CSDN 的排版/样式可能有问题&#xff0c;去我的博客查看原文系列吧&#xff0c;觉得有用的话&#xff0c;给我的库点个star&#xff0c;关注一下吧 创建项目 创建项目​ 在创建项目中可以找到相关内容&#xff0c;这里不再赘述 First commit​ 创建好项目之后&am…

40个IT运维加薪技术,每一个都含金量极高!运维必会的10个网络抓包/调试工具!系统崩溃的10种可能,运维人员必看!

40个IT运维加薪技术&#xff0c;每一个都含金量极高!运维必会的10个网络抓包/调试工具&#xff01;系统崩溃的10种可能&#xff0c;运维人员必看&#xff01; 在IT运维领域&#xff0c;想要加薪并不仅仅依赖于单一的技术&#xff0c;而是需要综合运用多种技能&#xff0c;不断提…

【LangChain】(一)掌握向量数据库的CRUD操作与语义检索技巧

文章目录 1. 向量数据库的基础知识1.1 为什么选择向量数据库&#xff1f; 2. CRUD操作详解2.1 向量数据库-新增数据示例代码&#xff1a; 2.2 向量数据库-删除数据示例代码&#xff1a; 2.3 向量数据库-更新数据2.4 向量数据库-查询数据2.4.1 相似度查询2.4.2 最大边际相关性查…

React(二) JSX中的this绑定问题;事件对象参数传递;条件渲染;列表渲染;JSX本质;购物车案例

文章目录 一、jsx事件绑定1. 回顾this的绑定方式2. jsx中的this绑定问题(1) 方式一&#xff1a;bind绑定(2) 方式二&#xff1a;使用 ES6 class fields 语法(3) 方式三&#xff1a;直接传入一个箭头函数(重要) 3.事件参数传递(1) 传递事件对象event(2) 传递其他参数 4. 事件绑定…

DeepBook V3和DEEP token正式上线

10月14日&#xff0c;DeepBook V3版本正式在主网上线&#xff0c;同时发布了DEEP token&#xff0c;标志着其发展历程中的一个重要里程碑。通过这次升级&#xff0c;用户和做市商将受益于精细调控的激励措施&#xff0c;为通过社区驱动的治理实现更大的去中心化奠定了基础。 D…

【大模型问答测试】大模型问答测试脚本实现(第一版)

背景 公司已经做了一段时间的大模型&#xff0c;每次测试或者回归的时候都需要针对问答进行测试回归&#xff0c;耗费大量的时间与精力&#xff0c;因此结合产品特点&#xff0c;开发自动化脚本替代人工的操作&#xff0c;提升测试回归效率 设计 使用pythonrequestExcel进行…

python+大数据+基于Spark的共享单车数据存储系统【内含源码+文档+部署教程】

博主介绍&#xff1a;✌全网粉丝10W,前互联网大厂软件研发、集结硕博英豪成立工作室。专注于计算机相关专业毕业设计项目实战6年之久&#xff0c;选择我们就是选择放心、选择安心毕业✌ &#x1f345;由于篇幅限制&#xff0c;想要获取完整文章或者源码&#xff0c;或者代做&am…

【jQuery】 jQuery基础及选择器介绍(基本选择器 层次选择器 属性选择器 过滤选择器)

文章目录 jQuery基础1. 优势2. 版本3. 基本语法4. 选择器基本选择器层次选择器属性选择器过滤选择器基本过滤选择器可见性过滤选择器 注意事项 jQuery基础 jQuery 是一个功能强大且易于使用的 JavaScript 库&#xff0c;它极大地简化了前端开发的工作。无论是 DOM 操作、事件处…

简单概述Ton链开发路径

区块链开发领域发展迅速&#xff0c;各种平台为开发人员提供不同的生态系统。其中一个更有趣且越来越相关的区块链是TON&#xff08;开放网络&#xff09;区块链。TON 区块链最初由 Telegram 构思&#xff0c;旨在提供快速、安全且可扩展的去中心化应用程序 (dApp)。凭借其独特…

加盟模式如何运营?有哪些好的技巧和方法!

对于很多品牌方来说&#xff0c;生意发展到一定程度&#xff0c;就考虑通过加盟的方式扩大市场份额。 本篇文章&#xff0c;将从3个角度来为大家分享&#xff0c;运营加盟模式的好方法和技巧&#xff01; 一、加盟前的准备 1、明确品牌定位与核心竞争力 确定你的企业在市场…

暖水毯/取暖毯语音识别控制芯片IC方案

暖水毯、取暖毯作为现代家居生活的温暖伴侣&#xff0c;其智能化升级已是大势所趋。在暖水毯与取暖毯中融入语音识别控制芯片IC方案&#xff0c;为用户的冬日取暖体验带来了革命性的变革。 一、暖水毯/取暖毯增加语音识别控制芯片方案&#xff0c;让产品能通过对话来调节&…