linux c 编写服务应用,并发测试时出现close wait

wangyou_1987 2018-09-20 10:33:41
并发测试服务程序时,出现close wait 后续交易就接收不了。已经在子进程退出时已经close了,怎么还会有closewait。请大神帮忙看一下。万分感谢。
int tcp_server2(char *hostip, int port)
{
int sockfd,lis_sockfd,clilen,childpid;
int ret = 0;
int optval = 1;
int dbopen = 0;

int s_idle_val = 15; /* 15 秒开始测试 */
int s_idle_len = sizeof(int);
int s_intvl_val = 5; /* 5 秒测一次 */
int s_intvl_len = sizeof(int);

struct sockaddr_in cli_addr,serv_addr;
fd_set sockfds;
struct timeval tcptime;
char address[128];
memset(address, 0, sizeof(address));

setpgrp ();
signal (SIGINT,SIG_IGN);
signal (SIGCLD,SIG_IGN);
if ((lis_sockfd = socket (AF_INET,SOCK_STREAM,0)) < 0)
{
return (-1);
}

memset ((char *)&serv_addr,0,sizeof (serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = htonl (INADDR_ANY);
serv_addr.sin_port = htons (port);
setsockopt(lis_sockfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(int));
if (bind (lis_sockfd, (struct sockaddr *)&serv_addr,sizeof (serv_addr )) < 0)
{
logmsg(LOGERR, __FILE__, __LINE__, "Net_Server: can't bind local address Error! errno=%d", errno);
return (-1);
}

listen (lis_sockfd, 500);
// dbOpen2("CATERINGORA","CATERINGORA", "");
for (;;) {

clilen = sizeof (cli_addr);
sockfd = accept (lis_sockfd, (struct sockaddr *)&cli_addr, &clilen);
if (sockfd < 0 ) {
logmsg(LOGERR,__FILE__,__LINE__,"Net_Server: can't accept connect errno=%d", errno);

continue;
}
logmsg(LOGERR, __FILE__, __LINE__, "accetpsuccess");

int val = 1;
if (setsockopt(sockfd, SOL_SOCKET, SO_KEEPALIVE, &val, sizeof(val))<0)
{
close(sockfd);
logmsg(LOGERR, __FILE__, __LINE__,"setsockopt():[SO_KEEPALIVE] Error! errno=%d", errno);
continue;
}

if (setsockopt(sockfd,IPPROTO_TCP,TCP_KEEPIDLE, (char *)&s_idle_val,s_idle_len)<0)
{
close(sockfd);
logmsg(LOGERR, __FILE__, __LINE__, "setsockopt():[TCP_KEEPIDLE] Error! errno=%d", errno);
continue;
}

if ( setsockopt(sockfd,IPPROTO_TCP,TCP_KEEPINTVL, (char *)&s_intvl_val,s_intvl_len) < 0 )
{
close(sockfd);
logmsg(LOGERR,__FILE__, __LINE__,"setsockopt:[TCP_KEEPINTVL] Error! errno=%d", errno);
continue;
}
logmsg(LOGERR, __FILE__, __LINE__, "gorkstart");
//if (concurrent_flag1 == 1)
{
if ((childpid = fork ()) < 0)
{

logmsg(LOGERR,__FILE__,__LINE__,"Net_Server: can't fork errno=%d", errno);
shutdown(sockfd, 2);
close(sockfd);
continue;
}
else if (childpid == 0)
{ /* Child process */
close(lis_sockfd);
logmsg(LOGERR, __FILE__, __LINE__, "子进程【%d]",getpid());
tcppro(sockfd,address, &dbopen);
usleep(500);
shutdown(sockfd, 2);
close(sockfd);
logmsg(LOGERR, __FILE__, __LINE__, "关闭socket");
exit (0);
}
}
close (sockfd);
}//while(1)
return 0;
}

int
tcppro(int sockfd, char *cliaddr, int* dbopen)
{
int ret, checktel, len;

unsigned char posbuf[PACKAGE_BUF_LEN] = {0};
unsigned char posbuf1[PACKAGE_BUF_LEN] = {0};
unsigned char posbuf2[PACKAGE_BUF_LEN] = {0x00,0x6E,0x60,0x00,0x00,0x00,0x03,0x60,0x32,0x00,0xF1,0x80,0x62,0x02,0x10,0x20,0x38,0x00,0x80,0x02,0xC0,0x20,0x10,0x70,0x01,0x06,0x00,0x00,0x16,0x11,0x52,0x14,0x09,0x07,0x00,0x41,0x30,0x30,0x30,0x30,0x30,0x30,0x30,0x30,0x31,0x30,0x30,0x30,0x30,0x30,0x30,0x30,0x30,0x37,0x34,0x34,0x30,0x30,0x30,0x33,0x00,0x42,0x5A,0x5A,0x41,0x41,0x30,0x30,0x31,0x33,0x41,0x42,0x30,0x31,0x35,0x36,0x32,0x31,0x33,0x30,0x31,0x30,0x35,0x32,0x33,0x31,0x30,0x30,0x30,0x31,0x41,0x46,0x30,0x30,0x38,0x32,0x30,0x31,0x38,0x30,0x39,0x30,0x37,0x23,0x00,0x11,0x00,0x00,0x00,0x01,0x00,0x40};

Hq_Store storeinf;

char soure[4] = {0};

signal(SIGCLD, SIG_IGN);
signal(SIGCHLD, SIG_IGN);
signal(SIGPIPE, SIG_IGN);
//while(1)
{

memset(posbuf, 0, sizeof(posbuf));
memset(posbuf1, 0, sizeof(posbuf1));

ret = crdrcv(sockfd, posbuf);


if(ret != 0)
{
logmsg(LOGDEBUG, __FILE__, __LINE__, "received message from front: failed!!");
//dbClose2();
return 0;
}
#if 10



if(*dbopen == 0)
{

logmsg(LOGERR, __FILE__, __LINE__, "dbuser:[%s] dbpwd:[%s] dbsrv:[%s]", dbuser, dbpwd, dbsrv);
ret = dbOpen2(dbuser, dbpwd, dbsrv);

if(ret)
{
logmsg(LOGERR, __FILE__, __LINE__, "打开数据库错");
return 0;
}
*dbopen =1;
}




checktel = 0;
len = 0;

memset(&storeinf, 0, sizeof(Hq_Store));

ret = DB_hq_store(soure, "0", &storeinf);


ret = pospro(posbuf, checktel, &storeinf);

if(ret < 0)
{
logmsg(LOGERR, __FILE__, __LINE__, "vou_pro handle failed !!\n");
return 0;
}
#endif
ret = crdsnd(sockfd, posbuf2);

if(ret != 0)
{
logmsg(LOGERR, __FILE__, __LINE__, "send respond message failed \n");
return 0;
}
}

return 0;

}
...全文
391 回复 打赏 收藏 转发到动态 举报
写回复
用AI写文章
回复
切换为时间正序
请发表友善的回复…
发表回复
这是一门linux下c++通讯架构实战课程,针对c/c++语言已经掌握的很熟并希望进一步深造以将来用c++在linux下从事网络通讯领域/网络服务器的开发和架构工作。这门课程学习难度颇高但也有着极其优渥的薪水(最少30K月薪,最高可达60-80K月薪),这门课程,会先从nginx源码的分析和讲解开始,逐步开始书写属于自己的高性能服务器框架代码,完善个人代码库,这些,将会是您日后能取得高薪的重要筹码。本课程原计划带着大家逐行写代码,但因为代码实在过于复杂和精细,带着写代码可能会造成每节课至少要4~5小的超长间,所以老师会在课前先写好代码,主要的间花费在逐行讲解这些代码上,这一点望同学们周知。如果你觉得非要老师领着写代码才行的话,老师会觉得你当前可能学习本门课程会比较吃力,请不要购买本课程,以免听不懂课程并给老师差评,差评也会非常影响老师课程的销售并造成其他同学的误解。 这门课程要求您具备下面的技能:(1)对c/c++语言掌握的非常熟练,语言本身已经不是继续学习的障碍,并不要求您一定熟悉网络或者linux;(2)对网络通讯架构领域有兴趣、勇于挑战这个高难度的开发领域并期望用大量的付出换取高薪;在这门课程中,实现了一个完整的项目,其中包括通讯框架和业务逻辑框架,浓缩总结起来包括如下几点:(1)项目本身是一个极完整的多线程高并发服务器程序;(2)按照包头包体格式正确的接收客户端发送过来的数据包, 完美解决收包的数据粘包问题;(3)根据收到的包的不同来执行不同的业务处理逻辑;(4)把业务处理产生的结果数据包正确返回给客户端;本项目用到的主要开发技术和特色包括:(1)epoll高并发通讯技术,用到的触发模式是epoll中的水平触发模式【LT】;(2)自己写了一套线程池来处理业务逻辑,调用适当的业务逻辑处理函数处理业务并返回给客户端处理结果;(3)线程之间的同步技术包括互斥量,信号量等等;(4)连接池中连接的延迟回收技术,这是整个项目中的精华技术,极大程度上消除诸多导致服务器程序工作不稳定的因素;(5)专门处理数据发送的一整套数据发送逻辑以及对应的发送线程;(6)其他次要技术,包括信号、日志打印、fork()子进程、守护进程等等;

23,126

社区成员

发帖
与我相关
我的任务
社区描述
Linux/Unix社区 应用程序开发区
社区管理员
  • 应用程序开发区社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