首页 > 编程语言 >java线程详解

java线程详解

时间:2023-07-26 17:55:12浏览次数:49  
标签:java Thread void t1 详解 线程 new public

java线程详解

线程

概念

说到线程,就不得不提进程,为什么呢,因为进程是操作系统进行分配资源和调度的最小单位,比如windows系统安装的应用软件(office、qq、微信等)启动时,由操作系统协调分配资源和调度执行称之为一个进程,进程间是相互独立和隔离的。而线程是进程最小执行单位,一个进程的执行过程中可以有多个线程,这样可以发挥多核CPU的能力,提高执行效率。
java中的线程不是由操作系统直接调度,而且通过java虚拟机与操作系统进行指令交互完成。所以对于java程序员来说,使用线程非常简单,只需要在语言层面编写完代码,交给虚拟机运行,剩下的脏活累活在底层就由java虚拟机完成,使用线程一时爽,一直使用一直爽(哈哈,虽然多线程能充分压榨CPU,但是用不好的话也会产生许多问题,比如并发导致的数据错误、系统负载飙升等)。

基本用法

有两种方式来创建线程
1、一种是实现Runnable接口,然后利用Thread类的构造函数传入Runnable接口创建Thread实例。
2、另外一种是继承Thread

1、实现Runnable

class WorkerThread1 implements Runnable {
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName() + "执行完成");
    }
}

2、继承Thread

class WorkerThread2 extends Thread {
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName() + "执行完成");
    }
}

测试用例:

public static void main(String[] args) {
    test1();
}

private static void test1() {
    new Thread(new WorkerThread1(), "t1").start();
    Thread t2 = new WorkerThread2();
    t2.setName("t2");
    t2.start();
}

输出:
注意这里的执行顺序不一定是t1先输出,也可能是t2先输出,因为线程启动后只是准备就绪,最终需要等待操作系统调度执行才能执行。

t1执行完成
t2执行完成

这两种方式都可以创建线程并执行,细心的读者可能看过源码发现其实Thread类也实现了Runnable接口。如果自己的类已经继承了别的类,那么可以实现Runnable接口创建线程,否则可以继承Thread类复写run()方法即可。

注意:要让线程执行需要调用start()方法,这样虚拟机才能创建一个线程等待操作系统调度,直接执行run()方法则是在当前线程直接调用该方法,同步执行,不会再创建线程。

线程状态及流转

线程的生命周期可用状态表示,总共有6种状态,
Thread类的源码里我们可以看到有个枚举类State

public enum State {
    // 新建状态,被new出来后,还未调用start方法
    NEW,
    // 可运行状态,线程已就绪,获取到CPU资源就运行,运行中就是Running状态
    RUNNABLE,
    // 阻塞状态,比如在等待锁对象,或者读取流等待
    BLOCKED,
    // 等待状态,比如在等待锁对象,需要被notify唤醒处于就绪状态,获取到CPU资源就运行
    WAITING,
    // 带有超时时间的等待,时间过后自动返回继续执行,比如sleep或者wait(long time)
    TIMED_WAITING,
    // 终止状态,自然停止或者抛出异常停止
    TERMINATED;
}

状态流转图:

属性及方法

属性

常用并且需要关注的属性如下:

private int priority;// 线程优先级,1<=priority<=10,优先级越大可能最终被操作系统优先调度的几率越大,但不一定会被先调用
private boolean daemon = false;// 是否后台线程,当所有的非后台线程结束时,所有的后台线程才结束,比如垃圾回收线程就是后台线程,负责在程序运行过程中,清理掉不在使用的对象
private Runnable target;// 目标类,要使用线程执行的业务逻辑写在run()方法里
private ThreadGroup group;// 线程组,所有的线程都必须属于某一个线程组,默认的是父线程组,未创建线程组时即main线程组
ThreadLocal.ThreadLocalMap threadLocals = null;// 线程独享本地变量表

ThreadLocal不了解的童鞋可以参考ThreadLocal源码分析

构造方法

其实最终都会调到一个init的方法,这里会初始化一些线程的基本信息。

