linux基础编程 套接字socket 完整的服务器端多线程socket程序【转】
转自:http://blog.csdn.net/ghostyu/article/details/7737203
此段程序来自我的一个项目中,稍微做了些修改,运行稳定,客户端程序比较简单所以未编写,可以用socket调试工具测试此段代码 费话不多说,直接上代码 [cpp] view plain copy #include<stdlib.h>
#include<stdio.h>
#include<string.h>
#include<errno.h>
#include<netdb.h>
#include<sys/types.h>
#include<netinet/in.h>
#include<sys/socket.h>
#include<sys/wait.h>
#include<unistd.h>
#include<pthread.h>
#include<fcntl.h> #define __DEBUG
#ifdef __DEBUG
#define DBG(fmt,args...) fprintf(stdout, fmt, ##args)
#else
#define DBG(fmt,args...)
#endif
#define ERR(fmt,args...) fprintf(stderr, fmt, ##args) /*
isStreamQuit == 0 :Start
isStreamQuit == 1 :Client Quit
isStreamQuit ==-1 :App Quit
*/
static int isStreamQuit = ;
static int gStreamServerFd = -;
int gStreamClientFd; #define PORT_SOCKET 9000 //server监听端口
#define RECV_BUF_MAX_LEN 512
#define TIME_WAIT 200000 /*200ms*/ /*
socket发送函数
*/
int ScketSend(unsigned char *buf,unsigned int len)
{ int ret = ;
if(gStreamClientFd <= )
return -;
ret = send(gStreamClientFd,buf,len,);
if(ret <= ){
return -;
}else if(ret == len){
return ;
}
while(ret < len){
ret += send(gStreamClientFd,buf+ret,len-ret,);
}
return ;
} /*
socket连接线程,阻塞等待客户端的连接,
连接后循环处理客户端发来的数据,
这里只能连接一个客户端,稍加修改,就可以支持多客户端连接
*/
void *SocketThread(void *arg)
{
fd_set SockRead;
struct timeval TimeOut;
int ret;
struct sockaddr clientAddr;
unsigned char *rcvBuf = (unsigned char *)malloc(RECV_BUF_MAX_LEN); socklen_t len;
len = sizeof(clientAddr);
while(){ int rcvLen;
/*阻塞等待客户端连接*/
gStreamClientFd = accept(gStreamServerFd, &clientAddr, &len);
if(gStreamClientFd == -){
ERR("ERROR in stream accept: %s\n",strerror(errno));
/*add errno handler here,in some case,the error is not fatal*/
/*code here*/
break;
}
/*isStreamQuit == -1,is the siganl to quit*/
if(isStreamQuit != -){
isStreamQuit = ;
}
DBG("accep a stream client\n");
/*客户端已连接,循环处理*/
while(isStreamQuit==){
FD_ZERO(&SockRead);
FD_SET(gStreamClientFd,&SockRead);
TimeOut.tv_sec = ;
TimeOut.tv_usec = TIME_WAIT;
/*
200ms超时判断是否有数据接收,有则接收,没有继续循环,
总是让程序不能一直阻塞在这里
*/
ret = select(gStreamClientFd+,&SockRead,NULL,NULL,&TimeOut);
if(ret == -){
ERR("ERROR in select\n");
break;
}else if(ret == ){
continue;
}
/*接收客户端发来的数据*/
rcvLen = recv(gStreamClientFd,rcvBuf,RECV_BUF_MAX_LEN,);
if(rcvLen < ){
ERR("ERROR in stream accept: %s\n",strerror(errno));
/*add errno handler here,in some case,the error is not fatal*/
/*code here*/
isStreamQuit = ;
break;
}else if(rcvLen == ){
DBG("stream client close\n");
/*code here*/
isStreamQuit = ;
break;
}
DBG("rcvLen: %d,rcvBuf: %s\n",rcvLen,rcvBuf); memset(rcvBuf,,RECV_BUF_MAX_LEN);
}
DBG("Close Stream Socket\n");
close(gStreamClientFd); /*OTHER Error or App Quit*/
if(isStreamQuit == -){
break;
} }
DBG("SocketThread exit\n");
free(rcvBuf);
close(gStreamServerFd);
pthread_exit(NULL);
} /*
socket连接初始化
*/
int SocketInit()
{
struct sockaddr_in server;
pthread_t tStreamSocketThr;
/*创建socket*/
gStreamServerFd = socket(AF_INET,SOCK_STREAM,);
if(gStreamServerFd == - ){
ERR("ERROR in create Stream Socket!\n");
return -;
}
memset(&server,,sizeof(server));
server.sin_family = AF_INET;
server.sin_port = htons(PORT_SOCKET);
server.sin_addr.s_addr = htonl(INADDR_ANY);
/*绑定监听端口*/
if(bind(gStreamServerFd ,(struct sockaddr*)&server,sizeof(server))== -){
ERR("ERROR in bind Stream Socket: %s\n",strerror(errno));
return -;
}
/*开始监听*/
if( listen(gStreamServerFd, ) == -){
ERR("ERROR in listen Stream Socket: %s\n",strerror(errno));
return -;
}
/*创建socket连接线程,用来处理监听的结果,比如有客户端连接等*/
pthread_create(&tStreamSocketThr,NULL,SocketThread,NULL);
/*release thread resource,so we not use pthread_join to waitting*/
pthread_detach(tStreamSocketThr);
return ;
} #if 1
int main()
{
if(SocketInit() == -){
return -;
}
while(){
sleep();
} /*notice to exit*/
isStreamQuit = -;
sleep();
if(gStreamServerFd){
close(gStreamServerFd);
}
DBG("SUCCESS in ConnectManageExit\n");
}
#endif 运行结果 [cpp] view plain copy [root@localhost src]# ./socket_connect
accep a stream client
rcvLen: ,rcvBuf: helle
Close Stream Socket
accep a stream client
rcvLen: ,rcvBuf: hello
Close Stream Socket
accep a stream client
rcvLen: ,rcvBuf: hello
rcvLen: ,rcvBuf: hello
rcvLen: ,rcvBuf: helle
Close Stream Socket
linux基础编程 套接字socket 完整的服务器端多线程socket程序【转】的更多相关文章
- python基础(29):网络编程(软件开发架构、网络基础、套接字初使用)
1. 软件开发架构 我们了解的程序之间通讯的应用可分为两种: 第一种是应用类:qq.微信.百度网盘.腾讯视频这一类是属于需要安装的桌面应用. 第二种是web类:比如百度.知乎.博客园等使用浏览器访问就 ...
- linux基础编程:IO模型:阻塞/非阻塞/IO复用 同步/异步 Select/Epoll/AIO(转载)
IO概念 Linux的内核将所有外部设备都可以看做一个文件来操作.那么我们对与外部设备的操作都可以看做对文件进行操作.我们对一个文件的读写,都通过调用内核提供的系统调用:内核给我们返回一个file ...
- UNIX网络编程——套接字选项(SO_REUSEADDR)
1.一般来说,一个端口释放后会等待两分钟之后才能再被使用,SO_REUSEADDR是让端口释放后立即就可以被再次使用. SO_REUSEADDR用于对TCP套接字处于TIME_WAIT状态下的sock ...
- 网络编程 套接字socket TCP UDP
网络编程与套接字 网络编程 网络编程是什么: 网络通常指的是计算机中的互联网,是由多台计算机通过网线或其他媒介相互链接组成的 编写基于网络的应用程序的过程序称之为网络编程. 网络编程最主要的工 ...
- Linux编程---套接字
网络相关的东西差点儿都是建立在套接字之上.所以这个内容对于程序猿来说还是蛮重要的啊. 事实上套接字也就是一个特殊的设备文件而已,我始终不能明确为什么要叫套接字.这么个奇怪的名字.只是还是就这样算了吧. ...
- [linux basic基础]----套接字
套接字是一种通信机制,凭借这种机制client/server系统的开发者既可以在本地机器上进行,也可以跨网络进行. 1,服务器应用程序用系统调用socket来创建一个套接字,他是系统分配给服务器进程的 ...
- Java网络编程--套接字Socket
一.套接字Socket IP地址标志Internet上的计算机,端口号标志正在计算机上运行的进程(程序). 端口号被规定为一个16位的0--65535之间的整数,其中,0--1023被预先定义的服务通 ...
- 流式套接字:基于TCP协议的Socket网络编程(案例3)
案例:在案例1的基础上将传输的字符串换成具体的对象. 客户端代码: package com.yh.SocketObject; import java.io.IOException; import ja ...
- UNIX网络编程——套接字选项(心跳检测、绑定地址复用)
/* 设置套接字选项周期性消息检测连通性 心跳包. 心博.主要用于长连接. * 参数:套接字, 1或0开启, 首次间隔时间, 两次间隔时间, 断开次数 */ void setKeepAlive( in ...
随机推荐
- JS运行在服务器端注意事项
<script runat="server" language="javascript"> </script> 1. ASP利于JS重载 ...
- 《python机器学习—预测分析核心算法》:构建预测模型的一般流程
参见原书1.5节 构建预测模型的一般流程 问题的日常语言表述->问题的数学语言重述重述问题.提取特征.训练算法.评估算法 熟悉不同算法的输入数据结构:1.提取或组合预测所需的特征2.设定训练目标 ...
- annoy安装
yum install gcc-c++ #linux下需安装c++编译器 sudo pip install annoy
- Win7系统下删除文件时出现“正在准备再循环”的解决方法
今天,笔者在备份文件的时候,将一个word文档从移动硬盘复制到桌面.经过一系列“复(meng)杂(bi)”的操作之后,笔者突然发现,文件无法删除了.当右键文件点击“删除”时,出现对话框显示“正在准备 ...
- Leetcode 684.冗余连接
冗余连接 在本问题中, 树指的是一个连通且无环的无向图. 输入一个图,该图由一个有着N个节点 (节点值不重复1, 2, ..., N) 的树及一条附加的边构成.附加的边的两个顶点包含在1到N中间,这条 ...
- BZOJ 4592 SHOI2015 脑洞治疗仪 线段树
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=4592 题意概述:需要维护一个01序列A,一开始A全部都是1.支持如下操作: 1.将区间[l ...
- 有向图的强连通分量——kosaraju算法
一.前人种树 博客:Kosaraju算法解析: 求解图的强连通分量
- 并查集——poj2492(带权并查集入门)
一.题目回顾 题目链接:传送门 题意:给定n只虫子,不同性别的可以在一起,相同性别的不能在一起.给你m对虫子,判断中间有没有同性别在一起的. 二.解题思路 种类并查集 和poj1073的本质一样 详见 ...
- Reactor模式是什么(转载)
一.Reactor模式是什么反应器设计模式(Reactor pattern)是一种为处理并发服务请求,并将请求提交到一个或者多个服务处理程序的事件设计模式.当客户端请求抵达后,服务处理程序使用多路分配 ...
- 201621123033 《Java程序设计》第11周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多线程相关内容. 2. 书面作业 本次PTA作业题集多线程 1. 源代码阅读:多线程程序BounceThread 1.1 BallR ...