首页 > 其他分享 >手写一个Callable和FutureTask,异步线程执行并得到结果,了解其原理

手写一个Callable和FutureTask,异步线程执行并得到结果,了解其原理

时间:2022-10-10 16:23:03浏览次数:81  
标签:Thread park LockSupport Callable unpark 线程 FutureTask public

一,先模拟源码的Callable创建自己的MyCallable

package com.example.test.demo.thread.callable;
 
public interface MyCallable<T> {
    T call();
}

二,创建自己的FutureTask

package com.example.test.demo.thread.callable;
 
/**
 * 因为要放在Thread中执行,所以要实现Runnable
 */
public class MyFutureTask<T> implements Runnable{
 
    final Object object = new Object();
    private T result;
    private MyCallable<T> callable;
 
    public MyFutureTask(MyCallable<T> callable) {
        this.callable = callable;
    }
 
    @Override
    public void run() {
        result = callable.call();
        // 线程执行完,换新get()方法中等待的线程
        // wait要放在synchronized
        synchronized (object) {
            object.notify();
        }
    }
 
    /**
     * 获取多线程返回值
     * @return 返回多线程执行结果
     */
    public T get () throws InterruptedException {
        // 必须等线程执行完才能返回
        // wait要放在synchronized
        synchronized (object) {
            object.wait();
            return result;
        }
    }
}

三,最后测试

package com.example.test.demo.thread.callable;
import com.example.test.pojo.User;
import java.util.Date;

