ACTIVE OBJECT 模式

2018-06-03  本文已影响0人  Mrs_Gao

ACTIVE OBJECT 模式

一.概述

主动对象模式基于命令模式,是实现多线程控制的一项古老的技术。该模式有多种使用方式,为许多工业系统提供了一个简单的多任务核心。

二.实现

假如现在需要延时执行一个任务,并且不能阻塞下一个任务的执行

package cn.zzf.active.object;
import java.util.LinkedList;

/**
 * @author GaoFeng2017
 * @date 2018-06-03 15:29:24
 **/

public class ActiveObjectEngine {

    private LinkedList<Command> taskList = new LinkedList<>();

    public void addCommand(Command command) {
        taskList.add(command);
    }

    public void run() {
        while (!taskList.isEmpty()) {
            Command command = taskList.removeFirst();
            command.execute();
        }
    }
}
package cn.zzf.active.object;

/**
 * @author GaoFeng2017
 * @date 2018-06-03 15:33:39
 **/

public interface Command {
    void execute();
}
package cn.zzf.active.object;

/**
 * @author GaoFeng2017
 * @date 2018-06-03 15:33:39
 **/

public interface Command {
    void execute();
}
package cn.zzf.active.object;

/**
 * @author GaoFeng2017
 * @date 2018-06-03 15:34:03
 **/

public class SleepCommand implements Command {
    private boolean start = false;
    private long startTime;
    private Command executeCommand;
    private int delay;
    private ActiveObjectEngine engine;

    public SleepCommand(Command executeCommand, int delay, ActiveObjectEngine engine) {
        this.executeCommand = executeCommand;
        this.delay = delay;
        this.engine = engine;
    }

    @Override
    public void execute() {
        if (!start) {
            start = true;
            startTime = System.currentTimeMillis();
            engine.addCommand(this);
        } else if (System.currentTimeMillis() - startTime < delay) {
            engine.addCommand(this);
        } else {
            engine.addCommand(executeCommand);
        }
    }
}

package cn.zzf.active.object;

/**
 * @author GaoFeng2017
 * @date 2018-06-03 15:45:37
 **/

public class Test {
    private boolean stop = false;

    @org.junit.Test
    public void test() {
        ActiveObjectEngine engine = new ActiveObjectEngine();
        Command command = new Command() {
            @Override
            public void execute() {
                stop = true;
            }
        };

        long time = System.currentTimeMillis();
        SleepCommand sleepCommand = new SleepCommand(command,1000,engine);
        engine.addCommand(sleepCommand);
        engine.run();
        System.out.println(stop);
        System.out.println(System.currentTimeMillis() - time);

    }

}

运行结果

true
1000

分析

三.案例

上面的实现已经演示了ACTIVE OBJECT模式是如何工作的,不过任务只被执行了一次,下面的案例将会让我们看到多个延时任务的循环执行,并且在指定延时后停止所有任务。

package cn.zzf.active.object;

/**
 * @author GaoFeng2017
 * @date 2018-06-03 17:00:10
 **/

public class DelayedTyper implements Command{

    private String name;
    private int delay;
    private static boolean stop = false;
    private static ActiveObjectEngine engine = new ActiveObjectEngine();

    @Override
    public void execute() {
        System.out.print(name);
        if (!stop) {
            repeatTask();
        }
    }

    public DelayedTyper(String name, int delay) {
        this.name = name;
        this.delay = delay;
    }


    public void repeatTask() {
        engine.addCommand(new SleepCommand(this,delay,engine));
    }

    public static ActiveObjectEngine getEngine() {
        return engine;
    }

    public static void main(String[] args) {

        engine.addCommand(new DelayedTyper("A",100));
        engine.addCommand(new DelayedTyper("B",300));
        engine.addCommand(new DelayedTyper("C",500));
        engine.addCommand(new DelayedTyper("D",700));

        engine.addCommand(new SleepCommand(() -> stop = true,5000,engine));
        engine.run();
    }
}

两次运行结果

ABCDAAABAACBADAABACAABAADABCAAABAACABDAAABACAABADAABCAAABAACDABAAABACAABDAAABCAAABADACABD
ABCDAAABACABADAAABCAABAADACABAAABACADABAABACAAABDAACABAAABACDAABAAABCAADABAACABAAABDACABD

分析
DelayTyper实现了Command,负责执行并添加任务,它的exceute方法不仅实现了延时后要执行的任务,而且会循环添加延时任务,直至stop被改变。具体步骤如下:

可以看到,两次运行结果并不一样,书上的解释是cpu时钟和实时时钟没有完美的同步,个人感觉是每次测试时,方法(比如addCommand)执行的时间可能存在差异,当然这种差异可能是毫秒级别的。

四.总结

ACTIVE OBJECT是COMMAND模式的一个应用场景,它使用单线程环境构建了一个多线程系统,巧妙的进行了任务轮询,这样的好处在于,下一个任务(command)不会被上一个任务阻塞,并且这些任务是共享一个运行时堆栈,减少了内存占用,不会有同步问题出现。

上一篇下一篇

猜你喜欢

热点阅读