模仿pgpool-II的方式,建立线程池

简介:
接上一篇,pgpool-II中是如何实现进程池的,自己实现线程池。可以正常运行。

server端代码:con-server03.c

复制代码
#include<netinet/in.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/signal.h>
#include<sys/wait.h>
#include<sys/time.h>

#define HELLO_WORLD_SERVER_PORT    6666
#define LENGTH_OF_LISTEN_QUEUE 20
#define BUFFER_SIZE 1024

int main(int argc, char **argv){
 struct sockaddr_in server_addr;
 bzero(&server_addr,sizeof(server_addr));
 server_addr.sin_family = AF_INET;
 server_addr.sin_addr.s_addr = htons(INADDR_ANY);
 server_addr.sin_port = htons(HELLO_WORLD_SERVER_PORT);

 int server_socket = socket(AF_INET,SOCK_STREAM,0);
 if( server_socket < 0){
        printf("Create Socket Failed!");
        exit(1);
 }

if( bind(server_socket,(struct sockaddr*)&server_addr,sizeof(server_addr))){
        printf("Server Bind Port : %d Failed!", HELLO_WORLD_SERVER_PORT);
        exit(1);
}

if ( listen(server_socket, LENGTH_OF_LISTEN_QUEUE) ){
        printf("Server Listen Failed!");
        exit(1);
}

//prepare for forking children
struct sockaddr_in client_addr;
socklen_t length = sizeof(client_addr);
int procnt;
int fds;
fd_set  readmask;
int newsock;
int fd = 0;

for (procnt=0; procnt<10; procnt++){
 int child_process_pid = fork(); 

 if(child_process_pid == 0 ){
   fprintf(stderr,"Child %d is born.\n",getpid());

    while(1){
     FD_ZERO(&readmask);
     FD_SET(server_socket, &readmask);

     struct timeval timeout;
     timeout.tv_sec=30;
     timeout.tv_usec=0;
     
     int ssocket=server_socket+1;
     fds = select( ssocket, &readmask, NULL, NULL,&timeout);

     /* timeout */
     if (fds == 0){
       sleep(5);
       continue;
     }

     if (FD_ISSET(server_socket, &readmask)){
       fd =server_socket;
     }
     memset( &client_addr, 0, sizeof(client_addr));

     newsock = accept(fd, (struct sockaddr *)&client_addr, &length);

     char buffer[BUFFER_SIZE];
     bzero(buffer, BUFFER_SIZE);
     strcpy(buffer,"Hello,World! FromServer! ");
     int pd;
     pd=getpid();
     char cpd[10];
     sprintf(cpd,"%d",pd);
     strcat(buffer,cpd);
     strcat(buffer,"\n");

     send(newsock,buffer,BUFFER_SIZE,0);
            
     bzero(buffer,BUFFER_SIZE);
     length = recv(newsock,buffer,BUFFER_SIZE,0);

     if (length < 0){
                printf("Server Recieve Data Failed!\n");
                exit(1);
       }

       fprintf(stderr,"got: %s in %d\n",buffer,getpid());
       close(newsock);
   }///while circle ends

   exit(0);///child ends.never goes here

  }else{
    //parent process, do nothing
  }
} /// for circle of forking ends

for(;;){
  sleep(10);
  fprintf(stderr,"I am main process waked from sleep.\n");
}

close(server_socket);
return 0;
}
复制代码
client端代码:con-client.c

复制代码
#include <netinet/in.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#define HELLO_WORLD_SERVER_PORT    6666
#define BUFFER_SIZE 1024

void talk_to_server(char ** argv){
   
    struct sockaddr_in client_addr;
    bzero(&client_addr,sizeof(client_addr));
    client_addr.sin_family = AF_INET;
    client_addr.sin_addr.s_addr = htons(INADDR_ANY);
    client_addr.sin_port = htons(0);

    int client_socket = socket(AF_INET,SOCK_STREAM,0);
    if( client_socket < 0){
        printf("Create Socket Failed!\n");
        exit(1);
    }
    
    if( bind(client_socket,(struct sockaddr*)&client_addr,sizeof(client_addr))){
        printf("Client Bind Port Failed!\n");
        exit(1);
    }

    struct sockaddr_in server_addr;
    bzero(&server_addr,sizeof(server_addr));
    server_addr.sin_family = AF_INET;

    if(inet_aton(argv[1],&server_addr.sin_addr) == 0){
        printf("Server IP Address Error!\n");
        exit(1);
    }
    server_addr.sin_port = htons(HELLO_WORLD_SERVER_PORT);
    socklen_t server_addr_length = sizeof(server_addr);

    if( connect( client_socket, (struct sockaddr*)&server_addr, 
          server_addr_length) < 0){
        printf("Can Not Connect To %s!\n",argv[1]);
        exit(1);
    }

    char buffer[BUFFER_SIZE];
    bzero(buffer,BUFFER_SIZE);

    int length = recv(  client_socket,  buffer,BUFFER_SIZE,0);
    if(length < 0){
        printf("Recieve Data From Server %s Failed!\n", argv[1]);
        exit(1);
    }
    printf("From Server %s :\t%s",argv[1],buffer);

    bzero(buffer,BUFFER_SIZE);
    char name[64];
    gethostname(name,sizeof(name));
    strcpy(buffer,name);
    send(client_socket,buffer,BUFFER_SIZE,0);
    close(client_socket);
}

