server

#include <stdio.h>
#include <winsock2.h>
#include <iostream> #pragma comment(lib, "WS2_32.lib") #define PORT 9999
#define DATA_BUFSIZE 8192 typedef struct _SOCKET_INFORMATION{
CHAR Buffer[DATA_BUFSIZE]; //发送和接收数据的缓冲区
WSABUF DataBuf; //定义发送和接收数据缓冲区的结构体,包括缓冲区长度和内容
SOCKET Socket; //与客户端进行通信的套接字
DWORD BytesSEND; //保存套接字发送的字节数
DWORD BytesRECV; //保存套接字接收的字节数
} SOCKET_INFOMATION, * LPSOCKET_INFORMATION; DWORD TotalSockets = ;
LPSOCKET_INFORMATION SocketArray[FD_SETSIZE]; bool CreateSocketInformation(SOCKET s)
{
LPSOCKET_INFORMATION SI; //用于保存套接字的信息
//为SI分配内存空间
if ((SI = (LPSOCKET_INFORMATION)GlobalAlloc(GPTR, sizeof SOCKET_INFOMATION)) == NULL)
{
printf("GlobalAlloc() failed with error %d\n", GetLastError());
return FALSE;
} //初始化SI的值
SI->Socket = s;
SI->BytesSEND = ;
SI->BytesRECV = ; SocketArray[TotalSockets] = SI;
TotalSockets++; return true;
} void FreeSocketInformation(DWORD Index)
{
LPSOCKET_INFORMATION SI = SocketArray[Index];
DWORD i; closesocket(SI->Socket);
GlobalFree(SI);
if (Index != (TotalSockets - ))
{
for (i = Index; i < TotalSockets; i++)
{
SocketArray[i] = SocketArray[i + ];
}
} TotalSockets--;
} int main()
{
SOCKET ListenSocket; //监听套接字
SOCKET AcceptSocket; //与客户端通信的套接字
SOCKADDR_IN InternetAddr; //服务器地址
WSADATA wsaData;
int Ret; FD_SET WriteSet; //获取可写套接字集合
FD_SET ReadSet; //获取可读性套接字集合
DWORD Total = ; //处于就绪状态的套接字
DWORD SendBytes; //发送套接字
DWORD RecvBytes; //接收的套接字 //初始化WinSock环境
if ((Ret = WSAStartup(MAKEWORD(, ), &wsaData) != ))
{
printf("WSAStartup() failed with error %d\n", Ret);
WSACleanup();
return -;
} //创建用于监听的套接字
if ((ListenSocket = WSASocket(AF_INET, SOCK_STREAM, IPPROTO_TCP, NULL, , WSA_FLAG_OVERLAPPED)) == INVALID_SOCKET)
{
printf("WSASocket() failed with error %d\n", WSAGetLastError());
return -;
} //设置监听地址和端口
InternetAddr.sin_family = AF_INET;
InternetAddr.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
InternetAddr.sin_port = htons(PORT); //绑定监听套接字到本地地址和端口
if (bind(ListenSocket, (PSOCKADDR)&InternetAddr, sizeof InternetAddr) == INVALID_SOCKET)
{
printf("bind() failed with error %d\n", WSAGetLastError());
return -;
} //开始监听
if (listen(ListenSocket, ))
{
printf("listen() failed with error %d\n", WSAGetLastError());
return -;
} //设置成非阻塞方式
ULONG NonBlock = ;
if (ioctlsocket(ListenSocket, FIONBIO, &NonBlock))
{
printf("ioctlsocket() failed with error %d\n", WSAGetLastError());
return -;
} CreateSocketInformation(ListenSocket);//ListenSocket套接字创建对应的SOCKET_INFORMATION,把ListenSocket添加到SocketArray数组中 while (true)
{
FD_ZERO(&ReadSet);//读套接字清零
FD_ZERO(&WriteSet);//写套接字清零 FD_SET(ListenSocket, &ReadSet);//向ReadSet中添加监听套接字ListenSocket for (DWORD i = ; i < TotalSockets; ++i)
{
LPSOCKET_INFORMATION SocketInfo = SocketArray[i];
FD_SET(SocketInfo->Socket, &ReadSet);
FD_SET(SocketInfo->Socket, &WriteSet);
} if((Total = select(, &ReadSet, &WriteSet, NULL, NULL)) == SOCKET_ERROR){//刚开始为什么监听套接字没有可写返回,而与客户端连接的套接字总是可写???
printf("select() returned with error %d\n", WSAGetLastError());
return -;
} for (DWORD i = ; i < TotalSockets; i++)
{
LPSOCKET_INFORMATION SocketInfo = SocketArray[i];//SocketArray保存了所有监听的可读和可写的套接字
if (FD_ISSET(SocketInfo->Socket, &ReadSet))//判断套接字是否可读
{
if (SocketInfo->Socket == ListenSocket)//对于监听套接字表示有新的客户端连接
{
Total--;
if ((AcceptSocket = accept(ListenSocket, NULL, NULL)) != INVALID_SOCKET)
{
NonBlock = ;
if (ioctlsocket(AcceptSocket, FIONBIO, &NonBlock) == SOCKET_ERROR)//设置AcceptSocket套接字为非阻塞套接字,这样与客户端通信就不会阻塞了
{
printf("ioctlsocket() failed with error %d\n", WSAGetLastError());
return -;
} if (CreateSocketInformation(AcceptSocket) == false)//将AcceptSocket添加到SocketArray数组中
{
return -;
}
}
else
{
if (WSAGetLastError() != WSAEWOULDBLOCK)
{
printf("accept() failed with error %d\n", WSAGetLastError());
return -;
}
}
}
else//接收数据
{
Total--;
memset(SocketInfo->Buffer, , sizeof SocketInfo->Buffer);
SocketInfo->DataBuf.buf = SocketInfo->Buffer;
SocketInfo->DataBuf.len = DATA_BUFSIZE; DWORD Flags = ;
if (WSARecv(SocketInfo->Socket, &(SocketInfo->DataBuf), , &RecvBytes, &Flags, NULL, NULL) == SOCKET_ERROR)
{
if (WSAGetLastError() != WSAEWOULDBLOCK)
{
printf("WSARecv() failed with error %d\n", WSAGetLastError());
FreeSocketInformation(i);
}
continue;
}
else
{
SocketInfo->BytesRECV = RecvBytes;
SocketInfo->DataBuf.buf[RecvBytes] = '\0';
if (RecvBytes == )
{
FreeSocketInformation(i);
continue;
}
else
{
std::cout << SocketInfo->DataBuf.buf << std::endl;// 如果成功接收数据,则打印收到的数据
}
}
}
}
else
{
if (FD_ISSET(SocketInfo->Socket, &WriteSet))//可写会一直被调用,判断大于其写缓冲的最低水位
{
Total--;
SocketInfo->DataBuf.buf = SocketInfo->Buffer + SocketInfo->BytesSEND;// 初始化缓冲区位置
SocketInfo->DataBuf.len = SocketInfo->BytesRECV - SocketInfo->BytesSEND;// 初始化缓冲区长度 if (SocketInfo->DataBuf.len > )// 如果有需要发送的数据,则发送数据
{
if (WSASend(SocketInfo->Socket, &(SocketInfo->DataBuf), , &SendBytes, , NULL, NULL) == SOCKET_ERROR)
{
if (WSAEWOULDBLOCK != WSAGetLastError())
{
printf("WSASend() failed with error %d\n", WSAGetLastError());
FreeSocketInformation(i);
}
continue;
}
else
{
SocketInfo->BytesSEND += SendBytes; //记录发送数据的字节数目
if (SocketInfo->BytesSEND == SocketInfo->BytesRECV)
{
SocketInfo->BytesSEND = ;
SocketInfo->BytesRECV = ;
}
}
}
}
}
}
}
system("pause");
return ; }

