java网络编程实现QQ发消息文件,图片等内容

前言:昨天我们已经初步学习了网络编程到底是怎么一回事儿,今天在此基础上继续学习网络编程,利用网络编程实现 QQ上的发送消息,群聊,私聊等功能。

首先我们来回顾一下昨天的两点基础知识:

1、建立服务器端
服务器建立通信ServerSocket
服务器建立Socket接收客户端连接
建立IO输入流读取客户端发送的数据
建立IO输出流向客户端发送数据消息

2、建立客户端
创建Socket通信,设置通信服务器的IP和Port
建立IO输出流向服务器发送数据消息
建立IO输入流读取服务器发送来的数据消息

群聊思路分析:
首先我们都知道,群聊是什么形式的,群聊就是一个人发的消息,所有人都可见,但是这要在程序中怎样体现这种关系呢?
首先我们必须明确知道的是,客户端只能向服务器端发送内容,服务器端只能接受客户端发送来的数据,客户端的其他需求可以在发送的字符里面体现。因为QQ的聊天中,分私聊和群聊,所以服务器端就要区分什么内容是群聊什么是私聊,谁私聊谁,为了解决这些问题,我们必须给客户端和服务器端一个统一的规范,这个统一的规范是让服务器端知道发送的是群聊消息还是私聊消息,谁私聊谁的问题。

下面我们定义的规范如下:


例如:p+ wll♥ 晚上好 p+——>服务器译为:私聊wll发送的内容是晚上好
a+ 大家好 a+——> 译为:是群发消息,发送的内容是:大家好

下面我们new一个interface,存放这些规范,如下所示:

   //定义一套规范
   public interface Chatinterface {
 //登录
 String  LOGIN_FLAG = "u+";
//私聊
 String PRIVATE_FLAG="P+";
 //群聊
 String PUBLIC_FLAG = "a+";
   // 分隔符
 String SPLTI_FLAG = "♥";
    //  成功
 String  SUCCESS = "1";
    //失败
 String  FAILE = "-1";
     }

根据昨天学习的知识,我们知道要定义服务器端和客户端,但是谁来管理登录,私聊群聊呢?所以这里我们要定义一个类来管理。
以下我们定义一个UserManager类来管理,相关注释如下所示:

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

//判断银狐是否已经登录
public 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 Socket socketName(String name) {
    return users.get(name);

}

//通过socket找到对应的名称
public 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();
  }
 }

