【socket】基于socket下进程上报温度

进程温度通信

fork系统调用

fork()函数又叫计算机程序设计中的分叉函数,fork是一个很有意思的函数,它可以建立一个新进程,把当前的进程分为父进程和子进程,新进程称为子进程,而原进程称为父进程。fork调用一次,返回两次,这两个返回分别带回它们各自的返回值,其中在父进程中的返回值是子进程的PID,而子进程中的返回值则返回 0。因此,可以通过返回值来判定该进程是父进程还是子进程。还有一个很奇妙的是:fork函数将运行着的程序分成2个(几乎)完全一样的进程,每个进程都启动一个从代码的同一位置开始执行的线程。这两个进程中的线程继续执行,就像是两个用户同时启动了该应用程序的两个副本。

新创建的子进程几乎但是不完全与父进程相同。子进程得到与父进程用户级虚拟地址空间相同(但是独立)的一份拷贝,包括文本,数据和bss段、堆以及用户栈。子进程还获得与父进程任何打开文件描述符相同的拷贝。这就是意味着当父进程调用fork时候,子进程还可以读写父进程中打开的任何文件。父进程和新创建的子进程之间最大区别在于他们有着不同的PID。
UNIX将复制父进程的地址空间内容给子进程,因此,子进程有了独立的地址空间。在不同的UNIX (Like)系统下,我们无法确定fork之后是子进程先运行还是父进程先运行,这依赖于系统的实现。
下面我们再以一个最简单的代码来更简单说明fork()函数:

#include <stdio.h>
#include <unistd.h>
int mian(void)
{
   fork();
   printf("hello");
   return 0;
}

每个子进程只有一个父进程,并且每个进程都可以通过getpid()获取自己的进程PID,也可以通过getppid()获取父进程的PID,这样在fork()时返回0给子进程是可取的。一个进程可以创建多个子进程,这样对于父进程而言,他并没有一个API函数可以获取其子进程的进程ID,所以父进程在通过fork()创建子进程的时候,必须通过返回值的形式告诉父进程其创建的子进程PID。这也是fork()系统调用两次返回值设计的原因。

fork()系统调用会创建一个新的子进程,这个子进程是父进程的一个副本。这也意味着,系统在创建新的子进程成功后,会将父进程的文本段、数据段、堆栈都复制一份给子进程,但子进程有自己独立的空间,子进程对这些内存的修改并不会影响父进程空间的相应内存。这时系统中出现两个基本完全相同的进程(父、子进程),这两个进程执行没有固定的先后顺序,哪个进程先执行要看系统的进程调度策略。如果需要确保让父进程或子进程先执行,则需要程序员在代码中通过进程间通信的机制来自己实现

代码实现

#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>
#include <signal.h>
#include <string.h>
#include <errno.h>
#include <getopt.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>

#include "sqlite3.h"
#include <sqlite3.h>

#define BACKLOG 13


void print_usage(char *progname);
void sig_stop(int signum);
int socket_listen(char *listen_ip, int listen_serv_port);
void sqlite_tem(char *buf);

static int g_stop = 0;

int main(int argc, char **argv)
{
    int                     rv;
    int                     ret;
    int                     opt;
    int                     serv_port;
    int                     log_fd;
    int                     ch = 1;
    int                     daemon_run = 0;
    int                     listen_fd = -1;
    int                     cli_fd = -1;
    pid_t                   pid = -1;
    struct sockaddr_in      ser_addr;
    struct sockaddr_in      cli_addr;
    socklen_t               cliaddr_len = 20;
    
    char                    *zErrMsg;
    sqlite3                 *db;
    char                    buf[1024];


    struct option            opts[] = { 
        {"daemon", no_argument, NULL, 'd'}, 
        {"serv_port", required_argument, NULL, 'p'}, 
        {"help", no_argument, NULL, 'h'}, 
        {NULL, 0, NULL, 0}
    }; 

    while ((opt = getopt_long(argc, argv, "dp:h", opts, &ch)) != -1)
    {
        switch(opt)
        { 
            case 'd':
                daemon_run = 1; 
                break;
            case 'p':
                serv_port = atoi(optarg); 
                break; 
            case 'h':
                print_usage(argv[0]);
                return 0;
        }
    } 


    if (!serv_port)
    {
        print_usage(argv[0]);
        return 0;
    }

    /*建立日志*/
    if (daemon_run)
    {
        printf("Program %s is running at the background now\n", argv[0]);


        log_fd = open("fork.log",  O_CREAT|O_RDWR, 0666);
        if (log_fd < 0)
        {
            printf("Open the logfile failure : %s\n", strerror(errno));
            return 0;
        }


        dup2(log_fd, STDOUT_FILENO);
        dup2(log_fd, STDERR_FILENO);


        if ((daemon(1, 1)) < 0)
        {
            printf("Deamon failure : %s\n", strerror(errno));
            return 0;
        }
    }


    signal(SIGUSR1, sig_stop);

    /*调用socket*/
    if( (listen_fd = socket_listen(NULL, serv_port)) < 0 )
    {
        printf("ERROR: %s server listen on serv_port %d failure\n", argv[0], serv_port);
        return -2;
    }
    printf("server start to listen on serv_port %d\n",  serv_port);

    
    while (!g_stop)
    {

        cli_fd = accept(listen_fd, (struct sockaddr *)&cli_addr, &cliaddr_len);
        if (cli_fd < 0)
        {
            printf("Accept the request from client failure:%s\n", strerror(errno));
            continue;
        }

        /*创建进程*/
        pid = fork();   
        if (pid < 0)
        {
            printf("Creat child process failure:%s\n", strerror(errno));
            continue;
        }
        else if (pid > 0) /*父进程关闭cli_fd,保留listen_fd,等待与其他客户端连接*/
        {
            close(cli_fd);  
            continue;
        }
        else if (pid == 0) /*子进程关闭ser_fd,保留listen_fd,保持与客户端连接*/
        {
            close(listen_fd);     

                while (1)       
                {
                    memset(buf, 0, sizeof(buf));

                    rv = read(cli_fd, buf, sizeof(buf));        
                    if (rv < 0)     
                    {
                        printf("Read information from client failure:%s\n", strerror(errno));
                        close(cli_fd);
                        exit(0);
                    }
                    else if (rv == 0)   
                    {
                        printf("The connection with client has broken!\n");
                        close(cli_fd);
                        exit(0);
                    }
                    else        
                    {
                        printf("%s\n",buf);
                        sqlite_tem(buf);
                        printf("Database inserted successfully!\n"); 
                    }
                }
        }
    }
    close(listen_fd);

    return 0;
}

