Linux下c++11多线程聊天室

刚看的c++11多线程,写个聊天室试试
编译的时候加上 c++11 和 多线程库
g++ -Wall -std=c++0x -pthread -o server server.cpp
server 和client 都是 q 退出
?1. [代码]server.cpp

#include <iostream>
#include <thread>
#include <arpa/inet.h>
#include <cstring>
#include <vector>
#include <algorithm>
#include <unistd.h>
using namespace std;
    vector<int> cli_vec;
    int listenfd, port;
    const char *ip = "127.0.0.1";
void SendMsg(char* buf, int connfd){
    //cout<<"now send..."<<endl;
    char *data = buf;
    int len;
    for(auto i=cli_vec.begin(); i!=cli_vec.end(); ++i){
        if(*i == connfd) continue;
        if((len=send(*i, data, strlen(data),0)) == -1){
            //cout<<"send error"<<endl;
        }else{
            //cout<<"Send: "<<data<<endl;
        }
    }
}
 
void RecvMsg(int connfd){
    int len;
    char buf[100];
    while(1){
        memset(buf, 0, sizeof(buf));
        //cout<<"now receive..."<<connfd<<endl;
        if((len = recv(connfd, buf,sizeof(buf), 0)) == -1){
            //cout<<"Falid to receive"<<endl;
        }
        if(0 == len){
            cout<<connfd<<" has exit the chatting room"<<endl;
            auto pos = find(cli_vec.begin(), cli_vec.end(), connfd);
            if(pos != cli_vec.end()){
                cli_vec.erase(pos);
            }
            break;
        }
        SendMsg(buf,connfd);
    }
}       
 
void WaitQuit(){
    char buf[100];
    while(1){
        cin>>buf;
        if(0 == strcmp(buf,"q")){
            for(auto i=cli_vec.begin(); i!=cli_vec.end(); i++){
                close(*i);
            }
            cli_vec.clear();
            close(listenfd);
            break;
        }
    }
    exit(0);
}
 