`2.建立客户端:
我们知道客户端首先要做的就是建议socket,然后向服务器端发送数据和接收服务器端的数据。第一步:new一个Client类,第二步:创建socket,第三步:创建发送和接收的数据流,格式如下:

//1.创建用于通信的socket
    //  指明和谁通信:ip地址  端口号
    Socket socket = new Socket("192.168.43.32",8989);
    //2. 接收服务器端的数据
    BufferedReader br = new BufferedReader(new InputStreamReader(socket.getInputStream()));

    //3.读取服务器端发来的数据
    String line = null;
    while ((line = br.readLine()) != null){
        System.out.println(line);
    }

3,多线程,不管是服务器端还是客户端都要接受数据和发送数据,但早发送数据的同时使可以接受数据的,早接收数据的同时,也是可以发送数据的,所以在客户端和服务器端都要使用多线程,服务器端的子线程用来处理客户端发送来的数据:判断是否已经登录,判断是群聊还是私聊,判断私聊给谁,
1.判断是否已经登录如下:

              判断用户名是否已经登录
                if(Server.manager.isLogined(name)){
                    //登录过了
                    ps.println(Chatinterface.FAILE);
                } else {
                    //没有登录过
                    Server.manager.save(name,socket);
                    ps.println(Chatinterface.SUCCESS);
                }
            }

2.判断是不是私聊:

      //判断是不是私聊
            else if(line.startsWith(Chatinterface.PRIVATE_FLAG)&&line.endsWith(Chatinterface.PRIVATE_FLAG)){
                //用户名和聊天内容
                //获取信息
                int len =line.length()-2;
                String msg = line.substring(2,len);
                //分割
                String[] itens = msg.split(Chatinterface.SPLTI_FLAG) ;
                String name = itens[0];
                String massage = itens[1];
                //通过用户名找到对应的socket
                Socket destsocket = Server.manager.socketName(name);
                PrintStream desPS = new PrintStream(destsocket.getOutputStream());

                //获取当前用户的名称
                String currentName = Server.manager.nameBySocket(socket);
                //发的送私聊信息
                desPS.println(currentName +"想你发来私聊"+massage);

            }

因为经过第一步,说明已经登录,经过第二步:说明不是群聊,所以就只用私聊了,处理私聊的大妈如下:

                int len =line.length()-2;
                String msg = line.substring(2,len);
                String currentNmae = Server.manager.nameBySocket(socket);
                //遍历所有的用户信息
                Collection<Socket> sockets = Server.manager.allUsers();
                for(Socket s :sockets){
                    PrintStream temps = new PrintStream(s.getOutputStream());
                    temps.println(currentNmae+"发来群聊"+msg);
               }

整个代码如下:
客服端:

public class Client {
public static void main(String[] args ){
    //BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
    BufferedReader br = null;
    PrintStream ps = null;
    BufferedReader bufferedReader = null;
    try ( Socket socket = new Socket("192.168.43.32",8898);){
        br = new BufferedReader(new InputStreamReader(System.in));
        ps= new PrintStream(socket.getOutputStream());
        bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
       while (true){
           //接受终端输入信息
          String line = JOptionPane .showInputDialog("请登录");
        //   String line = br.readLine();
           //拼接登录格式
          String  loginStr = Chatinterface.LOGIN_FLAG+line+Chatinterface.LOGIN_FLAG;
           //发送给服务器

           ps.println(loginStr);

           //接受服务器端返回的结果
           String  result = bufferedReader.readLine();
            if(result.equals(Chatinterface.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 = 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) {
           }
       }
  }
}

服务器端:

public class Server {
//用于保存每个与用户姓名和socket
public static  UseraManager manager = new UseraManager();
public static void main(String[] args){
    try( ServerSocket ss = new ServerSocket(8898);) {
       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= null;
    PrintStream ps = null;
    //登录
    //1.获取输入流
    try {
        br = new BufferedReader(new InputStreamReader(socket.getInputStream()));
        ps= new PrintStream(socket.getOutputStream() );
        String line =null;
        while ((line = br.readLine()) !=null){
            if(line.startsWith(Chatinterface.LOGIN_FLAG)&&(line.endsWith(Chatinterface.LOGIN_FLAG))){
      //   String [] items= line.substring(2).split("u+");
       //   String name =items[0];
       String name = line.substring(2,line.length()-3);
       //判断用户名是否已经登录
                if(Server.manager.isLogined(name)){
                    //登录过了
                    ps.println(Chatinterface.FAILE);
                } else {
                    //没有登录过
                    Server.manager.save(name,socket);
                    ps.println(Chatinterface.SUCCESS);
                }
            }
            //判断是不是私聊
            else if(line.startsWith(Chatinterface.PRIVATE_FLAG)&&line.endsWith(Chatinterface.PRIVATE_FLAG)){
                //用户名和聊天内容
                //获取信息
                int len =line.length()-2;
                String msg = line.substring(2,len);
                //分割
                String[] itens = msg.split(Chatinterface.SPLTI_FLAG) ;
                String name = itens[0];
                String massage = itens[1];
                //通过用户名找到对应的socket
                Socket destsocket = Server.manager.socketName(name);
                PrintStream desPS = new PrintStream(destsocket.getOutputStream());

                //获取当前用户的名称
                String currentName = Server.manager.nameBySocket(socket);
                //发的送私聊信息
                desPS.println(currentName +"想你发来私聊"+massage);

            }else{
                int len =line.length()-2;
                String msg = line.substring(2,len);
                String currentNmae = Server.manager.nameBySocket(socket);
                //遍历所有的用户信息
                Collection<Socket> sockets = Server.manager.allUsers();
                for(Socket s :sockets){
                    PrintStream temps = new PrintStream(s.getOutputStream());
                    temps.println(currentNmae+"发来群聊"+msg);
               }
            }
        }
    } catch (IOException e) {
        e.printStackTrace();
    }

  }
 }

运行结果:


总结:只是点开始综合了,但是逻辑还是简单的,前面的基础只是有点遗忘的,现在的去看看前面的,有种感觉就是,学习一个只是点的时候不知道他的具体作用,到真正的实战中才能知道对这个只是点到底了解多少,还需要继续努力。

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

推荐阅读更多精彩内容

  • 计算机网络概述 网络编程的实质就是两个(或多个)设备(例如计算机)之间的数据传输。 按照计算机网络的定义,通过一定...
    蛋炒饭_By阅读 1,220评论 0 10
  • 点击查看原文 Web SDK 开发手册 SDK 概述 网易云信 SDK 为 Web 应用提供一个完善的 IM 系统...
    layjoy阅读 13,758评论 0 15
  • 1.网络编程1.1计算机网络概述网络编程的实质就是两个(或多个)设备(例如计算机)之间的数据传输。按照计算机网络的...
    任振铭阅读 425评论 0 1
  • 网络编程的概述 网络编程的实质就是用来实现网络互连的不同计算机上运行的程序间可以进行数据交换。 一.OSI网络模型...
    思念挥霍阅读 375评论 0 0
  • 6.1 公钥密钥加密原理 6.1.1 基础知识 密钥:一般就是一个字符串或数字,在加密或者解密时传递给加密/解密算...
    AndroidMaster阅读 4,011评论 1 8