Socket聊天程序——多线程解决发送多条消息

参考博文https://www.cnblogs.com/unclejelly/p/4082095.html

开始遇到的问题——服务端与客户端的输入流有阻塞,导致聊天只能读一条写一条

原因:
比如服务端A的代码如下;

ServerSocket serverSocket = new ServerSocket(.....);
Socket socket = serverSocket.accpet();
InputStream is = socket.getInputStream();
OnputStream os = socket.getOnputStream();

os.write(); //服务端A的输出流等待服务端的输入内容,如果一直为空,则一直等待
is.read();  //服务端A的输入流等待客户端的回应内容,如果一直为空,则一直等待

Socket的输入流和系统的输入流都是阻塞的,就是会一直等待你的操作,不会立即返回,这就是阻塞(不满足我的条件,我一直等待)。

比如服务端的输入流会一直等待客户端的输出流的数据,直到有数据为止输入流才会返回并结束

现象:

服务器必须发送消息才能读取消息,客户端必须读取消息(消息内容不能为空)才能发送消息,这样给人感觉很不适

解决方法——多线程

将输出流和输入流分开,分别使用一个线程即可。

这样输入流的阻塞效果就不会影响到输出流的执行了

很明显,我们只是解决了发送多条消息,并没有解决输入流的阻塞问题,但我觉得这样并不影响

功能,但不确保是否有所纰漏。知道的朋友可以留下评论。

服务端

public class ServerSocketTest {

    public static void main(String[] args) throws IOException {
        InetAddress localHost = InetAddress.getLocalHost();
        //中间的“10”参数是backlog,限制连接服务端的客户端数量
        ServerSocket serverSocket = new ServerSocket(8080, 10, localHost); //初始化服务器的接口
        //设置accept()等待连接时间
//        serverSocket.setSoTimeout(3000);

        System.out.println("服务器等待连接中......");
        Socket server = serverSocket.accept(); //服务器通信和客户端一样使用Socket类
        System.out.println("检测到客户端连接!");

        Thread send = new Thread(new SendThread(server));
        Thread accept = new Thread(new AcceptThread(server));

        //线程优先级
        send.setPriority(Thread.MAX_PRIORITY);
        accept.setPriority(Thread.MIN_PRIORITY);

        send.start();
        accept.start();

    }
}

客户端

public class SocketTest {

    public static void main(String[] args) throws IOException {
        InetAddress localHost = InetAddress.getLocalHost();
        Socket client = new Socket(localHost, 8080);
//        SocketAddress remoteSocketAddress = client.getRemoteSocketAddress(); //获取远程主机信息——DESKTOP-4MCSL0K/192.168.93.1:8080

        Thread send = new Thread(new SendThread(client));
        Thread accept = new Thread(new AcceptThread(client));
        send.setPriority(Thread.MAX_PRIORITY);
        accept.setPriority(Thread.MIN_PRIORITY);
        send.start();
        accept.start();

    }
}

发送消息线程

public class SendThread implements Runnable{
    private static BufferedReader br = new BufferedReader(new InputStreamReader(System.in));

    private Socket socket;

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

    @Override
    public void run() {
        String msg = "";
        System.out.println("发送消息线程开启!");
        try {
        	//很奇怪,我用其他流不可以进行Socket通信
            DataOutputStream dataOutputStream = new DataOutputStream(socket.getOutputStream()); 
            while (!msg.equals("exit")){
                msg = br.readLine();
                dataOutputStream.writeUTF(msg);
            }
            System.out.println("发送消息线程已结束!");
            socket.close();

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

    }
}

接收消息线程

public class AcceptThread implements Runnable{

    private Socket socket;

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

    @Override
    public void run() {
        try {
            System.out.println("接收消息进程开启!");
            //很奇怪,我用其他流不可以进行Socket通信
            DataInputStream dataInputStream = new DataInputStream(socket.getInputStream());
            while (!socket.isClosed()){
                try {
                    String s = dataInputStream.readUTF();
                    System.out.println("[" + socket.getInetAddress().getHostAddress() + "] " + s);
                }catch (Exception e){
                    System.out.println("接收消息进程结束");
                    break;
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }

    }
}
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值