制作公司网站流程,制作网站代码大全,查域名ip地址查询,汶上手机网站建设一、线程5种状态 新建状态#xff08;New#xff09; 新创建了一个线程对象。 就绪状态#xff08;Runnable#xff09; 线程对象创建后#xff0c;其他线程调用了该对象的start()方法。该状态的线程位于可运行线程池中#xff0c;变得可运行#xff0c;等待获取CPU的使…一、线程5种状态 新建状态New 新创建了一个线程对象。 就绪状态Runnable 线程对象创建后其他线程调用了该对象的start()方法。该状态的线程位于可运行线程池中变得可运行等待获取CPU的使用权。 运行状态Running 就绪状态的线程获取了CPU执行程序代码。 阻塞状态Blocked 阻塞状态是线程因为某种原因放弃CPU使用权暂时停止运行。直到线程进入就绪状态才有机会转到运行状态。阻塞的情况分三种 等待阻塞运行的线程执行wait()方法JVM会把该线程放入等待池中。同步阻塞运行的线程在获取对象的同步锁时若该同步锁被别的线程占用则JVM会把该线程放入锁池中。其他阻塞运行的线程执行sleep()或join()方法或者发出了I/O请求时JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时线程重新转入就绪状态。 死亡状态Dead线程执行完了或者因异常退出了run()方法该线程结束生命周期。 二、Jstack中常见的线程状态 应用程序启动后我们对系统运行状况的观测大部分情况下是通过运行日志。但是若某一天发现日志中记录的行为与预想的不一致此时需要进一步的系统监控该怎么办Jstack是常用的排查工具它能输出在某一个时间java进程中所有线程的状态很多时候这些状态信息能给我们的排查工作带来有用的线索。 Jstack的输出中Java线程状态主要是以下几种 RUNNABLE 线程运行中或I/O等待BLOCKED 线程在等待monitor锁(synchronized关键字)TIMED_WAITING 线程在等待唤醒但设置了时限WAITING 线程在无限等待唤醒这里Jstack使用的关键字描述的线程状态与上一节中线程不太一样所以可能理解上的可能会出现混淆。虽然Java中的线程一样有上节中描述的5种状态但在实际情况下线程新建状态和死亡状态持续很短我们也并不太关心。大多时候我们关注的是运行状态/阻塞状态这里弄清楚Jstack的输出含义即可。下面用简单的代码产生出以上4中状态。 public static void main(String[] args) {System.out.println(Utils.pid());runnable(); // 1 // blocked(); // 2 // waiting(); // 3 // timedWaiting(); // 4 } public static String pid() { String name ManagementFactory.getRuntimeMXBean().getName(); return name.split()[0]; } 这里为了方便得到java进程id直接使用pid()函数输出。为了方便我们把观察线程固定为”main”因为JVM还有其他线程都会存在输出中我们可以通过关键字”main”找到我们要观察的线程。命令jstack -l [pid]。 1 让线程一直处于RUNNABLE public static void runnable() {long i 0; while (true) { i; } } 没什么好解释的死循环即可。 2 让线程一直处于BLOCKED public static void blocked() {final Object lock new Object(); new Thread() { public void run() { synchronized (lock) { System.out.println(i got lock, but dont release); try { Thread.sleep(1000L * 1000); } catch (InterruptedException e) { } } } }.start(); try { Thread.sleep(100); } catch (InterruptedException e) {} synchronized (lock) { try { Thread.sleep(30 * 1000); } catch (InterruptedException e) { } } } 主线程sleep先让另外一个线程拿到lock并长期持有locksleep会持有锁wait不会。此时主线程会BLOCK住等待lock被释放此时jstack的输出可以看到main线程状态是BLOCKED。这里要注意的是只有synchronized这种方式的锁monitor锁才会让线程出现BLOCKED状态等待ReentrantLock则不会。 3 让线程处于TIMED_WAITING状态 public static void timedWaiting() {final Object lock new Object(); synchronized (lock) { try { lock.wait(30 * 1000); } catch (InterruptedException e) { } } } 用Lock.tryLock(timeout, timeUnit)这种方式也会看到TIMED_WAITING状态这个状态说明线程当前的等待一定是可超时的。 4 让线程处于WAITING状态 public static void waiting() {final Object lock new Object(); synchronized (lock) { try { lock.wait(); } catch (InterruptedException e) { } } } 无超时的等待必须等待lock.notify()或lock.notifyAll()或接收到interrupt信号才能退出等待状态。同理ReentrantLock.lock()的无参方法调用也会使线程状态变成WAITING。 通过以上几个最简单的例子让线程达到jstack输出中常见的几种状态可以更好地理解jstack输出。转载于:https://www.cnblogs.com/barrywxx/p/8576358.html