2019-08-22 Day19 Socket实现QQ私聊与群聊


目的

利用学习的socket编程,完成简易QQ以实现私聊与群聊

Socket实现简易QQ

要求

1.每个客户端都有一个名称
2.一个客户端可以向另一个客户端发起私聊
3.一个客户端可以发起群聊

注意:
1.客户端只能向服务器端发送文件或字符
2.服务器端只能得到客户端发过来的数据

分析

程序流程

首先,运行服务器端;接着,运行客户端,客户端需要一个登陆机制,判断是否能登陆过;登陆失败就继续登陆,直到登陆成功;而登陆成功后,就可以选择发起私聊或者群聊

image.png

判断是否登录

判断是否已登录,就是与已存在的用户作对比;第一,需要保存下来已经登录过的用户的信息;然后,当有新用户登录时,就需要将其与已保存的用户对比;若已经登录,就登录失败,需重新登录;如果没有登陆过,就登录成功;因为要实现网络通信,必须得使用IP和端口号,所以可以借用用户名——socket这种映射关系,同时使用Map集合来保存多个映射关系,来达到目的

规范

因为我们使用的是Android Studio来编写这个代码,只有一个端口,没有一个具体的界面来实现私聊窗口或群聊窗口;而为了实现聊天功能,客户端和服务器端就必须制定一个规范,来让客户端的需求在发送的字符里面体现,如:


image.png

而之前就学习过,接口可以用来指定一套规范

代码实例

定义规范

定义一套规范,来帮助沟通客户端和服务器端

public interface ChatProtocol {
    //登录
    String LOGIN_FLAG = "u+";
    //私聊
    String PRITAVE_FLAG = "p+";
    //群聊
    String PUBLIC_FLAG = "a+";

    //分隔符
    String SPLIT_FLAG = "♥";

    //成功的状态
    String SUCCESS = "1";
    String FAILYRE = "-1";
}

定义UserManager类

管理所有登录用户的信息

import java.net.Socket;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;

/**
 * 管理所有的登录的用户Map<String, Socket>
 * 判断某个用户是否已经登录
 */

public class UserManager {
    // 保存所有用户信息
    private Map<String, Socket> users = new HashMap<>();

    /**
     * 判断用户是否已经登录
     */
    public synchronized boolean isLogined(String name){
        // 遍历数组
        for (String key: users.keySet()){
            if (key.equals(name)){
                return true;
            }
        }
        return false;
    }

    /**
     * 保存当前登录的用户信息
     */
    public void save(String name, Socket socket){
        users.put(name,socket);
    }

    /**
     * 通过用户名找到对应的socket
     */
    public synchronized Socket socketByName(String name){
        return users.get(name);
    }

    /**
     * 通过socket对象找到对应的名称
     */
    public synchronized String nameBySocket(Socket socket){
        for (String key: users.keySet()){
            // 对出这个key对应的socket
            if (socket == users.get(key)){
                return key;
            }
        }
        return null;
    }

    /**
     * 获取所有人的socket对象
     */
    public synchronized Collection<Socket> allUsers(){
        return users.values();
    }
}

服务器端

主线程

管理监听客户端的连接 判断是不是登录

public class Server {
    // 用于保存每一个用户对应的姓名和socket
    public static UserManager manager = new UserManager();

    public static void main(String[] args){
        // 胡藏剑ServerSocket
        try (ServerSocket ss = new ServerSocket(8888)){
            // 监听所有来链接的客户端
            while (true){
                Socket socket = ss.accept();

                //让子线程处理这个socket
                new ServerThread(socket).start();
            }
        } catch (IOException e) {

        }
    }
}

子线程

处理每个客户端的输入输出

class ServerThread extends Thread{
    private Socket socket;

    public ServerThread(Socket socket){
        this.socket = socket;
    }