int main()
{
    if((listenfd=socket(AF_INET,SOCK_STREAM,0)) == -1){
        cout<<"socket error"<<endl;
        return -1;
    }
    cout<<"请输入端口号:";
    cin>>port;
    struct sockaddr_in serv_addr,cli_addr;
    memset(&serv_addr, 0, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port   = htons(port);
    serv_addr.sin_addr.s_addr = inet_addr(ip); 
    if(bind(listenfd, (struct sockaddr*)&serv_addr, sizeof(serv_addr)) == -1){
        cout<<"bind error"<<endl;
        return -1;
    }http://www.enterdesk.com/special/dushen/?
    if(listen(listenfd, 10) == -1){
        cout<<"listen error"<<endl;
        return -1;赌神
    }   
    std::thread(WaitQuit).detach();
    cout<<"server is running..."<<endl;
    while(1){
        size_t cli_len = sizeof(cli_addr);
        memset(&cli_addr, 0, cli_len);
 
        int connfd = accept(listenfd, (struct sockaddr*)&cli_addr, &cli_len);
        if(connfd == -1){
            cout<<"accept error"<<endl;
            return -1;
        }
        cout<<connfd<<" has enter the chatting room"<<endl;
        cli_vec.push_back(connfd);
        std::thread(RecvMsg, connfd).detach();
    }
    return 0;
}
2. [代码]client.cpp

#include <iostream>
#include <arpa/inet.h> // socket 相关 ...
#include <unistd.h>    // close()
#include <thread> 
#include <string>
#include <cstring>
using namespace std;
 
int sockfd;
struct sockaddr_in serv_addr;
 
void SendMsg(const string &msg) {
    if(-1 == send(sockfd, msg.c_str(), msg.size(), 0))
        cout<<"Failed to send"<<endl;
}
 
void RMsg(){
    int len;
    char recv_buf[1024];
    while (1) {
        memset(recv_buf, 0, sizeof(recv_buf));
        len = recv(sockfd, recv_buf, sizeof(recv_buf), 0);
        if (-1 == len) {
            cout<<"Failed to recv"<<endl;
        }
        if(0 == len){
            close(sockfd);
            cout<<"与服务器断开链接!"<<endl;
            break;
        }
        cout<<"Receive message: "<<recv_buf<<endl;
    }
}
 
int main()
{
    string ip = "127.0.0.1";
    int port;
    cout<<"please input server port:";
    cin>>port;
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (-1 == sockfd) {
        cout<<"Failed to sockfd"<<endl;
        return -1;
    }
    memset(&serv_addr, 0, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(port);
    serv_addr.sin_addr.s_addr = inet_addr(ip.c_str());
 
    int stat = connect(sockfd, (struct sockaddr *) &serv_addr,sizeof(serv_addr));
    if (-1 == stat) {
        cout<<"Failed to connect"<<endl;
        return -1;
    }
    std::thread(RMsg).detach();
    while(1)
    {
        string send_msg;
         
        cin>>send_msg;
        if (send_msg == "q")
        {
            cout<<"结束客户端!"<<endl;
            break;
        }
        else
            SendMsg(send_msg);
    }
    close(sockfd);
    return 0;
}
?

时间: 2024-10-26 19:18:32

Linux下c++11多线程聊天室的相关文章

Linux下面基于TCP多线程聊天室(客户端)

不怎么会弄这个博客的排版,就直接将代码附上: 主要是使用多线程去等待接受数据和发送数据,下面是客户端的代码: tcpsed.h文件 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 #ifndef RTPSED_H #define RTPSED_H #define BUFFSIZE 512 int runcond; int socketfd; typedef struct TCP_send_arg { char *tcpserver

Linux下面基于TCP多线程聊天室(服务器)

接上篇博文,本文是服务器端的实现,主要实现的功能,就是现实客户端的连接,转发客户端发送的消息,以及客户端掉线提示等功能,同时可以在这这上面扩展和TCP以及线程相关的功能木块. tcpreceive.h 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 #ifndef TCPRECEIVE_H #define TCPRECEIVE_H #define BUFFSIZE 2048 #define listen_max 5 int cond; int

Linux以下基于TCP多线程聊天室(server)

接上篇博文,本文是server端的实现,主要实现的功能,就是现实client的连接.转发client发送的消息.以及client掉线提示等功能,同一时候能够在这这上面扩展和TCP以及线程相关的功能木块. tcpreceive.h 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 #ifndef TCPRECEIVE_H #define TCPRECEIVE_H #define BUFFSIZE 2048 #define listen_max 5

Linux下C语言多线程,网络通信简单聊天程序

原文:Linux下C语言多线程,网络通信简单聊天程序 功能描述:程序应用多线程技术,可是实现1对N进行网络通信聊天.但至今没想出合适的退出机制,除了用Ctr+C.出于演示目的,这里采用UNIX域协议(文件系统套接字),程序分为客户端和服务端.应用select函数来实现异步的读写操作. 先说一下服务端:首先先创建套接字,然后绑定,接下进入一个无限循环,用accept函数,接受“连接”请求,然后调用创建线程函数,创造新的线程,进入下一个循环.这样每当有一个新的“连接”被接受都会创建一个新的线程,实现

MFC笔记之多线程聊天室

新手刚接触,跟着孙鑫老师视频一步一步的做.从VC6.0到VS2010好像并不是那么顺利,下面记录下一点收获. 网络编程的一般步骤: 1声明套接字版本(WSAStartup);2创建套接字(socket);3绑定套接字(bind);4发送接收(sendto/recvfrom);5关闭(closesocket) 第1~3步代码如下: 1 CString error; 2 WORD wver; 3 wver=MAKEWORD(1,1); 4 WSAData data; 5 if(0!=WSAStart

多线程聊天室

基于Java的多线程聊天室 客户端: public class MultiThreadClient { private static class RecMsg implements Runnable{ private Socket client; public RecMsg(Socket client) { this.client = client; } @Override public void run() { Scanner sc = null; try { sc = new Scanner

linux下C语言多线程编程实例

学东西,往往实例才是最让人感兴趣的,老是学基础理论,不动手,感觉没有成就感,呵呵.下面先来一个实例.我们通过创建两个线程来实现对一个数的递加.或许这个实例没有实际运用的价值,但是稍微改动一下,我们就可以用到其他地方去拉.下面是我们的代码: /*thread_example.c : c multiple thread programming in linux *author : falcon *E-mail : [email protected] */ #include <pthread.h>

使用Html5下WebSocket搭建简易聊天室

一.Html5WebSocket介绍 WebSocket protocol 是HTML5一种新的协议(protocol).它是实现了浏览器与服务器全双工通信(full-duplex). 现在,很多网站为了实现即时通讯(real-time),所用的技术都是轮询(polling).轮询是在特定的的时间间隔(time interval)(如每1秒),由浏览器对服务器发出HTTP request,然后由服务器返回最新的数据给客服端的浏览器.这种传统的HTTP request d的模式带来很明显的缺点 –

linux下终端11选5平台出租常用命令和vi命令修改文件及保存的使用方法

首先11选5平台出租haozbbs.comQ1446595067介绍一下Ubuntu下各个目录的一般作用: /这就是根目录,一台电脑有且只有一个根目录,所有的文件都是从这里开始的.举个例子:当你在终端里输入"/home",你其实是在告诉电脑,先从/(根目录)开始,再进入到home目录./root系统管理员(root user)的目录.至于系统管理员的权限有多大我这里就不在废话了.因此,请小心使用root帐号./boot系统启动文件,所有与系统启动有关的文件都保存在这里 . /bin 这