1 服务器代码  Linux eclipse C++

 //============================================================================
// Name : epollServer.cpp
// Author : fangjunmin
// Version :
// Copyright : Your copyright notice
// Description : Hello World in C++, Ansi-style
//============================================================================ #include <sys/socket.h>
#include <netinet/in.h>
#include <iostream>
#include <stdio.h>
#include <arpa/inet.h>
#include <errno.h>
#include <sys/epoll.h>
#include <map>
#include <vector>
#include <memory.h>
#include "encode.h"
#include <stddef.h> using namespace std; int g_epfd = -;
int g_listen_fd = -;
u_short g_listen_port = ; typedef map<int, int> mapClient ;
typedef map<int, int>::iterator itmapClient ;
typedef map<int, int>::const_iterator citmapClient ; typedef vector<int> vecClient ;
typedef vecClient::iterator itvecClient ;
typedef vecClient::const_iterator citvecClient ; mapClient g_mapClient;
vecClient g_vecCLient; void InitListen();
void StartEpoll();
void BoardCast(const char* msg, long int nLen);
void removeFd(int fd); int main()
{
cout << "!!!Hello epoll!!!" << endl; // prints !!!Hello epoll!!! InitListen();
StartEpoll();
return ;
} void InitListen()
{
g_listen_fd = socket(AF_INET, SOCK_STREAM, IPPROTO_IP);
if (g_listen_fd == -)
{
return ;
} struct sockaddr_in sin;
//bzero(&sin, sizeof(struct sockaddr_in));
sin.sin_family = AF_INET;
sin.sin_addr.s_addr = INADDR_ANY;
sin.sin_port = htons(g_listen_port); if (bind(g_listen_fd, (struct sockaddr *) &sin, sizeof(struct sockaddr)) != )
{
printf("bind error: %d\n", errno);
return ;
} if (listen(g_listen_fd, ) != )
{
printf("listen error!\n");
return ;
}
} void StartEpoll()
{
int nSize = ;
g_epfd = epoll_create(nSize); epoll_event ev;
ev.data.fd=g_listen_fd;
ev.events = EPOLLIN | EPOLLET | EPOLLOUT;
int nAddResult = epoll_ctl(g_epfd ,EPOLL_CTL_ADD, g_listen_fd ,&ev); //将新的fd添加到epoll的监听队列中
if (nAddResult == -)
{
printf ("epoll_ctl error\n");
return ;
} while(true)
{
epoll_event events[nSize + ];
int nfds = epoll_wait(g_epfd, events, ,);
for(int i=; i<nfds; ++i)
{
printf("event num is: %d\n", nfds);
if(events[i].data.fd == g_listen_fd) //有新的连接
{
sockaddr_in clientaddr;
unsigned int clilen = sizeof(clientaddr);
int connfd = accept(g_listen_fd, (sockaddr *)&clientaddr, &clilen); //accept这个连接
printf("new connection! %d\n", connfd); epoll_event event;
event.data.fd=connfd;
event.events = EPOLLIN | EPOLLET | EPOLLOUT | EPOLLHUP;
epoll_ctl(g_epfd,EPOLL_CTL_ADD,connfd,&event); //将新的fd添加到epoll的监听队列中 g_vecCLient.push_back(connfd);
} else if( events[i].events & EPOLLIN ) //接收到数据,读socket
{
char szBuf[] = {};
int nRecvNum = recv(events[i].data.fd, szBuf, , ); //读 char szDest[] = {};
size_t nOutLen = ;
GB2312ToUtf8(szBuf, nRecvNum, szDest, nOutLen);
printf("recv size :%d, concent:%s\n", nRecvNum, szBuf);
printf("after conv size :%d, concent:%s\n", nOutLen , szDest); if(nRecvNum > )
{
BoardCast(szBuf, nRecvNum);
}
else if(nRecvNum == -)
{
perror(NULL);
removeFd(events[i].data.fd);
} // ev.data.ptr = md; //md为自定义类型,添加数据
// ev.events=EPOLLOUT|EPOLLET;
// epoll_ctl(epfd,EPOLL_CTL_MOD,sockfd,&ev);//修改标识符,等待下一个循环时发送数据,异步处理的精髓
}
else if(events[i].events & EPOLLOUT) //有数据待发送,写socket
{
printf("EPOLLOUT\n" );
// struct myepoll_data* md = (myepoll_data*)events[i].data.ptr; //取数据
// sockfd = md->fd;
// send( sockfd, md->ptr, strlen((char*)md->ptr), 0 ); //发送数据
// ev.data.fd=sockfd;
// ev.events=EPOLLIN|EPOLLET;
// epoll_ctl(epfd,EPOLL_CTL_MOD,sockfd,&ev); //修改标识符,等待下一个循环时接收数据
}
else if(events[i].events & EPOLLHUP) //socket disconnect
{
printf("socket disconnect socketid:%d\n", events[i].data.fd);
removeFd(events[i].data.fd);
}
else
{
//其他的处理
}
}
}
} void BoardCast(const char* msg, long int nLen)
{
for(citvecClient cit = g_vecCLient.begin(); cit != g_vecCLient.end(); cit++)
{
int nClientFd = *cit;
int nSendLen = send(nClientFd, (const void* )msg , nLen, );
printf("socket nSendLen:%d\n", nSendLen);
} } void removeFd(int fd)
{
epoll_ctl(g_epfd,EPOLL_CTL_DEL,fd, NULL); //delete socket in epoll
for(itvecClient it = g_vecCLient.begin(); it != g_vecCLient.end(); it++)
{
if(fd == *it)
{
g_vecCLient.erase(it);
return;
}
}
}

