Java BIO 使用socket实现服务端与客户端长连接

Client端

package com.athairen.bio;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
import java.nio.charset.StandardCharsets;
import java.util.Locale;
import java.util.Scanner;
import java.util.Stack;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadPoolExecutor;

public class BioClient {

    private static final String EXIT_STR="exit";

    public static void main(String[] args) throws IOException {
        ExecutorService threadPool= Executors.newCachedThreadPool();
        Socket socket=new Socket("127.0.0.1",6666);
        OutputStream outputStream = socket.getOutputStream();
        InputStream inputStream = socket.getInputStream();


        threadPool.execute(new Runnable() {
            @Override
            public void run() {
                byte[] buffer=new byte[1024];
                int length;
                try {
                    while ((length=inputStream.read(buffer))!=-1) {
                        String s=new String(buffer,0, length);
                        System.out.println(s);
                    }
                } catch (IOException e) {
                    e.printStackTrace();
                }

            }
        });
        System.out.println("输入消息:");
        String inputStr="";
        while (!inputStr.toLowerCase(Locale.ROOT).equals(EXIT_STR)){
            Scanner scanner=new Scanner(System.in);
            inputStr = scanner.next();
            outputStream.write(inputStr.getBytes(StandardCharsets.UTF_8));
            outputStream.flush();
        }

        inputStream.close();
        outputStream.close();
        socket.shutdownInput();
        socket.shutdownOutput();
        if(!socket.isClosed()){
            socket.close();
        }
        System.out.println("client programe exit");

    }

}

server端

package com.athairen.bio;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class BioServer {

    public static void main(String[] args) throws IOException {
        //create a thread pool
        ExecutorService threadPool = Executors.newCachedThreadPool();
        ServerSocket serverSocket = new ServerSocket(6666);
        System.out.println("server start");
        while (true) {
            final Socket client = serverSocket.accept();
            System.out.println("one client connected server");
            threadPool.execute(new Runnable() {
                @Override
                public void run() {
                    try {
                        handler(client);
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            });
        }
    }


    public static void handler(Socket client) throws IOException {

        if(!client.isConnected()){
            client.shutdownInput();
            client.shutdownOutput();
            if(!client.isClosed()){
                client.close();
            }
        }
        System.out.println("thread:: " + Thread.currentThread().getId() + " -- " + Thread.currentThread().getName());
        OutputStream outputStream = client.getOutputStream();
        InputStream inputStream = client.getInputStream();
        byte[] buffer = new byte[1024];
        int len;
        while ((len = inputStream.read(buffer)) != -1) {
            String s=new String(buffer, 0, len);
            System.out.println(s);
            outputStream.write(("hello,client,welcome!! your send content is: " + s).getBytes(StandardCharsets.UTF_8));
            outputStream.flush();
        }


    }

}

需要注意的地方:

短连接与长连接的不同:短连接将在shutdow流之后对应的socket也将会被关闭,所以我们在使用短连接时应该在流和socket关闭后,重新连接即new 一个新的socket,而长连接不应该关闭流和socket对象也不能shutdown流,而应该在写buffer时flush output流,注意!!,因此长连接的内存占用会高一些,也由于BIO的同步阻塞导致我们的服务器不能处理大量并发,因此更推荐使用NIO(同步非阻塞)框架netty,或AIO(异步非阻塞)。

注:

NIO更适合处理大量请求且少量资源(处理高并发字符串消息)的情况

AIO更适合处理大量请求且大量资源的情况(处理高并发且处理图片数据)的情况,但目前尚不成熟。

        

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您可以使用 JavaSocket 编程实现客户端服务端的通信,以实现基本的客户端-服务端模型。以下是一个简单的示例: 服务端代码: ```java import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; public class Server { public static void main(String[] args) { try { ServerSocket serverSocket = new ServerSocket(8888); // 创建 ServerSocket System.out.println("Server started"); while (true) { // 开启循环,等待客户端连接 Socket socket = serverSocket.accept(); // 监听客户端连接 System.out.println("Client connected: " + socket.getInetAddress().getHostAddress()); new ServerThread(socket).start(); // 创建一个新线程处理客户端请求 } } catch (IOException e) { e.printStackTrace(); } } } class ServerThread extends Thread { private Socket socket; public ServerThread(Socket socket) { this.socket = socket; } @Override public void run() { try { // 处理客户端请求,可以使用 socket.getInputStream() 和 socket.getOutputStream() 读写数据 } catch (IOException e) { e.printStackTrace(); } finally { try { socket.close(); // 关闭 socket } catch (IOException e) { e.printStackTrace(); } } } } ``` 客户端代码: ```java import java.io.IOException; import java.net.Socket; import java.net.UnknownHostException; public class Client { public static void main(String[] args) { try { Socket socket = new Socket("localhost", 8888); // 创建 Socket,连接到服务端 System.out.println("Connected to server"); // 可以使用 socket.getInputStream() 和 socket.getOutputStream() 读写数据 } catch (UnknownHostException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 这是一个简单的示例,您可以根据您的需求进行修改和扩展。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值