Java巧用lambda,异步方法优雅写法

了解lambda的基本原理

加入lambda之后,很多写法都变得简单起来,如创建一个线程对象,可以:

new Thread(new Runnable() {
    @Override
    public void run() {
        System.out.print("Hello");
    }
});

lambda写法:
new Thread(() -> System.out.print("Hello"));

单独将lambda拎出来:
Runnable runnable = () -> System.out.print("Hello");

其实lambda代表的就是一个接口的实现而已(匿名内部类)。而这种接口也叫函数式接口,会有@FunctionalInterface注解进行编译时检查。

或者直接把lambda看成一个方法,上述的 () -> System.out.print("Hello") 就是代表一个无入参、无返回值的一个方法(等同于public void run() {System.out.print("Hello")}),而Runnable runnable则是指向这个方法(类似函数指针),需要调用这个方法时,调用runnable.run()即可.

当有一个参数、无返回值则是Consumer<T>

// 对printStream对象的void print(String s)方法的引用
PrintStream printStream = System.out;
Consumer<String> consumer = printStream::print;
consumer.accept("Hello");

// 输出
Hello

对于有入参(一个或两个)、有返回值等情况,JDK也提供了对应的函数式接口:

接口 函数 说明
Consumer <T>void accept(T t) 无返回值、一个入参,T为入参类型
BiConsumer<T, U> void accept(T t, U u) 无返回值、两个入参,T为第一个入参类型、U为第二个入参类型
Supplier<T> T get() 有返回值、无入参,T为返回值类型
Function<T, R> R apply(T t) 有返回值、一个入参,T为入参类型,R为返回类型
BiFunction<T, U, R> R apply(T t, U u) 有返回值、两个入参,T为第一个入参类型、U为第二个入参类型,R为返回类型

以上为java.util.function包下的部分接口,剩余的基本上就是指定泛型类型的函数接口了,例如LongConsumer,无泛型,其实就是指定入参只能是Long。

上述已经代表了大部分的函数可以表示的形式了(值得注意的是,三个及以上的入参的函数式接口JDK并没有提供,需要时要自定义实现,实际上也很少用到)。

方法有静态方法(static)和非静态方法,但函数式接口关注的仅是入参、出参类型和个数而已:

public Class Test {
    public Long noStaticFoo(String str) {...}
    public static Long staticFoo(String str) {...}
    
    public static void main() {
        // 静态方法,类::
        Function<String, Long> staticFoo = Test::staticFoo;
        Long apply = staticFoo.apply("100");
        
        // 非静态方法,实例对象::
        Test test = new Test();
        Function<String, Long> noStaticFoo = test::noStaticFoo;
        Long apply2 = noStaticFoo.apply("100");
        
        /* 这种可以理解成两个入参、除了原来方法的入参外,还要指明实例对象(因为这是一个实例方法)*/
        BiFunction<Test, String, Long> noStaticFooTest = Test::noStaticFoo;
        Long apply3 = noStaticFooTest.apply(test, "100");
    }
}

当一个函数/方法可以被变量引用时,其实就可以利用这个特性做一些比较方便的事情了,例如Streams的相关API。效率也比反射中的Method要高。

优雅的异步方法调用写法

在java中的异步方法,原理基本上大同小异,其实就是新开一个线程(或者从线程池中获取线程),Spring也提供相应的@Async注解。这里期待的是,将业务代码与“系统是否异步执行”进行解耦。
假如我现在有个Service,里面有个方法是根据id远程拉取用户信息(HTTP):

    public class UserHttpService {
        public User getById(Long id) {
            ...
            return user;
        }
    }

有A、B、C三个地方有调用到UserHttpService#getById,我们发现A处同步发起了好几个HTTP请求(拉取用户信息只是其中一个),这时候我希望UserHttpService#getById能变成异步执行,提高效率。一种做法是@Async + 直接修改返回值为Future,但是问题来了,这样的话B、C两个地方都要做出修改,但是B、C只调了一个HTTP,没必要变成异步呀(事实上,在真实项目中,情况会比这个更加复杂)。另外一种做法则是结合lambda进行解耦。
先新增一个异步执行类:

public class AsyncExecutor {
        // 线程池,建议恰当配置(拒绝策略建议CallerRunsPolicy)和使用框架注入
        private ExecutorService executorService = Executors.newFixedThreadPool(10);

        /**
        * 单个入参,有返回值的异步执行方法 , public User getById(Long id)
        *
        * @param method 要执行的方法,如 , userHttpService::getById
        * @param param  入参值,如 100
        * @param <P>    入参类型,如 Long
        * @param <R>    返回值类型,如 User
        * @return Future对象,用以判断是否执行结束、获取返回结果
        */
        public <P, R> Future<R> async(Function<P, R> method, P param) {
            return executorService.submit(() -> method.apply(param));
        }
    }

这时候A处异步调用:

public class A {

