定名管道Linux

打印 上一主题 下一主题

主题 1640|帖子 1640|积分 4920

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有账号?立即注册

x
 管道是

毫不相干的进程进程间通讯::定名管道
管道

起首本身要用用户层缓冲区,还得把用户层缓冲区拷贝到管道里,(从键盘里输入数据到用户层缓冲区内里),然后用户层缓冲区通过体系调用(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

头文件


打印详细年代日 
年是从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

  1. #include "common.hpp"
  2. #include "log.hpp"
  3. int main()
  4. {
  5.     int fd = open(FIFO_FILE,O_WRONLY);
  6.     if(fd < 0)
  7.     {
  8.         perror("open");
  9.         exit(FIFO_OPEN_ERR);
  10.     }
  11.     string line;
  12.     while(true)
  13.     {
  14.         cout<< "Please Enter@ ";
  15.         // cin>> line;
  16.         getline(cin, line);
  17.         write(fd, line.c_str(),line.size());
  18.     }
  19.     close(fd);
  20.     return 0;
  21. }
复制代码
common.hpp

  1. #pragma noce
  2. #include<iostream>
  3. #include<vector>
  4. #include<string>
  5. #include<unistd.h>
  6. #include <sys/types.h>
  7. #include <sys/wait.h>
  8. #include <sys/stat.h>
  9. #include<fcntl.h>
  10. #include<stdio.h>
  11. using namespace std;
  12. #define FIFO_FILE "./myfifo"
  13. #define MODE 0664 //用于设置文件的权限,0664代表着8进制写法,4是其他用户可读不可写
  14. enum
  15. {
  16.     FIFO_CREATE_ERR = 1,
  17.     FIFO_DELETE_ERR,
  18.     FIFO_OPEN_ERR
  19. };
  20. class Init
  21. {
  22. public:
  23.     Init()
  24.     {
  25.         //创建管道
  26.         int n = mkfifo(FIFO_FILE,MODE);
  27.         if(n == -1)
  28.         {
  29.             perror("mkfofi");
  30.             exit(FIFO_CREATE_ERR);
  31.         }
  32.     }
  33.     ~Init()
  34.     {
  35.          //删除命名管道
  36.         int m = unlink(FIFO_FILE);
  37.         if(m == -1)
  38.         {
  39.             perror("unlink");
  40.             exit(FIFO_DELETE_ERR);
  41.         }      
  42.     }
  43. };
复制代码
log.hpp

  1. #pragma noce
  2. #include <stdarg.h>
  3. // #include "common.hpp"
  4. #include <iostream>
  5. #include <stdio.h>
  6. #include<string.h>//strerror(errno)头文件
  7. #include<stdlib.h>
  8. using namespace std;
  9. #define Info        0
  10. #define Debug       1
  11. #define Warning     2
  12. #define Error       3
  13. #define Fatal       4 // 致命的
  14. //打印方式
  15. #define Screen      1 //屏幕
  16. #define Onefile     2 //一个文件
  17. #define Classfile   3 //多个文件
  18. #define LogFile    "log.txt"
  19. class Log
  20. {
  21. public:
  22.     Log()
  23.     {
  24.         printMehod = Screen;
  25.         path = "./log/";
  26.     }
  27.     void Enable(int method)
  28.     {
  29.         printMehod = method;
  30.     }
  31.     string levelToString(int level)
  32.     {
  33.         switch (level)
  34.         {
  35.         case Info:
  36.             return "Info";
  37.         case Debug:
  38.             return "Debug";
  39.         case Warning:
  40.             return "Warning";
  41.         case Error:
  42.             return "Error";
  43.         case Fatal:
  44.             return "Fatal";
  45.         default:
  46.             return "";
  47.         }
  48.         return "";
  49.     }
  50.     // void logmessage(int level, const char *format, ...)
  51.     // {
  52.     //     time_t t = time(nullptr);
  53.     //     struct tm *ctime = localtime(&t);
  54.     //     char leftbuffer[1024];
  55.     //     snprintf(leftbuffer, sizeof(leftbuffer), "[%s][%d-%d-%d  %d:%d:%d]", levelToString(level).c_str(),
  56.     //             ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, ctime->tm_hour, ctime->tm_min, ctime->tm_sec);
  57.     //     va_list s;
  58.     //     va_start(s, format);
  59.     //     char rightbuffer[1024];
  60.     //     vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);
  61.     //     va_end(s);
  62.     //     // 格式:默认部分+自定义部分
  63.     //     char logtxt[1024 * 3];
  64.     //     snprintf(logtxt, sizeof(logtxt), "%s  %s\n", leftbuffer, rightbuffer);
  65.     //     //cout << logtxt << endl; // 暂时打印
  66.     //     printLog(level, logtxt);
  67.     // }
  68.     void operator()(int level, const char* format, ...)
  69.     {
  70.         time_t t = time(nullptr);
  71.         struct tm *ctime = localtime(&t);
  72.         char leftbuffer[1024];
  73.         snprintf(leftbuffer, sizeof(leftbuffer), "[%s][%d-%d-%d  %d:%d:%d]", levelToString(level).c_str(),
  74.                 ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, ctime->tm_hour, ctime->tm_min, ctime->tm_sec);
  75.         va_list s;
  76.         va_start(s, format);
  77.         char rightbuffer[1024];
  78.         vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);
  79.         va_end(s);
  80.         // 格式:默认部分+自定义部分
  81.         char logtxt[1024 * 3];
  82.         snprintf(logtxt, sizeof(logtxt), "%s  %s\n", leftbuffer, rightbuffer);
  83.         //cout << logtxt << endl; // 暂时打印
  84.         printLog(level, logtxt);
  85.     }
  86.     void printLog(int level, const string &logtxt)
  87.     {
  88.         switch(printMehod)
  89.         {
  90.             case Screen:
  91.                 cout<< logtxt <<endl;
  92.             break;
  93.             case Onefile:
  94.                 printOneFile(LogFile, logtxt);//"log.txt"
  95.             break;
  96.             case Classfile:
  97.                 printClassFile(level, logtxt);
  98.             break;
  99.             default:
  100.             break;
  101.         }
  102.     }
  103.     void printOneFile(const string &logname, const string &logtxt)
  104.     {
  105.         //          "./log/" "log.txt"
  106.         string _logname =path + logname;
  107.         int fd = open(_logname.c_str(), O_WRONLY|O_CREAT|O_APPEND, 0666);
  108.         if(fd < 0) return;
  109.         write(fd, logtxt.c_str(), logtxt.size());
  110.         close(fd);
  111.     }
  112.     void printClassFile(int level, const string &logtxt)
  113.     {
  114.         string filename = LogFile;//"log.txt"
  115.         filename += ".";//"log.txt."
  116.         filename += levelToString(level); //log.txt.Debug/Waring/Fatal
  117.         printOneFile(filename, logtxt);
  118.     }
  119.     ~Log()
  120.     {}
  121. private:
  122.     int printMehod;
  123.     string path;
  124. };
