网络服务器处理阻塞问题

最近客户采用采用多个进程来同时对snmp服务器来set/get  ,有时候会造成网络服务器 net-snmp 的阻塞而无服务器反馈信息

为了改善,自己写了一个简单服务器,为了提高处理效果,创建子进程来处理,并由父进程来回收

代码如下:

头文件:

socket_include.h

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>

  

server.c

#include <time.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <termios.h>
#include <string.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <sys/shm.h>
#include "socket_include.h"

#define MAXLINE 80
#define SERV_PORT 1234

struct serialdata{
	int cur[10][24];
	int mincur[10][24];
	int maxcur[10][24];
	int minTcur[10][3];
	int maxTcur[10][3];
	int minTvol[10][3];
	int maxTvol[10][3];
	int minTH[10][8];
	int maxTH[10][8];
	int Tcur[10][3];
	int Tvol[10][3];
	int temp[10][4];
	int hum[10][4];
	int water[10];
	int smoke[10];
	int door[10][2];
	int kwh[10][24];
	int Tpowerfactor[10][3];
	int Tkwh[10][3];
	int swicth[10][24];
	int outputnum[10];
	int online[10];
	int type[10];
	int m_s;
	int alramflag[10][24+3+3+9];//output cur vol th
	////////////////////////////////////////////////
	int numperswicth[6];
	int ondelaytime;
	int offdelaytime;
	char slavename[10][32];
	char outputname[10][24][32];

};

struct serialdata *memdata;
struct serialdata *shmdata()
{
	key_t shmkey;
	int shmid;
	shmkey=ftok("/",‘b‘);
	shmid=shmget(shmkey,sizeof(struct serialdata)*2,0666|IPC_CREAT);
	if(shmid==-1)
		return NULL;
	struct serialdata * addr;
		addr=(struct serialdata*)shmat(shmid,0,0);
	if(addr==(struct serialdata*)-1)
		return NULL;
	return addr;
}

void handler(int num) {
    //我接受到了SIGCHLD的信号啦
    int status;
    int pid = waitpid(-1, &status, WNOHANG);
    if (WIFEXITED(status)) {
        printf("The child %d exit with code %d\n", pid, WEXITSTATUS(status));
    }
}   

int main()
{
	memdata = NULL;
	memdata = shmdata();
	if(memdata==NULL)
	{
		return -1;
	}
	//网络服务

	pid_t pid;
	struct sockaddr_in servaddr, cliaddr;
    socklen_t cliaddr_len;
    int listenfd, connfd;
    char buf[MAXLINE];
    char str[INET_ADDRSTRLEN];
    int i, n;
	int index=0;
	int outindex=0;

	char cmd1[256];
	char show[255];

    listenfd = socket(AF_INET, SOCK_STREAM, 0);

    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port = htons(SERV_PORT);

    bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr));

    listen(listenfd, 20);

    printf("Accepting connections ...\n");
    while (1) {
        cliaddr_len = sizeof(cliaddr);
        connfd = accept(listenfd,
                (struct sockaddr *)&cliaddr, &cliaddr_len);

        n = read(connfd, buf, MAXLINE);//读取网络数据

		buf[n]=‘\0‘;
        printf("received from %s at PORT %d,buf:%s\n",
         inet_ntop(AF_INET, &cliaddr.sin_addr, str, sizeof(str)),
         ntohs(cliaddr.sin_port),buf);

		signal(SIGCHLD, handler); 

		pid=fork();
        if(pid<0)
        {

              perror("fork error");

        }
        else if(pid==0)//子进程处理数据
        {

		for(index=0;index<10;index++)
		{
			for(outindex=0;outindex<24;outindex++)
				{
						memset(cmd1,0,sizeof(char)*255);
						sprintf(cmd1,"OFF %d %d",index,outindex+1);
						if(strcmp(cmd1,buf)==0)
						{
							printf("is in readcmd succed!\n");
							char sworder[255];

							//关输出位
							for(n=0;n<2;n++)
							{
								memdata->swicth[index][outindex]=1;
								memset(sworder,0,sizeof(char)*255);
								printf("\n");
								sprintf(sworder,"/clever/bin/order %d %d 2 1",index,outindex+1);
								system(sworder);
								printf("the off order is dnoe.\n");
								printf("\n");
							}
							break;
						}

						memset(cmd1,0,sizeof(char)*255);
						sprintf(cmd1,"ON %d %d",index,outindex+1);
						if(strcmp(cmd1,buf)==0)
						{
							printf("is in readcmd succed!\n");

							//关输出位
							char sworder[255];
							for(n=0;n<2;n++)
							{
								memdata->swicth[index][outindex]=2;
								memset(sworder,0,sizeof(char)*255);
								printf("\n");
								sprintf(sworder,"/clever/bin/order %d %d 1 1",index,outindex+1);
								system(sworder);
								printf("the on order is dnoe.\n");
								printf("\n");
							}
							break;
						}

						memset(cmd1,0,sizeof(char)*255);
						sprintf(cmd1,"status %d %d",index,outindex+1);
						if(strcmp(cmd1,buf)==0)
						{
							memset(show,0,sizeof(char)*255);
							sprintf(show,"status:%s\n",memdata->swicth[index][outindex]==0?"--":
									(memdata->swicth[index][outindex]==1?"OFF":"ON"));
							write(connfd, show, strlen(show));
							break;
						}
				}
		}
		//处理完成

			 write(connfd, "exit sucessed!\n", strlen("exit sucessed!\n"));
			close(connfd);//短连接结束
            exit(0);
		}

/*
		for (i = 0; i < n; i++)
            buf[i] = toupper(buf[i]);
		write(connfd, buf, n);
*/
	   //write(connfd, "exit sucessed!\n", strlen("exit sucessed!\n"));
	  // close(connfd);

    }

	return 0;

}

