热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

JUC并发编程——线程的基本方法使用

目录一、线程名称设置和获取二、线程的sleep()三、线程的interrupt四、join()五、yield()六、wait(),notify(),notifyAll(

目录

一、线程名称设置和获取

二、线程的sleep()

三、线程的interrupt

四、join()

五、yield()

六、wait(),notify(),notifyAll()



一、线程名称设置和获取

1、线程名称一般在线程启动前设置,但也允许为正在运行的线程设置名称。在实际开发中,在使用多线程开发时,一定要自定义线程名称,偏于查找日志。

2、线程允许使用同样的名字,但应该尽量避免。

3、如果线程没有命名,系统会自动为线程设置名称。例如Thread-0,Thread-1。在SpringBoot中http-nio-端口号-exec-1,http-nio-端口号-exec-2。

线程命名

package com.xiaojie.juc.thread.base;/*** @author xiaojie* @version 1.0* @description: 线程命名和获取名称* @date 2021/12/17 22:54*/
public class ThreadNameDemo {public static void main(String[] args) {A a = new A();new Thread(a,"mythraed-a").start();new Thread(a,"mythraed-b").start();}static class A implements Runnable{@Overridepublic void run() {//获取线程名称System.out.println("当前线程的名称是>>>>>>>>:"+Thread.currentThread().getName());}}
}

线程池命名

package com.xiaojie.juc.thread.base;import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.atomic.AtomicInteger;/*** @author xiaojie* @version 1.0* @description: 线程池命名* @date 2021/12/17 23:09*/
public class ThreadPoolNameDemo {private static final AtomicInteger threadNumber &#61; new AtomicInteger(1);private static final String MYTHREAD_POOL_GROUP &#61; "mythread-pool-group-thread-";public static void main(String[] args) {ExecutorService executorService &#61; Executors.newFixedThreadPool(3, r -> new Thread(new ThreadGroup(""), r,MYTHREAD_POOL_GROUP &#43; threadNumber.getAndIncrement(),0));for (int i &#61; 0; i <10; i&#43;&#43;) {executorService.execute(() -> System.out.println("当前线程名称>>>>>>>>>:" &#43; Thread.currentThread().getName()));}//关闭线程池executorService.shutdown();}
}

二、线程的sleep()

package com.xiaojie.juc.thread.base;import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;/*** &#64;author xiaojie* &#64;version 1.0* &#64;description: 线程sleep方法* &#64;date 2021/12/17 23:46*/
public class ThreadSleepDemo {public static void main(String[] args) {ExecutorService executorService &#61; Executors.newFixedThreadPool(3);for (int i &#61; 0; i <10; i&#43;&#43;) {executorService.execute(() -> {try {if (Thread.currentThread().isInterrupted()) {System.out.println("Thread is Interrupted");}//一定要捕捉InterruptedException异常Thread.sleep(1000);} catch (InterruptedException e) {}System.out.println(Thread.currentThread().getName());});}//关闭线程池executorService.shutdown();}
}

sleep的作用是让目前正在执行的线程休眠&#xff0c;让CPU去执行其他任务&#xff0c;使线程从运行状态进去阻塞状态。当线程睡眠时间满后&#xff0c;线程并不一定是立刻马上开始执行&#xff0c;因为CPU可能正在执行其他任务&#xff0c;线程进入就绪状态&#xff0c;等待CPU分配时间片后才能执行。

三、线程的interrupt

Thread提供了stop()&#xff0c;但是是个过时方法&#xff0c;不推荐使用。stop()方法是强制中断线程&#xff0c;不管当前线程是否正在运行&#xff0c;有可能导致当前线程持有当前锁&#xff0c;强行中断后可能不会释放锁。interrupt()并不是用来中断一个线程&#xff0c;而是将线程设置为中断状态。

1、如果执行interrupt()时&#xff0c;当前线程处于阻塞状态&#xff08;wait(),sleep(),join()&#xff09;时&#xff0c;线程会立刻退出阻塞&#xff0c;抛出InterruptedException异常。

2、如果调用interrupt()时&#xff0c;线程处于运行状态&#xff0c;线程不受任何影响&#xff0c;只是会将中断标识设置为true,可以调用isInterrupted(),判断是否中断。

package com.xiaojie.juc.thread.base;/*** &#64;author xiaojie* &#64;version 1.0* &#64;description: 线程中断* &#64;date 2021/12/18 0:12*/
public class InterruptDemo {public static void main(String[] args) {A a &#61; new A();a.start();}static class A extends Thread {&#64;Overridepublic void run() {//手动中断线程Thread.currentThread().interrupt();try {while (true) {if (Thread.currentThread().isInterrupted()) {System.out.println("线程中断了");break;}}Thread.sleep(3000);System.out.println(Thread.currentThread().getName());} catch (InterruptedException e) {e.printStackTrace();System.out.println("线程中断了。。。。。when sleep");}}}
}

四、join()

 线程A在运行时&#xff0c;线程Bjoin(),之后&#xff0c;线程A会等待直到线程B运行完之后&#xff0c;才开始执行。如果线程B无限制运行&#xff0c;线程A可以设置等待时长join(1000)单位是毫秒。

package com.xiaojie.juc.thread.base;/*** &#64;author xiaojie* &#64;version 1.0* &#64;description:线程join模拟* &#64;date 2021/12/18 0:45*/
public class ThreadJoinDemo {public static void main(String[] args) {new A().start();}static class A extends Thread {&#64;Overridepublic void run() {try {B b &#61; new B();b.start();b.join();//B线程join之后&#xff0c;A线程需要等待B运行完之后才执行} catch (InterruptedException e) {e.printStackTrace();}System.out.println("线程A执行任务。。。。。。。");}}static class B extends Thread {&#64;Overridepublic void run() {try {Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("线程B执行任务。。。。。。。");}}
}

五、yield()

yield的作用是让正在执行的线程让出CPU的执行权限&#xff0c;使得CPU可以去执行其他任务。从线程状态来看&#xff0c;线程从运行变为就绪状态。线程在yield()方法之后&#xff0c;线程放弃和重占CPU是不确定的&#xff0c;可能是刚刚放弃CPU&#xff0c;马上又获得CPU执行权。yield不会阻塞线程&#xff0c;只会让当前线程暂停一下&#xff0c;让系统线程调度器重新调度一次。

package com.xiaojie.juc.thread.base;/*** &#64;author xiaojie* &#64;version 1.0* &#64;description: Yield实例* &#64;date 2021/12/18 1:13*/
public class ThreadYieldDemo {public static void main(String[] args) {Thread threadA &#61; new Thread(new A());threadA.setPriority(1);threadA.setName("threadA ");Thread threadB &#61; new Thread(new B());threadB.setName("threadB ");threadA.start();threadB.start();}static class A implements Runnable {&#64;Overridepublic void run() {//暂停一下,重新调度线程Thread.yield();System.out.println(Thread.currentThread().getName());}}static class B implements Runnable {&#64;Overridepublic void run() {System.out.println(Thread.currentThread().getName());}}
}

六、wait(),notify(),notifyAll()

这三个方法不是Thread独有的方法&#xff0c;是Object的方法&#xff01;&#xff01;&#xff01;

当一个对象实例调用了wait()方法后&#xff0c;当前线程会在这个对象上等待。直到其他线程调用这个实例的notify(),或者notifyAll()方法后才会唤醒。调用了wait()方法后&#xff0c;该线程就会进入该实例的等待队列&#xff0c;如果队列中有多个线程&#xff0c;notify()只会随机的唤醒一个线程。这个选择是不公平的&#xff0c;不是先等待的就会先唤醒。notifyAll(),是唤醒所有的等待线程。而且这些方法只能在synchronized代码块中使用。wait&#xff08;&#xff09;和sleep()都会让线程等待&#xff0c;但是wait()方法会释放目标对象的锁&#xff0c;而sleep()方法不会释放任何资源。

package com.xiaojie.juc.thread.base;/*** &#64;author xiaojie* &#64;version 1.0* &#64;description: wait demo* &#64;date 2021/12/18 1:43*/
public class ThreadWaitDemo {private static final Object obj &#61; new Object();public static void main(String[] args) {new Thread(new A()).start();new Thread(new B()).start();}static class A implements Runnable {&#64;Overridepublic void run() {synchronized (obj) {System.out.println("线程A开始等待。。。。。。");try {obj.wait();} catch (InterruptedException e) {e.printStackTrace();}System.out.println("线程A等待结束。。。。。。");System.out.println("线程A开始执行。。。。。。");}}}static class B implements Runnable {&#64;Overridepublic void run() {synchronized (obj) {try {obj.notify();Thread.sleep(2000);System.out.println("线程B运行。。。。。");} catch (InterruptedException e) {e.printStackTrace();}}}}
}

参考&#xff1a;《JAVA高并发核心编程&#xff08;卷2&#xff09;&#xff1a;多线程、锁、JMM、JUC、高并发设计》-尼恩编著


推荐阅读
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
  • Explore how Matterverse is redefining the metaverse experience, creating immersive and meaningful virtual environments that foster genuine connections and economic opportunities. ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 深入理解 SQL 视图、存储过程与事务
    本文详细介绍了SQL中的视图、存储过程和事务的概念及应用。视图为用户提供了一种灵活的数据查询方式,存储过程则封装了复杂的SQL逻辑,而事务确保了数据库操作的完整性和一致性。 ... [详细]
  • 本文详细介绍了Java中org.w3c.dom.Text类的splitText()方法,通过多个代码示例展示了其实际应用。该方法用于将文本节点在指定位置拆分为两个节点,并保持在文档树中。 ... [详细]
  • Python 异步编程:深入理解 asyncio 库(上)
    本文介绍了 Python 3.4 版本引入的标准库 asyncio,该库为异步 IO 提供了强大的支持。我们将探讨为什么需要 asyncio,以及它如何简化并发编程的复杂性,并详细介绍其核心概念和使用方法。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 技术分享:从动态网站提取站点密钥的解决方案
    本文探讨了如何从动态网站中提取站点密钥,特别是针对验证码(reCAPTCHA)的处理方法。通过结合Selenium和requests库,提供了详细的代码示例和优化建议。 ... [详细]
  • 本文详细介绍了 Dockerfile 的编写方法及其在网络配置中的应用,涵盖基础指令、镜像构建与发布流程,并深入探讨了 Docker 的默认网络、容器互联及自定义网络的实现。 ... [详细]
  • c# – UWP:BrightnessOverride StartOverride逻辑 ... [详细]
  • 使用 Azure Service Principal 和 Microsoft Graph API 获取 AAD 用户列表
    本文介绍了一段通用代码示例,该代码不仅能够操作 Azure Active Directory (AAD),还可以通过 Azure Service Principal 的授权访问和管理 Azure 订阅资源。Azure 的架构可以分为两个层级:AAD 和 Subscription。 ... [详细]
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
author-avatar
敦恒
招聘!招聘!招聘! 联系方式:56226135/shiziqian@hy9z.com 请优先邮件联系~谢谢!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有