当前位置: 首页 > news >正文

C/C++进程超详细详解【中部分】(系统性学习day07)

目录

前言

一、守护进程

1.概念

2.守护进程创建的原理(如图清晰可见)

 3.守护进程的实现(代码块)

二、dup和dup2

1,复制文件描述符

2.文件描述符重定向

三、系统日志

1,打开日志

2,向日志中写消息

3,关闭日志

四,文件锁

1.概念

2,给整个文件上锁

实例代码如下: 

3,给文件的某个区域上锁

 实例代码如下:

五,进程间通信

1.分类

2,无名管道

2.1 无名管道通信原理

2.2 用法

 实例代码如下:

总结


前言

上篇博客对C/C++进程的上部分进行了详细讲解,本篇博客将继续讲解和补充关于线程的知识点。


一、守护进程

1.概念

(1)守护进程,
    也就是通常所说的Daemon进程,是Linux中的后台服务进程。
    它是一个生存期较长的进程,通常独立于控制终端并且周期性的执行某种任务或等待处理某些发生的事件
    守护进程常常在系统引导装入时启动,在系统关闭时终止
    Linux系统有很多守护进程,大多数服务都是用守护进程实现的 

(2)终端
    在Linux中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个终端就称为这些进程的控制终端,当控制终端被关闭时,相应的进程都会被自动关闭。

守护进程能够突破这种限制,它从被执行开始运转,直到整个系统关闭才会退出。
如果想让某个进程不因为用户或终端或其他的变化而受到影响,就必须把这个进程变成一个守护进程。 

2.守护进程创建的原理(如图清晰可见)

 

 3.守护进程的实现(代码块)

void init_deamon(void)
{/*************** start ****************************/pid_t pid;int i,max_fd;//1,创建子进程if((pid = fork()) < 0){perror("fork");exit(1);}else if(pid > 0)exit(0);//2,创建新会话if(setsid() < 0){perror("setsid");exit(1);}//3,再创建子进程if((pid = fork()) < 0){perror("fork");exit(1);}else if(pid > 0)exit(0);//4,修改守护进程的工作目录chdir("/");//5,关闭进程父进程的所有的文件描述符max_fd = sysconf(_SC_OPEN_MAX);for (i = 0; i < max_fd;i++)close(i);//6,将标准输入,标准输出和标准错误重定向到/dev/nullopen("/dev/null",O_RDWR);dup(0);dup(0);//7,消除umask影响umask(0);/*************** end ****************************/
}

二、dup和dup2

1,复制文件描述符

int dup(int oldfd);
//参数 ---- 要复制的文件描述符
//返回值 ----成功:新的文件描述符,失败:-1

例如: 
int main(void)
{
    char str[] = "hello world";
    int fd1,fd2;


    fd1 = open("1.txt",O_RDWR|O_CREAT,0666);
    if(fd1 < 0){
        perror("open");
        exit(1);
    }

    write(fd1,str,strlen(str));

    fd2 = dup(fd1);   //复制文件描述fd1
    strcpy(str,"farsight");
    write(fd2,str,strlen(str));

    close(fd1);


    return 0;
}      
 

2.文件描述符重定向

int dup2(int oldfd, int newfd);
//参数1 --- 目标文件描述符
//参数2 --- 要重定向的文件描述符
//返回值 ---- 成功0,失败:-1

例如: 
int main(void)
{
    char str[] = "hello world";
    int fd1,fd2;


    fd1 = open("1.txt",O_RDWR|O_CREAT,0666);
    if(fd1 < 0){
        perror("open");
        exit(1);
    }
    fd2 = open("2.txt",O_RDWR|O_CREAT,0666);
    if(fd1 < 0){
        perror("open");
        exit(1);
    }

    write(fd1,str,strlen(str));

    dup2(fd1,fd2);  //将fd2重定向到fd1

    strcpy(str,"farsight");
    write(fd2,str,strlen(str));
    close(fd1);


    return 0;
}

三、系统日志

1,打开日志