ThreadGroup g:属于的线程组(每个线程必须属于一个线程组,没有设置线程组的话这里会默认使用父线程组,即main线程组)
Runnable target:要执行的方法
String name:线程名字,可单独设置,默认使用的是`Thread-自增序号`
long stackSize:这只线程占用的栈大小,一般不设置,由java虚拟机决定
AccessControlContext acc:这是一种安全访问机制,一般不用自己设置。

常用方法

// jvm方法,获取当前正在执行的线程
public static native Thread currentThread();
// 调用该方法所在的线程通知让出CPU资源,由操作系统重新分配线程执行,有可能还是该线程继续执行
public static native void yield();
// 线程休眠millis毫秒,让出CPU资源,但是不会释放持有的对象锁,当被中断时抛出异常
public static native void sleep(long millis) throws InterruptedException;
// 启动线程,更合理的说法是线程准备就绪,等待最终操作系统调度执行
public synchronized void start()
// 中断线程,其实只是设置中断标志为true,如果线程已经终止,调用该方法不会有任何作用,线程自身和安全验证通过的线程才有权限调该线程的这个方法,否则会返回异常SecurityException,还有以下几种特殊场景。
// 1、如果当前线程处理wait()、join()、sleep(long)方法时中断标志将被清除,并且抛出异常InterruptedException。(清除中断标志是因为抛出了异常后可由程序控制执行,中断标志没啥意义了)
// 2、继承了InterruptibleChannel如果IO阻塞状态,则channel会关闭并且中断标志设置为true,抛出异常ClosedByInterruptException
// 3、如果该线程阻塞在Selector,则中断标志设置为true并且select会立即返回一个非0的值表示select方法执行了
public void interrupt()
// 返回当前线程是否处于中断状态,并清除当前运行的线程的中断标志。
public static boolean interrupted() {
    return currentThread().isInterrupted(true);
}
// 返回当前线程是否处于中断状态,不会清除当前运行的线程的中断标志。线程终止后该方法始终返回false
public boolean isInterrupted() {
   return isInterrupted(false);
}

常用场景

重点看下interrupt()interrupted()isInterrupted()这三个方法的使用。
1、不使用变量终止线程
下面示例调用了interrupt()方法

public class ThreadTest {
    public static void main(String[] args) throws InterruptedException {
        Thread t = new Thread(new Runnable() {
            @Override
            public void run() {
                while (true) {
                    System.out.println(Thread.currentThread().getName() + ":" + System.currentTimeMillis());
                }
            }
        }, "t1");
        t.start();
        //主线程休2秒
        Thread.sleep(2000);
        t.interrupt();
        System.out.println(Thread.currentThread().getName() + "运行结束");
    }
}

程序运行部分结果

t1:1584607075440
t1:1584607075440
t1:1584607075440
t1:1584607075440
main运行结束
t1:1584607077740
t1:1584607077740
t1:1584607077740
t1:1584607077740
t1:1584607077741
t1:1584607077741
t1:1584607077741

这里我们看到调用了interrupt()方法,但是t线程并没有终止,因为这个方法只是设置中断标志为true,线程是否结束跟这个标志没有关系,需要其他逻辑判断。
如果我们把程序的while循环的条件调整如下这样,那么t线程调用isInterrupted()方法获取到中断标志为true,!true条件结果为false所以while循环结束,线程自然终止。

while (true) 
改成
while (!Thread.currentThread().isInterrupted())

2、线程组的操作
线程组主要是可以管理线程或线程组,一个大型的线程组看起来像是一棵树,可方便的对线程和子线程组进行监控和操作,比如统计活跃线程和线程组,或者中断和销毁线程组操作等。

public static void main(String[] args) {
    ThreadGroup tg1 = new ThreadGroup("tg1");
    ThreadGroup tg2 = new ThreadGroup("tg2");
    ThreadGroup tg3 = new ThreadGroup(tg2, "tg3");
    Thread t1 = new WorkerThread(tg1, "t1");
    Thread t2 = new WorkerThread(tg2, "t2");
    Thread t3 = new WorkerThread(tg1, "t3");
    Thread t4 = new WorkerThread(tg3, "t4");
    t1.start();
    t2.start();
    t3.start();
    t4.start();
    System.out.println("tgName:" + tg1.getName() + "\tactiveCount:" + tg1.activeCount());
    System.out.println("tgName:" + tg2.getName() + "\tactiveGroupCount:" + tg2.activeGroupCount());
    System.out.println("tgName:" + tg3.getParent().getName() + "\tactiveGroupCount:" + tg3.activeGroupCount());
}
    
