Java学习之路 —— 多线程

文章目录

  • 1. 线程创建方式
    • 1.1 继承Thread
    • 1.2 声明一个实现Runnable接口的类
    • 1.3 利用Callable接口、FutureTask类来实现
  • 2. 线程同步
    • 2.1 同步代码块
    • 2.2 同步方法
    • 2.3 Lock锁
  • 3. 线程同步
  • 4. 线程池

1. 线程创建方式

1.1 继承Thread

定义子类,继承Thread,创建对象,并调用start启动线程

  • 优点:编码简单
  • 缺点:线程类已经继承Thread,无法继承其他类,不利于功能的扩展
public class Main {// main方法是有一条默认的主线程执行的public static void main(String[] args) {// 1. 创建线程类的对象,代表一个线程Thread t = new MyThread();// 2. 启动线程t.start();for (int i = 0; i < 10000; i++) {System.out.println("主线程输出:" + i);}}
}

注意,是调用Thread的start方法,而不是run方法!!!

public class MyThread extends Thread{@Overridepublic void run() {for (int i = 0; i < 10000; i++) {System.out.println("子线程输出:" + i);}super.run();}
}

1.2 声明一个实现Runnable接口的类

  • 优点:只是实现了一个接口,还可以继承一个类,实现其他接口,扩展性强
  • 缺点:需要多创建一个Runnable对象