client

#include <winsock.h>
#include <iostream>
#define BUFSIZE 64
#define PORT 9999 #pragma comment(lib, "WS2_32.lib")
int main()
{
WSADATA wsaData;
SOCKET sClient; sockaddr_in addrServ;
char buf[BUFSIZE];
int retVal; if(WSAStartup(MAKEWORD(,), &wsaData) != )
{
std::cout << "WSAStartup失败!" << std::endl;
return -;
} sClient = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (INVALID_SOCKET == sClient)
{
std::cout << "socket() 错误!" << std::endl;
WSACleanup();
return -;
} addrServ.sin_family = AF_INET;
addrServ.sin_port = htons(PORT);
addrServ.sin_addr.S_un.S_addr = inet_addr("127.0.0.1"); retVal = connect(sClient, (LPSOCKADDR)&addrServ, sizeof(addrServ));
if (SOCKET_ERROR == retVal)
{
std::cout << "connect 错误!" << std::endl;
closesocket(sClient);
WSACleanup();
return -;
} while(true)
{
std::cout << "输入要发给服务器的内容:" << std::endl;
char msg[BUFSIZE];
std::cin.getline(msg, BUFSIZE);
ZeroMemory(buf, BUFSIZE);
strcpy(buf, msg);
retVal = send(sClient, buf, strlen(buf), );
if (SOCKET_ERROR == retVal)
{
std::cout << "发送失败" << std::endl;
closesocket(sClient);
WSACleanup();
return -;
} retVal = recv(sClient, buf, sizeof buf, );
std::cout << "从服务器端接收:" << buf << std::endl;
if (strcmp(buf, "quit") == )
{
std::cout << "quit" << std::endl;
break;
}
}
closesocket(sClient);
WSACleanup();
return ;
}

