Java多线程之Runable与Thread

简介:

Java多线程是Java开发中的基础内容,但是涉及到高并发就有很深的研究可做了。

最近看了下《Java并发实战》,发先有些地方,虽然可以理解,但是自己在应用中很难下手。

  所以还是先回顾一下基础知识:

Java中的线程

  线程的相关知识,了解操作系统的基本都能有所了解。

  线程有5中状态,基本变化如图所示:

如何在Java代码中创建线程

  众所周知,Java创建线程有两种方式:

  1 实现Runable接口

  2 继承Thread类

  那么这两种方式有什么区别呢?

  1 Runable属于接口,所以可以有多个实现;Thread只有一个。

  2 实现Runable的线程类,可以被多个线程实例共享数据。

  举个简单的例子,火车站售票处一共有3个售票口,但是只剩下5张票:

  如果单纯使用Thread实现3个售票口的售票过程:

package com.imooc.test;
class MyThread extends Thread{
    
    private int ticketsCount = 5;
    private String name;
    
    public MyThread(String name){
        this.name = name;
    }
    
    @Override
    public void run() {
        while(ticketsCount > 0){
            ticketsCount--;
            System.out.println(name+" 卖了一张票,还剩下:"+ticketsCount);
        }
    }
}
public class TicketsTestThread {
    public static void main(String[] args) {
        MyThread mt1 = new MyThread("窗口1");
        MyThread mt2 = new MyThread("窗口2");
        MyThread mt3 = new MyThread("窗口3");
        
        mt1.start();
        mt2.start();
        mt3.start();
    }
}

  执行结果如下:

窗口1 卖了一张票,还剩下:4
窗口2 卖了一张票,还剩下:4
窗口1 卖了一张票,还剩下:3
窗口1 卖了一张票,还剩下:2
窗口1 卖了一张票,还剩下:1
窗口1 卖了一张票,还剩下:0
窗口2 卖了一张票,还剩下:3
窗口2 卖了一张票,还剩下:2
窗口2 卖了一张票,还剩下:1
窗口2 卖了一张票,还剩下:0
窗口3 卖了一张票,还剩下:4
窗口3 卖了一张票,还剩下:3
窗口3 卖了一张票,还剩下:2
窗口3 卖了一张票,还剩下:1
窗口3 卖了一张票,还剩下:0

  可以看到每个线程拥有自己的5张票,其实是重复了!

  那么如果使用Runnable,则不会出现这种情况:

package com.imooc.test;
class MyRunnable implements Runnable{
    
    private int ticketsCount = 5;
    
    @Override
    public void run() {
        while(ticketsCount > 0){
            ticketsCount--;
            System.out.println(Thread.currentThread().getName()+" 卖了一张票,还剩下:"+ticketsCount);
        }
    }
}
public class TicketsTestRunnable {
    public static void main(String[] args) {
        MyRunnable mr = new MyRunnable();
        Thread th1 = new Thread(mr,"窗口1");
        Thread th2 = new Thread(mr,"窗口2");
        Thread th3 = new Thread(mr,"窗口3");
        
        th1.start();
        th2.start();
        th3.start();
    }
}

  执行结果:

窗口1 卖了一张票,还剩下:4
窗口3 卖了一张票,还剩下:2
窗口3 卖了一张票,还剩下:1
窗口3 卖了一张票,还剩下:0
窗口2 卖了一张票,还剩下:3

  这是因为创建Thread实例时,使用的是同一个MyRunnable类对象,所以会共享其中的数据。

用户线程与守护线程

  在Java线程中,共有两类线程:

  1 用户线程:用户代码生成

  2 守护线程:用于特定的功能,当用户线程都结束时,守护线程会随着JVM的停止而停止,因此守护线程不能用于IO操作。

  那么下面一个简单的守护线程的例子:

  创建一个守护线程,持续不断的向文件中写入数据。主线程中启动该线程,然后主线程在一定时间后,退出。

  观察守护线程的状态!

  代码如下:

package com.imooc.test;

import java.io.File;
import java.io.FileOutputStream;
import java.io.OutputStream;
import java.util.Scanner;

class DaemonThread implements Runnable{
    public void run() {
        System.out.println("进入守护线程:"+Thread.currentThread().getName());
        try {
            Write2File();
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("退出守护线程:"+Thread.currentThread().getName());
    }
    public void Write2File() throws Exception{
        File filename = new File("d:"+File.separator+"daemon.txt");
        OutputStream os = new FileOutputStream(filename,true);
        int count = 0;
        while(count < 999){
            os.write(("\r\nword "+count).getBytes());
            System.out.println("守护线程"+Thread.currentThread().getName()+
                    "写入了 "+count);
            count++;
            Thread.sleep(1000);
        }
    }
}
public class DaemonTest {
    public static void main(String[] args) {
        System.out.println("进入主线程"+Thread.currentThread().getName());
        DaemonThread dt = new DaemonThread();
        Thread th = new Thread(dt);
        th.setDaemon(true);
        th.start();
        
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } 
        
        System.out.println("退出主线程"+Thread.currentThread().getName());
    }
}

  当主线程睡眠了5秒后,便结束。此时JVM中没有其他的用户线程,于是守护线程也直接退出。

  执行结果如下:

进入主线程main
进入守护线程:Thread-0
守护线程Thread-0写入了 0
守护线程Thread-0写入了 1
守护线程Thread-0写入了 2
守护线程Thread-0写入了 3
守护线程Thread-0写入了 4
退出主线程main

  可以看到守护线程直接就中断退出了!

  鉴于守护线程的这种特性,常用于实时监控系统状态。比如数据库,JVM等等。

查看线程快照

  通过使用Jstack.exe程序,可以帮助用户查看线程状态。

  使用方法:

  1 查询线程PID

  2 在cmd中输入jstack -l pid

C:\Users\Administrator>jstack -l 5028
2015-04-01 17:43:30
Full thread dump Java HotSpot(TM) Client VM (24.60-b09 mixed mode, sharing):

"Thread-0" daemon prio=6 tid=0x00928800 nid=0x2798 waiting on condition [0x03d4f
000]
   java.lang.Thread.State: TIMED_WAITING (sleeping)
        at java.lang.Thread.sleep(Native Method)
        at com.imooc.test.DaemonThread.Write2File(DaemonTest.java:27)
        at com.imooc.test.DaemonThread.run(DaemonTest.java:12)
        at java.lang.Thread.run(Unknown Source)

   Locked ownable synchronizers:
        - None

"Service Thread" daemon prio=6 tid=0x008de000 nid=0xd64 runnable [0x00000000]
   java.lang.Thread.State: RUNNABLE

   Locked ownable synchronizers:
        - None

"C1 CompilerThread0" daemon prio=10 tid=0x008dc400 nid=0x2158 waiting on conditi
on [0x00000000]
   java.lang.Thread.State: RUNNABLE

   Locked ownable synchronizers:
        - None

"Attach Listener" daemon prio=10 tid=0x008f4800 nid=0x13e0 waiting on condition
[0x00000000]
   java.lang.Thread.State: RUNNABLE

   Locked ownable synchronizers:
        - None

"Signal Dispatcher" daemon prio=10 tid=0x00905800 nid=0x1c7c runnable [0x0000000
0]
   java.lang.Thread.State: RUNNABLE

   Locked ownable synchronizers:
        - None

"Finalizer" daemon prio=8 tid=0x00875800 nid=0x2460 in Object.wait() [0x03e0f000
]
   java.lang.Thread.State: WAITING (on object monitor)
        at java.lang.Object.wait(Native Method)
        - waiting on <0x23800fc8> (a java.lang.ref.ReferenceQueue$Lock)
        at java.lang.ref.ReferenceQueue.remove(Unknown Source)
        - locked <0x23800fc8> (a java.lang.ref.ReferenceQueue$Lock)
        at java.lang.ref.ReferenceQueue.remove(Unknown Source)
        at java.lang.ref.Finalizer$FinalizerThread.run(Unknown Source)

   Locked ownable synchronizers:
        - None

