java TCP做聊天程序是应该一个聊天窗口一个线程还是一个Socket一个线程

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了java TCP做聊天程序是应该一个聊天窗口一个线程还是一个Socket一个线程相关的知识,希望对你有一定的参考价值。

若代码中S用static修饰,开启两个客户端程序static会互相影响?
附代码
http://pan.baidu.com/s/1mikySkg

socket通讯是阻塞的,一个聊天就有一个socket,需要一直在哪里阻塞读取数据,所以就一个socket就要用一个线程了。如果用nio的话,通讯是非阻塞的,就不用一个聊天一个线程了。如果聊天人数大的话,建议用nio。现在netty框架就非常不错。 参考技术A java提供了两种IO方式,一种是基于阻塞式的(BIO),一种是基于非阻塞式的(NIO),基于阻塞式的socket会一直阻塞在等待客户端的链接上,如果客户端有链接过来则分配一个线程去处理,然后又继续阻塞;如果是基于NIO的socket,会用到缓冲技术,socket将会被放到通道上,即使客户端没有链接过来,也不会一直的去阻塞,会一直的去轮询,直到通道可以进行读写为止,所以可以做到一条线程去处理多个链接。。。如果是用BIO的方式就是一个scoket一条线程,如果是NIO的话就不用考虑。 参考技术B 你是用swing来做的么?
应该是这么说,一个socket一个线程。
服务器开启中。。。
客户端1启动的时候,创建socket客户端,发起请求,并且显示窗口。
你可以在服务端用线程池管理socket数量。。
参考技术C 其实无论怎么做都是可以的,只是那种更加优化.
通常的做法一个socket一个线程就可以了.根据命令请求解析内容.把不同内容放到不同的聊天窗口中.
另外如果只是聊天的话,建议用udp比较好.可以采用udp协议+应答机制.
参考技术D 理论上是一个用户一个socket,而不是窗口。
1、必须采用短连接;
2、采用TCP做聊天窗口不合理,建议UDP;
3、设置服务端的会话超时,以便于缓解会话连接数增加的压力;
4、线程的增加按业务需求而定,不是必须使用。

《Java项目实践》:简单聊天程序

《Java项目实践》:简单聊天程序

由于这个简单程序,还是涉及到很多的知识点,下面我们就一点一点的来完成。

我们熟悉的QQ聊天,就有一个界面,是吧,我们自己做一个简单的聊天程序,因此我们也就需要为Client写一个界面。因此,这就是我们第一步需要完成的任务。

第一步:为Client端写一个界面

完成一个界面有两种方法,一种是使用Frame对象来完成,另外一种是继承JFrame类来完成.本项目使用第二种。

第二种继承JFrame类完成的界面的程序如下:

    public class ChatClient extends JFrame{

        private TextField tf=null;
        private TextArea ta=null;
        public static void main(String[] args) {
            new ChatClient().launch();
        }

        private void launch() {
            this.setTitle("client端");
            this.setLocation(300, 400);
            this.setSize(300, 400);

            tf=new TextField();
            ta=new TextArea();

            this.add(BorderLayout.SOUTH,tf);
            this.add(BorderLayout.NORTH, ta);
            pack();//窗口自动适应大小,使窗口能正好显示里面所有的控件。
            this.setVisible(true);

        }

    }

第一种使用JFrame对象来实现与第二种类似,代码如下:

    public class ChatClient_v1 {

        private TextField tf =null;
        private TextArea ta = null;
        public static void main(String[] args) {
            new ChatClient_v1().launch();
        }

        private void launch() {
            JFrame frame=new JFrame();
            frame.setTitle("client端");
            frame.setLocation(300,400);
            frame.setSize(300,400);

            tf = new TextField();
            ta = new TextArea();
            frame.add(BorderLayout.SOUTH,tf);
            frame.add(BorderLayout.NORTH, ta);
            frame.pack();
            frame.setVisible(true);
        }

    }

运行结果如下:

技术分享

这样,我们的第一步就算完成了。

在QQ聊天中,当我们在输入消息窗口中输入好消息后,当我们按下回车键后,消息第一会出现在我们窗口的上面,第二会发送个服务器端。

下面我们先完成这两个功能。

2、第二步:为Client端的输入窗口添加回车事件