#include <syslog.h>
void openlog(const char *ident, int option, int facility);
//参数1 ------  //日志标签,自定义,方便查找日志信息
//参数2 ------  选项:
                LOG_CONS     如果消息不能发送给日志,则发送到控制台
                LOG_NDELAY   不延迟打开套接字,并发送消息
                LOG_NOWAIT   创建子进程,不阻塞发送消息给日志
                LOG_PERROR   发送日志,同时发送到标准错误文件
                LOG_PID      在消息中加入进程的ID
//参数3 ------   进程类型:
                    LOG_DAEMON      守护进程
                    LOG_FTP          tfp服务进程
                    LOG_KERN      内核进程
                    LOG_LPR       打印服务进程
                    LOG_MAIL      邮件服务进程

实例如下:
    openlog("mydaemon",LOG_PID,LOG_DAEMON);

2,向日志中写消息

void syslog(int priority, const char *format, ...);
//参数1 ----- 消息的优先级
               LOG_EMERG      非常紧急的错误
               LOG_ALERT      必须马上处理的错误
               LOG_CRIT       关键性错误
               LOG_ERR        一般错误
               LOG_WARNING    警告
               LOG_NOTICE     需要注意的消息
               LOG_INFO       正常消息
               LOG_DEBUG      调试消息
//参数2 -----向日志中写消息的格式
//变参 ----- 类似于printf的变参
例如: 
    syslog(LOG_ERR,"fopen:%s",strerror(errno));
    
运行测试:
     grep mydaemon /var/log/syslog -n
    203:Sep 26 23:36:26 ubuntu mydaemon[28968]: fopen:No such file or directory

3,关闭日志

 void closelog(void);

四,文件锁

1.概念

为了解决进程之间的互斥问题,引入咨询锁

采用锁文件的方式取代创建文件的方式
    需遵循“君子协定”
    共享锁(shared Lock)和互斥锁(exculusive lock)
    对整个文件上锁或者文件的某个部分上锁(记录锁定)

2,给整个文件上锁

#include <sys/file.h>
int flock(int fd, int operation);
//参数1 ---- 文件描述符
//参数2 ---- 锁的类型:LOCK_SH  LOCK_EX  LOCK_UN
//返回值 ---成功:0,失败:-1

实例代码如下: 

代码一:

int main(int argc,char **argv)
{int fd;int i;if(argc != 2){fprintf(stderr,"Usage: %s <filename>\n",argv[0]);exit(0);}if((fd = open(argv[1],O_RDWR)) < 0){perror("open");exit(1);}while(1){printf("等待获取锁\n");//获取互斥锁if(flock(fd,LOCK_EX) < 0){perror("flock");exit(1);}for(i = 0; i < 7; i++){printf("正在上厕所\n");sleep(1);}//释放锁if(flock(fd,LOCK_UN) < 0){perror("flock");exit(1);}printf("上完厕所出来了....\n");sleep(1);}return 0;
}

代码二:

int main(int argc,char **argv)
{int fd;int i;if(argc != 2){fprintf(stderr,"Usage: %s <filename>\n",argv[0]);exit(0);}if((fd = open(argv[1],O_RDWR)) < 0){perror("open");exit(1);}while(1){printf("等待着上厕所\n");//获取互斥锁if(flock(fd,LOCK_EX) < 0){perror("flock");exit(1);}for(i = 0; i < 7; i++){printf("正在上厕所...\n");sleep(1);}//释放锁if(flock(fd,LOCK_UN) < 0){perror("flock");exit(1);}printf("上完厕所!\n");sleep(1);}return 0;
}

3,给文件的某个区域上锁

#include <unistd.h>
#include <fcntl.h>
int fcntl(int fd, int cmd, ... /* arg */ );

 struct flock {
       short l_type;    /* Type of lock: F_RDLCK, F_WRLCK, F_UNLCK */
       short l_whence;  /* How to interpret l_start:
                           SEEK_SET, SEEK_CUR, SEEK_END */
       off_t l_start;   /* Starting offset for lock */
       off_t l_len;     /* Number of bytes to lock */
       pid_t l_pid;     /* PID of process blocking our lock
                           (set by F_GETLK and F_OFD_GETLK) */
   };

 实例代码如下:

//定义锁的结构体--设置锁的区域struct flock fl = {.l_whence = SEEK_SET,.l_start = 100,.l_len  = 1024,};while(1){printf("等待获取锁\n");//获取互斥锁fl.l_type = F_WRLCK;   //设置锁的类型if(fcntl(fd,F_SETLKW,&fl) < 0){perror("flock");exit(1);}for(i = 0; i < 7; i++){printf("正在上厕所\n");sleep(1);}//释放锁fl.l_type = F_UNLCK;    //解锁if(fcntl(fd,F_SETLK,&fl) < 0){perror("flock");exit(1);}printf("上完厕所出来了....\n");sleep(1);}

五,进程间通信

1.分类

 在linux中进程间通信分为三类:
(1)早期的进程间通信
    无名管道
    有名(命名)管道
    信号
(2)system V IPC
    消息队列
    共享内存
    信号灯(量)
(3)unix域套接字

2,无名管道

2.1 无名管道通信原理

2.2 用法

#include <unistd.h>
int pipe(int pipefd[2]);
//参数 ---- 保存管道两端文件描述符的数组
//返回值 ---成功:0,失败:-1

 实例代码如下:
int main(void)
{int fd[2];pid_t pid;char buf[100];//创建无名管道if(pipe(fd) < 0){   //pipe()会在内核中创建无名管道,然后将管道两端的文件描述符返回给当前进程perror("pipe");exit(1);}//创建子进程if((pid = fork()) < 0){perror("fork");exit(1);}else if(!pid){  //子进程执行:从键盘获取字符串,写到管道中close(fd[0]);  //关闭读端while(1){fgets(buf,sizeof(buf),stdin);write(fd[1],buf,strlen(buf));   //向管道中写数据}}else{  //父进程执行:从管道读数据,打印到屏幕上close(fd[1]); //关闭写端while(1){if(read(fd[0],buf,sizeof(buf)) < 0){perror("read");exit(1);}printf("%s",buf);}}return 0;
}


总结

        本篇文章针对进程进行超详细讲解和补充,希望能够帮到大家!

        以后还会给大家展现更多关于嵌入式和C语言的其他重要的基础知识,感谢大家支持懒大王!

       希望这篇博客能给各位朋友们带来帮助,最后懒大王请来过的朋友们留下你们宝贵的三连以及关注,感谢你们!

http://www.lryc.cn/news/180334.html

相关文章:

  • S型速度曲线轨迹规划(约束条件为速度和位移)
  • 从零手搓一个【消息队列】实现数据的硬盘管理和内存管理(线程安全)
  • 自动驾驶中的感知模型:实现安全与智能驾驶的关键
  • 【CVPR 2023】DSVT: Dynamic Sparse Voxel Transformer with Rotated Sets
  • MySQL超入门(1)__迅速上手掌握MySQL
  • 四、浏览器渲染过程,DOM,CSSDOM,渲染,布局,绘制详细介绍
  • 2021-06-10 51单片机设计一个蜂鸣器报警电路每秒
  • D‘Agostino-Pearson正态检验|偏度skewness和峰度kurtosis
  • 基于树莓派CM4制作img系统镜像批量制作TF卡
  • 【中秋国庆不断更】OpenHarmony组件内状态变量使用:@State装饰器
  • 【Java 进阶篇】MySQL多表关系详解
  • 【开发篇】十、Spring缓存:手机验证码的生成与校验
  • 【Aurora 8B/10B IP(1)--初步了解】
  • C++ vector容器的介绍与使用
  • openstack的组成
  • [React] React高阶组件(HOC)
  • 【逐步剖C++】-第二章-C++类和对象(中)
  • PL/SQL动态SQL
  • Python绘图系统24:添加辅助坐标轴
  • Java自学网站--十几个网站的分析与评测
  • java接口怎么写
  • 第8章 Spring(二)
  • 从0开始python学习-24.selenium 浏览器常见的操作
  • Canal实现数据同步
  • 数据库学习笔记——DDL
  • MATLAB算法实战应用案例精讲-【人工智能】边缘计算(附python代码实现)
  • 精彩回顾 | 迪捷软件亮相2023世界智能网联汽车大会
  • 【ShaderLab PBR 嗜血边缘角色_美式朋克风格_“Niohoggr“_角色渲染(第一篇)】
  • python经典百题之围圈报数
  • Google Earth Engine(GEE)案例——如何去除和过滤Landsat和sentinel等系列影像集合中的空影像(三种方法)