int main(int argc, char **argv){
    if (argc != 2){
        printf("Usage: ./%s ServerIPAddress\n",argv[0]);
        exit(1);
    }

    int i=0;
    for(i=0; i<10000; i++){
        talk_to_server(argv);
        sleep(10);
    }
    return 0;
}
复制代码
基本上来说,server端的实现方式,就是:

监听某个port, listen后,开始建立若干的子进程。在父进程里进入sleep/循环;

在子进程里面,select/accept ,accept后作动作,然后循环往复。

好像酒店大堂的门口,开了门纳客(监听);

然后派一批员工(子进程)等在门口(listen/accept),

若有客人来(网络请求),哪个员工(子进程)逮住了就给迎进酒店,领位(服务),然后循环往复。



本文转自健哥的数据花园博客园博客,原文链接:http://www.cnblogs.com/gaojian/archive/2012/08/06/2624979.html,如需转载请自行联系原作者
目录
相关文章
|
消息中间件 负载均衡 监控
skynet框架:批量服务管理方案
【10月更文挑战第7天】Skynet 框架下的批量服务管理方案通过高效、可靠及可扩展的设计,实现了对大量服务实例的集中管理。该方案涵盖服务注册与发现、消息队列、负载均衡及集群管理等关键技术,确保系统稳定运行并提升响应速度。其优势在于提高系统性能、增强可靠性、便于扩展及简化管理操作,适用于大规模分布式系统、微服务架构、实时数据处理及游戏服务器等多种场景。通过定制化优化,可充分发挥 Skynet 框架的优势,构建高效稳定的分布式系统。
716 6
|
Linux 数据处理
Linux命令`install`详解:不仅仅是安装工具
`install`命令在Linux中并非仅用于安装软件,而是用于精确复制文件和目录,设置权限及所有权。它能创建目标目录、处理符号链接并保留时间戳。例如,`install -m 644 source.txt /dest`用于复制文件并设置权限,`install -d -m 755 /dest/dir`创建目录。使用时要注意权限设置,避免误操作,并记录命令以备恢复。
|
Kubernetes 安全 网络协议
【kubernetes】二进制文件方式安装 Kubernetes 集群(二)
【kubernetes】二进制文件方式安装 Kubernetes 集群(二)
685 0
【kubernetes】二进制文件方式安装 Kubernetes 集群(二)
|
SQL 运维 算法
DTCC 2020 | 阿里云梁高中:DAS之基于Workload的全局自动优化实践
第十一届中国数据库技术大会(DTCC2020),在北京隆重召开。在12.23日性能优化与SQL审计专场上,邀请了阿里巴巴数据库技术团队高级技术专家梁高中为大家介绍DAS之基于Workload的全局自动优化实践。 SQL自动优化是阿里云数据库自治服务重要自治场景之一,该服务支撑阿里巴巴集团全网慢SQL的自动优化,目前已累计自动优化超4900万慢SQL。阿里在构建这一能力过程中有经验也有教训,期望从基于Workload的全局优化能力构建历程、智能化自动优化闭环实践两个方面和大家分享。
4935 2
DTCC 2020 | 阿里云梁高中:DAS之基于Workload的全局自动优化实践
|
安全 Java
为什么直接通过 @Autowired 注入的 HttpServletRequest 没有线程安全的问题
我们在各个地方注入依赖时,大多数情况下都是单例的。为什么直接通过 @Autowired 注入的 HttpServletRequest 没有线程安全的问题呢?带着这个问题我做了如下笔记。
502 0
|
存储 安全 Java
网络安全-反序列化漏洞简介、攻击与防御
网络安全-反序列化漏洞简介、攻击与防御
685 0
网络安全-反序列化漏洞简介、攻击与防御
|
Java Linux C语言
java中的synchronized和linux系统的futex到底什么个关系?
首先,futex不是个完整的锁,它是“支持实现userspace的锁的building block“。也就是说,如果你想实现一个mutex,但不想把整个mutex都弄到内核里面去,可以通过futex来实现。但futex本身主要就是俩系统调用futex_wait和futex_wake. 为了更好的解释这个问题,这里先梳理下锁本身是怎么工作的。
java中的synchronized和linux系统的futex到底什么个关系?
|
JavaScript 前端开发 大数据
ajax请求总是不成功?浏览器的同源策略和跨域问题详解
XMLHttpRequest cannot load http://oldwang.com/isdad. No 'Access-Control-Allow-Origin' header is present on the requested resource. Origin 'http://xiao
30972 1
|
关系型数据库 数据库连接 数据库
循序渐进丨MogDB 中 gs_dump 数据库导出工具源码概览
通过这种循序渐进的方式,您可以深入理解 `gs_dump` 的实现,并根据需要进行定制和优化。这不仅有助于提升数据库管理的效率,还能为数据迁移和备份提供可靠的保障。
400 6

热门文章

最新文章

下一篇
开通oss服务