java多线程

新建

new() ,Thread()进程的子类,start()

就绪

等待JVM线程调度器的调度

运行

获取CPU资源 run()

阻塞

sleep(),suspend(挂起)

  • 等待阻塞:运行状态中的线程执行 wait() 方法,使线程进入到等待阻塞状态。

  • 同步阻塞:线程在获取 synchronized 同步锁失败(因为同步锁被其他线程占用)。

  • 其他阻塞:通过调用线程的 sleep() 或 join() 发出了 I/O 请求时,线程就会进入到阻塞状态。当sleep() 状态超时,join() 等待线程终止或超时,或者 I/O 处理完毕,线程重新转入就绪状态。

priority(优先级)

创建线程

  • 通过实现 Runnable 接口;

需要重写run(),同时也需要调用Thread的start()

public class MyRunnable implements Runnable {
    public void run() {
        // ...
    }
}

public static void main(String[] args) {
    MyRunnable instance = new MyRunnable();
    Thread thread = new Thread(instance);
    thread.start();
}
  • 通过继承 Thread 类本身;

同样也是需要实现 run() 方法,因为 Thread 类也实现了 Runable 接口

public class MyThread extends Thread {
    public void run() {
        // ...
    }
}

public static void main(String[] args) {
    MyThread mt = new MyThread();
    mt.start();
}

  • 通过 Callable 和 Future 创建线程。

  • 实现接口会更好一些,因为:

Java 不支持多重继承,因此继承了 Thread 类就无法继承其它类,但是可以实现多个接口

类可能只要求可执行就行,继承整个 Thread 类开销过大。

互斥同步

java通过两种锁机制来控制多个线程对共享资源的互斥访问
JVM的synchronized
JDK的ReentranLock

Thread的几个重要方法。

  • start()方法,调用该方法开始执行该线程
  • stop()方法,调用该方法强制结束该线程执行
  • join方法,调用该方法等待该线程结束
  • sleep()方法,调用该方法该线程进入等待
  • run()方法,调用该方法直接执行线程的run()方法,但是线程调用start()方法时也会运行run()方法,区别就是一个是由线程调度运行run()方法,一个是直接调用了线程中的run()方法

wait()与sleep()的区别:

  • 简单来说wait()会释放对象锁而sleep()不会释放对象锁

start()方法的调用后并不是立即执行多线程代码,而是使得该线程变为可运行态(Runnable),什么时候运行是由操作系统决定的。

阻塞状态(Blocked):阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行。直到线程进入就绪状态,才有机会转到运行状态。阻塞的情况分三种:

  • 等待阻塞:运行的线程执行wait()方法,JVM会把该线程放入等待池中。(wait会释放持有的锁)
  • 同步阻塞:运行的线程在获取对象的同步锁时,若该同步锁被别的线程占用,则JVM会把该线程放入锁池中。
  • 其他阻塞:运行的线程执行sleep()或join()方法,或者发出了I/O请求时,JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。(注意,sleep是不会释放持有的锁)

notify(): 通知一个线程继续运行。

getID() : 获取线程的唯一标识

interrupt() : 停止线程,打一个停止的标记


Thread.currentThread().interrupted(); //中断程序
this.interrupted(); //是否已经中断,第一次调用之后状态标识便会被清除
this.isInterrupted(); //判断是否会中断,但是不会清除状态标识 

MyThread thread = new MyThread();
thread.start();
Thread.sleep();
thread.interrupt();

  1. stop() 暴力停止。可能会导致java.lang.ThreadDeath,已经过期了。
  2. 使用return停止线程
  • 暂停线程suspend();
  • 恢复线程resume();
  • 缺点就是不同步
  1. yieID() 放弃当前的CPU资源,,给其他任务去占用,放弃时间不确定

线程优先级的继承特性

A线程启动B线程,则B线程的优先级与A相同

为什么我们要调用Thread中的start()时才执行run()方法,而不是直接调用run()

调用start方法方可启动线程使得线程进入就绪状态,而run方法只是Thread线程中的一个普通调用,还是在主线程里面执行的

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容