复制代码
makefile

  1. .PHONY:all
  2. all:client server
  3. server:server.cc
  4.         g++ -o $@ $^ -g -std=c++11
  5.         mkdir log
  6. client:client.cc
  7.         g++ -o $@ $^ -g -std=c++11
  8. .PHONY:clean
  9. clean:
  10.         rm -f server client
复制代码
server.cc

  1. #include "common.hpp"
  2. #include "log.hpp"
  3. int main()
  4. {
  5.     //logmessage(Info, "hello");
  6.     //创建管道
  7.     Init init;
  8.     Log log;
  9.     //log.Enable(Onefile);
  10.     log.Enable(Classfile);
  11.     // //创建管道
  12.     // int n = mkfifo(FIFO_FILE,MODE);
  13.     // if(n == -1)
  14.     // {
  15.     //     perror("mkfofi");
  16.     //     exit(FIFO_CREATE_ERR);
  17.     // }
  18.     // sleep(5);
  19.     //打开管道
  20.     int fd = open(FIFO_FILE,O_RDONLY);
  21.     if(fd < 0)
  22.     {
  23.         //log.logmessage(Fatal, "error string:%s,error code:%d",strerror(errno), errno);
  24.         //优化后
  25.         log(Fatal, "error string:%s,error code:%d",strerror(errno), errno);
  26.         exit(FIFO_OPEN_ERR);
  27.     }
  28.     // log.logmessage(Info, "server open file done,error string:%s,error code:%d",strerror(errno), errno);
  29.     // log.logmessage(Warning, "server open file done,error string:%s,error code:%d",strerror(errno), errno);
  30.     //优化后
  31.     log(Info, "server open file done,error string:%s,error code:%d",strerror(errno), errno);
  32.     log(Warning, "server open file done,error string:%s,error code:%d",strerror(errno), errno);
  33.     //......
  34.     //开始通信
  35.     while(true)
  36.     {
  37.         char buffer[1024] = {0};
  38.         int x = read(fd, buffer, sizeof(buffer));
  39.         if(x > 0)
  40.         {
  41.             buffer[x] = 0;
  42.             cout<< "client say# " << buffer <<endl;
  43.         }
  44.         else if(x == 0)
  45.         {
  46.             //log.logmessage(Debug, "sclient quit too!,error string:%s,error code:%d",strerror(errno), errno);
  47.             //优化后
  48.             log(Debug, "sclient quit too!,error string:%s,error code:%d",strerror(errno), errno);
  49.             //cout<< "client quit too!\n" <<endl;
  50.             break;
  51.         }
  52.         else break;
  53.     }
  54.     close(fd);
  55.     // //删除命名管道
  56.     // int m = unlink(FIFO_FILE);
  57.     // if(n == -1)
  58.     // {
  59.     //     perror("unlink");
  60.     //     exit(FIFO_DELETE_ERR);
  61.     // }
  62.     return 0;
  63. }
复制代码


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

滴水恩情

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表