[面试]future模式

Future模式

什么是future模式?

传统单线程环境下,调用函数是同步的,必须等待程序返回结果后,才可进行其他处理。 Futrue模式下,调用方式改为异步。

Futrue模式的核心在于:充分利用主函数中的等待时间,利用等待时间处理其他任务,充分利用计算机资源。

简单描述一下future模式的实现

future模式有两种数据, 一种是真实数据RealData, 里面就是业务中想要得到的目标数据. 另一种是虚拟数据FutureData, 它是在使用future模式时立即返回的一个对象. 

调用方会首先拿到一个FutureData, 然后调用方就认为自己拿到该数据了, 没有进行阻塞, 继续去执行下面的逻辑处理. 如果真实数据准备好了, 就会把自己的引用赋给之前的那个FutureData, 并且置一个标记, 表示这个FutureData里面包含一个RealData, 拿到了想要的数据, 可以使用.

详细分的话, 会有下面这几种情况(假设RealData需要2秒才能创建好):

1. 调用方发送了自己需要RealData的请求的后, 会立即拿到一个FutureData, 但是根本就不着急使用, 所以, 第2秒的时候RealData创建完成后, 就会被绑定到对应的FutureData里. 假设第6秒调用方才开始使用RealData, 他会发现FutureData已经准保好了他想要的数据, 于是开心地使用就ok了.

2. 调用方发送了自己需要RealData的请求的后, 会立即拿到一个FutureData, 但是很着急使用, 因为接下来的处理过程依赖于RealData的内容. 于是在第0.5秒的时候, 调用方就想要获取RealData. 但是这个时候RealData并没有准备好, 此时的FutureData是一个空壳而已. 所以就在这里进行wait(或者忙等待). 直到RealData准备好,也就是再过1.5秒, 线程才会唤醒(或者打破忙等待).

请用代码实现一下Future模式

FutureData和RealData的统一抽象接口, Data类如下:

public interface Data {
    int getResult() throws InterruptedException;
}

RealData类

public class RealData implements Data {
    private int data;

    public RealData(int num) {
        //这里用sleep来模拟构造一个复杂对象的场景
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        
        this.data = num * 10;
    }

    @Override
    public int getResult() {
        return data;
    }
}

FutureData类

public class FutureData implements Data {
    // 真实数据RealData的引用.
    private RealData realData = null;

    public synchronized void setRealData(RealData realData) {
        // 如果this.realData不是空, 说明已经准备好了, 直接return
        if (this.realData != null)
            return;
        this.realData = realData;
        notifyAll();
    }

    @Override
    public synchronized int getResult() throws InterruptedException {
        // 如果this.realData是null, 说明数据还没准备好, 应该等待
        if (this.realData == null) {
            wait();
        }
        return realData.getResult();
    }
}

Client类

直接创建一个FutureData, 然后直接返回这个FutureData. 同事开辟一个线程来创建RealData, 并且在RealData创建完后绑定在FutureData中.

public class Client {
    public Data request(final int num) {
        // 当有请求的时候, 先创建一个虚拟对象.
        final FutureData futureData = new FutureData();

        // 然后开启一个新线程去创建RealData, 当RealData创建完成后, 绑定带FutureData里.
        new Thread(() -> {
            RealData realData = new RealData(num);
            futureData.setRealData(realData);
        }).start();

        // 不管RealData有没有创建完成, 都会直接返回这个FutureData.
        return futureData;
    }
}

Main

调用这个Future模型.

public class Main {
    public static void main(String[] args) throws InterruptedException {
        Client client = new Client();

        // 调用了之后会立即返回一个FutureData, 这个data就是FutureData
        Data data = client.request(4);

        // 用sleep来模拟主线程正在处理其他事情
        Thread.sleep(0);

        // getResult来获取真实数据
        //     |- 如果这时候真实数据没准备好, 那么就wait, 等待notify, 然后获取到真实数据
        //     |- 如果这时候真实数据准备好了, 那么就可以直接获取到了
        System.out.println("数据=" + data.getResult());
    }
}

 在RealData处进行了*10 的处理, 所以request(4), 最终会返回40.

使用过JDK自带的Future模式吗?

使用过, 例子如下:

定义一个RealData类

import java.util.concurrent.Callable;

public class RealData implements Callable<Integer> {
    private int data;

    public RealData(int data) {
        this.data = data * 10;
    }

    @Override
    public Integer call() {
        //利用sleep方法来表示真是业务是非常缓慢的
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return data;
    }
}

 Main

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.FutureTask;

public class Main {
    public static void main(String[] args) throws Exception {
        //线程池
        ExecutorService executor = Executors.newFixedThreadPool(1); //使用线程池

        // 之前自己实现的future模式中的 Data data = client.request(4) 这句相当于下面这两行代码
        //1. Data data
        FutureTask<Integer> futureTask = new FutureTask<>(new RealData(4));
        //2. 这里相当于 client.request(4);
        executor.submit(futureTask);

        //这里可以用一个sleep代替对其他业务逻辑的处理
        Thread.sleep(0);

        // 获取真实数据
        try {
            System.out.println("数据=" + futureTask.get());
        }finally {
            executor.shutdown();
        }
    }
}

 

posted @ 2018-06-29 18:12  GoldArowana  阅读(471)  评论(0编辑  收藏  举报