客户端: 

#include "socket_include.h"

#define MAXLINE 80
#define SERV_PORT 1234

int main(int argc, char *argv[])
{
    struct sockaddr_in servaddr;
    char buf[MAXLINE];
    int sockfd, n;
    char *str;

    if (argc != 2) {
        fputs("usage: ./client message\n", stderr);
        exit(1);
    }
    str = argv[1];

    sockfd = socket(AF_INET, SOCK_STREAM, 0);

    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    //inet_pton(AF_INET, "127.0.0.1", &servaddr.sin_addr);
    inet_pton(AF_INET, "192.168.1.239", &servaddr.sin_addr);
    servaddr.sin_port = htons(SERV_PORT);

    connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));

    write(sockfd, str, strlen(str));

    n = read(sockfd, buf, MAXLINE);
	buf[n]=‘\0‘;
    printf("Response from server:%s\n",buf);

    //write(STDOUT_FILENO, buf, n);

    close(sockfd);
    return 0;
}

  

测试程序:

#include <stdio.h>

int main()
{
	int i=0;
	for(i=0;i<100;i++)
	{
		printf("./client ‘ON 0 9‘\n");
		system("./client ‘ON 0 9‘");
		sleep(1);

		system("./client ‘status 0 9‘");
		sleep(1);

		printf("./client ‘OFF 0 9‘\n");
		system("./client ‘OFF 0 9‘");
		sleep(1);

		system("./client ‘status 0 9‘");
		sleep(1);

	}

	return 0;
}

  

时间: 2024-08-01 15:45:35

网络服务器处理阻塞问题的相关文章

网络服务器架构模型与比较

事件驱动为广大的程序员所熟悉,其最为人津津乐道的是在图形化界面编程中的应用:事实上,在网络编程中事件驱动也被广泛使用,并大规模部署在高连接数高吞吐量的服务器程序中,如 http 服务器程序.ftp 服务器程序等.相比于传统的网络编程方式,事件驱动能够极大的降低资源占用,增大服务接待能力,并提高网络传输效率. 关于本文提及的服务器模型,搜索网络可以查阅到很多的实现代码,所以,本文将不拘泥于源代码的陈列与分析,而侧重模型的介绍和比较.使用 libev 事件驱动库的服务器模型将给出实现代码. 本文涉及

几种经典的网络服务器架构模型的分析与比较

