多用户即时通讯系统04

4.编码实现03

4.5功能实现-群聊功能实现

4.5.1思路分析

群聊的实现思路和私聊的实现非常类似。

不同的是:私聊时,服务端接收到消息后,只需要找出接收方的socket并发送消息即可

群聊时,服务端在接收到消息后需要遍历集合中所有的线程,找出除了发送方的所有客户端的socket,并发送消息

群聊思路:

  • 客户端 - 发送者:
    • 用户在控制台输入信息,客户端接收内容
    • 将消息构建成Messgae对象,通过对应的socket发送给服务器
  • 服务器:
    • 读取客户端(发送者)发送给所有用户(接收者)的消息
    • 从管理线程的集合中,遍历所有线程,获取所有socket(除了发送者本身)
    • 将Message对象转发给所有的接收者
  • 客户端 - 所有接收者:
    • 所有接受者分别在线程(通信线程)中,读取到发送者的message消息,并显示即可

4.5.2代码实现

1.客户端:
1.修改MessageType接口

在接口中增加新的消息类型

String MESSAGE_TO_ALL_MES = "7";//表示群发消息包
2.修改MessageClientService类

在该类中增加sendMessageToAll方法,实现群发功能

/**
 * 群发消息功能
 * @param content 内容
 * @param senderId 发送者
 */
public void sendMessageToAll(String content,String senderId){
    //构建 message
    Message message = new Message();
    message.setMesType(MessageType.MESSAGE_TO_ALL_MES);//设置消息类型是群发消息
    message.setSender(senderId);
    message.setContent(content);
    message.setSendTime(new Date().toString());//发送时间也封装到 message对象中
    System.out.println(senderId + " 对大家说 " + content);

    //发送给服务端
    try {//在管理线程的集合中,通过userId来获取线程,通过线程来获取对应的socket,再通过socket获取输出流
        ObjectOutputStream oos =
                new ObjectOutputStream(ManageClientConnectServerThread.getClientConnectServerThread(senderId).getSocket().getOutputStream());
        oos.writeObject(message);
    } catch (IOException e) {
        e.printStackTrace();
    }
}
3.修改ClientConnectServerThread类

在该类的run方法中增加新的逻辑业务,增加接收群发消息类型的判断,并在控制台显示

 else if (message.getMesType().equals(MessageType.MESSAGE_TO_ALL_MES)) {
    //接收到的是群发的消息
    //就把服务器转发的消息,显示到控制台即可
    System.out.println("n" + message.getSendTime() + "n" + message.getSender()
            + " 对大家说: " + "n" + message.getContent());
} 

image-20220923163214511

4.修改QQView类

在该类的内层循环中,调用群发功能的方法:

case "2":
    System.out.println("请输入想对大家说的话");
    String s = Utility.readString(100);
    //调用一个方法,将消息封装成 message对象,发给服务端
    messageClientService.sendMessageToAll(s,userId);
    break;

image-20220923163420586

2.服务端:
1.修改MessageType接口

在接口中增加新的消息类型

String MESSAGE_TO_ALL_MES = "7";//表示群发消息包
2.修改ServerConnectClientThread类

在该类中增加新的业务逻辑

else if (message.getMesType().equals(MessageType.MESSAGE_TO_ALL_MES)) {
    //业务四:客户请求群发消息需要遍历管理线程的集合,把所有线程的socket都得到,然后将 message进行转发即可
    //得到hm
    HashMap<String, ServerConnectClientThread> hm = ManageClientThreads.getHm();
    //遍历
    Iterator<String> iterator = hm.keySet().iterator();
    while (iterator.hasNext()) {
        //取出所有userId
        String onlineUserId = iterator.next().toString();
        //取出除了发送者的所有用户id
        if (!onlineUserId.equals(message.getSender())) {
            //转发message
            //从集合中取出线程,在线程中取出socket,根据socket获得输出流,将socket的输出流转化为对象输出流
            ObjectOutputStream oos =
                    new ObjectOutputStream(hm.get(onlineUserId).getSocket().getOutputStream());
            oos.writeObject(message);
        }
    }
} 

image-20220923163737014

3.修改ManageClientThreads类

在该类中增加方法,获取集合

//返回hashmap
public static HashMap<String ,ServerConnectClientThread> getHm(){
    return hm;
}

运行程序:

1.运行服务端,进行监听

image-20220923164142319

2.运行三个客户端,登录三个用户

image-20220923164420287

3.在 用户uid=100 的账号发送群发消息

image-20220923164716432

4.其他用户也接收到了消息

image-20220923165201230image-20220923165230471

5.服务器端

image-20220923165354965

功能实现完毕

内容来源于网络如有侵权请私信删除

文章来源: 博客园

原文链接: https://www.cnblogs.com/liyuelian/p/16723966.html

你还没有登录,请先登录注册
  • 还没有人评论,欢迎说说您的想法!