一个线程两次调用start()方法会出现什么情况?
Java 的线程是不允许启动两次的,第二次调用必然会抛出 IllegalThreadStateException,这是一种运行时异常,多次调用 start 被认为是编程错误。
看源码
package com.wuyu.java;
public class JoinTest {
public static void main(String [] args) throws InterruptedException {
ThreadJoinTest t1 = new ThreadJoinTest("小明");
t1.start();
t1.start();
}
}
class ThreadJoinTest extends Thread{
public ThreadJoinTest(String name){
super(name);
}
@Override
public void run(){
for(int i=0;i<10;i++){
System.out.println(this.getName() + ":" + i);
}
}
}
线程到底是什么?
进程是程序的一次执行过程,是资源的基本单位; 进程包括线程,线程是调度和分派的基本单位。
从操作系统的角度,可以简单认为,线程是系统调度的最小单元,一个进程可以包含多个线程,作为任务的真正运作者,有自己的栈(Stack)、寄存器(Register)、本地存储(Thread Local)等,但是会和进程内其他线程共享文件描述符、虚拟地址空间等。
从线程生命周期的状态开始展开,那么在 Java 编程中,有哪些因素可能影响线程的状态呢?主要有:
线程自身的方法
,除了 start,还有多个 join 方法,等待线程结束;yield 是告诉调度器,主
动让出 CPU;另外,就是一些已经被标记为过时的 resume、stop、suspend 之类,据我所知,在 JDK 最新版本中,destory/stop 方法将被直接移除。
基类 Object 提供了一些基础的 wait/notify/notifyAll 方法
。如果我们持有某个对象的
Monitor 锁,调用 wait 会让当前线程处于等待状态,直到其他线程 notify 或者 notifyAll。
所以,本质上是提供了 Monitor 的获取和释放的能力,是基本的线程间通信方式。
并发类库中的工具,比如 CountDownLatch.await()
会让当前线程进入等待状态,直到
latch 被基数为 0,这可以看作是线程间通信的 Signal。