class MyRunnable implements Runnable {@Overridepublic void run() {for (int i = 0; i < 10000; i++) {System.out.println("子线程输出:" + i);}}
}
static void test_Runnable() {
//        Runnable target = new MyRunnable(); // 任务对象(不是线程对象)
//        new Thread(target).start();// lambdanew Thread(() -> {for (int i = 0; i < 10000; i++) {System.out.println("子线程输出:" + i);}for (int i = 0; i < 10; i++) {System.out.println("主线程输出:" + i);}}

1.3 利用Callable接口、FutureTask类来实现

  • 优点:可以返回线程执行完后的结果
  • 缺点:编码复杂
    在这里插入图片描述
class MyCallable implements Callable<String> {int n;public MyCallable(int n) {this.n = n;}@Overridepublic String call() throws Exception {// 描述线程的任务,返回线程执行后的结果// 求1-n的和int sum = 0;for(int i = 1; i <= n; i++)sum += n;return "1~n的总和是:" + sum;}
}
static void test_Callble() {// 1. 创建Callable对象MyCallable call = new MyCallable(50);// 2. 把Callable对象封装成FutureTask对象// 1. 是一个任务对象,实现了Runnable对象// 2. 可以在线程执行完后,调用get方法获取FutureTask<String> f1 = new FutureTask<>(call);new Thread(f1).start();for (int i = 0; i < 10; i++) {System.out.println("主线程输出:" + i);}// 获取线程执行完毕后的结果String s = null;try {// 会等到线程执行完毕,这行代码再执行s = f1.get();} catch (Exception e) {e.printStackTrace();}System.out.println(s);}

我发现啊,Java的多线程和C++的多线程不一样的点是,Java中创建子线程,如果主线程先执行完了,子线程没有执行完,子线程会继续执行;但是C++会被终止。

不过join方法都有着相同的用处,那就是阻塞主线程,等待线程执行完毕后,再执行主线程后面的代码。

2. 线程同步

2.1 同步代码块

synchronized(同步锁) {访问共享资源的核心代码
}

这个同步锁是一个字符串也可以(双引号在内存中存在常量区,只有一份),只要是一个唯一对象即可 。
但最好是用共享资源作为锁,比如说this

如果要调用静态方法,同步锁采用类名.class,锁住整个class。

2.2 同步方法

把访问共享资源的核心方法上锁,保证线程安全。这样能保证一个对象中,只有一个方法在执行,其他方法都无法执行。

修饰符 synchronized 返回值类型 方法名(形参列表){操作共享资源得到代码
}

2.3 Lock锁

Lock是一个接口类,可以用实现类ReentrantLock来实例化一个锁,来使用。

Lock lk = new ReentrantLock();
lk.lock();
lk.unlock();

3. 线程同步

来了来了,条件遍历来了。注意,一定要和锁搭配使用。
在这里插入图片描述
来一个经典的例子吧,2个线程交替打印A和B

package TestDemo;// 2个线程交替打印A和B10次
public class Test2 {public static int count;static final Object lock = new Object(); // 锁public static void main(String[] args){Thread t1 = new Thread(() -> {for (int i = 0; i < 10; i++) {synchronized (lock) {System.out.println(Thread.currentThread().getName() + "打印A");lock.notify();try {if(i < 9)lock.wait();} catch (Exception e) {e.printStackTrace();}}}}, "A线程");Thread t2 = new Thread(() -> {for (int i = 0; i < 10; i++) {synchronized (lock) {System.out.println(Thread.currentThread().getName() + "打印B");lock.notify();try {if(i < 9)lock.wait();} catch (Exception e) {e.printStackTrace();}}}}, "B线程");t1.start();t2.start();}
}

4. 线程池

JDK5提供了代表线程池的接口:ExecutorService。比较常用的实现类是ThreadPoolExecutor

  • public ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue, ThreadFactory threadFactory, RejectedExecutionHandler handler)
    • corePoolSize:指定线程池的核心线程数量
    • maximumPoolSize:指定线程池的最大线程数量
    • keepAliveTime:指定临时线程的存活时间
    • unit:指定临时线程存货的时间单位
    • workQueue:指定线程池的任务队列
    • threadFactory:指定线程池的任务工厂
    • handler:指定线程池的拒绝策略(任务队列满了后,新任务来了怎么处理)

在这里插入图片描述线程池默认是不会死亡的,除非调用shutdown(),或者shutdownNow()

  1. 临时线程什么时候创建?
    新任务提交时发现核心线程都在忙,任务队列也满了,并且还可以创建临时线程,此时才会创建临时线程,去执行新任务,而不是任务队列的任务(插队)。
  2. 什么时候会开始拒绝新任务?
    核心线程和临时线程都在忙,任务队列也满了,新的任务过来的时候才会开始拒绝任务。

Executors
是线程池的一个工具类,提供了很多静态方法用于返回不同特点的线程池对象。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/151415.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android描边外框stroke边线、rotate旋转、circle圆形图的简洁通用方案,基于Glide与ShapeableImageView,Kotlin

Android描边外框stroke边线、rotate旋转、circle圆形图的简洁通用方案&#xff0c;基于Glide与ShapeableImageView&#xff0c;Kotlin 利用ShapeableImageView专门处理圆形和外框边线的特性&#xff0c;通过Glide加载图片装载到ShapeableImageView。注意&#xff0c;因为要描边…

Docker 容器中的网络优化与 DNS 缓存清理

在使用Docker 18.03.1-ce版本在Ubuntu 18.04 LTS上运行多个Docker容器时&#xff0c;我发现当使用requests库发送请求到某个主机名时&#xff0c;响应速度非常慢。在本例中&#xff0c;每个容器都有自己的CherryPy服务器&#xff0c;并通过requests.get(http://main:8083)或req…

Nginx 413 Request Entity Too Large

当出现上图时候 更改nginx config 文件 在http{}或者server{}或者location{}中增加client_max_body_size 100m; 然后重启nginx 服务就好了

轻松搭建短域名短链接服务系统,可选权限认证,并自动生成证书认证把nginx的http访问转换为https加密访问,完整步骤和代码

轻松搭建短域名短链接服务系统&#xff0c;可选权限认证&#xff0c;并自动生成证书认证把nginx的http访问转换为https加密访问&#xff0c;完整步骤和代码。 在互联网信息爆炸的时代&#xff0c;网址复杂而冗长&#xff0c;很难在口头告知他人&#xff0c;也难以分享到社交媒体…

11 月 18 日 ROS 学习笔记——可视化和调试工具

文章目录 前言一、调试 ROS 节点1. gdb 调试器2. 在 ROS 节点启动时调用 gdb 调试器3. 在 ROS 节点启动时调用 valgrind 分析节点4. 设置 ROS 节点 core 文件转储5. 日志消息1). 输出日志消息2). 设置调试消息级别 二、检测系统状态1. rqt_graph2. 可视化坐标变换3. 保存与回放…

trzsz支持文件拖动到终端进行上传,类似lrzsz

考虑到 LapTop -> Host 1 -> Host 2 -> Docker -> TMUX&#xff0c;使用scp或sftp命令不方便&#xff1b;使用rz和sz命令就会方便很多&#xff0c;但是却又与 TMUX 不兼容&#xff08;备注&#xff1a;Tmux是一个终端复用工具&#xff0c;允许用户在一个终端窗口中…

初学UE5 C++②

目录 导入csv表格数据 创建、实例化、结构体 GameInstance Actor camera 绑定滚轮控制摇臂移动 碰撞绑定 角色碰撞设定 按钮 UI显示 单播代理 多播和动态多播 写一个接口 其他 NewObject 和 CreateDefaultSubobject区别 导入csv表格数据 创建一个object的C类 …

设计模式-责任链-笔记

动机&#xff08;Motivation&#xff09; 在软件构建过程中&#xff0c;一个请求可能被多个对象处理&#xff0c;但是每个请求在运行时只能有个接受者&#xff0c;如果显示指定&#xff0c;将必不可少地带来请求者与接受者的紧耦合。 如何使请求的发送者不需要指定具体的接受…

视频剪辑方法:一键批量调整色调的高效技巧

