免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 11369 | 回复: 8
打印 上一主题 下一主题

linux c socket多线程编程 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2011-11-28 10:52 |只看该作者 |倒序浏览
我现在写一个服务端程序,要求可以链接多个客户端
有报警信息时要能发送报警信息,如果客户端有数据发来时,接收客户端的信息
现在运行起来有问题,当我打开第一个客户端时运行正常,当我断开这个链接,再重新打开一个客户端时,发送时就出现了问题,情况如下:
第一个链接
There is a new socket link coming 4!
alarmtype = 50
data from socket 4, bytes = 1212
data from socket 4, bytes = 1212
Socket link 4 disconnect!
第二个链接
There is a new socket link coming 5!
alarmtype = 50
send error
: Bad file descriptor
data from socket 5, bytes = 1212
data from socket 5, bytes = 1212
代码如下:
void *sockalarm(void *arg)
{
  int sdbytes = 0;
int cnfd = (int)arg;
int connfd;
  rtcp_hdr_t rtcp_head;
  warn_hdr_t pkt;
struct sockaddr_in cliaddr;
socklen_t cliaddr_len;
pthread_t rcvthread[pthread_count];
pthread_t sendthread[pthread_count];

  struct sockaddr_in servaddr, cliaddr;
int sockfd;
pthread_t sockthread;

pthread_mutex_init(&mutex, NULL);
pthread_cond_init(&cond, NULL);

sockfd = 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(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));
listen(sockfd, MAX_NUM);
printf("Accepting connections ...\n");
cliaddr_len = sizeof(cliaddr);
   
while(1)
{
if(pthread_count < MAX_PTHR) {
if( (connfd = accept(cnfd, (struct sockaddr *)&cliaddr, &cliaddr_len)) < 0 ) {
perror("socket accept error\n");
} else {
printf("There is a new socket link coming %d!\n", connfd);
if( pthread_create(&rcvthread[pthread_count], NULL, rcvptz, (void *)connfd) ) {
perror("rcv pthread_create\n");
}
if( pthread_create(&sendthread[pthread_count], NULL, sendalarm, (void *)connfd) ) {
perror("send pthread_create\n");
}
pthread_count += 1;
}
} else {
printf("thread count has big enough!\n");
break;
}          
}
pthread_exit(NULL);
}

void *sendalarm(void *arg)
{
int sdbytes = 0;
int cnfd = (int)arg;
rtcp_hdr_t rtcp_head;
  warn_hdr_t pkt;

while (1)
{
pthread_mutex_lock(&mutex);
pthread_cond_wait(&cond, &mutex);
memset(&pkt, 0, sizeof(warn_hdr_t));
pkt.rtcp_head.ver = 2;
pkt.rtcp_head.pt = 204;
  pkt.rtcp_head.len = 1200;
  pkt.rtcp_head.ssrc = 11;
  pkt.rtcp_head.command = 2;
pkt.head = 0xDDCC;
pkt.len = 1188;
printf("alarmtype = %d\n", alarmtype);
pkt.type = alarmtype;
  sdbytes = send(cnfd, (char *)&pkt, RTP_LEN, MSG_DONTWAIT);
if(sdbytes < 0) {
  perror("send error\n");
break;
  }
pthread_mutex_unlock(&mutex);
sleep(1);
}
close(cnfd);
pthread_exit(NULL);
}

void *rcvptz(void *arg)
{
int i;
int rcvbytes = 0;
rtcp_hdr_t rtcp_head;
  warn_hdr_t pkt;
int cnfd = (int)arg;
char sockbuf[RTP_LEN] = {0};
char urtbuff[MAX_DATA] = {0};
  char netbuff[MAX_DATA] = {0};

while (1)
{
rcvbytes = recv(cnfd, sockbuf, RTP_LEN, 0);
if(rcvbytes > 0) {
printf("data from socket %d, bytes = %d\n", cnfd, rcvbytes);
if( rcvbytes > 12 ) {
  rtcp_head = *(rtcp_hdr_t *)sockbuf;
  if((rtcp_head.ver == 2) && (rtcp_head.pt == 204) && (rtcp_head.command == 1)) {
  for(i = 0; i < (rcvbytes - 12); i++) {
  urtbuff[i] = sockbuf[12+i];
// printf("%02x ", urtbuff[i]);
  }
  } else {       
  printf("the header of rtcp pcket is wrong\n");
  }   
  }
} else if(rcvbytes == 0) {
printf("Socket link %d disconnect!\n", cnfd);
break;
} else {
printf("Socket %d \n", cnfd);
perror("rcv error!\n");
break;
}
sleep(1);
}
close(cnfd);
pthread_exit(NULL);       
}

论坛徽章:
0
2 [报告]
发表于 2011-11-30 13:47 |只看该作者
感觉咋这里提供总没有人给答复
在csdn很快就会有人来,不管答案满不满意,反正会有人答复
让人感觉很好
这个问题已解决
链接中断后,直接退出了,没有解锁

论坛徽章:
0
3 [报告]
发表于 2011-11-30 13:48 |只看该作者
感觉咋这里提供总没有人给答复
在csdn很快就会有人来,不管答案满不满意,反正会有人答复
让人感觉很好
这个问题已解决
链接中断后,直接退出了,没有解锁

论坛徽章:
0
4 [报告]
发表于 2011-12-18 22:12 |只看该作者
代码用[code][/code]括起来,看起来舒服些。可能看看还要整理下代码,时间也不是那么充裕。

论坛徽章:
0
5 [报告]
发表于 2011-12-18 22:16 |只看该作者
代码用[code][/code]括起来,看起来舒服些。可能看看还要整理下代码,别人时间也不是那么充裕。

论坛徽章:
0
6 [报告]
发表于 2012-01-05 11:23 |只看该作者
int cnfd = (int)arg;
上面的转换不对,应该是
int* cnfd = (int*)arg;
send函数引用cnfd用*cnfd

论坛徽章:
0
7 [报告]
发表于 2012-05-18 13:09 |只看该作者
本帖最后由 weifeng270 于 2012-05-18 13:11 编辑

因为这是垃圾论坛。。

看看我发的贴子。也是自己解决的。

http://bbs.chinaunix.net/thread-3743754-1-1.html



论坛徽章:
1
戌狗
日期:2014-07-17 19:24:40
8 [报告]
发表于 2012-05-21 09:46 |只看该作者
代码写好点,这个实在没法看。

论坛徽章:
6
申猴
日期:2013-10-08 17:32:32金牛座
日期:2013-10-18 19:45:53天秤座
日期:2013-10-18 20:17:34处女座
日期:2014-02-11 10:10:29丑牛
日期:2014-02-15 10:44:15巳蛇
日期:2014-02-18 22:05:54
9 [报告]
发表于 2012-06-20 19:02 |只看该作者
请问void *rcvptz(void *arg)处理中,为何没有进行加锁处理?线程资源回收也没有处理,对send/recv函数处理,如果数据量过大,该处理方式没有完全处理
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP