windows下socket编程实现client和server双向通信
服务端代码server.c
// server.cpp : Defines the entry point for the console application.
// #include <stdio.h>
#include <Winsock2.h> //Socket的函数调用
#include <windows.h> #define BUF_SIZE 6400 // 缓冲区大小 #pragma comment (lib, "ws2_32") // 使用WINSOCK2.H时,则需要库文件WS2_32.LIB DWORD WINAPI Rcv( LPVOID lpParam )
{
SOCKET sClient = *(SOCKET*)lpParam;
int retVal;
char bufRecv[BUF_SIZE];
memset( bufRecv, 0, sizeof( bufRecv ) );
while(1)
{
retVal = recv( sClient, bufRecv, BUF_SIZE, 0 );
if ( retVal == SOCKET_ERROR ) {
printf( "recive faild!\n" );
break;
} else {
printf( "收到客户端消息:%s\n", bufRecv );
}
}
return 0;
} DWORD WINAPI Snd( LPVOID lpParam )
{
SOCKET sClient = *(SOCKET*)lpParam;
int retVal;
char bufSend[BUF_SIZE];
memset( bufSend, 0, sizeof( bufSend ) );
while(1)
{
gets( bufSend );
retVal = send( sClient, bufSend, strlen(bufSend)+sizeof(char), 0 );
if ( retVal == SOCKET_ERROR ) {
printf( "send faild!\n" );
break;
}
}
return 0;
} int main(int argc, char* argv[])
{
// 初始化套接字动态库
WSADATA wsaData;
if ( WSAStartup(MAKEWORD(2, 2), &wsaData) != 0 ) {
printf( "winsock load faild!\n" );
return 1;
} // 创建服务段套接字
SOCKET sServer = socket( AF_INET, SOCK_STREAM, IPPROTO_TCP );
if ( sServer == INVALID_SOCKET ) {
printf( "socket faild!\n" );
WSACleanup();
return -1;
} // 服务端地址
sockaddr_in addrServ; addrServ.sin_family = AF_INET;
addrServ.sin_port = htons( 9999 );
addrServ.sin_addr.s_addr = htonl( INADDR_ANY ); // 绑定套接字
if ( bind( sServer, ( const struct sockaddr* )&addrServ, sizeof(addrServ) ) == SOCKET_ERROR ) {
printf( "bind faild!\n" );
closesocket( sServer );
WSACleanup();
return -1;
} printf("Server is On IP:[%s],port:[%d]\n",inet_ntoa(addrServ.sin_addr),ntohs(addrServ.sin_port)); // 监听套接字 数字表示最多能监听客户个数
if ( listen( sServer, 5 ) == SOCKET_ERROR ) {
printf( "listen faild!\n" );
closesocket( sServer );
WSACleanup();
return -1;
} SOCKET sClient; // 客户端套接字 sockaddr_in addrClient;
int addrClientLen = sizeof( addrClient ); sClient = accept( sServer, ( sockaddr FAR* )&addrClient, &addrClientLen );
if ( sClient == INVALID_SOCKET ) {
printf( "accept faild!\n" );
closesocket( sServer );
WSACleanup();
return -1;
}
printf("accepted client IP:[%s],port:[%d]\n",inet_ntoa(addrClient.sin_addr),ntohs(addrClient.sin_port)); HANDLE hThread1, hThread2;
DWORD dwThreadId1, dwThreadId2; hThread1 = ::CreateThread(NULL, NULL, Snd, (LPVOID*)&sClient, 0, &dwThreadId1);
hThread2 = ::CreateThread(NULL, NULL, Rcv, (LPVOID*)&sClient, 0, &dwThreadId2); ::WaitForSingleObject(hThread1, INFINITE);
::WaitForSingleObject(hThread2, INFINITE);
::CloseHandle(hThread1);
::CloseHandle(hThread2); closesocket( sClient );
WSACleanup(); // 资源释放 return 0;
}
客户端代码client.c
// client.cpp : Defines the entry point for the console application.
// #include <stdio.h>
#include <Winsock2.h> //Socket的函数调用
#include <windows.h> #define BUF_SIZE 6400 #pragma comment (lib, "ws2_32") // 使用WINSOCK2.H时,则需要库文件WS2_32.LIB DWORD WINAPI Rcv( LPVOID lpParam )
{
SOCKET sHost = *(SOCKET*)lpParam;
int retVal;
char bufRecv[BUF_SIZE];
memset( bufRecv, 0, sizeof( bufRecv ) );
while(1)
{
retVal = recv( sHost, bufRecv, BUF_SIZE, 0 );
if ( retVal == SOCKET_ERROR ) {
printf( "recive faild!\n" );
break;
} else {
printf( "收到服务器消息:%s\n", bufRecv );
}
}
return 0;
} DWORD WINAPI Snd( LPVOID lpParam )
{
SOCKET sHost = *(SOCKET*)lpParam;
int retVal;
char bufSend[BUF_SIZE];
memset( bufSend, 0, sizeof( bufSend ) );
while(1)
{
gets( bufSend );
retVal = send( sHost, bufSend, strlen(bufSend)+sizeof(char), 0 );
if ( retVal == SOCKET_ERROR ) {
printf( "send faild!\n" );
break;
}
}
return 0;
} int main(int argc, char* argv[])
{
WSADATA wsaData;
if ( WSAStartup( MAKEWORD(2,2), &wsaData ) != 0 ) {
printf( "Winsock load faild!\n" );
return 1;
} // 服务器套接字
SOCKET sHost = socket( AF_INET, SOCK_STREAM, IPPROTO_TCP );
if ( sHost == INVALID_SOCKET ) {
printf( "socket faild!\n" );
WSACleanup();
return -1;
} SOCKADDR_IN servAddr;
servAddr.sin_family = AF_INET;
// 注意 当把客户端程序发到别人的电脑时 此处IP需改为服务器所在电脑的IP
servAddr.sin_addr.S_un.S_addr = inet_addr( "127.0.0.1" );
servAddr.sin_port = htons( 9999 ); // 连接服务器
if ( connect( sHost, (LPSOCKADDR)&servAddr, sizeof( servAddr ) ) == SOCKET_ERROR ) {
printf( "connect faild!\n" );
closesocket(sHost);
WSACleanup();
return -1;
}
printf("连接到服务器 IP:[%s],port:[%d]\n",inet_ntoa(servAddr.sin_addr),ntohs(servAddr.sin_port)); HANDLE hThread1, hThread2;
DWORD dwThreadId1, dwThreadId2; hThread1 = ::CreateThread( NULL, NULL, Snd, (LPVOID)&sHost, 0, &dwThreadId1 );
hThread2 = ::CreateThread( NULL, NULL, Rcv, (LPVOID)&sHost, 0, &dwThreadId2 ); ::WaitForSingleObject( hThread1, INFINITE );
::WaitForSingleObject( hThread2, INFINITE );
::CloseHandle(hThread1);
::CloseHandle(hThread2); closesocket(sHost);
WSACleanup();
return 0;
}
截图如下:编译好后首先是启动服务端(来监听),然后再启动客户端

