Elasticsearch 设计模式—命令模式
命令模式
不同的书籍不同的文章对于命令模式的解释不尽相同,但是总结起来,思想上缺失大同小异,这里用我自己的理解来
描述一下命令模式:
命令模式一般用来解决客户端/服务端的解耦问题,即将客户端的请求封装成一个抽象的对象,便于服务端将该请求进行参数话处理,从而实现“行为请求者”与“行为实现者”解耦
关于设计模式的详细介绍,请参考: 命令模式
一般来说, 命令模式会涉及到五个角色:
- 客户端角色(Client):创建一个具体命令(ConcreteCommand)对象并确定其接收者
- 命令角色(Command):声明了一个给所有具体命令类的抽象接口或抽象类
- 具体命令角色(ConcreteCommand):定义一个接收者和行为之间的弱耦合
- 请求者角色(Invoker):负责调用命令对象执行请求,相关的方法叫做行动方法
- 接收者角色(Receiver):负责具体实施和执行一个请求。任何一个类都可以成为接收者,实施和执行请求的方法叫做行动方法
其 UML 类图如下:
Elasticsearch 对于命令模式的应用
我们都知道 Elasticsearch 在启动的时候能接收许多不同的参数,以此来控制 Elasticsearch 的启动后的行为,Elasticsearch 是如何理解和处理这些参数呢,其实 Elasticsearch
内部就是采用了设计模式—命令模式来解决这些不同参数的问题.
下面我们来分析一下 Elasticsearch 是如何使用命名模式的.
命令角色(Command)
Elasticsearch 有一个抽象类 org.elasticsearch.cli.Command,是用来充当命令模式中的命令角色,其源码如下:
/**
* An action to execute within a cli.
*/
public abstract class Command implements Closeable {
/** A description of the command, used in the help output. */
protected final String description;
/** The option parser for this command. */
protected final OptionParser parser = new OptionParser();
private final OptionSpec<Void> helpOption = parser.acceptsAll(Arrays.asList("h", "help"), "show help").forHelp();
private final OptionSpec<Void> silentOption = parser.acceptsAll(Arrays.asList("s", "silent"), "show minimal output");
private final OptionSpec<Void> verboseOption =
parser.acceptsAll(Arrays.asList("v", "verbose"), "show verbose output").availableUnless(silentOption);
public Command(String description) {
this.description = description;
}
final SetOnce<Thread> shutdownHookThread = new SetOnce<>();
/** Parses options for this command from args and executes it. */
public final int main(String[] args, Terminal terminal) throws Exception {
if (addShutdownHook()) {
shutdownHookThread.set(new Thread(() -> {
try {
this.close();
} catch (final IOException e) {
try (
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw)) {
e.printStackTrace(pw);
terminal.println(sw.toString());
} catch (final IOException impossible) {
// StringWriter#close declares a checked IOException from the Closeable interface but the Javadocs for StringWriter
// say that an exception here is impossible
throw new AssertionError(impossible);
}
}
}));
Runtime.getRuntime().addShutdownHook(shutdownHookThread.get());
}
if (shouldConfigureLoggingWithoutConfig()) {
// initialize default for es.logger.level because we will not read the log4j2.properties
final String loggerLevel = System.getProperty("es.logger.level", Level.INFO.name());
final Settings settings = Settings.builder().put("logger.level", loggerLevel).build();
LogConfigurator.configureWithoutConfig(settings);
}
try {
mainWithoutErrorHandling(args, terminal);
} catch (OptionException e) {
printHelp(terminal);
terminal.println(Terminal.Verbosity.SILENT, "ERROR: " + e.getMessage());
return ExitCodes.USAGE;
} catch (UserException e) {
if (e.exitCode == ExitCodes.USAGE) {
printHelp(terminal);
}
terminal.println(Terminal.Verbosity.SILENT, "ERROR: " + e.getMessage());
return e.exitCode;
}
return ExitCodes.OK;
}
/**
* Indicate whether or not logging should be configured without reading a log4j2.properties. Most commands should do this because we do
* not configure logging for CLI tools. Only commands that configure logging on their own should not do this.
*
* @return true if logging should be configured without reading a log4j2.properties file
*/
protected boolean shouldConfigureLoggingWithoutConfig() {
return true;
}
/**
* Executes the command, but all errors are thrown.
*/
void mainWithoutErrorHandling(String[] args, Terminal terminal) throws Exception {
final OptionSet options = parser.parse(args);
if (options.has(helpOption)) {
printHelp(terminal);
return;
}
if (options.has(silentOption)) {
terminal.setVerbosity(Terminal.Verbosity.SILENT);
} else if (options.has(verboseOption)) {
terminal.setVerbosity(Terminal.Verbosity.VERBOSE);
} else {
terminal.setVerbosity(Terminal.Verbosity.NORMAL);
}
execute(terminal, options);
}
/** Prints a help message for the command to the terminal. */
private void printHelp(Terminal terminal) throws IOException {
terminal.println(description);
terminal.println("");
printAdditionalHelp(terminal);
parser.printHelpOn(terminal.getWriter());
}
/** Prints additional help information, specific to the command */
protected void printAdditionalHelp(Terminal terminal) {}
@SuppressForbidden(reason = "Allowed to exit explicitly from #main()")
protected static void exit(int status) {
System.exit(status);
}
/**
* Executes this command.
*
* Any runtime user errors (like an input file that does not exist), should throw a {@link UserException}. */
protected abstract void execute(Terminal terminal, OptionSet options) throws Exception;
/**
* Return whether or not to install the shutdown hook to cleanup resources on exit. This method should only be overridden in test
* classes.
*
* @return whether or not to install the shutdown hook
*/
protected boolean addShutdownHook() {
return true;
}
@Override
public void close() throws IOException {
}
}
该类(命令角色)的作用有如下几个:
-
声明该命令的描述信息,以方便在打印命令参数的时候格式化输出
-
定义一个 execute()抽象方法,用于其具体子类,即具体的命令的执行
具体命令角色(ConcreteCommand)
Elasticsearch 中有一个 EnvironmentAwareCommand 抽象类,是 Command 的子类,并具体实现了 Command 的 execute()抽象方法方法,具体实现如下:
@Override
protected void execute(Terminal terminal, OptionSet options) throws Exception {
final Map<String, String> settings = new HashMap<>();
for (final KeyValuePair kvp : settingOption.values(options)) {
if (kvp.value.isEmpty()) {
throw new UserException(ExitCodes.USAGE, "setting [" + kvp.key + "] must not be empty");
}
if (settings.containsKey(kvp.key)) {
final String message = String.format(
Locale.ROOT,
"setting [%s] already set, saw [%s] and [%s]",
kvp.key,
settings.get(kvp.key),
kvp.value);
throw new UserException(ExitCodes.USAGE, message);
}
settings.put(kvp.key, kvp.value);
}
putSystemPropertyIfSettingIsMissing(settings, "path.conf", "es.path.conf");
putSystemPropertyIfSettingIsMissing(settings, "path.data", "es.path.data");
putSystemPropertyIfSettingIsMissing(settings, "path.home", "es.path.home");
putSystemPropertyIfSettingIsMissing(settings, "path.logs", "es.path.logs");
execute(terminal, options, createEnv(terminal, settings));
}
其中 execute 接收两个参数,这里解释一下:
-
Terminal terminal 代表一个终端操作,可读可写
-
OptionSet options 一系列命令的集合
我们来看下具体的命令角色是如何完成调用接收者相应的操作的,源码如下:
@Override
protected void execute(Terminal terminal, OptionSet options, Environment env) throws UserException {
if (options.nonOptionArguments().isEmpty() == false) {
throw new UserException(ExitCodes.USAGE, "Positional arguments not allowed, found " + options.nonOptionArguments());
}
if (options.has(versionOption)) {
if (options.has(daemonizeOption) || options.has(pidfileOption)) {
throw new UserException(ExitCodes.USAGE, "Elasticsearch version option is mutually exclusive with any other option");
}
terminal.println("Version: " + org.elasticsearch.Version.CURRENT
+ ", Build: " + Build.CURRENT.shortHash() + "/" + Build.CURRENT.date()
+ ", JVM: " + JvmInfo.jvmInfo().version());
return;
}
final boolean daemonize = options.has(daemonizeOption);
final Path pidFile = pidfileOption.value(options);
final boolean quiet = options.has(quietOption);
try {
init(daemonize, pidFile, quiet, env);
} catch (NodeValidationException e) {
throw new UserException(ExitCodes.CONFIG, e.getMessage());
}
}
该方法中有一个 init(daemonize, pidFile, quiet, env)方法,该方法则是由命令角色中的接收者角色来完成,那这个接收者角色是谁呢?
接收者角色
我们继续看 init(daemonize, pidFile, quiet, env)的方法是谁在负责调度呢? 源码如下:
void init(final boolean daemonize, final Path pidFile, final boolean quiet, Environment initialEnv)
throws NodeValidationException, UserException {
try {
Bootstrap.init(!daemonize, pidFile, quiet, initialEnv);
} catch (BootstrapException | RuntimeException e) {
// format exceptions to the console in a special way
// to avoid 2MB stacktraces from guice, etc.
throw new StartupException(e);
}
}
可以看到,真正处理 init()方法参数的核心方法是:Bootstrap.init(!daemonize, pidFile, quiet, initialEnv);
也就是说,真正的接收者角色是 Bootstrap,它在接收到具体的命令对象的时候,负责来完成 Elasticsearch 环境的初始化工作.
请求者(Invoker)角色
我们再回顾下请求者(Invoker)角色的作用:负责调用命令对象执行请求
那么在 Elasticsearch 中,谁负责调用命令对象呢?
答案是:Elasticsearch 类.
Elasticsearch 既然是请求者(Invoker)角色,那它是如何持有命令角色呢?这里 Elasticsearch 的处理方式是继承具体的命令角色,这样其自身
就可以有充当具体命令的角色,又充当请求者角色,而且 Elasticsearch 还完成了组装命令对象和接收者.因此这里
Elasticsearch 既是请求者角色又是客户端角色.
解决问题
如果细看 Elasticsearch 来解析和使用这些参数的整个过程的话,会发现整个调用链比较长,处理过程比较复杂,而且会对某些参数进行处理
比如有些参数是帮助参数(-help),则这些参数需要打印其其他具体命令参数的使用描述信息,有些参数是控制启动行为的参数,比如-d,则需要
对其进行处理,而且在整个调用过程中,会有一些内部日志的记录以及缓存.使用命令模式则很好的解决了这些问题.
由此可见命令模式在下面这些场景中可以按需使用:
- 整个调用过程比较繁杂,或者存在多处这种调用。这时,使用 Command 类对该调用加以封装,便于功能的再利用
- 调用前后需要对调用参数进行某些处理。
- 调用前后需要进行某些额外处理,比如日志,缓存,记录历史操作等
命令模式的使用效果
- 将调用操作的对象和知道如何实现该操作的对象解耦
- Command 是头等对象。他们可以像其他对象一样被操作和扩展
- 你可将多个命令装配成一个复合命令
- 增加新的 Command 很容易,因为这无需改变现有的类
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于