/*帮助信息*/
void print_usage(char *progname)
{
    printf("-d(--daemon):let program run in the background\n"); 
    printf("-p(--serv_port):enter server serv_port\n");
    printf("-h(--help):print this help information\n"); 

    return ;
}

/*安装信号*/
void sig_stop(int signum)
{
    if (SIGUSR1 == signum)      
    {
        g_stop = 1;     
    }
    return ;
}

/*socket函数*/
int socket_listen(char *listen_ip, int listen_serv_port)
{
    int                     rv = 0;
    int                     on = 1;
    int                     listen_fd;
    struct sockaddr_in      servaddr;

    if ( (listen_fd = socket(AF_INET, SOCK_STREAM, 0)) < 0 )
    {   
        printf("Use socket() to create a TCP socket failure: %s\n", strerror(errno));
        return -1; 
    }   

    setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));

    memset(&servaddr, 0, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(listen_serv_port);


    if( !listen_ip )
    {   
        servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    }   
    else
    {   
        if( inet_pton(AF_INET, listen_ip, &servaddr.sin_addr) <= 0 )
        {   
            printf("Inet_pton() set listen IP address failure\n");
            rv = -2; 
            goto CleanUp;
        }   
    }

    if( bind(listen_fd, (struct sockaddr *) &servaddr, sizeof(servaddr)) < 0 )
    {   
        printf("Use bind() to bind the TCP socket failure: %s\n", strerror(errno));
        rv = -3; 
        goto CleanUp;
    }

    if( listen(listen_fd, 64) < 0 )
    {   
        printf("Use bind() to bind the TCP socket failure: %s\n", strerror(errno));
        rv = -4; 
        goto CleanUp;
    }   

CleanUp:
    if( rv < 0 )
        close(listen_fd);
    else
        rv = listen_fd;
    return rv;
}

/*数据库*/
void sqlite_tem(char *buf)
{  
    int             nrow=0;
    int             ncolumn = 0;
    char          **azResult=NULL;
    int             rv; 
    sqlite3        *db=NULL;
    char           *zErrMsg = 0;
    char            sql1[100];
    char           *ipaddr=NULL;
    char           *datetime=NULL;
    char           *temper=NULL;
    char           *sql = "create table if not exists temperature(ipaddr char(30), datetime char(50), temper  char(30))"; 


    ipaddr = strtok(buf,"/");
    datetime = strtok(NULL, "/");
    temper = strtok(NULL, "/");

    rv = sqlite3_open("tempreture.db", &db);
    if(rv)
    {
        printf("Can't open database:%s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return;
    }
    printf("opened a sqlite3 database named tempreture.db successfully!\n");



    int ret = sqlite3_exec(db,sql, NULL, NULL, &zErrMsg);
    if(ret != SQLITE_OK)
    {
        printf("create table fail: %s\n",zErrMsg);
    }

    if(snprintf(sql1,sizeof(sql1), "insert into temper values('%s','%s','%s')", ipaddr, datetime, temper) < 0)
    {
        printf("Failed to write data\n");
    }
    //printf("Write data successfully!\n");

    sqlite3_exec(db, sql1, 0, 0, &zErrMsg);
    sqlite3_free(zErrMsg);
    sqlite3_close(db);
    return;
}

在该程序中,父进程accept()接收到新的连接后,就调用fork()系统调用来创建子进程来处理与客户端的通信。因为子进程会继承父进程处于listen状态的socket 文件描述符(sockfd),也会继承父进程accept()返回的客户端socket 文件描述符(cli_fd),但子进程只处理与客户端的通信,这时他会将父进程的listen的文件描述符sockfd关闭;同样父进程只处理监听的事件,所以会将cli_fd关闭。

此时父子进程同时运行完成不同的任务,子进程只负责跟已经建立的客户端通信,而父进程只用来监听到来的socket客户端连接。所以当有新的客户端到来时,父进程就有机会来处理新的客户连接请求了,同时每来一个客户端都会创建一个子进程为其服务。

posted @ 2022-09-25 19:39  西故黄鹤楼  阅读(30)  评论(0编辑  收藏  举报