JavaSE多线程

Java并发编程

当一个程序启动后,操作系统就会为这个程序创建一个进程并分配内存空间。如果这个程序是一个Java程序,那么它的内存空间会分为堆区、栈区、元数据区、本地方法栈、程序计数器等。

当进程创建完后,栈中的主线程也就是main方法就会开始执行。在并发编程的需求下,需要创建子线程,首先在堆中创建一个线程对象,然后启动它。

  • 进程,是对运行时程序的封装,是操作系统进行资源调度和分配的基本单位,实现了操作系统的并发。
  • 线程,是进程的子任务,是CPU调度和分派的基本单位,实现了进程内部的并发。

一、线程的创建

创建Java线程有三种方式

1. 继承Thread类

public class MyThread extends Thread {
    @Override
    public void run() {
        for (int i = 0; i < 100; i++) {
            System.out.println(getName() + "第" + i + "次执行");
        }
    }
}
//创建MyThread对象
MyThread t1=new  MyThread();
MyThread t2=new  MyThread();
//设置线程的名字
t1.setName("T1线程");
t2.setName("T2线程");
//启动线程
t1.start();
t2.start();

2. 实现Runnable接口

public class MyRunnable implements Runnable {
    @Override
    public void run() {
        for (int i = 0; i < 10; i++) {
            try {//sleep会发生异常要显示处理
                Thread.sleep(20);//暂停20毫秒
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(getName() + "第" + i + "次执行");
        }
    }
}
//创建MyRunnable类
MyRunnable mr = new MyRunnable();
//创建Thread类的有参构造,并设置线程名
Thread t1 = new Thread(mr, "T1线程");
Thread t2 = new Thread(mr, "T2线程");
//启动线程
t1.start();
t2.start();

3. 使用FutureTask类

FutureTask是RunnableFuture的实现类,RunnableFuture又是Runnable和Future的子接口

FutureTask类的构造方法可以传入Callable接口实现类的实例,也可以传入Runnable接口的实例与线程执行后的返回值

这种创建线程的方法可以在主线程中获取子线程的返回值以及抛出的异常

public class CallerTask implements Callable<String> {
    public String call() throws Exception {
        return "task done";
    }
}
//创建异步任务
FutureTask<String> task = new FutureTask<String>(new CallerTask());
//启动线程
new Thread(task).start();
try {
    //等待执行完成,并获取返回结果
    String result = task.get();
    //String result = task.get(5,TimeUnit.MINUTES);
    System.out.println(result);
} catch (InterruptedException e) {
    e.printStackTrace();
} catch (ExecutionException e) {
    //通过调用getCause获取线程抛出的异常
    Throwable cause = e.getCause();
    e.printStackTrace();
} catch (TimeoutException e) {
    //如果调用带有参数的get方法,当线程超时后会抛出此异常
    e.printStackTrace();
}
  • run()方法和start()方法有什么区别?
    • run():封装线程执行的代码,直接调用相当于调用普通方法。
    • start():启动线程,然后由JVM 调用此线程的 run() 方法。

二、线程的状态

在java.lang.Thread类中的内部枚举,定义了Java线程的六种状态。

public enum State {
    NEW,
    RUNNABLE,
    BLOCKED,
    WAITING,
    TIMED_WAITING,
    TERMINATED;
}

1. NEW(新建状态)

处于NEW状态的线程此时尚未启动。这里的尚未启动指的是还没调用Thread实例的start()方法。

private void testStateNew() {
    Thread thread = new Thread(() -> {});
    System.out.println(thread.getState()); // 输出 NEW 
}

2. RUNNABLE(可运行状态)

Java线程的RUNNABLE状态包括了传统操作系统线程的readyrunning两个状态的。

处于RUNNABLE状态的线程在Java虚拟机中运行,也有可能在等待CPU分配资源。

Thread源码里对RUNNABLE状态的定义:

/**
 * Thread state for a runnable thread.  A thread in the runnable
 * state is executing in the Java virtual machine but it may
 * be waiting for other resources from the operating system
 * such as processor.
 */
 //可运行的线程状态。处于可运行状态的线程正在 Java 虚拟机中执行,也可能正在等待来自操作系统(如处理器)的其他资源。

3. BLOCKED(阻塞状态)

处于BLOCKED状态的线程正等待锁的释放以进入同步区。

4. WAITING(等待状态)

处于等待状态的线程变成RUNNABLE状态需要其他线程唤醒。

调用如下3个方法会使线程进入等待状态:

  • Object.wait():使当前线程处于等待状态直到另一个线程唤醒它;
  • Thread.join():等待线程执行完毕,底层调用的是Object实例的wait方法;
  • LockSupport.park():除非获得调用许可,否则禁用当前线程进行线程调度。

5. TIMED_WAITING(超时等待状态)

线程等待一个具体的时间,时间到后会被自动唤醒。

调用如下方法会使线程进入超时等待状态:

  • Thread.sleep(long millis):使当前线程睡眠指定时间;
  • Object.wait(long timeout):线程休眠指定时间,等待期间可以通过notify()/notifyAll()唤醒;
  • Thread.join(long millis):等待当前线程最多执行millis毫秒,如果millis为0,则会一直执行;
  • LockSupport.parkNanos(long nanos): 除非获得调用许可,否则禁用当前线程进行线程调度指定时间;
  • LockSupport.parkUntil(long deadline):同上,也是禁止线程进行调度指定时间;

6. TERMINATED(终止状态)

此时线程执行完毕。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,193评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,306评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,130评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,110评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,118评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,085评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,007评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,844评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,283评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,508评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,395评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,985评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,630评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,797评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,653评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,553评论 2 352