windows下socket编程实现client和server双向通信的更多相关文章
- windows下socket编程:区分shutdown()及closesocket()
以下描述主要是针对windows平台下的TCP socket而言. 首先需要区分一下关闭socket和关闭TCP连接的区别,关闭TCP连接是指TCP协议层的东西,就是两个TCP端之间交换了一些协议包( ...
- linux下socket编程实例
linux下socket编程实例一.基本socket函数Linux系统是通过提供套接字(socket)来进行网络编程的.网络的socket数据传输是一种特殊的I/O,socket也是一种文件描述符.s ...
- Windows 下用 gogs 配置局域网 git server
大道曙光 Windows 下用 gogs 配置局域网 git server 最近要用 C# 开发一个新的项目,所以需要在 Windows 局域网环境下构建一个 git server. 在 Window ...
- 初尝Windows 下批处理编程
本文叫“ 初尝Windows 下批处理编程”是为了延续上一篇“初尝 Perl”,其实对于博主而言批处理以及批处理编程早就接触过了. 本文包括以下内容 1.什么是批处理 2.常用批处理命令 3.简介批处 ...
- Linux下Socket编程的端口问题( Bind error: Address already in use )
Linux下Socket编程的端口问题( Bind error: Address already in use ) 在进行linux网络编程时,每次修改了源代码并再次编译运行时,常遇到下面的地使用错误 ...
- 初探WINDOWS下IME编程
初探WINDOWS下IME编程作者:广东南海市昭信科技有限公司-李建国 大家知道,DELPHI许多控件有IME属性.这么好用的东西VC可没自带,怎么办呢?其实,可通过注册表,用API实现.下面说一下本 ...
- Windows下串口编程
造冰箱的大熊猫@cnblogs 2019/1/27 将Windows下串口编程相关信息进行下简单小结,以备后用. 1.打开串口 打开串口使用CreateFile()函数.以打开COM6为例: HAN ...
- Linux下socket编程基本知识
本文档主要讲解了Linux下socket编程的一些基本知识,主要包括套接字和字节序的概念,以及一些常用的结构体和函数. 本文是在网易云课堂学习过程中的记录,这个老师讲得很不错,推荐大家围观. Linu ...
- 一个linux下socket编程的例子,client连server
关于socket编程,以下文章写得比较好:http://www.cnblogs.com/xudong-bupt/archive/2013/12/29/3483059.html 1. accept()函 ...
随机推荐
- nginx服务器图片防盗链的方法
nginx服务器图片防盗链的方法<pre> location ~* \.(gif|jpg|png|jpeg)$ { expires 30d; valid_referers *.shuche ...
- 关于i7 8700以上系列主机,安装虚拟机Win7下连接U盘,故障处理的补充说明
正如前文“虚拟机下怎么连接U盘,如何使用U盘?一策书(湘岳阳万江波)的随笔”所言,我在win10下的虚拟机win7(i7 9700),而且听说了是不支持Win7的,其原因是不支持USB的驱动问题. 我 ...
- 视觉融合定位github
1. obr_slam有关的非常有用的github链接: https://github.com/Ewenwan/MVision/tree/master/vSLAM/oRB_SLAM2 2. gnss, ...
- Docker底层原理介绍
1.docker介绍 1.1什么是docker Docker 是一个开源的应用容器引擎,基于 Go 语言 并遵从Apache2.0协议开源. Docker 可以让开发者打包他们的应用以及依赖包到一个轻 ...
- Charles设置断点- (超详解)
1.选择你要设置断点的接口 2.右键选择 Breakpoints 3.断点的相关配置, Proxy ——>Breakpoint Settings 5.双击刚刚已经设置的断点接口,进行设置 6. ...
- ABA问题的产生及解决
什么是ABA问题 在CAS算法中,需要取出内存中某时刻的数据(由用户完成),在下一时刻比较并交换(CPU保证原子操作),这个时间差会导致数据的变化. 1.线程1从内存位置V中取出A2.线程2从内存位置 ...
- Shiro 使用 JWT Token 配置类参考
项目中使用了 Shiro 进行验证和授权,下面是 Shiro 配置类给予参考. 后来并没有使用 Shiro,感觉使用 JWT 还是自己写拦截器比较灵活,使用 Shiro 后各种地方需要魔改,虽然功能也 ...
- Sitecore性化 - 您需要了解的4件事
Sitecore非常强大,是一个数字体验平台.它可以帮助您取悦并留住客户.它可以帮助您衡量和评估广告系列.它使你成为一个更好的营销人员.它可以帮助您获得结果! 它结合了易于使用的网站内容管理系统和数字 ...
- 百度前端技术学院-task1.4源代码
任务描述 实现如 示例图(点击打开) 的效果 灰色元素水平垂直居中,有两个四分之一圆位于其左上角和右下角. 任务注意事项 思考不同情况下(如灰色高度是根据内容动态变化的)水平垂直居中的解决方案. 动手 ...
- MVC+Ninject+三层架构+代码生成 -- 总结(六、邏輯層)
1.邏輯層的方法應該與數據層的方法一一對應.邏輯層返回的結果都是用接口IResult封裝,用於項目轉換時,能減少變化的代碼量. 2.邏輯層都需要繼承 BaseLogic 類 public class ...