原文出处:http://blog.csdn.net/lmh12506/article/details/7753978 前言 事件驱动为广大的程序员所熟悉,其最为人津津乐道的是在图形化界面编程中的应用:事实上,在网络编程中事件驱动也被广泛使用,并大规模部署在高连接数高吞吐量的服务器程序中,如 http 服务器程序.ftp 服务器程序等.相比于传统的网络编程方式,事件驱动能够极大的降低资源占用,增大服务接待能力,并提高网络传输效率. 关于本文提及的服务器模型,搜索网络可以查阅到很多的实现代码,所以,

使用事件驱动模型实现高效稳定的网络服务器程序

使用事件驱动模型实现高效稳定的网络服务器程序 几种网络服务器模型的介绍与比较 围绕如何构建一个高效稳定的网络服务器程序,本文从一个最简单的服务器模型开始,依次介绍了使用多线程的服务器模型.使用非阻塞接口的服务器模型.利用select()接口实现的基于事件驱动的服务器模型,和使用libev事件驱动库的服务器模型.通过比较各个模型,得出事件驱动模型更适合构建高效稳定的网络服务器程序的结论. 前言 事件驱动为广大的程序员所熟悉,其最为人津津乐道的是在图形化界面编程中的应用:事实上,在网络编程中事件驱动

自己动手开发网络服务器(一)

这个读书笔记是学习Let's Build A Web Server系列.原文地址: https://ruslanspivak.com/lsbaws-part1/ 包含3个部分  python有很多web框架,django,flask,tornodo,web.py.我们可以基于这些框架来开发我们的网站.这些框架其实是给我们封装了很多底层的实现.比如WSGI,模板映射等功能.为了更好的理解和开发web服务器.我们必须了解这些实现的细节和原理,这样才能更好的理解和优化 作者还专门通过写了一个小故事来说

网络IO之阻塞、非阻塞、同步、异步总结

1.前言 在网络编程中,阻塞.非阻塞.同步.异步经常被提到.unix网络编程第一卷第六章专门讨论五种不同的IO模型,Stevens讲的非常详细,我记得去年看第一遍时候,似懂非懂,没有深入理解.网上有详细的分析:http://blog.csdn.net/historyasamirror/article/details/5778378.我结合网上博客和书总结一下,加以区别,加深理解. 2.数据流向 网络IO操作实际过程涉及到内核和调用这个IO操作的进程.以read为例,read的具体操作分为以下两个

使用Python SocketServer快速实现多线程网络服务器

Python SocketServer使用介绍 1.简介:          SocketServer是python的一个网络服务器框架,可以减少开发人员编写网络服务器程序的工作量. SocketServer总共有4个server基类. TCPServer:负责处理TCP协议. UDPServer:负责处理UDP协议. UnixStreamServer:只适用于类unix平台,不常用. UnixDatagramServer:只适用于类unix平台,不常用. 这4个类会同步处理每一个request

中小型企业网络服务器建设

一.网络拓扑图 中小型企业网络服务器建设

Python3-socketserver模块-网络服务器框架

Python3中的socketserver模块简化了编写网络服务器的任务 在实际的开发中,特别是多并发的情况下,socket模块显然对我们的用处不大,因为如果你要通过socket模块来实现并发的socket通信,简直太麻烦了,socketserver模块则是Python提供给你的现成的接口,你只需要编写少量的代码,就可以实现你的需求 首先,您必须通过对BaseRequestHandler类进行子类化并覆盖其handle()方法来创建请求处理程序类:此方法将处理传入请求.其次,您必须实例化一个服务

租用网络服务器怎么去选择

怎么租用网络服务器,租用网络服务器找谁?租用网络服务器,不选最贵的,只选最对的.租用方式灵活多样,各种解决方案 应有尽有,当然如果你的需求不是特殊的,一般选择常规的就完全可以满足,并且价格方面也是比较适中的,除了租用外,如果发展稳定了,也可以考虑自己买服务 器放到机房去托管,这也就叫做服务器托管. 怎么租用网络服务器流程 怎么租用网络服务器 1.选择一家正规的服务器机房提供商(可以让熟人推荐,或者自己从网络上搜索寻找筛查). 2.与服务器机房提供商进行沟通,主要是从机房的服务,服务器的配置,带宽