public class MyCallableDemo {
    public static void main(String[] args) throws InterruptedException {
        // 1. 创建callable
        MyCallable<User> callable = new MyCallable<User>() {
            @Override
            public User call() {
                // 模拟执行耗时 3秒
                try {
                    Thread.sleep(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                return new User("用户" + new Date().getTime(), "29");
            }
        };
        // 2.创建futureTask
        MyFutureTask<User> future = new MyFutureTask<>(callable);
 
        // 3.放到线程中执行
        new Thread(future).start();
        // 4.获取返回结果
        User user = future.get();
        // 5.打印
        System.out.println(user);
    }
}

最后,这里是用wait和notify模拟的,还可以使用LockSupport来实现

LockSupport.park();
LockSupport.unpark();

文章转载自:https://blog.csdn.net/wjy_0208/article/details/118613879

四,(扩展)LockSupport的park()和unpark()的简单使用

concurrent包是基于AQS (AbstractQueuedSynchronizer)框架的,AQS框架借助于两个类:

  • Unsafe(提供CAS操作)
  • *LockSupport*(提供park/unpark操作)

因此,LockSupport非常重要。

两个重点

(1)操作对象

归根结底,LockSupport.park()和LockSupport.unpark(Thread thread)调用的是Unsafe中的native代码:

//LockSupport中
public static void unpark(Thread thread) {
        if (thread != null)
            UNSAFE.unpark(thread);
    }
//LockSupport中
public static void unpark(Thread thread) {
        if (thread != null)
            UNSAFE.unpark(thread);
    }

Unsafe类中的对应方法:

    //park
    public native void park(boolean isAbsolute, long time);
    
    //unpack
    public native void unpark(Object var1);

park函数是将当前调用Thread阻塞,而unpark函数则是将指定线程Thread唤醒。

与Object类的wait/notify机制相比,park/unpark有两个优点:

  • 以thread为操作对象更符合阻塞线程的直观定义
  • 操作更精准,可以准确地唤醒某一个线程(notify随机唤醒一个线程,notifyAll唤醒所有等待的线程),增加了灵活性。

(2)关于“许可”

  • 在上面的文字中,我使用了阻塞和唤醒,是为了和wait/notify做对比。

  • 其实park/unpark的设计原理核心是“许可”:park是等待一个许可,unpark是为某线程提供一个许可。
    如果某线程A调用park,那么除非另外一个线程调用unpark(A)给A一个许可,否则线程A将阻塞在park操作上。

  • 有一点比较难理解的,是unpark操作可以再park操作之前。
    也就是说,先提供许可。当某线程调用park时,已经有许可了,它就消费这个许可,然后可以继续运行。这其实是必须的。考虑最简单的生产者(Producer)消费者(Consumer)模型:Consumer需要消费一个资源,于是调用park操作等待;Producer则生产资源,然后调用unpark给予Consumer使用的许可。非常有可能的一种情况是,Producer先生产,这时候Consumer可能还没有构造好(比如线程还没启动,或者还没切换到该线程)。那么等Consumer准备好要消费时,显然这时候资源已经生产好了,可以直接用,那么park操作当然可以直接运行下去。如果没有这个语义,那将非常难以操作。

  • 但是这个“许可”是不能叠加的,“许可”是一次性的。
    比如线程B连续调用了三次unpark函数,当线程A调用park函数就使用掉这个“许可”,如果线程A再次调用park,则进入等待状态。

简单代码实现:

import java.util.concurrent.locks.LockSupport;
 
public class LockTest {
    public static void main(String[] args) {
        Thread A = new Thread(() -> {
            System.out.println("准备阻塞");
            LockSupport.park();
            System.out.println("阻塞完成");
        }, "线程A");
        
        Thread B = new Thread(() -> {
            try {
                Thread.sleep(2000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("准备唤醒线程A");
            LockSupport.unpark(A);
        }, "线程B");
        A.start();
        B.start();
    }
}

LockSupport.park()可以被interrupt()方法中断:

    public static void main(String[] args) {
        Thread a = new Thread(new Runnable() {
            @Override
            public void run() {
                LockSupport.park();
                System.out.println(Thread.interrupted());
            }
        });
        a.start();
//        LockSupport.unpark(a);
        a.interrupt();
    }

执行后结果如下:

文章转载自:https://blog.csdn.net/thetimelyrain/article/details/114587111

标签:Thread,park,LockSupport,Callable,unpark,线程,FutureTask,public
From: https://www.cnblogs.com/fantongxue/p/16776113.html

相关文章

  • Spring Boot 2.x基础教程:如何隔离@Async异步任务的线程池
    通过上一篇:配置@Async异步任务的线程池的介绍,你应该已经了解到异步任务的执行背后有一个线程池来管理执行任务。为了控制异步任务的并发不影响到应用的正常运作,我们必须要......
  • 31、并发编程(进程、线程、协程)
    31.1、操作系统:1、为什么要有操作系统:(1)介绍:现代计算机系统是由一个或者多个处理器,主存,磁盘,打印机,键盘,鼠标显示器,网络接口以及各种其他输入输出设备组成的复杂系统,每位程序员......
  • 【Java高级】程序、进程和线程
    1.程序一段静态的代码(未启动的APP),可以完成特定任务。2.进程正在运行的程序(挂在后台的APP),系统给进程分配了内存。3.线程程序内部的一条执行路径(比如QQ中执行聊天功能,发......
  • 【博学谷学习记录】超强总结,用心分享 。多线程相关点知识学习。
    一、实现多线程1.1了解多线程多线程是指从软件或硬件上实现多个线程并发执行的技术。具有多线程能力的计算机因有硬件支持而能够在同一时间执行多个线程,提......
  • python多线程
    importtimeimportdatetimeimportthreadingdefdotask():whileTrue:print(datetime.datetime.now().strftime('%Y-%m-%d%H:%M:%S'))time.sleep(5)......
  • 线程通信练习--生产者与消费者
    packageexer4;/***@author高槐玉*#Description生产者,消费者案例*#Date:2022/10/9/14点08分*#Shangguigu:*/classClerk{privateintgoods=0......
  • 三个线程顺序打印ABC?我有十二种做法,彻底掌握多线程同步通信机制
    大家好,我是老三,这篇文章分享一道非常不错的题目:三个线程按序打印ABC。很多读者朋友应该都觉得这道题目不难,这次给大家带来十二种做法,一定有你没有见过的新姿势。1.synchron......
  • Java 多线程(三)静态代理模式
    静态代理模式:1.真实角色和代理角色实现同一个接口2.代理角色要代理真实角色3.代理角色可以做真实角色做不了的事4.真实角色专注做自己的事publicclassStaticProxy......
  • 达梦数据库体系结构(物理结构、逻辑结构、内存结构、线程结构)
    DM目录数据库安装目录下图展示为DM8数据库目录。  /dm8/bin 目录存放DM数据库的可执行文件,例如disql命令、dminit命令、dmrman工具等。  /dm8/deskto......
  • Java实现多线程
    Java实现多线程的方式有4种分别是继承Thread类,实现Runnable,Callable接口和通过线程池提交线程任务。其中实现Callable接口的方式可以获取返回值。1.继承Thread类通过继......