    private AsyncExecutor asyncExecutor;

    private UserHttpService userHttpService;

    public void foo() {
      ...
      // 异步调用
      Future<User> userFuture = asyncExecutor.async(userHttpService::getById, id);

      ... 其他操作(如再发起http请求)
 
      // 获取结果
      User user = userFuture.get();
      ...
    }
}

对于原来的UserHttpService并不需要做任何修改,只要在需要的地方(A)指定为异步即可。

对于其他的无入参、两入参、无返回值等的方法形式,也可以类似处理:

public class AsyncExecutor {
      // 线程池,建议恰当配置(拒绝策略建议CallerRunsPolicy)和使用框架注入
      private ExecutorService executorService = Executors.newFixedThreadPool(10);
     
     /**
     * 无入参,无返回值的异步执行方法 , void noStaticFoo()
     *
     * @param method 要执行的方法,如 user::noStaticFoo;
     * @return Future对象,用以判断是否执行结束
     */
    public Future async(Runnable method) {
        return executorService.submit(method);
    }

    /**
     * 有单个入参,无返回值的异步执行方法,如 void noStaticFoo(Long id)
     *
     * @param method 要执行的方法,如, user::noStaticFoo
     * @param param  方法执行的入参,如id
     * @param <P>    入参类型,如Long
     * @return Future对象,用以判断是否执行结束
     */
    public <P> Future async(Consumer<P> method, P param) {
        return executorService.submit(() -> method.accept(param));
    }

    /**
     * 有两个参数但是无返回值的异步执行方法, 如void noStaticFoo(Long id,Entity entity)
     *
     * @param method 要执行的方法,如 , user::noStaticFoo
     * @param param1 第一个入参值,如id
     * @param param2 二个入参值,如entity
     * @param <P1>   第一个入参类型
     * @param <P2>   第二个入参类型
     * @return Future对象,用以判断是否执行结束
     */
    public <P1, P2> Future async(BiConsumer<P1, P2> method, P1 param1, P2 param2) {
        return executorService.submit(() -> method.accept(param1, param2));
    }

    /**
     * 无参数有返回值的异步执行方法 , Entity noStaticFoo()
     *
     * @param method 要执行的方法,如 , user::noStaticFoo
     * @param <R>    返回值类型,如 Entity
     * @return Future对象,用以判断是否执行结束、获取返回结果
     */
    public <R> Future<R> async(Supplier<R> method) {
        return executorService.submit(method::get);
    }

    /**
     * 单个入参,有返回值的异步执行方法 , Entity noStaticFoo(Long id)
     *
     * @param method 要执行的方法,如 , user::noStaticFoo
     * @param param  入参值,如 id
     * @param <P>    入参类型,如Long
     * @param <R>    返回值类型,如 Entity
     * @return Future对象,用以判断是否执行结束、获取返回结果
     */
    public <P, R> Future<R> async(Function<P, R> method, P param) {
        return executorService.submit(() -> method.apply(param));
    }

    /**
     * 两个入参,有返回值的异步执行方法 , Entity noStaticFoo(Long id)
     *
     * @param method 要执行的方法,如 , user::noStaticFoo
     * @param param1 第一个入参值,如id
     * @param param2 二个入参值,如entity
     * @param <P1>   第一个入参类型
     * @param <P2>   第二个入参类型
     * @param <R>    返回值类型,如 Entity
     * @return Future对象,用以判断是否执行结束、获取返回结果
     */
    public <P1, P2, R> Future<R> async(BiFunction<P1, P2, R> method, P1 param1, P2 param2) {
        return executorService.submit(() -> method.apply(param1, param2));
    }
}

JDK1.8 有提供CompletableFuture,也是类似异步处理的方法,默认线程池为ForkJoinPool(默认最大工作线程数=CPU总核心数-1),该线程池擅长于计算密集型任务,IO密集型任务请尽量使用自己合理配置的线程池。

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

推荐阅读更多精彩内容

  • 进程和线程 进程 所有运行中的任务通常对应一个进程,当一个程序进入内存运行时,即变成一个进程.进程是处于运行过程中...
    胜浩_ae28阅读 5,102评论 0 23
  • 进程和线程 进程 所有运行中的任务通常对应一个进程,当一个程序进入内存运行时,即变成一个进程.进程是处于运行过程中...
    小徐andorid阅读 2,805评论 3 53
  • 大家都想能够尽快升职加薪,下面我就个人经验给大家一些建议: 上班前的15分钟很重要 我曾经在的一个公司是一个创业型...
    欧尼在线阅读 1,166评论 0 7
  • 会计学的基本公式: 资产 = 负债 + 所有者权益 这个公式是一个时点公式,即表达的是一个存量的概念。此公式可以转...
    陆非阅读 1,649评论 1 8
  • 无爱亦无恨,无得便无失。无求方无欲,无畏才无欺!
    秋一叶西阅读 205评论 0 0