【java多线程】多线程编程实例

之前学习多线程的时候一直不知道多线程道理有哪些应用,最近发现了一个java多线程的应用,那就是使用BIO的方式开发一个客户端与服务器的简单通信模型。

所谓BIO就是同步阻塞式通信方式。所谓同步就是指在逻辑上,多个客户端可以同时发送消息;所谓阻塞,则是指服务器在建立连接后,如果没有接收到客户端的消息,那么服务端的线程将会被阻塞,直到接收到消息被唤醒。(以上是我的理解,如有不对,欢迎与我讨论)

在服务端编程时,我们可以直接在建立一个链接后就创建一个线程。但是这种方法是有缺陷的,比如当连接很多的时候,那么不停的创建和销毁线程,这个开销是很大的,并且也不方便我们进行管理(比如统计有多少连接,在服务器压力大的时候采取什么策略等等),于是我们引入了线程池。

那么,在面试中被问到线程池的作用就可以回答:减小开销,方便管理。

下面就是简单实例

client.java

// 客户端
public class client {
    public static void main(String[] args) {
        try {
            // 1.简历一个与服务端的Socket对象:套接字
            Socket socket = new Socket("127.0.0.1", 9999);
            // 2.从socket管道中获取一个输出流,写数据给服务端
            OutputStream os = socket.getOutputStream() ;
            // 3.把输出流包装成一个打印流
            PrintWriter pw = new PrintWriter(os);
            // 4.反复接收用户的输入
            BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
            String line = null ;
            while((line = br.readLine()) != null){
                pw.println(line);
                pw.flush();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

线程池处理类HandlerSocketThreadPool.java

// 线程池处理类
public class HandlerSocketThreadPool {

    // 线程池
    private ExecutorService executor;

    public HandlerSocketThreadPool(int maxPoolSize, int queueSize){

        this.executor = new ThreadPoolExecutor(
                3, // 8
                maxPoolSize,
                120L,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<Runnable>(queueSize) );
    }

    public void execute(Runnable task){
        this.executor.execute(task);
    }
    
}

yishserver.java

// 服务端
public class server {
    public static void main(String[] args) {
        try {
            System.out.println("----------服务端启动成功------------");
            ServerSocket ss = new ServerSocket(9999);

            // 一个服务端只需要对应一个线程池
            HandlerSocketThreadPool handlerSocketThreadPool =
                    new HandlerSocketThreadPool(3, 1000);

            // 客户端可能有很多个
            while(true){
                Socket socket = ss.accept() ; // 阻塞式的!
                System.out.println("有人上线了!!");
                // 每次收到一个客户端的socket请求,都需要为这个客户端分配一个
                // 独立的线程 专门负责对这个客户端的通信!!
                handlerSocketThreadPool.execute(new ReaderClientRunnable(socket));
            }

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


class ReaderClientRunnable implements Runnable{

    private Socket socket ;

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

    @Override
    public void run() {
        try {
            // 读取一行数据
            InputStream is = socket.getInputStream() ;
            // 转成一个缓冲字符流
            Reader fr = new InputStreamReader(is);
            BufferedReader br = new BufferedReader(fr);
            // 一行一行的读取数据
            String line = null ;
            while((line = br.readLine())!=null){ // 阻塞式的!!
                System.out.println("服务端收到了数据:"+line);
            }
        } catch (Exception e) {
            System.out.println("有人下线了");
        }

    }
}

以上代码来自于黑马Java-IO模式精讲(AIO&BIO&NIO),Java-IO基础+高级+实战全套教程,黑马程序员行业大牛深度精讲_哔哩哔哩_bilibili

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值