Java线程池中哪些事??

2023-11-14

需要提前把线程准备好!!创建线程不是直接从系统申请,而是从池子里拿!

等到线程不用了,也是还给池子!!

池子的目的是为了提高效率

线程的创建虽然比进程轻量,但是在频繁创建的情况下,开销也是不可忽略的!!则希望还能进一步提高效率,那么,就需要用到线程池了!!

为啥从池子里拿线程比从系统创建线程更高效??

原因:从线程池那线程,存粹的用户态操作,从系统创建线程,涉及到用户态和内核态之间的切换,真正的创建,是要在内核态完成的~~

纯用户态操作时间是可控的!

涉及到内核态操作,时间就不太可控了!

标准库里提供了现成的线程池~~

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class Main {
    public static void main(String[] args) {
        ExecutorService pool= Executors.newFixedThreadPool(10);
        //此处并非直接new ExecutorService 对象,而是通过Executors类,里面的静态方法,完成对象的构造
        //工厂模式:创建对象不在new,而是使用一些其他的方法(通常为静态方法),协助我们把对象创建出来
        //Executors.newFixedThreadPool(10)构造一个固定为10线程的线程池
        //Executors.newFixedThreadPool()不设置固定值,按需创建,用完了不会立即销毁,留着以后接着用!
        pool.submit(new Runnable() {
            @Override
            public void run() {
                System.out.println("hello");
            }
        });
    }
}

需要大家主注意一下笔者的这些备注~

上述的这些工厂方法,其实都是基于一个类:ThreadPoolExecutor的封装(参数很多,还挺抽象的)

总分:多线程代码,还是挺复杂的,稍微不留点神额,就会有问题!

但是,在Java圈子里,多线程仍然是一个比较主流的并发编程模型~

线程池:从线程池取线程,是属于纯用户态操作,不涉及到和内存的交互!!

ThreadPoolExecutor:标准库提供的四种拒绝策略

ThreadPoolExecutor.AbortPolicy:如果满了,继续添加任务,添加操作直接抛出异常

ThreadPoolExecutor.CallerRunsPolicy:添加的线程自己负责执行这个任务

ThreadPoolExecutor.DiscardOldestPolicy:丢弃最老的任务(最先安排的任务)

ThreadPoolExecutor.DiscardPolicy:丢弃最新的任务

接下来,我们来自己实现一个线程池:

import java.util.concurrent.BlockingDeque;
import java.util.concurrent.LinkedBlockingDeque;

class MyThreadPool {
    //手动实现一个线程池
    //阻塞队列
    private BlockingDeque<Runnable> queue=new LinkedBlockingDeque<>();
    public void submit(Runnable runnable) throws InterruptedException{
        queue.put(runnable);
    }

    //此处实现一个固定线程数的线程池
    public MyThreadPool(int n){//n表示线程数量
        for (int i = 0; i < n; i++) {
            Thread t=new Thread(()->{
                try {
                    while (true){
                        //while可确保不停的取任务
                        Runnable runnable=queue.take();
                        runnable.run();
                    }
                }catch (InterruptedException o){
                    o.printStackTrace();
                }
            });
            //启动线程
            t.start();
        }
    }
}


public class Main1 {
    public static void main(String[] args) throws InterruptedException {
        MyThreadPool pool=new MyThreadPool(10);//创建线程池
        for (int i = 0; i < 100; i++) {
            int number=i;
            //匿名内部类,而lambad本质就是匿名内部类的简化写法!!

            pool.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println("hello + main");
                }
            });
        }
        Thread.sleep(1000);
    }
}

根据代码的运行结果:

可以看到:线程池中任务执行的顺序和添加顺序不一定相同!!非常正常!!原因:创建的这10个线程是无序调度的!

问题1:

在上述的代码中:用到了number=i;直接用不行吗??

每次循环都是创建新的number,没有人修改该number,匿名内部类也是要捕获外界变量,而变量捕获要求得是final,而此处的 i 在跟着for循环在不停的修改,因此,创建一个新的变量,让起每次for循环都变成新的!!

问题2:

当前代码中,搞了个10个线程的线程池,在实际开发中,一个线程池的线程数量设置多少是比较合适的??

肯定线程并不是越多越好!!线程的执行本质上是要在CPU上调度的(资源)!最好的做法是做测试的方式来确定!(运行下程序,计算下时间,同时监测资源使用状态)

到此为止:多线程大致结束!!

多线程初阶,主要介绍了线程的概念,及其多线程编程,多线程编程的注意事项,代码案列,算是最核心的部分了!!

面试常考+工作常用!!

多线程进阶(锦上添花),学有余力可以搞搞!!(能理解最好,不能理解就背)

主要围绕一些更深奥的面试题来展开的!

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Java线程池中哪些事?? 的相关文章