为TextField添加回车事件有两种方法,具体看这篇博文:http://blog.sina.com.cn/s/blog_908652640100ut12.html

public class ChatClient extends JFrame{

        private TextField tf=null;
        private TextArea ta=null;
        public static void main(String[] args) {
            new ChatClient().launch();
        }

        private void launch() {
            this.setTitle("client端");
            this.setLocation(300, 400);
            this.setSize(300, 400);
            /*
             * 关闭窗口的两种方法
             * */
            //this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            this.addWindowListener(new WindowAdapter(){//为了关闭窗口
               public void windowClosing(WindowEvent e)
               {
                   System.exit(0);
               }
              });

            tf=new TextField();
            ta=new TextArea();

            //为TextField添加回车事件响应
            tf.addActionListener(new ActionListener() {

                @Override
                public void actionPerformed(ActionEvent event) {
                    String content = tf.getText();
                    /*
                     * 判断TextArea中是否有内容,如果有,则需要先加入一个换行符,
                     * 然后再加入内容,否则直接加入内容
                     * */
                    if(ta.getText().trim().length()!=0){
                        ta.setText(ta.getText()+"\n"+content);
                    }
                    else{
                        ta.setText(content);
                    }

                    tf.setText("");
                }
            });

            this.add(BorderLayout.SOUTH,tf);
            this.add(BorderLayout.NORTH, ta);
            pack();//窗口自动适应大小,使窗口能正好显示里面所有的控件。
            this.setVisible(true);

        }

    }

技术分享

这样就完成了为TextField添加回车事件。

为了和QQ聊天更加贴近,我们还对其显示在TextArea中进行了一些细节的处理(例如,TextArea中的内容为原先的内容经过换行后添加新的内容等)。本想和QQ一样,Client端发送的内容显示在TextArea的右边,但是没有找到相应的方法。

下面进行下一步的实现,将内容发送到Server端。

但是,将Client端的内容发送的Server端,有如下的几个小的步骤需要实现,例如,Client端和Server端先需要建立连接,然后才能通信,是吧。因此,我们想完成建立连接的过程。

第三步:Client端和Server端建立连接

首先先创建一个Server类,代码如下:

    public class ChatServer {

        private ServerSocket server=null;


        public static void main(String[] args) {
            new ChatServer().start();
        }

        private void start(){
            boolean b_serverStart = false;
            try {
                //监听本地地址且端口号为8888
                server = new ServerSocket(8888);
                b_serverStart = true;
            } catch (IOException e) {
                e.printStackTrace();
            }

            while(b_serverStart){
                //等待客户端的连接
                Socket client = null;
                try {
                    client = server.accept();
                } catch (IOException e) {
                    e.printStackTrace();
                }  
                System.out.println("与客户端连接成功!");  
            }

        }

    }

而Client端主要添加了一个connect方法

    private void connect() {
            //客户端请求与本机在8888端口建立TCP连接 
            try {
                client = new Socket("127.0.0.1", 8888);
                client.setSoTimeout(10000);  
            } catch (UnknownHostException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }

Client端完整代码如下:

    public class ChatClient extends JFrame{

        private TextField tf = null;
        private TextArea ta = null;
        private Socket client =null;

        public static void main(String[] args) {
            new ChatClient().launch();
        }

        private void launch() {
            this.setTitle("client端");
            this.setLocation(300, 400);
            this.setSize(300, 400);
            /*
             * 关闭窗口的两种方法
             * */
            //this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            this.addWindowListener(new WindowAdapter(){//为了关闭窗口
               public void windowClosing(WindowEvent e)
               {
                   System.exit(0);
               }
              });

            tf=new TextField();
            ta=new TextArea();

            //为TextField添加回车事件响应
            tf.addActionListener(new ActionListener() {

                @Override
                public void actionPerformed(ActionEvent event) {
                    String content = tf.getText();
                    /*
                     * 判断TextArea中是否有内容,如果有,则需要先加入一个换行符,
                     * 然后再加入内容,否则直接加入内容
                     * */
                    if(ta.getText().trim().length()!=0){
                        ta.setText(ta.getText()+"\n"+content);
                    }
                    else{
                        ta.setText(content);
                    }

                    tf.setText("");
                }
            });

            this.add(BorderLayout.SOUTH,tf);
            this.add(BorderLayout.NORTH, ta);
            pack();//窗口自动适应大小,使窗口能正好显示里面所有的控件。
            this.setVisible(true);

            connect();

        }

        private void connect() {
            //客户端请求与本机在8888端口建立TCP连接 
            try {
                client = new Socket("127.0.0.1", 8888);
                client.setSoTimeout(10000);  
            } catch (UnknownHostException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }

    }

