用Socket实现QQ群聊和私聊功能

目的:

  • 知道怎么使用Socket实现群聊和私聊
  • 使用一个子线程来实现方法
  • 定义一个规范来实现你的功能
  • $D$2%0Y2C6FCD0VNJNTHPKS.jpg

技术:

  • 客户端只能向服务器端发送文件或者字符
  • 服务器端只能够得到客户端发来的数据
  • 必须客户端和服务器端有一个规范
    客户端的需求可以在发送的字符里面体现
    1.登录 u+ 姓名 u+
    2.返回结果 成功1 失败-1
    3.私聊 p+姓名♥聊天内容 p+
    4.群聊 a+ 聊天内容 a+
    5.发文件 f+
    6.发语音 v+
public interface ChartProtocal {
    //登录
    String LOGTN_FLAG="u+";
    //私聊
    String PRIVATE_FLAG="p+";
    //群聊
    String PUBLIC_FLAG="a+";
    //分隔符
    String SPLIT_FLAG="♥";
    //成功状态
    String SUCCESS="1";
    String FAILURE="-1";
}
  • 定义UserManager类 管理理所有登录⽤用的信息
public class UserManager {
    //用于保存每一个用户对应的姓名和Socket
    private  Map<String, Socket> users=new HashMap<>();

    //判断某个用户是否已经登录
    public synchronized boolean isLodined(String name){
        //遍历数组
        for (String key:users.keySet()){
            if (key.equals(name)){
                return true;
            }
        }
        return false;
    }
    //保存当前登录的用户信息
    public synchronized 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();
    }
}
  • 定义Server类来实现服务器端
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){

        }
    }
}
  • 定义Client来实现客户端
public class Client {
    public static void main(String[] args){
        BufferedReader br=null;
        PrintStream ps=null;
        BufferedReader brServer=null;
       //连接服务器
        try( Socket socket=new Socket("127.0.0.1",8888)){
           //登录
            //从接收终端输入流
            ps=new PrintStream(socket.getOutputStream());
            //发送给服务器端的输出流
            brServer=new BufferedReader(new InputStreamReader(socket.getInputStream()));
            //接收服务器端的输入流
            br=new BufferedReader(new InputStreamReader(System.in));
            while (true){
                //接收终端输入的信息
                String line= JOptionPane.showInputDialog("请输入姓名");
                //拼接登录格式
                String loginStr=ChartProtocal.LOGTN_FLAG+line+ChartProtocal.LOGTN_FLAG;
                //发送给服务器端
                ps.println(loginStr);
                //接收服务器端的结果
                String reault=brServer.readLine();
                //判断登录结果
                if (reault.equals(ChartProtocal.SUCCESS)){
                    System.out.println(line+"登录成功");
                    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=null;
            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){

            }
        }

    }
}
  • 定义一个子线程来实现服务器端的操作
    • 登录的操作:
class ServerThread extends Thread{
    private Socket socket;

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

    @Override
    public void run() {

        BufferedReader br=null;
        PrintStream ps=null;
        try {
            //1.获取对应的输入流对象
             br=new BufferedReader(new InputStreamReader(socket.getInputStream()));
             ps=new PrintStream(socket.getOutputStream());
             String line=null;
             while ((line=br.readLine())!=null){
                 //登录u+...u+
                 if (line.startsWith(ChartProtocal.LOGTN_FLAG)&&line.endsWith(ChartProtocal.LOGTN_FLAG)){
//                     String[] items=line.substring(2).split("u+");
//                     String name=items[0];
                     String name=line.substring(2,line.length()-2);
                     //判断用户是否登录
                     if (Server.manager.isLodined(name)){
                         //登录过了
                         ps.println(ChartProtocal.FAILURE);
                     }else {
                         //没有登录
                         //保存当前登录的用户信息
                         Server.manager.save(name,socket);
                         System.out.println(name+"已登录");
                         ps.println(ChartProtocal.SUCCESS);

                     }
                 }
  • 私信的操作:
                 //p+...♥...p+
                 //判断是不是私聊
                 else if (line.startsWith(ChartProtocal.PRIVATE_FLAG)&&line.endsWith(ChartProtocal.PRIVATE_FLAG)){
                    //获取信息
                     String msg=line.substring(2,line.length()-2);
                     //分割
                     String[] items=msg.split(ChartProtocal.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 {
                     //a+...a+
                     //群聊
                     //处理数据
                     String msg=line.substring(2,line.length()-2);
                     //获取当前用户的名称
                     String currentName=Server.manager.nameBySocket(socket);
                     //获取所有的用户信息
                     Collection<Socket> sockets=Server.manager.allusers();
                     for (Socket s:sockets){
                         PrintStream temps=new PrintStream(s.getOutputStream());
                         temps.println(currentName+"发来群聊:"+msg);
                         //temps.close();
                     }
                 }
             }
        } catch (IOException e) {
            e.printStackTrace();
        }finally {

        }
    }
}

最后的结果:

1566463805836.gif

总结:

在昨天的基础上进一步进行了今天这个群聊和私聊的操作。在自己重新写了一遍昨天的代码过后再来听今天的课程还算是能够跟上,虽然说还是有许多地方不懂尤其是当创建多个类然后这边看一下那边看一下感觉很容易就懵了,不过能够看了一遍代码然后自己慢慢的跟着写出来其实这也是一种成就感,感觉挺厉害的一些代码写完然后计算机就会跟着你的代码来实现非常的棒。

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

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,084评论 1 32
  • 下雨的天空 沉重的眼眸 昨夜我是否曾哭泣 寂寥的世界 沉寂的心灵 我是否在回忆昨晚 听雨落的声音 看远处的孤景 我...
    米夏小雨阅读 355评论 0 1
  • 大家知道吗?在风水师眼中,就算是一个小小的植物也能够具有满满的生气活力,而且有些植物一不小心还可以给你带来财运哦!...
    0a911c2e79ad阅读 223评论 0 1
  • 从严密的逻辑思维里跳出来,本周来感受一下张爱玲的笔触。我一直喜欢在文学作品里找共鸣,有时为了一只跳骚爬上我生命华美...
    Yuki酱就这样阅读 130评论 0 1
  • 一 要想坚持一件事情,最归根结底的就是你想要做这样一件事情。 怎么样确定一件事情是你想要做的,我有一个最基本的准则...
    青年_1ea0阅读 756评论 1 2