在视频剪辑的过程中&#xff0c;色调调整是一项非常重要的工作。它能够改变影片的氛围、情感和视觉效果&#xff0c;更好地沉浸在影片的情境中。然而&#xff0c;对于许多视频剪辑师来说&#xff0c;批量调整色调是一项非常繁琐的任务&#xff0c;需要耗费大量的时间和精力。色…

Docker Desktop 配置阿里云镜像加速

阿里云搜索镜像&#xff0c;打开容器镜像服务&#xff0c;复制镜像加速器地址 Docker Desktop 右上角设置&#xff0c;选择 Docker Engine&#xff0c;在配置中添加阿里云的镜像地址&#xff0c;右下 Apply & restart 即可。 "registry-mirrors": ["https…

vmware workstation pro 17.5 安装 macos 13.5.2 虚拟机超详细图文教程

前言 本文很细&#xff0c;甚至有点墨迹&#xff0c;主要为了方便从来没用过 vmware 的新人&#xff0c;其实大部分步骤和正常安装虚拟机没有区别&#xff0c;详细贴图以方便大家对比细节 参考文章 感谢大佬们的无私分享 https://blog.csdn.net/qq_19731521/article/details…

idea中误删.iml和.idea文件,如何处理

目录 一、问题描述 二、解决方案 1、理论知识 &#xff08;1&#xff09;.iml 文件 &#xff08;2&#xff09;.idea文件 2、操作环境 3、操作步骤 &#xff08;1&#xff09;找到【Maven】工具按钮 &#xff08;2&#xff09;点图标&#xff0c;重复导入maven项目&am…

大批量合并识别成一个表或文档的方法

金鸣表格文字识别系统功能强大&#xff0c;其中可以将上百张图片或上百页PDF中的表格文字合并识别成一个表格或文档的功能尤其受到广大用户的欢迎&#xff0c;那应该怎么操作呢&#xff1f; 一、打开金鸣表格文字识别软件&#xff0c;点击左上角的“表格识别”&#xff0c;选择…

多因素方差分析(Multi-way Analysis of Variance) R实现

1, data0507 flower 是某种植物在两个海拔和两个气温下的开花高度&#xff0c;采用合适 的统计方法&#xff0c;检验该种植物的开花高度在不同的海拔之间和不同的气温之间有无差异&#xff1f;如果有差异&#xff0c;具体如何差异的&#xff1f;&#xff08;说明依据、结论等关…

Codeforces Round 908 (Div. 2)

一个教训&#xff1a;做题的时候一定要自己模拟一遍所有样例&#xff0c;这样思路出来的很快&#xff01;&#xff01;&#xff01; C. Anonymous Informant Example input Copy 6 5 3 4 3 3 2 3 3 100 7 2 1 5 5 6 1 1 1 1 1 1000000000 1 8 48 9 10 11 12 13 14 …

【springboot笔记】程序可用性检测ApplicationAvailability

1.背景 springboot-3.1.5 ApplicationAvailability LivenessState ReadinessState AvailabilityChangeEvent 我们可以通过ApplicationAvailability获取当前应用程序的可用性&#xff0c;这个可用性包括ApplicationContext和对外请求路由两种。 LivenessState 是表示Applicatio…

打开文件 和 文件系统的文件产生关联

补充1&#xff1a;硬件级别磁盘和内存之间数据交互的基本单位 OS的内存管理 内存的本质是对数据临时存/取&#xff0c;把内存看成很大的缓冲区 物理内存和磁盘交互的单位是4KB&#xff0c;磁盘中未被打开的文件数据块也是4KB&#xff0c;所以磁盘中页帧也是4KB&#xff0c;内存…

吴恩达《机器学习》8-7:多元分类

在机器学习领域&#xff0c;经常会遇到不止两个类别的分类问题。这时&#xff0c;需要使用多类分类技术。本文将深入探讨多类分类&#xff0c;并结合学习内容中的示例&#xff0c;了解神经网络在解决这类问题时的应用。 一、理解多类分类 多类分类问题是指当目标有多个类别时…

Vue3 常用组件

一、Fragment组件 Vue2 的template 模板中必须要有一个根标签&#xff0c;而我们在Vue3 的模板中不需要使用根标签就能渲染&#xff0c;因为Vue3 在内部会将多个标签包含在一个Fragment 虚拟元素中。 好处就在于可以减少标签的层级&#xff0c;减小内存占用。 二、Teleport组…

使用cli批量下载GitHub仓库中所有的release

文章目录 1\. 引言2\. 工具官网3\. 官方教程4\. 测试用的网址5\. 安装5.1. 使用winget安装5.2. 查看gh是否安装成功了 6\. 使用6.1. 进行GitHub授权6.1.1. 授权6.1.2. 授权成功6.2 查看指定仓库中的所有版本的release6.2.1. 默认的30个版本6.2.2. 自定义的100个版本6.3 下载特定…