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()函 ...
随机推荐
- setInterval()调用其他函数时候报错
(function(){ function shortcut() { // 配件优化 window.topValue = 0// 上次滚动条到顶部的距离 window.interval = null; ...
- MySQL8的密码策略
解释: 由于valiadte_password策略.密码强度需要非常高,所以有时候密码都无法成功修改.了解完下面变量就能解决了. validate_password.policy:密码策略,检查用户的 ...
- 将Prometheus alerts保存到elasticsearch
Prometheus产生的告警通常会发送到alertmanager,当使用alertmanager时,其告警信息仅存在于alertmanager的内存中,无法持久化.故实现了小工具,用于将Promet ...
- [转] Performance_js中计算网站性能监控利器
1.Performance方法 Performance提供的方法可以灵活使用,获取到页面加载等标记的耗时情况. performance.now() //返回当前到页面打开时刻的耗时,精确到千分之一毫秒 ...
- 第一个APP上架IOS审核相关的记录
以前一直没做过APP开发,第一版是用WAP版做的,采用了light7框架制作,没有UI设计. 升级到第二版之后,使用了HBUILDER的方式开发,https://dcloud.io/ 官方在这里. 目 ...
- redis持久化rdb和aof之间的优势劣势
1.RDB(Redis Database) a.基本概念 概念: 在指定的时间间隔内将内存中的数据集快照写入磁盘, 也就是行话讲的Snapshot快照,它恢复时是将快照文件直接读到内存里. Redis ...
- Flask应用启动流程
目录 flask应用启动流程 WSGI 启动流程 flask应用启动流程 WSGI 所有的 python web 框架都要遵循 WSGI 协议 在这里还是要简单回顾一下 WSGI 的核心概念. WSG ...
- 来看一下Java中“-”与equeals的区别
简介: == ==是比较两个变量的值,如果是基本数据类型,那么就是比较的基本数据的大小值 情况一 int a=1; int b=1; System.out.println(a==b); 以上图中:== ...
- Vue定义组件和生命周期函数及实例演示!
定义全局组件 Vue.component("name",{...}) 定义局部组件 let Com = {....} new Vue({ data : ..., ..., comp ...
- 使用<label>标签修改input[type="checkbox"]的样式
因为<label>的特性有两点 : ①不呈现任何效果, ②用户点击该标签, 浏览器能自动将焦点转移到相关的表单控件上. <form> <input type=" ...