0


命名管道Linux

管道是

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

管道

首先自己要用用户层缓冲区,还得把用户层缓冲区拷贝到管道里,(从键盘里输入数据到用户层缓冲区里面),然后用户层缓冲区通过系统调用(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. }
标签: linux c++ c

本文转载自: https://blog.csdn.net/2401_83427936/article/details/142603367
版权归原作者 Ljw... 所有, 如有侵权,请联系我们删除。

“命名管道Linux”的评论:

还没有评论