"Reference Handler" daemon prio=10 tid=0x00870800 nid=0x21c8 in Object.wait() [0
x03b6f000]
   java.lang.Thread.State: WAITING (on object monitor)
        at java.lang.Object.wait(Native Method)
        - waiting on <0x23800db0> (a java.lang.ref.Reference$Lock)
        at java.lang.Object.wait(Object.java:503)
        at java.lang.ref.Reference$ReferenceHandler.run(Unknown Source)
        - locked <0x23800db0> (a java.lang.ref.Reference$Lock)

   Locked ownable synchronizers:
        - None

"main" prio=6 tid=0x0099c000 nid=0x14b8 waiting on condition [0x0052f000]
   java.lang.Thread.State: TIMED_WAITING (sleeping)
        at java.lang.Thread.sleep(Native Method)
        at com.imooc.test.DaemonTest.main(DaemonTest.java:40)

   Locked ownable synchronizers:
        - None

"VM Thread" prio=10 tid=0x0086f000 nid=0x22dc runnable

"VM Periodic Task Thread" prio=10 tid=0x00927000 nid=0x131c waiting on condition


JNI global references: 111


C:\Users\Administrator>

  其中详细的描述了线程的名字,是否为守护线程,以及状态等等。

参考

  【1】慕课网Thread VS Runnable:http://www.imooc.com/learn/312

本文转自博客园xingoo的博客,原文链接:Java多线程之Runable与Thread,如需转载请自行联系原博主。
相关文章
|
18天前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
80 17
|
29天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者
|
14天前
|
缓存 安全 算法
Java 多线程 面试题
Java 多线程 相关基础面试题
|
1月前
|
消息中间件 缓存 安全
Java多线程是什么
Java多线程简介:本文介绍了Java中常见的线程池类型,包括`newCachedThreadPool`(适用于短期异步任务)、`newFixedThreadPool`(适用于固定数量的长期任务)、`newScheduledThreadPool`(支持定时和周期性任务)以及`newSingleThreadExecutor`(保证任务顺序执行)。同时,文章还讲解了Java中的锁机制,如`synchronized`关键字、CAS操作及其实现方式,并详细描述了可重入锁`ReentrantLock`和读写锁`ReadWriteLock`的工作原理与应用场景。
|
28天前
|
Java 程序员 调度
【JavaEE】线程创建和终止,Thread类方法,变量捕获(7000字长文)
创建线程的五种方式,Thread常见方法(守护进程.setDaemon() ,isAlive),start和run方法的区别,如何提前终止一个线程,标志位,isinterrupted,变量捕获
|
28天前
|
安全 Java API
【JavaEE】多线程编程引入——认识Thread类
Thread类,Thread中的run方法,在编程中怎么调度多线程
|
5月前
|
存储 监控 Java
Java多线程优化:提高线程池性能的技巧与实践
Java多线程优化:提高线程池性能的技巧与实践
154 1
|
8月前
|
设计模式 监控 Java
Java多线程基础-11:工厂模式及代码案例之线程池(一)
本文介绍了Java并发框架中的线程池工具,特别是`java.util.concurrent`包中的`Executors`和`ThreadPoolExecutor`类。线程池通过预先创建并管理一组线程,可以提高多线程任务的效率和响应速度,减少线程创建和销毁的开销。
264 2
|
8月前
|
Java 数据库
【Java多线程】对线程池的理解并模拟实现线程池
【Java多线程】对线程池的理解并模拟实现线程池
84 1
|
5月前
|
安全 算法 Java
17 Java多线程(线程创建+线程状态+线程安全+死锁+线程池+Lock接口+线程安全集合)(下)
17 Java多线程(线程创建+线程状态+线程安全+死锁+线程池+Lock接口+线程安全集合)
93 6

热门文章

最新文章