class WorkerThread extends Thread {
    public WorkerThread(ThreadGroup tg, String name) {
        super(tg, name);
    }
    @Override
    public void run() {
        while (true) {
        }
    }
}

输出结果如下:

tgName:tg1	activeCount:2
tgName:tg2	activeGroupCount:1
tgName:tg2	activeGroupCount:0

总结

线程的使用可以提高系统的处理能力,比较复杂的业务可以使用多线程协作缩短执行时间,线程的结束不能仅仅依靠中断,需要根据业务实现自己的逻辑。另外引入多线程会增加系统的复杂度,线程的管理问题也会比较麻烦,同时线程数过多会频繁的让CPU切换,增加系统负载,这种时候可以选择使用线程池来解决问题,后续会再详细介绍。

标签:java,Thread,void,t1,详解,线程,new,public
From: https://www.cnblogs.com/star95/p/17583193.html

相关文章

  • 定位详解
    定位这个东西,blablabla,什么absolute,fixed...一头雾水定位=定位模式+边偏移定位模式......
  • 学习Java第6天
    java语法Java注释单行注释://多行注释:/**/文档注释:/***/标识符与关键字java所有的组成部分都需要名字。类名、变量名以及方法名都被称为标识符标识符注意点所有的标识符都应该以字母(A-Z或者a-z),美元符($)、或者下划线(_)开始首字母以后可以是字母(A-Z或者a-z),......
  • linux更改java版本
    查看系统版本uname-a查看java当前版本whichjavaecho$JAVA_HOME下载新的jdkhttp://www.oracle.com/technetwork/cn/java/javase/downloads解压tar-zxvfjdk-8u181-linux-x64.tar.gz修改JAVA_HOMEvim/etc/profile更新环境source/etc/profile......
  • java主线程等待多个子线程中任意一个有结果后,主线程继续执行
    1.背景2.代码packagecom.qianxingniwo.ls;importorg.junit.Test;importjava.util.concurrent.atomic.AtomicReference;importjava.util.concurrent.locks.LockSupport;/***@Copyright(C)XXXXX技有限公司*@Author:ldp*@Date:2023/7/2615:30*@Descri......
  • Java中代码Bug记录--泛型失效、数组删除、HashMap死循环
    最近在工作的过程中,遇到了不少奇怪自己或者同事的Bug,都是一些出乎意料的,不太容易发现的,记录一下来帮助可能也遇到了这些Bug的人1.编译时泛型校验失效Map<String,String>nameToType=newHashMap<>();nameToType.put("testName",123);//java:不兼容的类型:int无法转......
  • [Java] Stream流求和、排序、分组
    List、Set集合通过Stream流求和一、泛型为Integer、Long、Double、BigDecimal求和Integersum=scores.stream().reduce(Integer::sum).orElse(0);Longsum=scores.stream().reduce(Long::sum).orElse(0L);Doublesum=scores.stream().reduce(Double::sum).orElse(0.00)......
  • JavaScript命令模式:优雅地管理代码
    JavaScript命令模式在JavaScript中,命令模式是一种行为设计模式,它允许我们将请求封装为一个对象,从而使我们能够将请求的不同参数、方法和对象进行参数化。这种模式的主要目的是将请求的发送者和接收者解耦,从而使代码更加灵活和可维护。命令模式的实现在JavaScript中,我们可以使用......
  • Java SE 6 新特性: 对脚本语言的支持
    [-]脚本引擎ServiceProviderJava脚本API概述脚本引擎脚本引擎就是指脚本的运行环境,它能能够把运行其上的解释性语言转换为更底层的汇编语言,没有脚本引擎,脚本就无法被运行。JavaSE6引入了对JavaSpecificationRequest(JSR)223的支持,JSR223 旨在定义一个统......
  • 【面试必背知识】Java 中常见的异常有哪些?
    ......
  • 【Java面试题】Spring是如何解决循环依赖问题?
    ......