随机推荐

  • MOS管的作用及原理介绍

    MOS管的英文全称叫MOSFET Metal Oxide Semiconductor Field Effect Transistor 即金属氧化物半导体型场效应管 属于场效应晶体管中的绝缘栅型 因此 MOS管有时被称为场效应管 在一般电子电
  • Modbus RTU 工业通讯技术实现

    Modbus 是一个工业上常用的通讯协议 一种通讯约定 ModBus 协议是应用层报文传输协议 OSI 模型第7层 它定义了一个与通信层无关的协议数据单元 PDU 即PDU 功能码 数据域 ModBus 协议能够应用在不同类型的总线或网络
  • javaWeb中如何防止两个人同时操作同一条记录(限制单人操作),使用redis的解决方式

    最近在做demo的时候 碰到这么一个问题 当页面上同一角色有两个人登录的时候 他们同时操作统一条记录 简言之就是 同一记录同一时刻多人操作 这种情况在我的业务中回导致数据的冗余 干扰正常的程序运行 当时想到的解决办是锁表或者使用缓存 当然其
  • h0105 (10 分) c/c++

    c 代码 include
  • 互联网摸鱼日报(2023-03-29)

    互联网摸鱼日报 2023 03 29 InfoQ 热门话题 阿里再启组织变革 六大业务集团全面独立经营 张勇 具备条件的业务都可能独立上市 新一代Serverless事件中间件EventMesh正式毕业为Apache顶级项目 杭银消金基于
  • AVR单片机ATemga328P中断原理的介绍

    1 一AVR单片机中断原理的介绍 ATmega328P微控制器具有两个外部中断引脚 分别是INT0和INT1 外部中断0 INT0 它对应的引脚是PD2 数字引脚2 INT0可以用于响应外部信号的边沿触发 上升沿 下降沿或任意边沿 并触发相
  • 深聊全链路压测之:第二十讲

    日志隔离落地方案 1 引言 2 Demo预演 2 1 技术方案选型 2 2 Demo系统预演 2 3 扩展知识 日志分离 3 总结 1 引言 这节课 我们来学习如何基于微服务技术落地日志隔离 从第14讲开始 我们就详细的落地了基于微服务技术
  • linux服务器怎么添加路由,linux系统中添加路由的方法

    linux系统中添加路由的方法 发布时间 2020 06 17 11 38 59 来源 亿速云 阅读 95 作者 Leah 这篇文章将为大家详细讲解有关linux系统中添加路由的方法 小编觉得挺实用的 因此分享给大家做个参考 希望大家阅读完
  • Android webview实现h5视频全屏播放兼容Android7.0,自己添加webview库兼容全部版本

    2017年Android5 0主流机型webview的兼容性出现大问题导致很多公司app都不使用h5播放视频 关于谷歌自带的webview有各种兼容问题 最典型的的就是视频全屏onShowCustomView经常会点击不回调 可能是谷歌推崇
  • sonar scanner配置

    sonar scanner配置 这里记录如何配置sonar scanner扫描C C 项目代码 话不多说 先上官网链接 文章目录 sonar scanner配置 1 环境 1 1 SonarSource Build Wrapper 1 2
  • UDP及TCP通信对比讲解

    概述 TCP是面向连接的协议 也就是说在通信发送数据前 必须和对方建立连接 以数据流的模式传播 传输过程中不会有数据丢失 速率上比UDP要慢不少 适用于对数据准确性要求高 速度可以相对慢的场景 如发送或接收邮件 打电话 微信消息传输等等 U
  • 一看就懂的网络协议五层模型(一)

    我们每天使用互联网 你是否想过 它是如何实现的 全世界几十亿台电脑 连接在一起 两两通信 上海的某一块网卡送出信号 洛杉矶的另一块网卡居然就收到了 两者实际上根本不知道对方的物理位置 你不觉得这是很神奇的事情吗 互联网的核心是一系列协议 总
  • 【VSCode远程调试】【Python】Linux/Ubuntu远程服务器使用plt.show()没有反应

    plt模块结合numpy数组的可视化的常见用法 import matplotlib pyplot as plt img np array 1 2 3 4 plt imshow img plt show 以上写法在linux或者ubuntu系
  • mysql重命名一个表

    使用rename table rename table table1 altered to table 这里的sql语句会将table1 altered 重命名为table rename语句还可以将表移到另一个数据库中 rename tab
  • 14. 函数返回值为引用?

    函数返回值可以是引用吗 当然可以 只是在函数返回引用的时候需要注意几点 以下给出讨论 函数在返回值的时候 会产生一个临时变量作为函数返回值的副本 而函数在返回引用的时候 不会产生副本 那么既然是引用 那么到底是引用谁呢 首先 我们知道要清楚
  • 解决dubbo注册zookepper服务IP乱入问题的三种方式

    最近做一个项目引入了dubbo zookepper的分布式服务治理框架 在应用的发布的时候出现一个怪问题 zookepper服务是起在开发服务器192 168 23 180上 本机起应用服务提供者注册到192 168 23 180上的dub
  • 第一课:k8s介绍安装

    第一课 k8s介绍安装 tags golang 2019尚硅谷 categories K8s 文章目录 第一课 k8s介绍安装 第一节 K8s发展流程 1 1 相关概念 1 2 K8s发展历史 1 3 K8s特点 1 4 课程架构 需要掌握
  • 【廖雪峰python入门笔记】list_倒序访问

    倒序访问list 我们还是用一个list按分数从高到低表示出班里的3个同学 L Adam Lisa Bart 这时 老师说 请分数最低的同学站出来 要写代码完成这个任务 我们可以先数一数这个 list 发现它包含3个元素 因此 最后一个元素
  • MySql中json类型数据的查询以及在MyBatis-Plus中的使用

    表结构和初始数据 新建表结构 CREATE TABLE json test id int NOT NULL AUTO INCREMENT roles json DEFAULT NULL COMMENT 角色 project json DEF
  • Java线程池中哪些事??

    需要提前把线程准备好 创建线程不是直接从系统申请 而是从池子里拿 等到线程不用了 也是还给池子 池子的目的是为了提高效率 线程的创建虽然比进程轻量 但是在频繁创建的情况下 开销也是不可忽略的 则希望还能进一步提高效率 那么 就需要用到线程池