zoukankan      html  css  js  c++  java
  • Java8 对多个异步任务进行流水线操作(笔记)

         现在我们要对商店商品进行折扣服务.每个折扣代码对应不同的折扣率,使用一个枚举变量Discount.Code来实现这一想法,具体代码如下所示.
    以枚举类型定义的折扣代码
    /**
     * 折扣服务api
     *
     * @author Darcy
     *         Created by Administrator on 2017/3/17.
     */
    public class Discount {
        public enum Code {
            NONE(0), SILVER(0), GOLD(10), PLATINUM(15), DIAMOND(20);
            private final int percentage;
    
            Code(int percentage) {
                this.percentage = percentage;
            }
        }
    
        public static String applyDiscount(Quote quote) {
            return quote.getShopName() + " price is " + Discount.apply(quote.getPrice(), quote.getDiscountCode());
        }
    
        private static double apply(double price, Code code) {
            delay();
            return price * (100 - code.percentage) / 100;
        }
    
        /**
         * 模拟计算,查询数据库等耗时
         */
        public static void delay() {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
    
    修改商店返回价格的格式:
    
    public String getPrice(String product) {
        double price = calculatePrice(product);
        Discount.Code code = Discount.Code.values()[
                random.nextInt(Discount.Code.values().length)];
        return String.format("%s:%.2f:%s", name, price, code);
    }
    
    
        * 实现折扣服务
    
    我们的商店已经能从不同的商店获得商品价格,解析结果字符串,针对每个字符串,查询折扣服务器取的折扣代码.这个流程决定了请求商品的最终折扣价格.我们将对商店返回的字符串的解析操作封装到了下面的Quote类中:
    /**
     * 商店返回消息实体,不可变对象模式 线程安全
     * @author Darcy
     *         Created by Administrator on 2017/3/17.
     */
    public final class Quote {
        private final String shopName;
        private final double price;
        private final Discount.Code discountCode;
    
        public Quote(String shopName, double price, Discount.Code discountCode) {
            this.shopName = shopName;
            this.price = price;
            this.discountCode = discountCode;
        }
    
        public static Quote parse(String s) {
            String[] split = s.split(":");
            String shopName = split[0];
            double price = Double.parseDouble(split[1]);
            Discount.Code discountCode = Discount.Code.valueOf(split[2]);
            return new Quote(shopName, price, discountCode);
        }
    
        public String getShopName() {
            return shopName;
        }
    
        public double getPrice() {
            return price;
        }
    
        public Discount.Code getDiscountCode() {
            return discountCode;
        }
    }
    
         Discount服务还提供了一个applyDiscount方法,它接收一个Quote对象,返回一个字符串,表示生成该Quote的shop中的折扣价格,代码如下:
    public static String applyDiscount(Quote quote) {
        return quote.getShopName() + " 商品原价: " + quote.getPrice() + " 折扣后价格: " + Discount.apply(quote.getPrice(), quote.getDiscountCode());
    }
    
    private static double apply(double price, Code code) {
        delay();
        return price * (100 - code.percentage) / 100;
    }
    
    
        * 使用Discount服务
    
    /**
     * 商店折扣价格查询器
     *
     * @param product 商品
     * @return
     */
    public static List<String> findprices(String product) {
        return shops
                .stream()
                .map(shop ->  shop.getPrice(product))
                .map(Quote::parse)
                .map(Discount::applyDiscount)
                .collect(Collectors.toList());
    }
    
    执行结果:
    
    换成并行流:
    /**
     * 商店折扣价格查询器
     *
     * @param product 商品
     * @return
     */
    public static List<String> findprices(String product) {
        return shops
                .parallelStream()
                .map(shop ->  shop.getPrice(product))
                .map(Quote::parse)
                .map(Discount::applyDiscount)
                .collect(Collectors.toList());
    }
    
    执行结果:
    看到差距了吧
    
    
        * 构建同步和异步操作
    
    我们再次使用ComoletableFuture提供的特性.以异步方式重新实现findPrices方法,详细代码如下:
    /**
     * 商店折扣价格查询器(CompletableFuture方式)
     *
     * @param product 商品
     * @return
     */
    public static List<String> findPrices(String product) {
        List<CompletableFuture<String>> collect = shops
                .stream()
                //以异步凡是取得每个shop中指定产品的原始价格
                .map(shop -> CompletableFuture.supplyAsync(() -> shop.getPrice(product), executor))
                //Quote对象存在时,对其返回值进行转换
                .map(future -> future.thenApply(Quote::parse))
                //使用另一个异步任务构建期望的Future,申请折扣 thenCompose 将多个future组合 一个一个执行
                .map(future -> future.thenCompose(quote -> CompletableFuture.supplyAsync(() -> Discount.applyDiscount(quote), executor)))
                .collect(Collectors.toList());
        return collect
                .stream()
                //等待流中所有的future执行完毕,并提取各自的返回值
                .map(CompletableFuture::join)
                .collect(Collectors.toList());
    }
    
    
        * 对最佳价格查询器应用的优化
    
    上面的所有例子中都是通过响应之前添加1秒延迟的等待时间模拟方法的远程调用,毫无疑问,现实生活中,你的应用访问各个远程服务器时很可能遭遇无法预知的延迟,触发原因多种多样,从服务器的负荷到网络的延迟,有些甚至是源于远程服务如何评估你应用的商业价值,
    由于这些原因,你希望购买的商品在某些原因的查询速度要比另一些商店更快,我们模拟了操作:
    
    /**
     * 模拟不同的商店 延迟不一样的情况
     */
    public static void randomDelay() {
        int delay = 500 + random.nextInt(2000);
        try {
            Thread.sleep(delay);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
    
    重构findPrices方法 返回一个由Future构成的流:
    /**
     * 重构findPrices方法 返回一个由Future构成的流
     *
     * @param product 商品
     * @return
     */
    public static Stream<CompletableFuture<String>> findProcesStream(String product) {
        return shops
                .stream()
                //以异步凡是取得每个shop中指定产品的原始价格
                .map(shop -> CompletableFuture.supplyAsync(() -> shop.getPrice(product), executor))
                //Quote对象存在时,对其返回值进行转换
                .map(future -> future.thenApply(Quote::parse))
                //使用另一个异步任务构建期望的Future,申请折扣 thenCompose 将多个future组合 一个一个执行
                .map(future -> future.thenCompose(quote -> CompletableFuture.supplyAsync(() -> Discount.applyDiscount(quote), executor)));
    }
    
    付诸实现:
    long start = System.nanoTime();
    CompletableFuture[] futures = findProcesStream("myPhones27s")
            //Java 8的CompletableFuture通过thenAccept方法  他接收CompletableFuture执行完毕的返回值作为参数.
            .map(f -> f.thenAccept(
                    s -> System.out.println(s + " (done in " +
                            ((System.nanoTime() - start) / 1_000_000) + " msecs)")))
            .toArray(CompletableFuture[]::new);
    //allOf工厂方法接收一个由CompletableFuture构成的数组,数组中所有的CompletableFuture对象执行完毕后,它返回一个
    //CompletableFuture<Void>对象,这意味着你需要等待最初Stream中所有的CompletableFuture对象执行完毕
    //angOf该方法接收一个CompletableFuture对象构成的数组,返回由第一个执行完毕的CompletableFuture对象的返回值构成的CompletableFuture<Object>
    CompletableFuture.allOf(futures).join();
    System.out.println("All shops have now responded in  " + ((System.nanoTime() - start) / 1_000_000) + " msecs");
    
    执行结果:
     
  • 相关阅读:
    基于移动最小二乘曲面的点云对齐(一) 隐式平面的生成
    最小二乘法的原理及改进
    三维扫描原理及精度控制
    C语言实现界面(不通过MFC避免遗忘)
    2020年04月11日腾讯一面总结
    什么样的钱该花
    js修改伪类before边框颜色
    12.19技术知识点整理
    在windows下搭建类mac前端开发环境
    工作中遇到的bug修复
  • 原文地址:https://www.cnblogs.com/ten951/p/6590661.html
Copyright © 2011-2022 走看看