非阻塞socket学习,select基本用法的更多相关文章

  1. Linux 网络编程七(非阻塞socket:epoll--select)

    阻塞socket --阻塞调用是指调用结果返回之前,当前线程会被挂起.函数只有在得到结果之后才会返回. --对于文件操作 read,fread函数调用会将线程阻塞(平常使用read感觉不出来阻塞, 因 ...

  2. JAVA基础知识之网络编程——-基于NIO的非阻塞Socket通信

    阻塞IO与非阻塞IO 通常情况下的Socket都是阻塞式的, 程序的输入输出都会让当前线程进入阻塞状态, 因此服务器需要为每一个客户端都创建一个线程. 从JAVA1.4开始引入了NIO API, NI ...

  3. 【2018.08.13 C与C++基础】网络通信:阻塞与非阻塞socket的基本概念及简单实现

    一.前言 最近在做Matalb/Simulink与C/C++的混合编程,主要是完成TCP.UDP.SerialPort等常见通信方式的中间件设计,为Simulink模型提供数据采集及解析模块. 问题在 ...

  4. (转)非阻塞Connect对于select时应注意问题

    对于面向连接的socket类型(SOCK_STREAM,SOCK_SEQPACKET)在读写数据之前必须建立连接,首先服务器端socket必须在一个客户端知道的地址进行监听,也就是创建socket之后 ...

  5. 非阻塞socket与epoll

    阻塞socket. –阻塞调用是指调用结果返回之前,当前线程会被挂起.函数只有在得到结果之后才会返回. –对于文件操作read,fread函数调用会将线程阻塞. –对于socket,accept与re ...

  6. Linux - 非阻塞socket编程处理EAGAIN错误

            在linux进行非阻塞的socket接收数据时经常出现Resource temporarily unavailable,errno代码为11(EAGAIN),这表明你在非阻塞模式下调用 ...

  7. linux网络编程中阻塞和非阻塞socket的区别

    读操作 对于阻塞的socket,当socket的接收缓冲区中没有数据时,read调用会一直阻塞住,直到有数据到来才返 回.当socket缓冲区中的数据量小于期望读取的数据量时,返回实际读取的字节数.当 ...

  8. 阻塞和非阻塞socket的区别

    读操作 对于阻塞的socket,当socket的接收缓冲区中没有数据时,read调用会一直阻塞住,直到有数据到来才返回.当socket缓冲区中的数据量小于期望读取的数据量时,返回实际读取的字节数.当s ...

  9. 斐讯面试记录—阻塞Socket和非阻塞Socket

    文章出自:http://blog.csdn.net/VCSockets/ 1.TCP中的阻塞Socket和非阻塞Socket 阻塞与非阻塞是对一个文件描述符指定的文件或设备的两种工作方式. 阻塞的意思 ...

随机推荐

  1. alarm

    AlarmManager的使用机制有的称呼为全局定时器,有的称呼为闹钟.通过对它的使用,它的作用和Timer有点相似.都有两种相似的用法:(1)在指定时长后执行某项操作 (2)周期性的执行某项操作 在 ...

  2. android Baseadapter 和 ViewHolder的使用

    昨晚学习了徐大神的关于BaseAdapter的讲解,让我受益匪浅特来博客留下印记 说到baseadapter大家一定都不陌生,下面这张图就展示了数据.listview.baseadapter 之间的关 ...

  3. Day Five (beta)

    站立式会议 站立式会议内容总结 331 今天:爬虫抓取豆瓣数据; 明天:数据展示在话题详情页,话题页修改 442 今天:了解聊天模块 遇到的问题:无 明天:编写聊天模块 439 今天:学习 遇到困难 ...

  4. 用ElasticSearch和Protovis实现数据可视化

    搜索引擎最重要的目的,嗯,不出意料就是搜索.你传给它一个请求,然后它依照相关性返回你一串匹配的结果.我们可以根据自己的内容创造各种请求结构,试验各种不同的分析器,搜索引擎都会努力尝试提供最好的结果. ...

  5. iOS开发中的错误整理,IOS9中canOpenURL调用失败分析

    由于IOS加入对用户隐私以及禁止扫描系统信息的控制,目前通过canOpenURL的方法来判断用户是否安装特定app,则会出现-canOpenURL: failed for URL: "ABC ...

  6. javascript 红宝书笔记之数据类型

      typeof   检测给定变量的数据类型,通过typeof来区分函数和其它对象   var message = 'some string'; console.log(typeof(message) ...

  7. win7如何让局域网其他电脑通过IP直接访问自己电脑的网站

    一.打开控制面板 二.打开防火墙 三.点击右侧高级设置 四.点击防火墙属性 五.点击防火墙状态选择为关闭,确定 六.点击右侧允许或功能通过windows防火墙 七.执行第六部会打开防火墙通信例外窗口, ...

  8. JS-slider.js实现鼠标拖动滑块控制取值特效

    制作效果,如下图,鼠标点击颜色标能左右拖动并设置文本框中的值 源码: <div id="example"> <div id="slideContaine ...

  9. 【收藏】Android更新UI的几种常见方法

    ----------------将会调用onDraw()重绘控件---------------- 1.view.invalidate刷新UI(主线程)   2.view.postInvalidate刷 ...

  10. sublime text 3之快捷键操作

    1.安装插件 https://packagecontrol.io/installation 2.将 Tab缩进(制表符缩进) 改为 4个空格 打开Preferences -> Settings- ...