2 Windows下的客户端代码  Windows、visual studio 、C++

主线程  读线程  写线程

 #include "stdio.h"
#include <WinSock2.h>
#include "cInitSock.h"
#include <string>
#pragma comment(lib,"WS2_32") using namespace std;
//struct stMsg
//{
// int sCmdID;
// int sLen;
// int nValue;
// string strContent;
//
// stMsg()
// {
// sCmdID = 0;
// sLen = 0;
// nValue = 0;
// strContent = "";
// }
//}; char g_szIp[] = "192.168.10.32";
short g_port = ;
SOCKET g_sock; void ReadHandle();
void WriteHandle();
CRITICAL_SECTION g_cs; static int s_na = ; void main()
{
cInitSock initSock;
InitializeCriticalSection(&g_cs);
g_sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); sockaddr_in sockAddr;
memset(&sockAddr,,sizeof(sockAddr)); sockAddr.sin_family = AF_INET;
sockAddr.sin_addr.S_un.S_addr = inet_addr(g_szIp);
sockAddr.sin_port = htons((u_short)g_port);
int nRestle = ;
nRestle= connect(g_sock, (sockaddr *)&sockAddr, sizeof(sockAddr) ); DWORD dErr = GetLastError();
if (nRestle != )
{
printf("连接服务器失败 \n");
return;
}
printf("connect success \n"); int nReadThreadID;
int nWriteThreadID;
CreateThread(, , (LPTHREAD_START_ROUTINE) ReadHandle, , , (LPDWORD)&nReadThreadID);
CreateThread(, , (LPTHREAD_START_ROUTINE) WriteHandle, , , (LPDWORD)&nWriteThreadID); printf("客户端开启成功\n");
while()
{
;
} } void ReadHandle()
{
char buf[] = "";
int buflen = ; while ()
{
memset(buf, , );
buflen = recv(g_sock, buf, , );
if ( buflen )
{
EnterCriticalSection((LPCRITICAL_SECTION)&g_cs);
printf("recv msg : %s,msgSiez:%d\n", buf, buflen);
LeaveCriticalSection((LPCRITICAL_SECTION)&g_cs);
}
}
} void WriteHandle()
{
char buf[] = "";
int buflen = ; while ()
{
gets_s(buf);
//printf("the gets buf is:%s:end\n", buf);
buflen = send(g_sock, (const char* )&buf, strlen(buf), );
if ( buflen > )
{
EnterCriticalSection((LPCRITICAL_SECTION)&g_cs);
printf("send msg : %s,msgSiez:%d\n", buf, buflen);
LeaveCriticalSection((LPCRITICAL_SECTION)&g_cs);
}
if (buflen == -)
{
EnterCriticalSection((LPCRITICAL_SECTION)&g_cs);
printf("send msg error: %s\n", buf);
LeaveCriticalSection((LPCRITICAL_SECTION)&g_cs);
}
}
}