    @Override
    public void run() {
        BufferedReader br ;
        PrintStream ps ;

        try {
            // 登录
            // 1.得到对应的输入流对象
            br = new BufferedReader(new InputStreamReader(socket.getInputStream()));

            // 得到对应的输出流
            ps = new PrintStream(socket.getOutputStream());

            String line ;
            while ((line = br.readLine()) != null){
                // 登录 u+......u+
                if (line.startsWith(ChatProtocol.LOGIN_FLAG)&&line.endsWith(ChatProtocol.LOGIN_FLAG)){
                    // String[] items = line.substring(2).split("u+");
                    // String name = items[0];
                    // 获取名字
                    String name = line.substring(2,line.length()-2);

                    // 判断用户是否已经登录
                    // 发生结果给客户端
                    if (Server.manager.isLogined(name)){
                        // 已经登录
                        ps.println(ChatProtocol.FAILYRE);
                    }else{
                        // 没有登录
                        // 保存当前登录的用户信息
                        Server.manager.save(name,socket);

                        // 发生结果给客户端
                        ps.println(ChatProtocol.SUCCESS);
                    }
                }

处理私聊

                // 判断是不是私聊
                else if (line.startsWith(ChatProtocol.PRITAVE_FLAG)&&line.endsWith(ChatProtocol.PRITAVE_FLAG)){
                    // 获取信息
                    int endIndex = line.length()-2;
                    String msg = line.substring(2,endIndex);

                    // 分割
                    String[] items = msg.split(ChatProtocol.SPLIT_FLAG);

                    // 用户名
                    String name = items[0];

                    // 聊天内容
                    String message = items[1];

                    // 通过用户名找到对应socket
                    Socket desSocket = Server.manager.socketByName(name);
                    PrintStream desPs = new PrintStream(desSocket.getOutputStream());

                    // 获取当前用户的名称
                    String currentName = Server.manager.nameBySocket(socket);

                    // 发生私聊消息
                    desPs.println(currentName+"向你发来私聊:"+message);
                }

处理群聊

                 else{
                    // 群聊
                    // 处理数据
                    int endIndex = line.length()-2;
                    String msg = line.substring(2,endIndex);

                    // 获取当前用户的名称
                    String currentName = Server.manager.nameBySocket(socket);

                    // 遍历所有的用户信息
                    Collection<Socket> sockets = Server.manager.allUsers();

                    for (Socket s : sockets){
                        PrintStream tempps = new PrintStream(s.getOutputStream());
                        tempps.println(currentName+"发来群聊:"+msg);
                    }
                }

            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

客户端

主线程

处理登录 接收终端输入 传到服务器

public class Client {
    public static void main(String[] args){
        BufferedReader br ;
        PrintStream ps ;
        BufferedReader brServer ;

        // 连接服务器
        try (Socket socket = new Socket("10.129.19.144",8888)){
            // 登录
            // 接收终端输入流
            br = new BufferedReader(new InputStreamReader(System.in));

            // 发给服务器端的输出流
            ps = new PrintStream(socket.getOutputStream());

            // 接收服务器端的输入流
            brServer = new BufferedReader(new InputStreamReader(socket.getInputStream()));

            while (true){
                // 接收终端输入信息
                String line = JOptionPane.showInputDialog("请输入姓名");

                // 拼接登录格式
                String loginStr = ChatProtocol.LOGIN_FLAG+line+ChatProtocol.LOGIN_FLAG;

                // 发生给服务器端
                ps.println(loginStr);

                // 接收服务器端返回的结果
                String result = brServer.readLine();

                // 判断登录结果
                if (result.equals(ChatProtocol.SUCCESS)){
                    System.out.println("登录成功");
                    break;
                }else{
                    System.out.println("用户名已存在 请重新登录");
                }
            }

            // 登录成功
            // 开启线程 处理服务器端的输入
            new ClientThread(socket).start();

            // 接收终端输入 发送给服务器端
            String line ;
            while ((line = br.readLine()) != null){
                // 发送给服务器
                ps.println(line);
            }

        } catch (IOException e) {

        }
    }
}

子线程

接收服务器端输入 发送到客户端

class ClientThread extends Thread{
    private Socket socket;

    public ClientThread(Socket socket){
        this.socket = socket;
    }

    @Override
    public void run() {
        BufferedReader br = null;
        try {
            br = new BufferedReader(new InputStreamReader(socket.getInputStream()));
            String line ;
            while((line = br.readLine()) != null){
                System.out.println(line);
            }
        } catch (IOException e) {
            System.out.println("网络出错");
        }finally {
            try {
                if (br != null) {
                    br.close();
                }

                if (socket != null){
                    socket.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

心得体会

今天也算是很有收获的一天,虽然有些地方没有怎么弄到,但是晚上写简书的时候还渐渐理解了

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容

  • QQ群: 功能:私聊 好友 拍照 拍视频 视频通话 发文件 QQ空间 1.每个客户端有个名称 2.私聊 向一个客...
    信1张欣欣阅读 729评论 0 0
  • 一、Python简介和环境搭建以及pip的安装 4课时实验课主要内容 【Python简介】: Python 是一个...
    _小老虎_阅读 5,734评论 0 10
  • 目的: 知道怎么使用Socket实现群聊和私聊 使用一个子线程来实现方法 定义一个规范来实现你的功能 $D$2%0...
    凤橙_52c9阅读 1,277评论 0 2
  • 一、简历准备 1、个人技能 (1)自定义控件、UI设计、常用动画特效 自定义控件 ①为什么要自定义控件? Andr...
    lucas777阅读 5,190评论 2 54
  • 世界上没有一个标准能来衡量所有的人,每个人都有着不同的身世和经历。每个人也都有着自己不完美的地方,而正是这些不完美...
    梦想晒月光阅读 5,278评论 2 0