这样就建立了一个Server端和Client的连接。

当Server端启动成功后,支持任意多个Client端来建立连接。

以上就是Server端和Client端建立连接,接下来需要完成的任务就是Client端给Server端发送数据

第四步:Client端给Server端发送数据

在Client端,主要是在TextField的的Listener中的代码上面添加一些代码,代码如下:

    private class ClientListener implements ActionListener{

                @Override
                public void actionPerformed(ActionEvent event) {
                    String content = tf.getText();
                    /*
                     * 判断TextArea中是否有内容,如果有,则需要先加入一个换行符,
                     * 然后再加入内容,否则直接加入内容
                     * */
                    if(ta.getText().trim().length()!=0){
                        ta.setText(ta.getText()+"\n"+content);
                    }
                    else{
                        ta.setText(content);
                    }

                    tf.setText("");
                //往服务器端发送数据 
                    try {
                        bw = new BufferedWriter(
                                new OutputStreamWriter(client.getOutputStream()));
                    } catch (IOException e) {
                        e.printStackTrace();
                    }

                    try {
                        bw.write(content);
                        bw.flush();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    try {
                        bw.close();
                        client.close();
                    } catch (IOException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }


                }
            }

Client端完整代码如下:

    public class ChatClient extends JFrame{

        private TextField tf = null;
        private TextArea ta = null;
        private Socket client =null;
        private BufferedWriter bw = null;

        public static void main(String[] args) {
            new ChatClient().launch();
        }

        private void launch() {
            this.setTitle("client端");
            this.setLocation(300, 400);
            this.setSize(300, 400);
            /*
             * 关闭窗口的两种方法
             * */
            //this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            this.addWindowListener(new WindowAdapter(){//为了关闭窗口
               public void windowClosing(WindowEvent e)
               {
                   System.exit(0);
               }
              });

            tf=new TextField();
            ta=new TextArea();

            //为TextField添加回车事件响应
            tf.addActionListener(new ClientLisenter());

            this.add(BorderLayout.SOUTH,tf);
            this.add(BorderLayout.NORTH, ta);
            pack();//窗口自动适应大小,使窗口能正好显示里面所有的控件。
            this.setVisible(true);

            connect();

        }

        private void connect() {
            //客户端请求与本机在8888端口建立TCP连接 
            try {
                client = new Socket("127.0.0.1", 8888);
                client.setSoTimeout(10000);  
            } catch (UnknownHostException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }

        private class ClientLisenter implements ActionListener{

                @Override
                public void actionPerformed(ActionEvent event) {
                    String content = tf.getText();
                    /*
                     * 判断TextArea中是否有内容,如果有,则需要先加入一个换行符,
                     * 然后再加入内容,否则直接加入内容
                     * */
                    if(ta.getText().trim().length()!=0){
                        ta.setText(ta.getText()+"\n"+content);
                    }
                    else{
                        ta.setText(content);
                    }

                    tf.setText("");

                    try {
                        bw = new BufferedWriter(
                                new OutputStreamWriter(client.getOutputStream()));
                    } catch (IOException e) {
                        e.printStackTrace();
                    }

                    try {
                        bw.write(content);
                        bw.flush();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    try {
                        bw.close();
                        client.close();
                    } catch (IOException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }


                }
            }


    }

而在Server端的代码如下,代码思想为:Server端每接收一个Client的连接请求,就开启一个线程来进行数据的接收处理。

    public class ChatServer {

        private ServerSocket server = null;

        public static void main(String[] args) {
            new ChatServer().start();
        }

        private void start(){
            boolean b_serverStart = false;
            try {
                //监听本地地址且端口号为8888
                server = new ServerSocket(8888);
                b_serverStart = true;
            } catch (IOException e) {
                //解决打开多个Server端的情况
                System.out.println("请关闭已经打开的Server端,重新开启");
                System.exit(0);
            }

            while(b_serverStart){
                //等待客户端的连接,如果没有获取连接  
                Socket client = null;
                try {
                    client = server.accept();
                } catch (IOException e) {
                    e.printStackTrace();
                }  
                System.out.println("与客户端连接成功!");  
              //为每个客户端连接开启一个线程  
                new Thread(new ServerThread(client)).start();  
            }

            try {
                server.close();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }

        private class ServerThread implements Runnable{
            private Socket client = null;
            private DataInputStream dis = null;
            private boolean b_connect = false;

            public  ServerThread(Socket client) {
                this.client=client;
                try {
                    dis = new DataInputStream(this.client.getInputStream());
                    b_connect=true;
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            @Override
            public void run() { 

                try{
                    while(b_connect){  
                        //接收从客户端发送过来的数据  
                        String str= dis.readUTF();
                        if(str == null || "".equals(str)){  
                            b_connect = false;  
                        }else{  
                            if("bye".equals(str)){  
                                b_connect = false;  
                            }else{  
                                //将接收到的字符串前面加上输出到控制台
                                System.out.println(str);
                            }  
                        }  
                    }
                }
                catch(IOException e){
                    System.out.println("client端关闭了");
                    //e.printStackTrace();
                }
                finally{
                    try{
                        if(dis!=null) dis.close();
                        if(client!=null) client.close();
                    }
                    catch(IOException e){
                        e.printStackTrace();
                    }               
                }
            }           
        }

    }

但是,里面有一个bug,Server端只能接收Client端的发送的第一条语句,但发送第二条语句时,就报如下的错误:

    java.net.SocketException: Socket is closed
    at java.net.Socket.getOutputStream(Unknown Source)
    at client.ChatClient$ClientLienter.actionPerformed(ChatClient.java:92)
    at java.awt.TextField.processActionEvent(Unknown Source)
    at java.awt.TextField.processEvent(Unknown Source)
    at java.awt.Component.dispatchEventImpl(Unknown Source)
    at java.awt.Component.dispatchEvent(Unknown Source)
    at java.awt.EventQueue.dispatchEventImpl(Unknown Source)
    at java.awt.EventQueue.access$500(Unknown Source)
    at java.awt.EventQueue$3.run(Unknown Source)
    at java.awt.EventQueue$3.run(Unknown Source)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
    at java.awt.EventQueue$4.run(Unknown Source)
    at java.awt.EventQueue$4.run(Unknown Source)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
    at java.awt.EventQueue.dispatchEvent(Unknown Source)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(Unknown Source)
    at java.awt.EventDispatchThread.pumpEventsForFilter(Unknown Source)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(Unknown Source)
    at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
    at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
    at java.awt.EventDispatchThread.run(Unknown Source)
    java.io.IOException: Stream closed
    at java.io.BufferedWriter.ensureOpen(Unknown Source)
    at java.io.BufferedWriter.write(Unknown Source)
    at java.io.Writer.write(Unknown Source)
    at client.ChatClient$ClientLienter.actionPerformed(ChatClient.java:98)

即java.net.SocketException: Socket is closed 和java.io.IOException: Stream closed。

因此,将ChatClient端最后的代码注释掉就OK了。

                try {
                    br.close();
                    client.close();  
                } catch (IOException e) {
                    e.printStackTrace();
                } 

现在又出现了新的问题,当我们输入多条消息时,Server端不读取消息,而是等数据达到一定长度或者是Client关闭后,才读取消息。

将BufferedReader/BufferedWrite换成 DataInputStream/DataOutputStream就可以了。

完整代码如下:

    public class ChatClient extends JFrame{

        private TextField tf = null;
        private TextArea ta = null;
        private Socket client =null;
        private DataOutputStream  dos = null;

        public static void main(String[] args) {
            new ChatClient().launch();
        }

        private void launch() {
            this.setTitle("client端");
            this.setLocation(300, 400);
            this.setSize(300, 400);
            /*
             * 关闭窗口的两种方法
             * */
            //this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            this.addWindowListener(new WindowAdapter(){//为了关闭窗口
               public void windowClosing(WindowEvent e)
               {
                   disconnect();
                   System.exit(0);
               }
              });

            tf=new TextField();
            ta=new TextArea();

            //为TextField添加回车事件响应
            tf.addActionListener(new ClientLisenter());

            this.add(BorderLayout.SOUTH,tf);
            this.add(BorderLayout.NORTH, ta);
            pack();//窗口自动适应大小,使窗口能正好显示里面所有的控件。
            this.setVisible(true);

            connect();

        }

        private void connect() {
            //客户端请求与本机在8888端口建立TCP连接 
            try {
                client = new Socket("127.0.0.1", 8888);
                client.setSoTimeout(10000); 
                dos = new DataOutputStream(client.getOutputStream());
            } catch (UnknownHostException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }

        private void disconnect(){  

                try {
                    if(dos!=null) dos.close();
                    if(client!=null) client.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }

        }

        private class ClientLisenter implements ActionListener{

                @Override
                public void actionPerformed(ActionEvent event) {
                    String content = tf.getText();
                    /*
                     * 判断TextArea中是否有内容,如果有,则需要先加入一个换行符,
                     * 然后再加入内容,否则直接加入内容
                     * */
                    if(ta.getText().trim().length()!=0){
                        ta.setText(ta.getText()+"\n"+content);
                    }
                    else{
                        ta.setText(content);
                    }

                    tf.setText("");

                    try {
                        dos.writeUTF(content);
                        dos.flush();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }

                }
            }


    }

以上就是Client端发送数据到Server端的代码实现。

最后来实现下Server端发送数据到Client端。

第五步:Server端发送数据到Client端

有了上面的基础,Server端发送数据到Client端就比较好实现了

Client端的代码主要是在上一个版本的基础上添加了一个receiver 函数,这个函数使用了一个while循环来做。

    public class ChatClient extends JFrame{

        private TextField tf = null;
        private TextArea ta = null;
        private Socket client =null;
        private DataOutputStream  dos = null;
        private DataInputStream dis = null;

        private boolean b_conn =false;

        public static void main(String[] args) {
            new ChatClient().launch();
        }

        private void launch() {
            this.setTitle("client端");
            this.setLocation(300, 400);
            this.setSize(300, 400);
            /*
             * 关闭窗口的两种方法
             * */
            //this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            this.addWindowListener(new WindowAdapter(){//为了关闭窗口
               public void windowClosing(WindowEvent e)
               {
                   disconnect();
                   System.exit(0);
               }
              });

            tf=new TextField();
            ta=new TextArea();

            //为TextField添加回车事件响应
            tf.addActionListener(new ClientLisenter());

            this.add(BorderLayout.SOUTH,tf);
            this.add(BorderLayout.NORTH, ta);
            pack();//窗口自动适应大小,使窗口能正好显示里面所有的控件。
            this.setVisible(true);

            connect();

            receiver();

        }

        private void receiver() {
            while(b_conn){
                String str = null;
                try {
                    //从服务器端接收数据有个时间限制(系统自设,也可以自己设置),超过了这个时间,便会抛出该异常  
                    str = dis.readUTF();
                } catch (IOException e) {
                    System.out.println("Time out, No response"); 
                }
                System.out.println(str);
            }
        }

        private void connect() {
            //客户端请求与本机在8888端口建立TCP连接 
            try {
                client = new Socket("127.0.0.1", 8888);
                client.setSoTimeout(10000); 
                dos = new DataOutputStream(client.getOutputStream());
                dis = new DataInputStream(this.client.getInputStream());
                b_conn = true;
            } catch (UnknownHostException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }

        private void disconnect(){  

                try {
                    b_conn = false;
                    if(dos!=null) dos.close();
                    if(client!=null) client.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }

        }

        private class ClientLisenter implements ActionListener{

                @Override
                public void actionPerformed(ActionEvent event) {
                    String content = tf.getText();
                    /*
                     * 判断TextArea中是否有内容,如果有,则需要先加入一个换行符,
                     * 然后再加入内容,否则直接加入内容
                     * */
                    if(ta.getText().trim().length()!=0){
                        ta.setText(ta.getText()+"\n"+content);
                    }
                    else{
                        ta.setText(content);
                    }

                    tf.setText("");

                    try {
                        dos.writeUTF(content);
                        dos.flush();

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

                }
            }


    }   

Server端的代码如下:

    public class ChatServer {

        private ServerSocket server = null;

        public static void main(String[] args) {
            new ChatServer().start();
        }

        private void start(){
            boolean b_serverStart = false;
            try {
                //监听本地地址且端口号为8888
                server = new ServerSocket(8888);
                b_serverStart = true;
            } catch (IOException e) {
                //解决打开多个Server端的情况
                System.out.println("请关闭已经打开的Server端,重新开启");
                System.exit(0);
            }

            while(b_serverStart){
                //等待客户端的连接,如果没有获取连接  
                Socket client = null;
                try {
                    client = server.accept();
                } catch (IOException e) {
                    e.printStackTrace();
                }  
                System.out.println("与客户端连接成功!");  
              //为每个客户端连接开启一个线程  
                new Thread(new ServerThread(client)).start();  
            }

            try {
                server.close();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }

        private class ServerThread implements Runnable{
            private Socket client = null;
            private DataInputStream dis = null;
            private DataOutputStream dos = null;
            private boolean b_connect = false;

            public  ServerThread(Socket client) {
                this.client=client;
                try {
                    dis = new DataInputStream(this.client.getInputStream());
                    dos = new DataOutputStream(client.getOutputStream()); 
                    b_connect=true;
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            @Override
            public void run() { 

                try{
                    while(b_connect){  
                        //接收从客户端发送过来的数据  
                        String str= dis.readUTF();
                        if(str == null || "".equals(str)){  
                            b_connect = false;  
                        }else{  
                            if("bye".equals(str)){  
                                b_connect = false;  
                            }else{  
                                //将接收到的字符串前面加上输出到控制台
                                System.out.println(str);
                                //将Client端发送过来的数据+时间发送回去
                                dos.writeUTF("server在时间"+new Date().toString()+"收到信息:"+str);
                            }  
                        }  
                    }
                }
                catch(IOException e){
                    System.out.println("client端关闭了");
                    //e.printStackTrace();
                }
                finally{
                    try{
                        if(dis!=null) dis.close();
                        if(dos!=null) dos.close();
                        if(client!=null) client.close();
                    }
                    catch(IOException e){
                        e.printStackTrace();
                    }               
                }
            }           
        }

    }

以上就实现了简单Client端和Server端的通信。

最后一步:Server端实现消息的转发

前一个版本实现了Server端和Client的通信,但是,离我们的聊天程序还差一点点。

QQ聊天的流程时这样,

技术分享

因此,在Server端,还需要将Client端发送来的数据转发给其它的客户端。

在Server端,

1、首先需要将连接此服务器端的所有Client端记录下来。

2、然后,如果有Client发送消息,则将消息广播出去。

有一个细节需要注意:如果某个Client端关闭了,则在Server端保存有所有Client端记录的结果中需要将此Client端的记录移出。

实现的代码如下:

    public class ChatServer {

        private ServerSocket server = null;

        //记录所有连接上此服务器的客户端
        private List<ServerThread> serverThreads = new ArrayList<ServerThread>();

        public static void main(String[] args) {
            new ChatServer().start();
        }

        private void start(){
            boolean b_serverStart = false;
            try {
                //监听本地地址且端口号为8888
                server = new ServerSocket(8888);
                b_serverStart = true;
            } catch (IOException e) {
                //解决打开多个Server端的情况
                System.out.println("请关闭已经打开的Server端,重新开启");
                System.exit(0);
            }

            while(b_serverStart){
                //等待客户端的连接,如果没有获取连接  
                Socket client = null;
                try {
                    client = server.accept();
                } catch (IOException e) {
                    e.printStackTrace();
                }  
                System.out.println("与客户端连接成功!");  
              //为每个客户端连接开启一个线程  
                ServerThread st = new ServerThread(client);
                serverThreads.add(st);
                new Thread(st).start();  
            }

            try {
                server.close();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }

        private class ServerThread implements Runnable{
            private Socket client = null;
            private DataInputStream dis = null;
            private DataOutputStream dos = null;
            private boolean b_connect = false;

            public  ServerThread(Socket client) {
                this.client=client;
                try {
                    dis = new Dat

以上是关于java TCP做聊天程序是应该一个聊天窗口一个线程还是一个Socket一个线程的主要内容,如果未能解决你的问题,请参考以下文章

关于Java聊天收到消息自动弹出窗口代码

Java案例:基于TCP的简单聊天程序

基于TCP协议的服务器中转简易聊天

Java实现一个简单的网络聊天程序

求一个C#最简单的TCP传输信息例子(实现聊天和传送文件,再简单点实现聊天也行)

使用python实现一个hello/hi的简单的网络聊天程序