简单通讯聊天 群聊功能 Windows下的客户端 Linux下的epoll服务器的更多相关文章

  1. Java-->实现群聊功能(C/S模式--TCP协议)

    --> Java 对TCP协议的支持: --> java.net包中定义了两个类ServerSocket 和Socket ,分别用来实现双向连接的server 端和client 端. -- ...

  2. day04-1群聊功能

    多用户即时通讯系统04 4.编码实现03 4.5功能实现-群聊功能实现 4.5.1思路分析 群聊的实现思路和私聊的实现非常类似. 不同的是:私聊时,服务端接收到消息后,只需要找出接收方的socket并 ...

  3. 基于ejabberd简单实现xmpp群聊离线消息

    首先,xmpp服务器是基于ejabberd.离线消息模块是mod_interact,原地址地址:https://github.com/adamvduke/mod_interact: 修改后实现群聊离线 ...

  4. 解决Windows下文件在Linux下打开出现乱码的问题

    目录 问题 原理 解决 总结 参考资料 问题 前几天生病了,Java一直在看代码但是没跟着打,于是决定偷一波小小的懒,直接把教材的代码从Windows通过共享文件夹放到了Linux里面.但是编译的时候 ...

  5. windows下plsql连接linux下的oracle数据库

    windows下plsql连接linux下的oracle数据库 经过多方查找,终于找到解决办法,特此记录下来,共享之. PL/SQL Develorper:目前未发现可以在Linux系统中安装的版本. ...

  6. ASP.NET SignalR 与LayIM配合,轻松实现网站客服聊天室(四) 添加表情、群聊功能

    休息了两天,还是决定把这个尾巴给收了.本篇是最后一篇,也算是草草收尾吧.今天要加上表情功能和群聊.基本上就差不多了,其他功能,读者可以自行扩展或者优化.至于我写的代码方面,自己也没去重构.好的,我们开 ...

  7. netty无缝切换rabbitmq、activemq、rocketmq实现聊天室单聊、群聊功能

    netty的pipeline处理链上的handler:需要IdleStateHandler心跳检测channel是否有效,以及处理登录认证的UserAuthHandler和消息处理MessageHan ...

  8. Java网络编程Demo,使用TCP 实现简单群聊功能Groupchat,创建一个服务端,使多个客户端都能收到消息

    效果图: 开启服务端 客户端一 客户端二 客户端三 实现代码: 客户端类 import java.io.IOException; import java.net.ServerSocket; impor ...

  9. Asp.net SignalR 应用并实现群聊功能 开源代码

    ASP.NET SignalR 是为 ASP.NET 开发人员提供的一个库,可以简化开发人员将实时 Web 功能添加到应用程序的过程.实时 Web 功能是指这样一种功能:当所连接的客户端变得可用时服务 ...

随机推荐

  1. string::find_last_not_of

    #include <iostream>#include <string> using namespace std;int main(){ string s1("abc ...

  2. DevExpress ASP.NET v19.1版本亮点:Pivot Grid等控件

    行业领先的.NET界面控件DevExpress 发布了v19.1版本,本文将以系列文章的方式为大家介绍DevExpress ASP.NET Controls v19.1中新增的一些控件及增强的控件功能 ...

  3. 前端js之JQuery

    目录 jQuery介绍 jQuery的优势 jQuery内容 jQuery对象 jQuery基础语法结构 jQuery 使用注意事项 查找标签 基本选择器 层级选择器 基本选择器 属性选择器 表单筛选 ...

  4. 字典树Trie--实现敏感词过滤

    序言 Trie树 资料 https://blog.csdn.net/m0_37907797/article/details/103272967?utm_source=apphttps://blog.c ...

  5. 关系型数据库(七),复杂SQL语句

    目录 1.SQL语句由六部分组成 2.Group By 和HAVING 七.复杂SQL语句 1.SQL语句由六部分组成 Select  *** From  *** Where  *** Group B ...

  6. python中oepen及fileobject初步整理之划水篇

    open选项 参考官方文档,很多东西也没有看懂,将自己理解的部分先整理到这里,以后还是要参阅官方文档的. open (file, mode='r', buffering=-1, encoding=No ...

  7. selenium实现chrome分屏截图的合并

    selenium的截图功能在chrome下无法实现,但是可以操作滚动条来一屏一屏的截图,然后再合并成一张图,合并图片的代码在网上找的,十分感谢那位朋友,具体解决方案如下:直接上代码: def capt ...

  8. Oracle用函数或PIVOT实现行转列

    原数据: 目标数据: 实现代码: SELECT YEAR AS "年", SUM (DECODE (QUATER, 1, RATE, 0)) AS "季度一", ...

  9. JavaWeb_初识监听器Listener

    监听器(listener):对项目起到监听的作用,它能感知到包括request(请求域),session(会话域)和applicaiton(应用程序)的初始化和属性的变化 监听器是Servlet规范中 ...

  10. 构建springboot的几种方式 在线构建 STS构建 Idea 内置构建 Maven 构建

    SpringBoot项目的几种创建方式,启动.和访问   最常用的4种方式,但除了这些以外,还有其他方式: ①在线创建 ②STS构建 ③Intell  Idea内置构建工具 ④Maven创建 STS官 ...