689.d9ae26a7.js 443.2 KB
Newer Older
茶陵後's avatar
茶陵後 已提交
1
(window.webpackJsonp=window.webpackJsonp||[]).push([[689],{1137:function(t,e,a){"use strict";a.r(e);var n=a(56),s=Object(n.a)({},(function(){var t=this,e=t.$createElement,a=t._self._c||e;return a("ContentSlotsDistributor",{attrs:{"slot-key":t.$parent.slotKey}},[a("h1",{attrs:{id:"spring-statemachine-文档"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#spring-statemachine-文档"}},[t._v("#")]),t._v(" Spring StateMachine 文档")]),t._v(" "),a("h2",{attrs:{id:"序言"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#序言"}},[t._v("#")]),t._v(" 序言")]),t._v(" "),a("p",[t._v("状态机的概念很可能比这个引用文档的任何读者都要古老,而且肯定比 Java 语言本身还要古老。对有限自动机的描述可以追溯到 1943 年,当时 Warren McCulloch 先生和 Walter Pitts 先生写了一篇关于它的论文。后来,George H.Mealy 在 1955 年提出了一个状态机概念(称为“Mealy Machine”)。一年后的 1956 年,Edward F.Moore 发表了另一篇论文,他在论文中描述了所谓的“摩尔机器”。如果你曾经读过任何有关状态机的东西,那么 Mealy 和 Moore 这两个名字应该是在某个时候出现的。")]),t._v(" "),a("p",[t._v("本参考文档包含以下部分:")]),t._v(" "),a("p",[a("a",{attrs:{href:"#introduction"}},[t._v("导言")]),t._v("包含此参考文档的介绍。")]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine"}},[t._v("Using Spring Statemachine")]),t._v("描述了 Spring statemachine 的用法。")]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples"}},[t._v("状态机示例")]),t._v("包含更详细的状态机示例。")]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-faq"}},[t._v("FAQ")]),t._v("包含常见问题。")]),t._v(" "),a("p",[a("a",{attrs:{href:"#appendices"}},[t._v("Appendices")]),t._v("包含有关已用材料和状态机的通用信息。")]),t._v(" "),a("h1",{attrs:{id:"引言"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#引言"}},[t._v("#")]),t._v(" 引言")]),t._v(" "),a("p",[t._v("Spring StateMachine 是一种允许应用程序开发人员在 Spring 应用程序中使用传统状态机概念的框架。SSM 提供以下功能:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("易于使用的平面(一级)状态机,用于简单的用例。")])]),t._v(" "),a("li",[a("p",[t._v("分层状态机结构,以简化复杂的状态配置.")])]),t._v(" "),a("li",[a("p",[t._v("状态机区域提供甚至更复杂的状态配置。")])]),t._v(" "),a("li",[a("p",[t._v("触发器、转换、保护和动作的使用。")])]),t._v(" "),a("li",[a("p",[t._v("类型安全配置适配器。")])]),t._v(" "),a("li",[a("p",[t._v("状态机事件监听器。")])]),t._v(" "),a("li",[a("p",[t._v("Spring 将 bean 与状态机相关联的 IOC 集成。")])])]),t._v(" "),a("p",[t._v("在继续之前,我们建议先查看一下附录"),a("a",{attrs:{href:"#glossary"}},[t._v("Glossary")]),t._v(""),a("a",{attrs:{href:"#crashcourse"}},[t._v("状态机速成课程")]),t._v(",以了解状态机是什么。文档的其余部分希望你熟悉状态机的概念。")]),t._v(" "),a("h2",{attrs:{id:"背景"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#背景"}},[t._v("#")]),t._v(" 背景")]),t._v(" "),a("p",[t._v("状态机是强大的,因为它们的行为总是被保证是一致的,并且由于操作规则是在机器启动时用石头写成的,因此相对容易地进行调试。其思想是,你的应用程序现在处于并且可能存在于有限数量的状态中。然后会发生一些事情,将你的应用程序从一个州带到另一个州。状态机由触发器驱动,触发器基于事件或计时器。")]),t._v(" "),a("p",[t._v("在应用程序之外设计高级逻辑,然后以各种不同的方式与状态机交互,这要容易得多。你可以通过发送事件、监听状态机所做的工作或请求当前状态来与状态机交互。")]),t._v(" "),a("p",[t._v("传统上,当开发人员意识到代码库开始看起来像一盘满是意大利面条的盘子时,状态机就会被添加到现有的项目中。意大利面条代码看起来像是一个永无止境的,if,else 和 break 子句的层次结构,当事情开始看起来太复杂时,编译器可能应该要求开发人员回家。")]),t._v(" "),a("h2",{attrs:{id:"使用场景"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用场景"}},[t._v("#")]),t._v(" 使用场景")]),t._v(" "),a("p",[t._v("当出现以下情况时,项目是使用状态机的一个很好的候选者:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("你可以将应用程序或其结构的一部分表示为状态。")])]),t._v(" "),a("li",[a("p",[t._v("你希望将复杂的逻辑分割成更小的可管理的任务。")])]),t._v(" "),a("li",[a("p",[t._v("应用程序已经遇到了并发性问题,例如,某些事情是异步发生的。")])])]),t._v(" "),a("p",[t._v("你已经在尝试实现一个状态机,当你:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("使用布尔标志或枚举来建模情况。")])]),t._v(" "),a("li",[a("p",[t._v("具有仅对应用程序生命周期的某些部分具有意义的变量。")])]),t._v(" "),a("li",[a("p",[t._v("在一个 if-else 结构(或者更糟糕的是,多个这样的结构)中循环,检查是否设置了特定的标志或枚举,然后在你的标志和枚举的某些组合存在或不存在时,针对该做什么做出进一步的例外。")])])]),t._v(" "),a("h1",{attrs:{id:"入门"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#入门"}},[t._v("#")]),t._v(" 入门")]),t._v(" "),a("p",[t._v("如果你刚刚开始使用 Spring Statemachine,这是适合你的一节!这里,我们回答基本的“"),a("code",[t._v("what?")]),t._v("”、“"),a("code",[t._v("how?")]),t._v("”和“"),a("code",[t._v("why?")]),t._v("”问题。我们从温和地介绍 Spring 静态机械开始。然后,我们构建了我们的第一个 Spring Statemachine 应用程序,并讨论了一些核心原则。")]),t._v(" "),a("h2",{attrs:{id:"系统需求"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#系统需求"}},[t._v("#")]),t._v(" 系统需求")]),t._v(" "),a("p",[t._v("Spring StateMachine3.0.1 是用 JDK8(所有工件都具有 JDK7 兼容性)和 Spring Framework5.3.8 构建和测试的。在其核心系统中,它不需要 Spring 框架之外的任何其他依赖关系。")]),t._v(" "),a("p",[t._v("其他可选部分(例如"),a("a",{attrs:{href:"#sm-distributed"}},[t._v("使用分布状态")]),t._v(")对 ZooKeeper 具有依赖性,而"),a("a",{attrs:{href:"#statemachine-examples"}},[t._v("状态机示例")]),t._v("则对"),a("code",[t._v("spring-shell")]),t._v(""),a("code",[t._v("spring-boot")]),t._v("具有依赖性,这会将其他依赖性拉出框架本身之外。此外,可选的安全和数据访问功能具有对 Spring 安全和 Spring 数据模块的依赖性。")]),t._v(" "),a("h2",{attrs:{id:"模块"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#模块"}},[t._v("#")]),t._v(" 模块")]),t._v(" "),a("p",[t._v("下表描述了 Spring Statemachine 可用的模块。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th",[t._v("Module")]),t._v(" "),a("th",[t._v("说明")])])]),t._v(" "),a("tbody",[a("tr",[a("td",[a("code",[t._v("spring-statemachine-core")])]),t._v(" "),a("td",[t._v("Spring 机械的核心系统。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-recipes-common")])]),t._v(" "),a("td",[t._v("不需要依赖于核心"),a("br"),t._v("框架之外的常见菜谱。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-kryo")])]),t._v(" "),a("td",[t._v("Spring statemachine 的"),a("code",[t._v("Kryo")]),t._v("序列化器。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-data-common")])]),t._v(" "),a("td",[t._v("用于"),a("code",[t._v("Spring Data")]),t._v("的公共支持模块。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-data-jpa")])]),t._v(" "),a("td",[a("code",[t._v("Spring Data JPA")]),t._v("的支持模块。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-data-redis")])]),t._v(" "),a("td",[a("code",[t._v("Spring Data Redis")]),t._v("的支持模块。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-data-mongodb")])]),t._v(" "),a("td",[a("code",[t._v("Spring Data MongoDB")]),t._v("的支持模块。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-zookeeper")])]),t._v(" "),a("td",[t._v("分布式状态机的 ZooKeeper 集成。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-test")])]),t._v(" "),a("td",[t._v("支持状态机测试的模块.")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-cluster")])]),t._v(" "),a("td",[t._v("支持 Spring 云集群的模块。"),a("br"),t._v("注意, Spring 云集群已被 Spring 集成所取代。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-uml")])]),t._v(" "),a("td",[t._v("使用 Eclipse Papyrus 进行 UI UML 建模的支持模块。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-autoconfigure")])]),t._v(" "),a("td",[t._v("支持 Spring 启动的模块。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-bom")])]),t._v(" "),a("td",[t._v("材料清单 POM。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("spring-statemachine-starter")])]),t._v(" "),a("td",[t._v("Spring 引导启动器。")])])])]),t._v(" "),a("h2",{attrs:{id:"使用-gradle"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用-gradle"}},[t._v("#")]),t._v(" 使用 Gradle")]),t._v(" "),a("p",[t._v("下面的清单显示了一个典型的"),a("code",[t._v("build.gradle")]),t._v("文件,该文件是通过在"),a("a",{attrs:{href:"https://start.spring.io",target:"_blank",rel:"noopener noreferrer"}},[t._v("https://start.spring.io"),a("OutboundLink")],1),t._v("处选择各种设置来创建的:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("buildscript {\n\text {\n\t\tspringBootVersion = '2.4.8'\n\t}\n\trepositories {\n\t\tmavenCentral()\n\t\tmaven { url \"https://repo.spring.io/snapshot\" }\n\t\tmaven { url \"https://repo.spring.io/milestone\" }\n\t}\n\tdependencies {\n\t\tclasspath(\"org.springframework.boot:spring-boot-gradle-plugin:${springBootVersion}\")\n\t}\n}\n\napply plugin: 'java'\napply plugin: 'eclipse'\napply plugin: 'org.springframework.boot'\napply plugin: 'io.spring.dependency-management'\n\ngroup = 'com.example'\nversion = '0.0.1-SNAPSHOT'\nsourceCompatibility = 1.8\n\nrepositories {\n\tmavenCentral()\n\tmaven { url \"https://repo.spring.io/snapshot\" }\n\tmaven { url \"https://repo.spring.io/milestone\" }\n}\n\next {\n\tspringStatemachineVersion = '3.0.1'\n}\n\ndependencies {\n\tcompile('org.springframework.statemachine:spring-statemachine-starter')\n\ttestCompile('org.springframework.boot:spring-boot-starter-test')\n}\n\ndependencyManagement {\n\timports {\n\t\tmavenBom \"org.springframework.statemachine:spring-statemachine-bom:${springStatemachineVersion}\"\n\t}\n}\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("用要使用的版本替换"),a("code",[t._v("0.0.1-SNAPSHOT")]),t._v("")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("对于普通的项目结构,你可以使用以下命令构建该项目:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# ./gradlew clean build\n")])])]),a("p",[t._v("预期的 Spring 引导打包的 FAT JAR 将是"),a("code",[t._v("build/libs/demo-0.0.1-SNAPSHOT.jar")]),t._v("")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("对于"),a("br"),t._v("产品开发,不需要"),a("code",[t._v("libs-milestone")]),t._v(""),a("code",[t._v("libs-snapshot")]),t._v("存储库。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"使用-maven"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用-maven"}},[t._v("#")]),t._v(" 使用 Maven")]),t._v(" "),a("p",[t._v("下面的示例显示了一个典型的"),a("code",[t._v("pom.xml")]),t._v("文件,该文件是通过在"),a("a",{attrs:{href:"https://start.spring.io",target:"_blank",rel:"noopener noreferrer"}},[t._v("https://start.spring.io"),a("OutboundLink")],1),t._v("处选择各种选项创建的:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('<?xml version="1.0" encoding="UTF-8"?>\n<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"\n\txsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">\n\t<modelVersion>4.0.0</modelVersion>\n\n\t<groupId>com.example</groupId>\n\t<artifactId>demo</artifactId>\n\t<version>0.0.1-SNAPSHOT</version>\n\t<packaging>jar</packaging>\n\n\t<name>gs-statemachine</name>\n\t<description>Demo project for Spring Statemachine</description>\n\n\t<parent>\n\t\t<groupId>org.springframework.boot</groupId>\n\t\t<artifactId>spring-boot-starter-parent</artifactId>\n\t\t<version>2.4.8</version>\n\t\t<relativePath/> \x3c!-- lookup parent from repository --\x3e\n\t</parent>\n\n\t<properties>\n\t\t<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>\n\t\t<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>\n\t\t<java.version>1.8</java.version>\n\t\t<spring-statemachine.version>3.0.1</spring-statemachine.version>\n\t</properties>\n\n\t<dependencies>\n\t\t<dependency>\n\t\t\t<groupId>org.springframework.statemachine</groupId>\n\t\t\t<artifactId>spring-statemachine-starter</artifactId>\n\t\t</dependency>\n\n\t\t<dependency>\n\t\t\t<groupId>org.springframework.boot</groupId>\n\t\t\t<artifactId>spring-boot-starter-test</artifactId>\n\t\t\t<scope>test</scope>\n\t\t</dependency>\n\t</dependencies>\n\n\t<dependencyManagement>\n\t\t<dependencies>\n\t\t\t<dependency>\n\t\t\t\t<groupId>org.springframework.statemachine</groupId>\n\t\t\t\t<artifactId>spring-statemachine-bom</artifactId>\n\t\t\t\t<version>${spring-statemachine.version}</version>\n\t\t\t\t<type>pom</type>\n\t\t\t\t<scope>import</scope>\n\t\t\t</dependency>\n\t\t</dependencies>\n\t</dependencyManagement>\n\n\t<build>\n\t\t<plugins>\n\t\t\t<plugin>\n\t\t\t\t<groupId>org.springframework.boot</groupId>\n\t\t\t\t<artifactId>spring-boot-maven-plugin</artifactId>\n\t\t\t</plugin>\n\t\t</plugins>\n\t</build>\n\n\t<repositories>\n\t\t<repository>\n\t\t\t<id>spring-snapshots</id>\n\t\t\t<name>Spring Snapshots</name>\n\t\t\t<url>https://repo.spring.io/snapshot</url>\n\t\t\t<snapshots>\n\t\t\t\t<enabled>true</enabled>\n\t\t\t</snapshots>\n\t\t</repository>\n\t\t<repository>\n\t\t\t<id>spring-milestones</id>\n\t\t\t<name>Spring Milestones</name>\n\t\t\t<url>https://repo.spring.io/milestone</url>\n\t\t\t<snapshots>\n\t\t\t\t<enabled>false</enabled>\n\t\t\t</snapshots>\n\t\t</repository>\n\t</repositories>\n\n\t<pluginRepositories>\n\t\t<pluginRepository>\n\t\t\t<id>spring-snapshots</id>\n\t\t\t<name>Spring Snapshots</name>\n\t\t\t<url>https://repo.spring.io/snapshot</url>\n\t\t\t<snapshots>\n\t\t\t\t<enabled>true</enabled>\n\t\t\t</snapshots>\n\t\t</pluginRepository>\n\t\t<pluginRepository>\n\t\t\t<id>spring-milestones</id>\n\t\t\t<name>Spring Milestones</name>\n\t\t\t<url>https://repo.spring.io/milestone</url>\n\t\t\t<snapshots>\n\t\t\t\t<enabled>false</enabled>\n\t\t\t</snapshots>\n\t\t</pluginRepository>\n\t</pluginRepositories>\n\n</project>\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("用要使用的版本替换"),a("code",[t._v("0.0.1-SNAPSHOT")]),t._v("")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("对于普通的项目结构,你可以使用以下命令构建该项目:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# mvn clean package\n")])])]),a("p",[t._v("预期的 Spring 引导打包的 fat-jar 将是"),a("code",[t._v("target/demo-0.0.1-SNAPSHOT.jar")]),t._v("")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("对于"),a("br"),t._v("产品开发,不需要"),a("code",[t._v("libs-milestone")]),t._v(""),a("code",[t._v("libs-snapshot")]),t._v("存储库。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"开发你的第一个-spring-statemachine-应用程序"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#开发你的第一个-spring-statemachine-应用程序"}},[t._v("#")]),t._v(" 开发你的第一个 Spring Statemachine 应用程序")]),t._v(" "),a("p",[t._v("你可以从创建一个简单的 Spring boot"),a("code",[t._v("Application")]),t._v("类开始,该类实现"),a("code",[t._v("CommandLineRunner")]),t._v("。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@SpringBootApplication\npublic class Application implements CommandLineRunner {\n\n    public static void main(String[] args) {\n        SpringApplication.run(Application.class, args);\n    }\n\n}\n")])])]),a("p",[t._v("然后需要添加状态和事件,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States {\n    SI, S1, S2\n}\n\npublic enum Events {\n    E1, E2\n}\n")])])]),a("p",[t._v("然后需要添加状态机配置,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class StateMachineConfig\n        extends EnumStateMachineConfigurerAdapter<States, Events> {\n\n    @Override\n    public void configure(StateMachineConfigurationConfigurer<States, Events> config)\n            throws Exception {\n        config\n            .withConfiguration()\n                .autoStartup(true)\n                .listener(listener());\n    }\n\n    @Override\n    public void configure(StateMachineStateConfigurer<States, Events> states)\n            throws Exception {\n        states\n            .withStates()\n                .initial(States.SI)\n                    .states(EnumSet.allOf(States.class));\n    }\n\n    @Override\n    public void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n            throws Exception {\n        transitions\n            .withExternal()\n                .source(States.SI).target(States.S1).event(Events.E1)\n                .and()\n            .withExternal()\n                .source(States.S1).target(States.S2).event(Events.E2);\n    }\n\n    @Bean\n    public StateMachineListener<States, Events> listener() {\n        return new StateMachineListenerAdapter<States, Events>() {\n            @Override\n            public void stateChanged(State<States, Events> from, State<States, Events> to) {\n                System.out.println("State change to " + to.getId());\n            }\n        };\n    }\n}\n')])])]),a("p",[t._v("然后需要实现"),a("code",[t._v("CommandLineRunner")]),t._v("和 autowire"),a("code",[t._v("StateMachine")]),t._v("。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Autowired\nprivate StateMachine<States, Events> stateMachine;\n\n@Override\npublic void run(String... args) throws Exception {\n    stateMachine.sendEvent(Events.E1);\n    stateMachine.sendEvent(Events.E2);\n}\n")])])]),a("p",[t._v("根据你是使用"),a("code",[t._v("Gradle")]),t._v("还是"),a("code",[t._v("Maven")]),t._v("构建应用程序,你可以分别使用"),a("code",[t._v("java -jar build/libs/gs-statemachine-0.1.0.jar")]),t._v(""),a("code",[t._v("java -jar target/gs-statemachine-0.1.0.jar")]),t._v("来运行它。")]),t._v(" "),a("p",[t._v("这个命令的结果应该是正常的引导输出。但是,你还应该找到以下几行:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("State change to SI\nState change to S1\nState change to S2\n")])])]),a("p",[t._v("这些行表示你构造的机器正在从一种状态移动到另一种状态,正如它应该的那样。")]),t._v(" "),a("h1",{attrs:{id:"最新更新"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#最新更新"}},[t._v("#")]),t._v(" 最新更新")]),t._v(" "),a("h2",{attrs:{id:"in1-1"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#in1-1"}},[t._v("#")]),t._v(" in1.1")]),t._v(" "),a("p",[t._v("Spring StateMachine1.1 专注于安全性和与 Web 应用程序的更好的互操作性。它包括以下内容:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("增加了对 Spring 安全性的全面支持。见"),a("a",{attrs:{href:"#sm-security"}},[t._v("状态机安全")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("与“@withstatemachine”的上下文集成已大大增强。见"),a("a",{attrs:{href:"#sm-context"}},[t._v("上下文整合")]),t._v("")])]),t._v(" "),a("li",[a("p",[a("code",[t._v("StateContext")]),t._v("现在是一级公民,允许你与状态机进行交互。参见[使用"),a("code",[t._v("StateContext")]),t._v("]。")])]),t._v(" "),a("li",[a("p",[t._v("围绕持久性的特性已经通过对 Redis 的内置支持得到了增强。见"),a("a",{attrs:{href:"#sm-persist-redis"}},[t._v("使用 Redis")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("一个新的特性有助于持久化操作。参见[使用"),a("code",[t._v("StateMachinePersister")]),t._v("]。")])]),t._v(" "),a("li",[a("p",[t._v("配置模型类现在在一个公共 API 中。")])]),t._v(" "),a("li",[a("p",[t._v("基于计时器的事件的新功能。")])]),t._v(" "),a("li",[a("p",[t._v(""),a("code",[t._v("Junction")]),t._v("伪态。见"),a("a",{attrs:{href:"#statemachine-config-states-junction"}},[t._v("连接状态")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("新的出口点和入口点是假状态。见"),a("a",{attrs:{href:"#statemachine-config-states-exitentry"}},[t._v("出境点和入境点状态")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("配置模型验证器。")])]),t._v(" "),a("li",[a("p",[t._v("新样品。见"),a("a",{attrs:{href:"#statemachine-examples-security"}},[t._v("Security")]),t._v(""),a("a",{attrs:{href:"#statemachine-examples-eventservice"}},[t._v("活动服务")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("使用 Eclipse Papyrus 的 UI 建模支持。见"),a("a",{attrs:{href:"#sm-papyrus"}},[t._v("Eclipse 建模支持")]),t._v("")])])]),t._v(" "),a("h2",{attrs:{id:"in1-2"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#in1-2"}},[t._v("#")]),t._v(" in1.2")]),t._v(" "),a("p",[t._v("Spring StateMachine1.2 侧重于通用增强、更好的 UML 支持以及与外部配置存储库的集成。它包括以下内容:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("支持 UML 子机。见"),a("a",{attrs:{href:"#sm-papyrus-submachineref"}},[t._v("使用子机引用")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("将机器配置保留在外部存储库中的新的存储库抽象。见"),a("a",{attrs:{href:"#sm-repository"}},[t._v("存储库支持")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("对国家行动的新支持。见"),a("a",{attrs:{href:"#state-actions"}},[t._v("国家行动")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("新的转换错误动作概念。见"),a("a",{attrs:{href:"#statemachine-config-transition-actions-errorhandling"}},[t._v("转换动作错误处理")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("新的动作错误概念。见"),a("a",{attrs:{href:"#statemachine-config-state-actions-errorhandling"}},[t._v("状态动作错误处理")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("Spring 启动支持的初步工作。见"),a("a",{attrs:{href:"#sm-boot"}},[t._v("Spring Boot Support")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("支持跟踪和监视。见"),a("a",{attrs:{href:"#sm-monitoring"}},[t._v("监视状态机")]),t._v("")])])]),t._v(" "),a("h3",{attrs:{id:"in1-2-8"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#in1-2-8"}},[t._v("#")]),t._v(" in1.2.8")]),t._v(" "),a("p",[t._v("Spring Statemachine1.2.8 所包含的功能比在点释放中通常看不到的多一点,但是这些变化不值得 Spring Statemachine1.3 的叉子。它包括以下内容:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("JPA 实体类已经更改了表名。见"),a("a",{attrs:{href:"#sm-repository-config-jpa"}},[t._v("JPA")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("一个新的样本。见"),a("a",{attrs:{href:"#statemachine-examples-datapersist"}},[t._v("数据持续存在")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("用于持久性的新实体类。见"),a("a",{attrs:{href:"#sm-repository-persistence"}},[t._v("存储库持久性")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("过渡冲突政策。见"),a("a",{attrs:{href:"#statemachine-config-commonsettings"}},[t._v("配置公共设置")])])])]),t._v(" "),a("h2",{attrs:{id:"in2-0"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#in2-0"}},[t._v("#")]),t._v(" in2.0")]),t._v(" "),a("p",[t._v("Spring StateMachine2.0 关注于 Spring Boot2.x 支持。")]),t._v(" "),a("h3",{attrs:{id:"in2-0-0"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#in2-0-0"}},[t._v("#")]),t._v(" in2.0.0")]),t._v(" "),a("p",[t._v("Spring Statemachine2.0.0 包括以下内容:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("监视和跟踪的格式已经更改。见"),a("a",{attrs:{href:"#sm-boot-monitoring"}},[t._v("监测和追踪")]),t._v("")])]),t._v(" "),a("li",[a("p",[a("code",[t._v("spring-statemachine-boot")]),t._v("模块已重命名为"),a("code",[t._v("spring-statemachine-autoconfigure")]),t._v("")])])]),t._v(" "),a("h2",{attrs:{id:"in3-0"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#in3-0"}},[t._v("#")]),t._v(" in3.0")]),t._v(" "),a("p",[t._v("Spring Statemachine3.0.0 侧重于添加反应性支持。从"),a("code",[t._v("2.x")]),t._v("移动到"),a("code",[t._v("3.x")]),t._v("会引入一些突破性的变化,在"),a("a",{attrs:{href:"#appendix-reactormigrationguide"}},[t._v("反应堆迁移指南")]),t._v("中有详细说明。")]),t._v(" "),a("p",[t._v("使用"),a("code",[t._v("3.0.x")]),t._v(",我们已经不推荐所有阻塞方法,这些方法将在未来的版本中的某个时刻被删除。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("请仔细阅读附录"),a("a",{attrs:{href:"#appendix-reactormigrationguide"}},[t._v("反应堆迁移指南")]),t._v(",因为它将引导你完成向"),a("br"),t._v("迁移的过程,对于内部不处理的情况。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("在这一点上,大多数文档已经被更改为展示反应性接口,而我们仍然保留一些注释,以供仍在使用旧的阻塞方法的用户使用。")]),t._v(" "),a("h1",{attrs:{id:"使用-spring-安定"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用-spring-安定"}},[t._v("#")]),t._v(" 使用 Spring 安定")]),t._v(" "),a("p",[t._v("参考文档的这一部分解释了 Spring StateMachine 向任何基于 Spring 的应用程序提供的核心功能。")]),t._v(" "),a("p",[t._v("它包括以下主题:")]),t._v(" "),a("ul",[a("li",[a("p",[a("a",{attrs:{href:"#sm-config"}},[t._v("机器构型")]),t._v("描述了通用配置支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-machineid"}},[t._v("状态机 ID")]),t._v("描述了机器 ID 的使用。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-factories"}},[t._v("国营机器工厂")]),t._v("描述了通用状态机工厂支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-deferevents"}},[t._v("使用延迟事件")]),t._v("描述了延迟事件支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-scopes"}},[t._v("使用作用域")]),t._v("描述了范围支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-actions"}},[t._v("使用动作")]),t._v("描述了对操作的支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-guards"}},[t._v("使用防护装置")]),t._v("描述了防护支撑。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-extendedstate"}},[t._v("使用扩展状态")]),t._v("描述了扩展的状态支持。")])]),t._v(" "),a("li",[a("p",[t._v("[使用"),a("code",[t._v("StateContext")]),t._v("]描述状态上下文支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-triggers"}},[t._v("触发转换")]),t._v("描述了触发器的使用。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-listeners"}},[t._v("监听状态机事件")]),t._v("描述了状态机侦听器的使用。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-context"}},[t._v("上下文整合")]),t._v("描述了通用的 Spring 应用程序上下文支持。")])]),t._v(" "),a("li",[a("p",[t._v("[使用"),a("code",[t._v("StateMachineAccessor")]),t._v("]描述了对状态机内部访问器的支持。")])]),t._v(" "),a("li",[a("p",[t._v("[使用"),a("code",[t._v("StateMachineInterceptor")]),t._v("]描述了状态机错误处理支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-security"}},[t._v("状态机安全")]),t._v("描述了状态机的安全支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-error-handling"}},[t._v("状态机错误处理")]),t._v("描述了状态机拦截器的支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-service"}},[t._v("状态机服务")]),t._v("描述状态机服务支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-persist"}},[t._v("保持状态机")]),t._v("描述状态机持久支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-boot"}},[t._v("Spring Boot Support")]),t._v("描述了 Spring 引导支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-monitoring"}},[t._v("监视状态机")]),t._v("描述了监视和转换支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-distributed"}},[t._v("使用分布状态")]),t._v("描述分布式状态机支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-test"}},[t._v("测试支持")]),t._v("描述了状态机测试支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-papyrus"}},[t._v("Eclipse 建模支持")]),t._v("描述了状态机 UML 建模支持。")])]),t._v(" "),a("li",[a("p",[a("a",{attrs:{href:"#sm-repository"}},[t._v("存储库支持")]),t._v("描述状态机存储库配置支持。")])])]),t._v(" "),a("h2",{attrs:{id:"机械构型"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#机械构型"}},[t._v("#")]),t._v(" 机械构型")]),t._v(" "),a("p",[t._v("使用状态机时的常见任务之一是设计其运行时配置。本章重点讨论如何配置 Spring StateMachine,以及如何利用 Spring 的轻量级 IoC 容器来简化应用程序内部,使其更易于管理。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("本节中的配置示例没有完成功能。也就是说,"),a("br"),t._v("总是需要同时定义状态和转换。"),a("br"),t._v("否则,状态机配置将是格式错误的。我们有"),a("br"),t._v(",只是通过将其他需要的部分"),a("br"),t._v("留出来,使代码片段不那么冗长。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"使用enable注释"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用enable注释"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("enable")]),t._v("注释")]),t._v(" "),a("p",[t._v("我们使用两个熟悉的 Spring "),a("em",[t._v("使能者")]),t._v("注释来简化配置:"),a("code",[t._v("@EnableStateMachine")]),t._v(""),a("code",[t._v("@EnableStateMachineFactory")]),t._v("。当将这些注释放置在"),a("code",[t._v("@Configuration")]),t._v("类中时,将启用状态机所需的一些基本功能。")]),t._v(" "),a("p",[t._v("当需要配置来创建"),a("code",[t._v("StateMachine")]),t._v("实例时,可以使用"),a("code",[t._v("@EnableStateMachine")]),t._v("。通常,"),a("code",[t._v("@Configuration")]),t._v("类扩展了适配器("),a("code",[t._v("EnumStateMachineConfigurerAdapter")]),t._v(""),a("code",[t._v("StateMachineConfigurerAdapter")]),t._v("),它允许你覆盖配置回调方法。我们会自动检测你是否使用这些适配器类,并相应地修改运行时配置逻辑。")]),t._v(" "),a("p",[t._v("当需要配置来创建"),a("code",[t._v("StateMachineFactory")]),t._v("实例时,可以使用"),a("code",[t._v("@EnableStateMachineFactory")]),t._v("")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("下面几节将展示这些方法的使用示例。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"配置状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置状态"}},[t._v("#")]),t._v(" 配置状态")]),t._v(" "),a("p",[t._v("在本指南的后面,我们将介绍更复杂的配置示例,但我们首先从简单的内容开始。对于大多数简单的状态机,你可以使用"),a("code",[t._v("EnumStateMachineConfigurerAdapter")]),t._v("并定义可能的状态并选择初始和可选的结束状态。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config1Enums\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.S1)\n\t\t\t\t.end(States.SF)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n}\n")])])]),a("p",[t._v("你还可以使用"),a("code",[t._v("StateMachineConfigurerAdapter")]),t._v("将字符串而不是枚举作为状态和事件,如下一个示例所示。大多数配置示例使用枚举,但通常来说,你可以交换字符串和枚举。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config1Strings\n\t\textends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.end("SF")\n\t\t\t\t.states(new HashSet<String>(Arrays.asList("S1","S2","S3","S4")));\n\t}\n\n}\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("使用枚举带来了一组更安全的状态和事件类型,但"),a("br"),t._v("限制了编译时间的可能组合。字符串没有这个"),a("br"),t._v("限制,允许你使用更动态的方式来构建状态"),a("br"),t._v("机器配置,但不允许相同级别的安全。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"配置层次结构状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置层次结构状态"}},[t._v("#")]),t._v(" 配置层次结构状态")]),t._v(" "),a("p",[t._v("可以通过使用多个"),a("code",[t._v("withStates()")]),t._v("调用来定义分层状态,其中可以使用"),a("code",[t._v("parent()")]),t._v("来指示这些特定状态是某些其他状态的子状态。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config2\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.S1)\n\t\t\t\t.state(States.S1)\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States.S1)\n\t\t\t\t\t.initial(States.S2)\n\t\t\t\t\t.state(States.S2);\n\t}\n\n}\n")])])]),a("h3",{attrs:{id:"配置区域"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置区域"}},[t._v("#")]),t._v(" 配置区域")]),t._v(" "),a("p",[t._v("没有特殊的配置方法来将一组状态标记为正交状态的一部分。简单地说,当同一个层次状态机有多个状态集,每个状态集都有一个初始状态时,就会创建正交状态。因为单个状态机只能有一个初始状态,所以多个初始状态意味着一个特定的状态必须有多个独立的区域。下面的示例展示了如何定义区域:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config10\n\t\textends EnumStateMachineConfigurerAdapter<States2, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States2, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States2.S1)\n\t\t\t\t.state(States2.S2)\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S2)\n\t\t\t\t\t.initial(States2.S2I)\n\t\t\t\t\t.state(States2.S21)\n\t\t\t\t\t.end(States2.S2F)\n\t\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S2)\n\t\t\t\t\t.initial(States2.S3I)\n\t\t\t\t\t.state(States2.S31)\n\t\t\t\t\t.end(States2.S3F);\n\t}\n\n}\n")])])]),a("p",[t._v("当持久化具有区域的机器时,或者通常依赖于任何功能来重置机器时,你可能需要为一个区域提供一个专用的 ID。默认情况下,此 ID 是生成的 UUID。如下例所示,"),a("code",[t._v("StateConfigurer")]),t._v("有一个名为"),a("code",[t._v("region(String id)")]),t._v("的方法,它允许你为区域设置 ID:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config10RegionId\n\t\textends EnumStateMachineConfigurerAdapter<States2, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States2, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States2.S1)\n\t\t\t\t.state(States2.S2)\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S2)\n\t\t\t\t\t.region("R1")\n\t\t\t\t\t.initial(States2.S2I)\n\t\t\t\t\t.state(States2.S21)\n\t\t\t\t\t.end(States2.S2F)\n\t\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S2)\n\t\t\t\t\t.region("R2")\n\t\t\t\t\t.initial(States2.S3I)\n\t\t\t\t\t.state(States2.S31)\n\t\t\t\t\t.end(States2.S3F);\n\t}\n\n}\n')])])]),a("h3",{attrs:{id:"配置转换"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置转换"}},[t._v("#")]),t._v(" 配置转换")]),t._v(" "),a("p",[t._v("我们支持三种不同类型的转换:"),a("code",[t._v("external")]),t._v(""),a("code",[t._v("internal")]),t._v(""),a("code",[t._v("local")]),t._v("。转换由信号(发送到状态机的事件)或计时器触发。下面的示例展示了如何定义所有三种类型的转换:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config3\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.S1)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source(States.S1).target(States.S2)\n\t\t\t\t.event(Events.E1)\n\t\t\t\t.and()\n\t\t\t.withInternal()\n\t\t\t\t.source(States.S2)\n\t\t\t\t.event(Events.E2)\n\t\t\t\t.and()\n\t\t\t.withLocal()\n\t\t\t\t.source(States.S2).target(States.S3)\n\t\t\t\t.event(Events.E3);\n\t}\n\n}\n")])])]),a("h3",{attrs:{id:"配置守卫"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置守卫"}},[t._v("#")]),t._v(" 配置守卫")]),t._v(" "),a("p",[t._v("你可以使用保护来保护状态转换。你可以使用"),a("code",[t._v("Guard")]),t._v("接口来执行计算,其中方法可以访问"),a("code",[t._v("StateContext")]),t._v("。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config4\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source(States.S1).target(States.S2)\n\t\t\t\t.event(Events.E1)\n\t\t\t\t.guard(guard())\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source(States.S2).target(States.S3)\n\t\t\t\t.event(Events.E2)\n\t\t\t\t.guardExpression("true");\n\n\t}\n\n\t@Bean\n\tpublic Guard<States, Events> guard() {\n\t\treturn new Guard<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\t\t\treturn true;\n\t\t\t}\n\t\t};\n\t}\n\n}\n')])])]),a("p",[t._v("在前面的示例中,我们使用了两种不同类型的保护配置。首先,我们创建了一个简单的"),a("code",[t._v("Guard")]),t._v("作为 Bean,并将其附加到状态"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v("之间的转换。")]),t._v(" "),a("p",[t._v("其次,我们使用 SPEL 表达式作为保护,要求表达式必须返回"),a("code",[t._v("BOOLEAN")]),t._v("值。在幕后,这个基于表达式的保护是"),a("code",[t._v("SpelExpressionGuard")]),t._v("。我们将其附加到状态"),a("code",[t._v("S2")]),t._v(""),a("code",[t._v("S3")]),t._v("之间的转换。两个后卫的估值总是"),a("code",[t._v("true")]),t._v("")]),t._v(" "),a("h3",{attrs:{id:"配置操作"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置操作"}},[t._v("#")]),t._v(" 配置操作")]),t._v(" "),a("p",[t._v("你可以定义要用转换和状态执行的操作。动作总是作为源自触发器的转换的结果运行的。下面的示例展示了如何定义一个动作:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config51\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source(States.S1)\n\t\t\t\t.target(States.S2)\n\t\t\t\t.event(Events.E1)\n\t\t\t\t.action(action());\n\t}\n\n\t@Bean\n\tpublic Action<States, Events> action() {\n\t\treturn new Action<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\t\t// do something\n\t\t\t}\n\t\t};\n\t}\n\n}\n")])])]),a("p",[t._v("在前面的示例中,单个"),a("code",[t._v("Action")]),t._v("被定义为名为"),a("code",[t._v("action")]),t._v("的 Bean,并与从"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v("的转换相关联。下面的示例展示了如何多次使用一个动作:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config52\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.S1, action())\n\t\t\t\t.state(States.S1, action(), null)\n\t\t\t\t.state(States.S2, null, action())\n\t\t\t\t.state(States.S2, action())\n\t\t\t\t.state(States.S3, action(), action());\n\t}\n\n\t@Bean\n\tpublic Action<States, Events> action() {\n\t\treturn new Action<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\t\t// do something\n\t\t\t}\n\t\t};\n\t}\n\n}\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("通常,你不会为不同的"),a("br"),t._v("阶段定义相同的"),a("code",[t._v("Action")]),t._v("实例,但是我们在这里这样做是为了避免在代码"),a("br"),t._v("片段中产生太多噪声。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("在前面的示例中,单个"),a("code",[t._v("Action")]),t._v("由名为"),a("code",[t._v("action")]),t._v("的 Bean 定义并与状态相关联的"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v(""),a("code",[t._v("S3")]),t._v("。我们需要弄清楚这是怎么回事:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("我们为初始状态定义了一个动作,"),a("code",[t._v("S1")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("我们为 state"),a("code",[t._v("S1")]),t._v("定义了一个进入动作,并将退出动作保留为空。")])]),t._v(" "),a("li",[a("p",[t._v("我们为 state"),a("code",[t._v("S2")]),t._v("定义了一个退出操作,并将该进入操作保留为空。")])]),t._v(" "),a("li",[a("p",[t._v("我们定义了状态"),a("code",[t._v("S2")]),t._v("的单个状态动作。")])]),t._v(" "),a("li",[a("p",[t._v("我们定义了状态"),a("code",[t._v("S3")]),t._v("的进入和退出操作。")])]),t._v(" "),a("li",[a("p",[t._v("注意,状态"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("initial()")]),t._v(""),a("code",[t._v("state()")]),t._v("函数一起使用两次。只有当你想要用初始状态定义进入或退出操作时,你才需要这样做。")])])]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v(""),a("code",[t._v("initial()")]),t._v("函数定义动作,只在启动状态机或子状态时运行特定的"),a("br"),t._v("动作。此操作"),a("br"),t._v("是仅运行一次的初始化操作。如果状态机返回"),a("br"),t._v("并在初始和非初始状态之间向前转换,则将运行定义有"),a("br"),t._v(""),a("code",[t._v("state()")]),t._v("的操作。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h4",{attrs:{id:"状态动作"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态动作"}},[t._v("#")]),t._v(" 状态动作")]),t._v(" "),a("p",[t._v("与进入和退出操作相比,运行状态操作是不同的,因为执行发生在输入状态之后,如果在特定操作完成之前发生了状态退出,则可以取消执行。")]),t._v(" "),a("p",[t._v("通过订阅反应堆的默认并行调度器,使用正常的无功流执行状态操作。这意味着,无论你在操作中做什么,都需要能够捕获"),a("code",[t._v("InterruptedException")]),t._v(",或者更一般地,定期检查"),a("code",[t._v("Thread")]),t._v("是否被中断。")]),t._v(" "),a("p",[t._v("下面的示例展示了使用默认"),a("code",[t._v("IMMEDIATE_CANCEL")]),t._v("的典型配置,当运行中的任务的状态完成时,该配置将立即取消该任务:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class Config1 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<String, String> config) throws Exception {\n\t\tconfig\n\t\t\t.withConfiguration()\n\t\t\t\t.stateDoActionPolicy(StateDoActionPolicy.IMMEDIATE_CANCEL);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states) throws Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.state("S2", context -> {})\n\t\t\t\t.state("S3");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions) throws Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("S1")\n\t\t\t\t.target("S2")\n\t\t\t\t.event("E1")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("S2")\n\t\t\t\t.target("S3")\n\t\t\t\t.event("E2");\n\t}\n}\n')])])]),a("p",[t._v("你可以将策略设置为"),a("code",[t._v("TIMEOUT_CANCEL")]),t._v(",并为每台机器设置一个全局超时。这将更改状态行为,以便在请求取消之前等待动作完成。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineConfigurationConfigurer<String, String> config) throws Exception {\n\tconfig\n\t\t.withConfiguration()\n\t\t\t.stateDoActionPolicy(StateDoActionPolicy.TIMEOUT_CANCEL)\n\t\t\t.stateDoActionPolicyTimeout(10, TimeUnit.SECONDS);\n}\n")])])]),a("p",[t._v("如果"),a("code",[t._v("Event")]),t._v("直接将机器带入一种状态,以便特定操作可以使用事件头,则还可以使用专用事件头来设置特定的超时(在"),a("code",[t._v("millis")]),t._v("中定义)。为此,你可以使用保留的标头值"),a("code",[t._v("StateMachineMessageHeaders.HEADER_DO_ACTION_TIMEOUT")]),t._v("。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateMachine<String, String> stateMachine;\n\nvoid sendEventUsingTimeout() {\n\tstateMachine\n\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t.withPayload("E1")\n\t\t\t.setHeader(StateMachineMessageHeaders.HEADER_DO_ACTION_TIMEOUT, 5000)\n\t\t\t.build()))\n\t\t.subscribe();\n\n}\n')])])]),a("h4",{attrs:{id:"转换动作错误处理"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#转换动作错误处理"}},[t._v("#")]),t._v(" 转换动作错误处理")]),t._v(" "),a("p",[t._v("你总是可以手动捕获异常。但是,对于为转换定义的操作,你可以定义一个错误操作,如果出现异常,则调用该操作。然后,从传递给该操作的"),a("code",[t._v("StateContext")]),t._v("中可以获得异常。下面的示例展示了如何创建处理异常的状态:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config53\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source(States.S1)\n\t\t\t\t.target(States.S2)\n\t\t\t\t.event(Events.E1)\n\t\t\t\t.action(action(), errorAction());\n\t}\n\n\t@Bean\n\tpublic Action<States, Events> action() {\n\t\treturn new Action<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\t\tthrow new RuntimeException("MyError");\n\t\t\t}\n\t\t};\n\t}\n\n\t@Bean\n\tpublic Action<States, Events> errorAction() {\n\t\treturn new Action<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\t\t// RuntimeException("MyError") added to context\n\t\t\t\tException exception = context.getException();\n\t\t\t\texception.getMessage();\n\t\t\t}\n\t\t};\n\t}\n\n}\n')])])]),a("p",[t._v("如果需要,你可以手动为每个动作创建类似的逻辑。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\tthrows Exception {\n\ttransitions\n\t\t.withExternal()\n\t\t\t.source(States.S1)\n\t\t\t.target(States.S2)\n\t\t\t.event(Events.E1)\n\t\t\t.action(Actions.errorCallingAction(action(), errorAction()));\n}\n")])])]),a("h4",{attrs:{id:"状态动作错误处理"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态动作错误处理"}},[t._v("#")]),t._v(" 状态动作错误处理")]),t._v(" "),a("p",[t._v("与处理状态转换中的错误的逻辑类似的逻辑也可用于状态的进入和状态的退出。")]),t._v(" "),a("p",[t._v("对于这些情况,"),a("code",[t._v("StateConfigurer")]),t._v("具有称为"),a("code",[t._v("stateEntry")]),t._v(""),a("code",[t._v("stateDo")]),t._v(""),a("code",[t._v("stateExit")]),t._v("的方法。这些方法定义了一个"),a("code",[t._v("error")]),t._v("动作和一个正常(非错误)"),a("code",[t._v("action")]),t._v("动作。下面的示例展示了如何使用这三种方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config55\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.S1)\n\t\t\t\t.stateEntry(States.S2, action(), errorAction())\n\t\t\t\t.stateDo(States.S2, action(), errorAction())\n\t\t\t\t.stateExit(States.S2, action(), errorAction())\n\t\t\t\t.state(States.S3);\n\t}\n\n\t@Bean\n\tpublic Action<States, Events> action() {\n\t\treturn new Action<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\t\tthrow new RuntimeException("MyError");\n\t\t\t}\n\t\t};\n\t}\n\n\t@Bean\n\tpublic Action<States, Events> errorAction() {\n\t\treturn new Action<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\t\t// RuntimeException("MyError") added to context\n\t\t\t\tException exception = context.getException();\n\t\t\t\texception.getMessage();\n\t\t\t}\n\t\t};\n\t}\n}\n')])])]),a("h3",{attrs:{id:"配置伪状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置伪状态"}},[t._v("#")]),t._v(" 配置伪状态")]),t._v(" "),a("p",[a("em",[t._v("伪态")]),t._v("配置通常是通过配置状态和转换来完成的。伪状态作为状态自动添加到状态机中。")]),t._v(" "),a("h4",{attrs:{id:"初始状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#初始状态"}},[t._v("#")]),t._v(" 初始状态")]),t._v(" "),a("p",[t._v("可以使用"),a("code",[t._v("initial()")]),t._v("方法将特定状态标记为初始状态。例如,这个初始操作对于初始化扩展状态变量是很好的。下面的示例展示了如何使用"),a("code",[t._v("initial()")]),t._v("方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config11\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.S1, initialAction())\n\t\t\t\t.end(States.SF)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n\t@Bean\n\tpublic Action<States, Events> initialAction() {\n\t\treturn new Action<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\t\t// do something initially\n\t\t\t}\n\t\t};\n\t}\n\n}\n")])])]),a("h4",{attrs:{id:"终止状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#终止状态"}},[t._v("#")]),t._v(" 终止状态")]),t._v(" "),a("p",[t._v("可以使用"),a("code",[t._v("end()")]),t._v("方法将特定状态标记为结束状态。对于每个子机器或区域,你最多可以执行一次。下面的示例展示了如何使用"),a("code",[t._v("end()")]),t._v("方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config1Enums\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.S1)\n\t\t\t\t.end(States.SF)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n}\n")])])]),a("h4",{attrs:{id:"国家历史"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#国家历史"}},[t._v("#")]),t._v(" 国家历史")]),t._v(" "),a("p",[t._v("你可以为每个单独的状态机定义一次状态历史。你需要选择其状态标识符并设置"),a("code",[t._v("History.SHALLOW")]),t._v(""),a("code",[t._v("History.DEEP")]),t._v("。下面的示例使用"),a("code",[t._v("History.SHALLOW")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config12\n\t\textends EnumStateMachineConfigurerAdapter<States3, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States3, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States3.S1)\n\t\t\t\t.state(States3.S2)\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States3.S2)\n\t\t\t\t\t.initial(States3.S2I)\n\t\t\t\t\t.state(States3.S21)\n\t\t\t\t\t.state(States3.S22)\n\t\t\t\t\t.history(States3.SH, History.SHALLOW);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States3, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withHistory()\n\t\t\t\t.source(States3.SH)\n\t\t\t\t.target(States3.S22);\n\t}\n\n}\n")])])]),a("p",[t._v("此外,正如前面的示例所示,你可以在同一台机器中定义从历史状态到状态顶点的缺省转换。例如,如果从未输入过机器,那么这种转换将作为默认情况发生——因此,没有可用的历史记录。如果未定义缺省状态转换,则完成对区域的正常输入。如果机器的历史记录是最终状态,也可以使用此默认转换。")]),t._v(" "),a("h4",{attrs:{id:"选择状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#选择状态"}},[t._v("#")]),t._v(" 选择状态")]),t._v(" "),a("p",[t._v("需要在状态和转换中定义选择才能正常工作。可以使用"),a("code",[t._v("choice()")]),t._v("方法将特定状态标记为选择状态。当为此选择配置转换时,此状态需要匹配源状态。")]),t._v(" "),a("p",[t._v("你可以通过使用"),a("code",[t._v("withChoice()")]),t._v("来配置转换,其中你定义了源状态和"),a("code",[t._v("first/then/last")]),t._v("结构,这等同于正常的"),a("code",[t._v("if/elseif/else")]),t._v("。使用"),a("code",[t._v("first")]),t._v(""),a("code",[t._v("then")]),t._v(",你可以指定一个保护,就像使用带有"),a("code",[t._v("if/elseif")]),t._v("子句的条件一样。")]),t._v(" "),a("p",[t._v("转换需要能够存在,因此你必须确保使用"),a("code",[t._v("last")]),t._v("。否则,该配置是不成型的。下面的示例展示了如何定义选择状态:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config13\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.SI)\n\t\t\t\t.choice(States.S1)\n\t\t\t\t.end(States.SF)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withChoice()\n\t\t\t\t.source(States.S1)\n\t\t\t\t.first(States.S2, s2Guard())\n\t\t\t\t.then(States.S3, s3Guard())\n\t\t\t\t.last(States.S4);\n\t}\n\n\t@Bean\n\tpublic Guard<States, Events> s2Guard() {\n\t\treturn new Guard<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t};\n\t}\n\n\t@Bean\n\tpublic Guard<States, Events> s3Guard() {\n\t\treturn new Guard<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\t\t\treturn true;\n\t\t\t}\n\t\t};\n\t}\n\n}\n")])])]),a("p",[t._v("操作可以与选择伪状态的传入和传出转换一起运行。正如下面的示例所示,定义了一个虚拟 lambda 操作,该操作将导致进入选择状态,并且为一个传出转换(其中还定义了一个错误操作)定义了一个类似的虚拟 lambda 操作:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config23\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.SI)\n\t\t\t\t.choice(States.S1)\n\t\t\t\t.end(States.SF)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source(States.SI)\n\t\t\t\t.action(c -> {\n\t\t\t\t\t\t// action with SI-S1\n\t\t\t\t\t})\n\t\t\t\t.target(States.S1)\n\t\t\t\t.and()\n\t\t\t.withChoice()\n\t\t\t\t.source(States.S1)\n\t\t\t\t.first(States.S2, c -> {\n\t\t\t\t\t\treturn true;\n\t\t\t\t\t})\n\t\t\t\t.last(States.S3, c -> {\n\t\t\t\t\t\t// action with S1-S3\n\t\t\t\t\t}, c -> {\n\t\t\t\t\t\t// error callback for action S1-S3\n\t\t\t\t\t});\n\t}\n}\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("具有相同 API 格式的连接意味着动作可以被定义"),a("br"),t._v("类似。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h4",{attrs:{id:"结态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#结态"}},[t._v("#")]),t._v(" 结态")]),t._v(" "),a("p",[t._v("你需要在状态和转换两方面定义一个连接,以使其正常工作。可以使用"),a("code",[t._v("junction()")]),t._v("方法将特定状态标记为选择状态。当为此选择配置转换时,此状态需要与源状态匹配。")]),t._v(" "),a("p",[t._v("你可以通过使用"),a("code",[t._v("withJunction()")]),t._v("来配置转换,其中你定义了源状态和"),a("code",[t._v("first/then/last")]),t._v("结构(相当于正常的"),a("code",[t._v("if/elseif/else")]),t._v(")。使用"),a("code",[t._v("first")]),t._v(""),a("code",[t._v("then")]),t._v(",你可以指定一个保护,就像使用带有"),a("code",[t._v("if/elseif")]),t._v("子句的条件一样。")]),t._v(" "),a("p",[t._v("转换需要能够存在,因此你必须确保使用"),a("code",[t._v("last")]),t._v("。否则,该配置是不成型的。下面的示例使用了一个结点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config20\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.SI)\n\t\t\t\t.junction(States.S1)\n\t\t\t\t.end(States.SF)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withJunction()\n\t\t\t\t.source(States.S1)\n\t\t\t\t.first(States.S2, s2Guard())\n\t\t\t\t.then(States.S3, s3Guard())\n\t\t\t\t.last(States.S4);\n\t}\n\n\t@Bean\n\tpublic Guard<States, Events> s2Guard() {\n\t\treturn new Guard<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t};\n\t}\n\n\t@Bean\n\tpublic Guard<States, Events> s3Guard() {\n\t\treturn new Guard<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\t\t\treturn true;\n\t\t\t}\n\t\t};\n\t}\n\n}\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("选择和连接之间的区别纯粹是学术性的,因为两者都是"),a("br"),t._v(""),a("code",[t._v("first/then/last")]),t._v("结构实现的。然而,在理论上,基于 UML 建模的"),a("br"),t._v(""),a("code",[t._v("choice")]),t._v("只允许一个传入转换,而"),a("code",[t._v("junction")]),t._v("允许多个传入转换。在代码级别上,"),a("br"),t._v("功能几乎相同。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h4",{attrs:{id:"fork-state"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#fork-state"}},[t._v("#")]),t._v(" fork state")]),t._v(" "),a("p",[t._v("你必须在状态和转换中定义一个 fork,才能使其正常工作。可以使用"),a("code",[t._v("fork()")]),t._v("方法将特定状态标记为选择状态。当为此 fork 配置转换时,此状态需要匹配源状态。")]),t._v(" "),a("p",[t._v("目标状态需要是一个区域中的超级状态或直接状态。使用超级状态作为目标,所有区域都会进入初始状态。以单个国家为目标可以更好地控制进入地区。下面的示例使用分叉:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config14\n\t\textends EnumStateMachineConfigurerAdapter<States2, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States2, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States2.S1)\n\t\t\t\t.fork(States2.S2)\n\t\t\t\t.state(States2.S3)\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S3)\n\t\t\t\t\t.initial(States2.S2I)\n\t\t\t\t\t.state(States2.S21)\n\t\t\t\t\t.state(States2.S22)\n\t\t\t\t\t.end(States2.S2F)\n\t\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S3)\n\t\t\t\t\t.initial(States2.S3I)\n\t\t\t\t\t.state(States2.S31)\n\t\t\t\t\t.state(States2.S32)\n\t\t\t\t\t.end(States2.S3F);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States2, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withFork()\n\t\t\t\t.source(States2.S2)\n\t\t\t\t.target(States2.S22)\n\t\t\t\t.target(States2.S32);\n\t}\n\n}\n")])])]),a("h4",{attrs:{id:"加入状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#加入状态"}},[t._v("#")]),t._v(" 加入状态")]),t._v(" "),a("p",[t._v("你必须在状态和转换中定义一个连接,才能使其正常工作。你可以使用"),a("code",[t._v("join()")]),t._v("方法将附节状态标记为选择状态。在转换配置中,此状态不需要匹配源状态或目标状态。")]),t._v(" "),a("p",[t._v("你可以选择一个目标状态,当所有源状态都已加入时,转换将在其中进行。如果使用状态托管区域作为源,则区域的结束状态被用作连接。否则,你可以从一个地区中选择任何一个州。以下 Exmaple 使用连接:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config15\n\t\textends EnumStateMachineConfigurerAdapter<States2, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States2, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States2.S1)\n\t\t\t\t.state(States2.S3)\n\t\t\t\t.join(States2.S4)\n\t\t\t\t.state(States2.S5)\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S3)\n\t\t\t\t\t.initial(States2.S2I)\n\t\t\t\t\t.state(States2.S21)\n\t\t\t\t\t.state(States2.S22)\n\t\t\t\t\t.end(States2.S2F)\n\t\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S3)\n\t\t\t\t\t.initial(States2.S3I)\n\t\t\t\t\t.state(States2.S31)\n\t\t\t\t\t.state(States2.S32)\n\t\t\t\t\t.end(States2.S3F);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States2, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withJoin()\n\t\t\t\t.source(States2.S2F)\n\t\t\t\t.source(States2.S3F)\n\t\t\t\t.target(States2.S4)\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source(States2.S4)\n\t\t\t\t.target(States2.S5);\n\t}\n}\n")])])]),a("p",[t._v("你还可以让多个转换起源于一个连接状态。在这种情况下,我们建议你使用保护并定义你的保护,以便在任何给定的时间只有一个保护的值"),a("code",[t._v("TRUE")]),t._v("。否则,过渡行为是不可预测的。下面的示例显示了这一点,在该示例中,保护程序检查扩展状态是否具有变量:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config22\n\t\textends EnumStateMachineConfigurerAdapter<States2, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States2, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States2.S1)\n\t\t\t\t.state(States2.S3)\n\t\t\t\t.join(States2.S4)\n\t\t\t\t.state(States2.S5)\n\t\t\t\t.end(States2.SF)\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S3)\n\t\t\t\t\t.initial(States2.S2I)\n\t\t\t\t\t.state(States2.S21)\n\t\t\t\t\t.state(States2.S22)\n\t\t\t\t\t.end(States2.S2F)\n\t\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States2.S3)\n\t\t\t\t\t.initial(States2.S3I)\n\t\t\t\t\t.state(States2.S31)\n\t\t\t\t\t.state(States2.S32)\n\t\t\t\t\t.end(States2.S3F);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States2, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withJoin()\n\t\t\t\t.source(States2.S2F)\n\t\t\t\t.source(States2.S3F)\n\t\t\t\t.target(States2.S4)\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source(States2.S4)\n\t\t\t\t.target(States2.S5)\n\t\t\t\t.guardExpression("!extendedState.variables.isEmpty()")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source(States2.S4)\n\t\t\t\t.target(States2.SF)\n\t\t\t\t.guardExpression("extendedState.variables.isEmpty()");\n\t}\n}\n')])])]),a("h4",{attrs:{id:"出入点状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#出入点状态"}},[t._v("#")]),t._v(" 出入点状态")]),t._v(" "),a("p",[t._v("你可以使用出入点和进入点来执行更多的控制出入点和进入点。下面的示例使用"),a("code",[t._v("withEntry")]),t._v(""),a("code",[t._v("withExit")]),t._v("方法来定义入口点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class Config21 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t.withStates()\n\t\t\t.initial("S1")\n\t\t\t.state("S2")\n\t\t\t.state("S3")\n\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent("S2")\n\t\t\t\t.initial("S21")\n\t\t\t\t.entry("S2ENTRY")\n\t\t\t\t.exit("S2EXIT")\n\t\t\t\t.state("S22");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t.withExternal()\n\t\t\t.source("S1").target("S2")\n\t\t\t.event("E1")\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source("S1").target("S2ENTRY")\n\t\t\t.event("ENTRY")\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source("S22").target("S2EXIT")\n\t\t\t.event("EXIT")\n\t\t\t.and()\n\t\t.withEntry()\n\t\t\t.source("S2ENTRY").target("S22")\n\t\t\t.and()\n\t\t.withExit()\n\t\t\t.source("S2EXIT").target("S3");\n\t}\n}\n')])])]),a("p",[t._v("如前面所示,你需要将特定状态标记为"),a("code",[t._v("exit")]),t._v(""),a("code",[t._v("entry")]),t._v("状态。然后创建一个正常的转换到这些状态,并指定"),a("code",[t._v("withExit()")]),t._v(""),a("code",[t._v("withEntry()")]),t._v(",这些状态分别在其中退出和进入。")]),t._v(" "),a("h3",{attrs:{id:"配置公共设置"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置公共设置"}},[t._v("#")]),t._v(" 配置公共设置")]),t._v(" "),a("p",[t._v("可以使用"),a("code",[t._v("ConfigurationConfigurer")]),t._v("设置公共状态机配置的一部分。有了它,你可以为状态机设置"),a("code",[t._v("BeanFactory")]),t._v("和自动启动标志。它还允许你注册"),a("code",[t._v("StateMachineListener")]),t._v("实例,配置转换冲突策略和区域执行策略。下面的示例展示了如何使用"),a("code",[t._v("ConfigurationConfigurer")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config17\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<States, Events> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withConfiguration()\n\t\t\t\t.autoStartup(true)\n\t\t\t\t.machineId("myMachineId")\n\t\t\t\t.beanFactory(new StaticListableBeanFactory())\n\t\t\t\t.listener(new StateMachineListenerAdapter<States, Events>())\n\t\t\t\t.transitionConflictPolicy(TransitionConflictPolicy.CHILD)\n\t\t\t\t.regionExecutionPolicy(RegionExecutionPolicy.PARALLEL);\n\t}\n}\n')])])]),a("p",[t._v("默认情况下,状态机"),a("code",[t._v("autoStartup")]),t._v("标志是禁用的,因为所有处理子状态的实例都由状态机本身控制,不能自动启动。另外,机器是否应该自动启动,由用户自己决定要安全得多。此标志仅控制顶级状态机的自动启动。")]),t._v(" "),a("p",[t._v("在配置类中设置"),a("code",[t._v("machineId")]),t._v("只是为了方便你想要或需要在配置类中设置"),a("code",[t._v("machineId")]),t._v("")]),t._v(" "),a("p",[t._v("注册"),a("code",[t._v("StateMachineListener")]),t._v("实例在一定程度上也是为了方便,但如果你希望在状态机生命周期期间捕获回调,例如获得状态机的启动和停止事件的通知,则需要注册。请注意,如果启用"),a("code",[t._v("autoStartup")]),t._v(",则无法侦听状态机的启动事件,除非你在配置阶段注册了侦听器。")]),t._v(" "),a("p",[t._v("当可以选择多个转换路径时,可以使用"),a("code",[t._v("transitionConflictPolicy")]),t._v("。一个常见的用例是,当机器包含从子状态和父状态引出的匿名转换,并且你想要定义一个策略来选择其中一个。这是机器实例中的全局设置,默认设置为"),a("code",[t._v("CHILD")]),t._v("")]),t._v(" "),a("p",[t._v("你可以使用"),a("code",[t._v("withDistributed()")]),t._v("来配置"),a("code",[t._v("DistributedStateMachine")]),t._v("。它允许你设置"),a("code",[t._v("StateMachineEnsemble")]),t._v(",它(如果存在的话)自动用"),a("code",[t._v("DistributedStateMachine")]),t._v("包装任何创建的"),a("code",[t._v("StateMachine")]),t._v(",并启用分布式模式。下面的示例展示了如何使用它:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config18\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<States, Events> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withDistributed()\n\t\t\t\t.ensemble(stateMachineEnsemble());\n\t}\n\n\t@Bean\n\tpublic StateMachineEnsemble<States, Events> stateMachineEnsemble()\n\t\t\tthrows Exception {\n\t\t// naturally not null but should return ensemble instance\n\t\treturn null;\n\t}\n}\n")])])]),a("p",[t._v("有关分布状态的更多信息,请参见"),a("a",{attrs:{href:"#sm-distributed"}},[t._v("使用分布状态")]),t._v("")]),t._v(" "),a("p",[a("code",[t._v("StateMachineModelVerifier")]),t._v("接口在内部用于对状态机的结构进行一些明智的检查。它的目的是尽早快速失败,而不是让常见的配置错误进入状态机。默认情况下,将自动启用验证器,并使用"),a("code",[t._v("DefaultStateMachineModelVerifier")]),t._v("实现。")]),t._v(" "),a("p",[t._v("使用"),a("code",[t._v("withVerifier()")]),t._v(",如果需要,可以禁用验证器或设置自定义验证器。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config19\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<States, Events> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withVerifier()\n\t\t\t\t.enabled(true)\n\t\t\t\t.verifier(verifier());\n\t}\n\n\t@Bean\n\tpublic StateMachineModelVerifier<States, Events> verifier() {\n\t\treturn new StateMachineModelVerifier<States, Events>() {\n\n\t\t\t@Override\n\t\t\tpublic void verify(StateMachineModel<States, Events> model) {\n\t\t\t\t// throw exception indicating malformed model\n\t\t\t}\n\t\t};\n\t}\n}\n")])])]),a("p",[t._v("有关配置模型的更多信息,请参见"),a("a",{attrs:{href:"#devdocs-configmodel"}},[t._v("statemachine 配置模型")]),t._v("")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[a("code",[t._v("withSecurity")]),t._v(""),a("code",[t._v("withMonitoring")]),t._v(""),a("code",[t._v("withPersistence")]),t._v("配置方法"),a("br"),t._v("分别在"),a("a",{attrs:{href:"#sm-security"}},[t._v("状态机安全")]),t._v(""),a("a",{attrs:{href:"#sm-monitoring"}},[t._v("监视状态机")]),t._v("和[using"),a("code",[t._v("StateMachineRuntimePersister")]),t._v("](#sm-persistue-statemachinerunitemepersister)中有记载。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"配置模型"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置模型"}},[t._v("#")]),t._v(" 配置模型")]),t._v(" "),a("p",[a("code",[t._v("StateMachineModelFactory")]),t._v("是一个钩子,它允许你在不使用手动配置的情况下配置一个 Statemachine 模型。本质上,它是一种集成到配置模型中的第三方集成。你可以使用"),a("code",[t._v("StateMachineModelConfigurer")]),t._v(""),a("code",[t._v("StateMachineModelFactory")]),t._v("连接到配置模型中。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic static class Config1 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineModelConfigurer<String, String> model) throws Exception {\n\t\tmodel\n\t\t\t.withModel()\n\t\t\t\t.factory(modelFactory());\n\t}\n\n\t@Bean\n\tpublic StateMachineModelFactory<String, String> modelFactory() {\n\t\treturn new CustomStateMachineModelFactory();\n\t}\n}\n")])])]),a("p",[t._v("Follwoing 示例使用"),a("code",[t._v("CustomStateMachineModelFactory")]),t._v("来定义两个状态("),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v(")和这些状态之间的事件("),a("code",[t._v("E1")]),t._v("):")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('public static class CustomStateMachineModelFactory implements StateMachineModelFactory<String, String> {\n\n\t@Override\n\tpublic StateMachineModel<String, String> build() {\n\t\tConfigurationData<String, String> configurationData = new ConfigurationData<>();\n\t\tCollection<StateData<String, String>> stateData = new ArrayList<>();\n\t\tstateData.add(new StateData<String, String>("S1", true));\n\t\tstateData.add(new StateData<String, String>("S2"));\n\t\tStatesData<String, String> statesData = new StatesData<>(stateData);\n\t\tCollection<TransitionData<String, String>> transitionData = new ArrayList<>();\n\t\ttransitionData.add(new TransitionData<String, String>("S1", "S2", "E1"));\n\t\tTransitionsData<String, String> transitionsData = new TransitionsData<>(transitionData);\n\t\tStateMachineModel<String, String> stateMachineModel = new DefaultStateMachineModel<String, String>(configurationData,\n\t\t\t\tstatesData, transitionsData);\n\t\treturn stateMachineModel;\n\t}\n\n\t@Override\n\tpublic StateMachineModel<String, String> build(String machineId) {\n\t\treturn build();\n\t}\n}\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("定义一个定制模型通常不是人们想要的,"),a("br"),t._v("尽管这是可能的。然而,它是允许"),a("br"),t._v("外部访问此配置模型的核心概念。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("你可以在"),a("a",{attrs:{href:"#sm-papyrus"}},[t._v("Eclipse 建模支持")]),t._v("中找到使用此模型工厂集成的示例。你可以在"),a("a",{attrs:{href:"#devdocs"}},[t._v("开发人员文档")]),t._v("中找到有关自定义模型集成的更多通用信息。")]),t._v(" "),a("h3",{attrs:{id:"要记住的事情"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#要记住的事情"}},[t._v("#")]),t._v(" 要记住的事情")]),t._v(" "),a("p",[t._v("在从配置中定义操作、保护或任何其他引用时,需要记住 Spring Framework 是如何与 bean 一起工作的。在下一个示例中,我们定义了一个正常配置,其状态"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v("之间有四个转换。所有转换都由"),a("code",[t._v("guard1")]),t._v(""),a("code",[t._v("guard2")]),t._v("保护。你必须确保"),a("code",[t._v("guard1")]),t._v("是作为真实 Bean 创建的,因为它是用"),a("code",[t._v("@Bean")]),t._v("注释的,而"),a("code",[t._v("guard2")]),t._v("不是。")]),t._v(" "),a("p",[t._v("这意味着事件"),a("code",[t._v("E3")]),t._v("将得到"),a("code",[t._v("guard2")]),t._v("条件为"),a("code",[t._v("TRUE")]),t._v(",而"),a("code",[t._v("E4")]),t._v("将得到"),a("code",[t._v("guard2")]),t._v("条件为"),a("code",[t._v("FALSE")]),t._v(",因为这些条件来自对那些函数的普通方法调用。")]),t._v(" "),a("p",[t._v("然而,因为"),a("code",[t._v("guard1")]),t._v("被定义为"),a("code",[t._v("@Bean")]),t._v(",所以它由 Spring 框架代理。因此,对其方法的额外调用只会导致该实例的一个实例化。Event"),a("code",[t._v("E1")]),t._v("将首先获得带有条件"),a("code",[t._v("TRUE")]),t._v("的代理实例,而 Event"),a("code",[t._v("E2")]),t._v("将获得带有"),a("code",[t._v("TRUE")]),t._v("条件的相同实例,当方法调用被定义为"),a("code",[t._v("FALSE")]),t._v("时。这不是 Spring 特定于状态机的行为。相反,它是 Spring Framework 如何与 bean 一起工作的。下面的示例展示了这种安排的工作原理:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config1\n\t\textends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.state("S2");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("S1").target("S2").event("E1").guard(guard1(true))\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("S1").target("S2").event("E2").guard(guard1(false))\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("S1").target("S2").event("E3").guard(guard2(true))\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("S1").target("S2").event("E4").guard(guard2(false));\n\t}\n\n\t@Bean\n\tpublic Guard<String, String> guard1(final boolean value) {\n\t\treturn new Guard<String, String>() {\n\t\t\t@Override\n\t\t\tpublic boolean evaluate(StateContext<String, String> context) {\n\t\t\t\treturn value;\n\t\t\t}\n\t\t};\n\t}\n\n\tpublic Guard<String, String> guard2(final boolean value) {\n\t\treturn new Guard<String, String>() {\n\t\t\t@Override\n\t\t\tpublic boolean evaluate(StateContext<String, String> context) {\n\t\t\t\treturn value;\n\t\t\t}\n\t\t};\n\t}\n}\n')])])]),a("h2",{attrs:{id:"状态机-id"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态机-id"}},[t._v("#")]),t._v(" 状态机 ID")]),t._v(" "),a("p",[t._v("在方法中,各种类和接口使用"),a("code",[t._v("machineId")]),t._v("作为变量或参数。本节将更仔细地了解"),a("code",[t._v("machineId")]),t._v("与正常的机器操作和实例化之间的关系。")]),t._v(" "),a("p",[t._v("在运行时期间,"),a("code",[t._v("machineId")]),t._v("实际上没有任何大的操作作用,除了区分机器之间的区别——例如,在跟踪日志或进行更深入的调试时。如果没有一种简单的方法来识别这些实例,那么拥有大量不同的机器实例很快就会让开发人员迷失在翻译过程中。因此,我们添加了设置"),a("code",[t._v("machineId")]),t._v("的选项。")]),t._v(" "),a("h3",{attrs:{id:"使用-enablestatemachine"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用-enablestatemachine"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("@EnableStateMachine")])]),t._v(" "),a("p",[t._v("在 Java 配置中将"),a("code",[t._v("machineId")]),t._v("设置为"),a("code",[t._v("mymachine")]),t._v(",然后公开日志的该值。同样的"),a("code",[t._v("machineId")]),t._v("也可以从"),a("code",[t._v("StateMachine.getId()")]),t._v("方法获得。下面的示例使用"),a("code",[t._v("machineId")]),t._v("方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Override\npublic void configure(StateMachineConfigurationConfigurer<String, String> config)\n\t\tthrows Exception {\n\tconfig\n\t\t.withConfiguration()\n\t\t\t.machineId("mymachine");\n}\n')])])]),a("p",[t._v("下面的日志输出示例显示了"),a("code",[t._v("mymachine")]),t._v("ID:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("11:23:54,509  INFO main support.LifecycleObjectSupport [main] -\nstarted S2 S1  / S1 / uuid=8fe53d34-8c85-49fd-a6ba-773da15fcaf1 / id=mymachine\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("手动构建器(参见"),a("a",{attrs:{href:"#state-machine-via-builder"}},[t._v("通过构建器的状态机")]),t._v(")使用相同的配置"),a("br"),t._v("接口,这意味着行为是等效的。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"使用-enablestatemachinefactory"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用-enablestatemachinefactory"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("@EnableStateMachineFactory")])]),t._v(" "),a("p",[t._v("如果使用"),a("code",[t._v("StateMachineFactory")]),t._v("并使用该 ID 请求一台新机器,则可以看到相同的"),a("code",[t._v("machineId")]),t._v("正在被配置,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('StateMachineFactory<String, String> factory = context.getBean(StateMachineFactory.class);\nStateMachine<String, String> machine = factory.getStateMachine("mymachine");\n')])])]),a("h3",{attrs:{id:"使用statemachinemodelfactory"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statemachinemodelfactory"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateMachineModelFactory")])]),t._v(" "),a("p",[t._v("在幕后,所有机器配置首先被转换为"),a("code",[t._v("StateMachineModel")]),t._v(",这样"),a("code",[t._v("StateMachineFactory")]),t._v("就不需要知道配置的起源,因为机器可以从 Java 配置、UML 或存储库构建。如果你想发疯,也可以使用自定义"),a("code",[t._v("StateMachineModel")]),t._v(",这是定义配置的最低级别。")]),t._v(" "),a("p",[t._v("所有这些都与"),a("code",[t._v("machineId")]),t._v("有什么关系?"),a("code",[t._v("StateMachineModelFactory")]),t._v("还具有具有以下签名的方法:"),a("code",[t._v("StateMachineModel<S, E> build(String machineId)")]),t._v(""),a("code",[t._v("StateMachineModelFactory")]),t._v("实现可以选择使用。")]),t._v(" "),a("p",[a("code",[t._v("RepositoryStateMachineModelFactory")]),t._v("(参见"),a("a",{attrs:{href:"#sm-repository"}},[t._v("存储库支持")]),t._v(")使用"),a("code",[t._v("machineId")]),t._v("来支持持久存储中的不同配置\n通过 Spring 数据存储库接口进行存储。例如,"),a("code",[t._v("StateRepository")]),t._v(""),a("code",[t._v("TransitionRepository")]),t._v("都有一个方法("),a("code",[t._v("list<T> FindbyMachineID")]),t._v("), to build different states and transitions by a "),a("code",[t._v("MachineID")]),t._v(". With"),a("code",[t._v("RepositorystateMachineModelFactory")]),t._v(", if "),a("code",[t._v("MachineID")]),t._v(" 被用作空或空,它默认为存储库配置(在备份-持久性模型中),而没有已知的机器 ID。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("目前,"),a("code",[t._v("UmlStateMachineModelFactory")]),t._v("不区分"),a("br"),t._v("不同的机器 ID,因为 UML 源总是来自相同的"),a("br"),t._v("文件。这种情况可能会在未来的版本中发生变化。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"状态机工厂"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态机工厂"}},[t._v("#")]),t._v(" 状态机工厂")]),t._v(" "),a("p",[t._v("在某些情况下,需要动态地创建状态机,而不是在编译时定义静态配置。例如,如果有一些定制组件使用它们自己的状态机,并且这些组件是动态创建的,那么就不可能在应用程序启动期间构建静态状态机。在内部,状态机总是通过工厂接口构建的。这就给了你一个以编程方式使用此功能的选项。状态机工厂的配置与本文档中各种示例中所示的配置完全相同,其中状态机配置是硬编码的。")]),t._v(" "),a("h3",{attrs:{id:"通过适配器出厂"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#通过适配器出厂"}},[t._v("#")]),t._v(" 通过适配器出厂")]),t._v(" "),a("p",[t._v("实际上,通过使用"),a("code",[t._v("@EnableStateMachine")]),t._v("创建状态机是通过工厂工作的,所以"),a("code",[t._v("@EnableStateMachineFactory")]),t._v("只是通过其接口公开了该工厂。下面的示例使用"),a("code",[t._v("@EnableStateMachineFactory")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachineFactory\npublic class Config6\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.S1)\n\t\t\t\t.end(States.SF)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n}\n")])])]),a("p",[t._v("既然你已经使用"),a("code",[t._v("@EnableStateMachineFactory")]),t._v("创建了一个工厂,而不是一个状态机 Bean,那么你可以插入它并使用它来请求新的状态机。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class Bean3 {\n\n\t@Autowired\n\tStateMachineFactory<States, Events> factory;\n\n\tvoid method() {\n\t\tStateMachine<States,Events> stateMachine = factory.getStateMachine();\n\t\tstateMachine.startReactively().subscribe();\n\t}\n}\n")])])]),a("h4",{attrs:{id:"适配器出厂限制"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#适配器出厂限制"}},[t._v("#")]),t._v(" 适配器出厂限制")]),t._v(" "),a("p",[t._v("Factory 当前的限制是,它与状态机关联的所有操作和保护都共享同一个实例。这意味着,根据你的操作和保护,你需要专门处理由不同状态机调用相同 Bean 的情况。这一限制将在未来的版本中得到解决。")]),t._v(" "),a("h3",{attrs:{id:"通过构建器的状态机"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#通过构建器的状态机"}},[t._v("#")]),t._v(" 通过构建器的状态机")]),t._v(" "),a("p",[t._v("使用适配器(如上面所示)有其通过 Spring "),a("code",[t._v("@Configuration")]),t._v("类和应用程序上下文工作的要求所施加的限制。虽然这是一个配置状态机的非常清晰的模型,但它限制了编译时的配置,而这并不总是用户想要做的。如果需要构建更多的动态状态机,那么可以使用一个简单的构建器模式来构建类似的实例。通过使用字符串作为状态和事件,你可以使用此 Builder 模式在 Spring 应用程序上下文之外构建完全动态的状态机。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('StateMachine<String, String> buildMachine1() throws Exception {\n\tBuilder<String, String> builder = StateMachineBuilder.builder();\n\tbuilder.configureStates()\n\t\t.withStates()\n\t\t\t.initial("S1")\n\t\t\t.end("SF")\n\t\t\t.states(new HashSet<String>(Arrays.asList("S1","S2","S3","S4")));\n\treturn builder.build();\n}\n')])])]),a("p",[t._v("构建器在幕后使用与"),a("code",[t._v("@Configuration")]),t._v("模型用于适配器类相同的配置接口。同样的模型也适用于通过构建器的方法来配置转换、状态和公共配置。这意味着,无论使用普通的"),a("code",[t._v("EnumStateMachineConfigurerAdapter")]),t._v("还是"),a("code",[t._v("StateMachineConfigurerAdapter")]),t._v(",都可以通过构建器动态地使用。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("目前,"),a("code",[t._v("builder.configureStates()")]),t._v(""),a("code",[t._v("builder.configureTransitions()")]),t._v(""),a("br"),t._v(""),a("code",[t._v("builder.configureConfiguration()")]),t._v("接口方法不能被"),a("br"),t._v("链接在一起,这意味着生成器方法需要单独调用。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("下面的示例使用构建器设置了许多选项:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("StateMachine<String, String> buildMachine2() throws Exception {\n\tBuilder<String, String> builder = StateMachineBuilder.builder();\n\tbuilder.configureConfiguration()\n\t\t.withConfiguration()\n\t\t\t.autoStartup(false)\n\t\t\t.beanFactory(null)\n\t\t\t.listener(null);\n\treturn builder.build();\n}\n")])])]),a("p",[t._v("你需要了解何时需要将公共配置与从构建器实例化的机器一起使用。你可以使用从"),a("code",[t._v("withConfiguration()")]),t._v("返回的配置器来设置"),a("code",[t._v("autoStart")]),t._v(""),a("code",[t._v("BeanFactory")]),t._v("。你也可以使用一个来注册"),a("code",[t._v("StateMachineListener")]),t._v("。如果通过使用"),a("code",[t._v("@Bean")]),t._v("将从构建器返回的"),a("code",[t._v("StateMachine")]),t._v("实例注册为 Bean,则"),a("code",[t._v("BeanFactory")]),t._v("将自动附加。如果在 Spring 应用程序上下文之外使用实例,则必须使用这些方法来设置所需的设施。")]),t._v(" "),a("h2",{attrs:{id:"使用延迟事件"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用延迟事件"}},[t._v("#")]),t._v(" 使用延迟事件")]),t._v(" "),a("p",[t._v("当发送事件时,它可能会触发"),a("code",[t._v("EventTrigger")]),t._v(",如果状态机处于成功计算触发器的状态,那么这可能会导致转换发生。通常情况下,这可能会导致一种情况,即一个事件不被接受,并被放弃。但是,你可能希望将此事件推迟到状态机进入另一种状态。在这种情况下,你可以接受该事件。换句话说,一项活动是在一个不方便的时间到来的。")]),t._v(" "),a("p",[t._v("Spring Statemachine 提供了一种机制,用于将事件延迟到以后的处理中。每个州都可以有一个延迟事件的列表。如果当前状态的“延迟事件”列表中的事件发生,则该事件将被保存(延迟)以备将来处理,直到输入一个未在其“延迟事件”列表中列出该事件的状态。当输入这样的状态时,状态机会自动召回所有已保存的不再延迟的事件,然后消耗或丢弃这些事件。超状态有可能在由子状态延迟的事件上定义转换。遵循相同的层次状态机概念,子态优先于超态,事件被推迟,并且超态的转换不运行。对于正交区域,其中一个正交区域延迟一个事件,而另一个接受该事件,该接受具有优先权,并且该事件被消耗而不是延迟。")]),t._v(" "),a("p",[t._v("事件延迟最明显的用例是,当一个事件导致转换到特定状态,然后状态机返回到其原始状态时,第二个事件将导致相同的转换。下面的示例展示了这种情况:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class Config5 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("READY")\n\t\t\t\t.state("DEPLOYPREPARE", "DEPLOY")\n\t\t\t\t.state("DEPLOYEXECUTE", "DEPLOY");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("READY").target("DEPLOYPREPARE")\n\t\t\t\t.event("DEPLOY")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("DEPLOYPREPARE").target("DEPLOYEXECUTE")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("DEPLOYEXECUTE").target("READY");\n\t}\n}\n')])])]),a("p",[t._v("在前面的示例中,状态机的状态为"),a("code",[t._v("READY")]),t._v(",这表示机器已准备好处理将其带到"),a("code",[t._v("DEPLOY")]),t._v("状态的事件,而实际部署将在该状态中进行。运行部署操作后,机器将返回到"),a("code",[t._v("READY")]),t._v("状态。如果机器使用同步执行器,以"),a("code",[t._v("READY")]),t._v("状态发送多个事件不会造成任何麻烦,因为事件发送会在事件调用之间阻塞。但是,如果执行器使用线程,其他事件可能会丢失,因为机器不再处于可以处理事件的状态。因此,推迟这些事件中的一些可以让机器保留它们。下面的示例展示了如何配置这样的安排:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class Config6 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("READY")\n\t\t\t\t.state("DEPLOY", "DEPLOY")\n\t\t\t\t.state("DONE")\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent("DEPLOY")\n\t\t\t\t\t.initial("DEPLOYPREPARE")\n\t\t\t\t\t.state("DEPLOYPREPARE", "DONE")\n\t\t\t\t\t.state("DEPLOYEXECUTE");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("READY").target("DEPLOY")\n\t\t\t\t.event("DEPLOY")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("DEPLOYPREPARE").target("DEPLOYEXECUTE")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("DEPLOYEXECUTE").target("READY")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("READY").target("DONE")\n\t\t\t\t.event("DONE")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("DEPLOY").target("DONE")\n\t\t\t\t.event("DONE");\n\t}\n}\n')])])]),a("p",[t._v("在前面的示例中,状态机使用嵌套状态而不是平坦状态模型,因此"),a("code",[t._v("DEPLOY")]),t._v("事件可以在子状态中直接延迟。它还显示了在子状态中推迟"),a("code",[t._v("DONE")]),t._v("事件的概念,如果发送"),a("code",[t._v("DONE")]),t._v("事件时状态机恰好处于"),a("code",[t._v("DEPLOYPREPARE")]),t._v("状态,则该状态将覆盖"),a("code",[t._v("DEPLOY")]),t._v(""),a("code",[t._v("DONE")]),t._v("状态之间的匿名转换。在"),a("code",[t._v("DEPLOYEXECUTE")]),t._v("状态下,当"),a("code",[t._v("DONE")]),t._v("事件没有延迟时,此事件将在超级状态下处理。")]),t._v(" "),a("h2",{attrs:{id:"使用作用域"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用作用域"}},[t._v("#")]),t._v(" 使用作用域")]),t._v(" "),a("p",[t._v("状态机中对作用域的支持非常有限,但是你可以通过使用普通 Spring "),a("code",[t._v("@Scope")]),t._v("注释来启用"),a("code",[t._v("session")]),t._v("作用域,方法有以下两种:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("如果状态机是通过使用构建器手动构建的,并以"),a("code",[t._v("@Bean")]),t._v("的形式返回到上下文中。")])]),t._v(" "),a("li",[a("p",[t._v("通过配置适配器。")])])]),t._v(" "),a("p",[t._v("这两个参数都需要"),a("code",[t._v("@Scope")]),t._v("才能存在,将"),a("code",[t._v("scopeName")]),t._v("设置为"),a("code",[t._v("session")]),t._v(",将"),a("code",[t._v("proxyMode")]),t._v("设置为"),a("code",[t._v("ScopedProxyMode.TARGET_CLASS")]),t._v("。以下示例展示了这两种用例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\npublic class Config3 {\n\n\t@Bean\n\t@Scope(scopeName="session", proxyMode=ScopedProxyMode.TARGET_CLASS)\n\tStateMachine<String, String> stateMachine() throws Exception {\n\t\tBuilder<String, String> builder = StateMachineBuilder.builder();\n\t\tbuilder.configureConfiguration()\n\t\t\t.withConfiguration()\n\t\t\t\t.autoStartup(true);\n\t\tbuilder.configureStates()\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.state("S2");\n\t\tbuilder.configureTransitions()\n\t\t\t.withExternal()\n\t\t\t\t.source("S1")\n\t\t\t\t.target("S2")\n\t\t\t\t.event("E1");\n\t\tStateMachine<String, String> stateMachine = builder.build();\n\t\treturn stateMachine;\n\t}\n\n}\n')])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\n@Scope(scopeName="session", proxyMode=ScopedProxyMode.TARGET_CLASS)\npublic static class Config4 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<String, String> config) throws Exception {\n\t\tconfig\n\t\t\t.withConfiguration()\n\t\t\t\t.autoStartup(true);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states) throws Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.state("S2");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions) throws Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("S1")\n\t\t\t\t.target("S2")\n\t\t\t\t.event("E1");\n\t}\n\n}\n')])])]),a("p",[t._v("提示:有关如何使用会话范围定义,请参见"),a("a",{attrs:{href:"#statemachine-examples-scope"}},[t._v("Scope")]),t._v("")]),t._v(" "),a("p",[t._v("一旦将状态机的作用域设为"),a("code",[t._v("session")]),t._v(",则将其自动连线到"),a("code",[t._v("@Controller")]),t._v("中,每会话提供一个新的状态机实例。当"),a("code",[t._v("HttpSession")]),t._v("无效时,每个状态机都会被销毁。下面的示例展示了如何在控制器中使用状态机:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Controller\npublic class StateMachineController {\n\n\t@Autowired\n\tStateMachine<String, String> stateMachine;\n\n\t@RequestMapping(path="/state", method=RequestMethod.POST)\n\tpublic HttpEntity<Void> setState(@RequestParam("event") String event) {\n\t\tstateMachine\n\t\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t\t.withPayload(event).build()))\n\t\t\t.subscribe();\n\t\treturn new ResponseEntity<Void>(HttpStatus.ACCEPTED);\n\t}\n\n\t@RequestMapping(path="/state", method=RequestMethod.GET)\n\t@ResponseBody\n\tpublic String getState() {\n\t\treturn stateMachine.getState().getId();\n\t}\n}\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v(""),a("code",[t._v("session")]),t._v("范围内使用状态机需要仔细的计划,"),a("br"),t._v("主要是因为它是一个相对较重的组件。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("Spring StateMachine POM 与 Spring MVC类没有依赖关系,你将需要使用会话范围来处理这些类。但是,如果你正在使用 Web 应用程序,那么你已经直接从 Spring MVC 或 Spring Boot 中提取了这些依赖项。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"使用动作"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用动作"}},[t._v("#")]),t._v(" 使用动作")]),t._v(" "),a("p",[t._v("动作是你可以用来与状态机交互和协作的最有用的组件之一。你可以在状态机及其状态生命周期的不同位置运行操作——例如,进入或退出状态或在转换期间。下面的示例展示了如何在状态机中使用操作:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\tthrows Exception {\n\tstates\n\t\t.withStates()\n\t\t\t.initial(States.SI)\n\t\t\t.state(States.S1, action1(), action2())\n\t\t\t.state(States.S2, action1(), action2())\n\t\t\t.state(States.S3, action1(), action3());\n}\n")])])]),a("p",[t._v("在前面的示例中,"),a("code",[t._v("action1")]),t._v(""),a("code",[t._v("action2")]),t._v("bean 分别附加到"),a("code",[t._v("entry")]),t._v(""),a("code",[t._v("exit")]),t._v("状态。下面的示例定义了这些操作(以及"),a("code",[t._v("action3")]),t._v("):")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean\npublic Action<States, Events> action1() {\n\treturn new Action<States, Events>() {\n\n\t\t@Override\n\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t}\n\t};\n}\n\n@Bean\npublic BaseAction action2() {\n\treturn new BaseAction();\n}\n\n@Bean\npublic SpelAction action3() {\n\tExpressionParser parser = new SpelExpressionParser();\n\treturn new SpelAction(\n\t\t\tparser.parseExpression(\n\t\t\t\t\t"stateMachine.sendEvent(T(org.springframework.statemachine.docs.Events).E1)"));\n}\n\npublic class BaseAction implements Action<States, Events> {\n\n\t@Override\n\tpublic void execute(StateContext<States, Events> context) {\n\t}\n}\n\npublic class SpelAction extends SpelExpressionAction<States, Events> {\n\n\tpublic SpelAction(Expression expression) {\n\t\tsuper(expression);\n\t}\n}\n')])])]),a("p",[t._v("你可以直接将"),a("code",[t._v("Action")]),t._v("实现为匿名函数,或者创建自己的实现,并将适当的实现定义为 Bean。")]),t._v(" "),a("p",[t._v("在前面的示例中,"),a("code",[t._v("action3")]),t._v("使用 SPEL 表达式将"),a("code",[t._v("Events.E1")]),t._v("事件发送到状态机。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[a("code",[t._v("StateContext")]),t._v("在[使用"),a("code",[t._v("StateContext")]),t._v("]中进行了描述。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"带动作的-spel-表达式"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#带动作的-spel-表达式"}},[t._v("#")]),t._v(" 带动作的 spel 表达式")]),t._v(" "),a("p",[t._v("你也可以使用 SPEL 表达式作为完整"),a("code",[t._v("Action")]),t._v("实现的替换。")]),t._v(" "),a("h3",{attrs:{id:"反应动作"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#反应动作"}},[t._v("#")]),t._v(" 反应动作")]),t._v(" "),a("p",[t._v("正常的"),a("code",[t._v("Action")]),t._v("接口是一种简单的函数方法,它取"),a("code",[t._v("StateContext")]),t._v("并返回"),a("em",[t._v("无效")]),t._v("。在你阻塞方法本身之前,这里没有任何阻塞,这是一个问题,因为 Framework 无法知道它内部到底发生了什么。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public interface Action<S, E> {\n\tvoid execute(StateContext<S, E> context);\n}\n")])])]),a("p",[t._v("为了克服这个问题,我们在内部更改了"),a("code",[t._v("Action")]),t._v("处理,以处理普通 Java 的"),a("code",[t._v("Function")]),t._v(",并返回"),a("code",[t._v("StateContext")]),t._v("。通过这种方式,我们可以调用 Action,并以一种反应式的方式完全执行 Action,仅在订阅时执行,并以一种非阻塞的方式等待完成。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public interface ReactiveAction<S, E> extends Function<StateContext<S, E>, Mono<Void>> {\n}\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("在内部,旧的"),a("code",[t._v("Action")]),t._v("接口用一个可运行的反应器 Mono 包装,因为它"),a("br"),t._v("共享相同的返回类型。我们无法控制你用那种方法做什么!")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"使用保护"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用保护"}},[t._v("#")]),t._v(" 使用保护")]),t._v(" "),a("p",[t._v(""),a("a",{attrs:{href:"#statemachine-config-thingstoremember"}},[t._v("要记住的事情")]),t._v("中所示,"),a("code",[t._v("guard1")]),t._v(""),a("code",[t._v("guard2")]),t._v("bean 分别附加到进入和退出状态。下面的示例还对事件使用了保护:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\tthrows Exception {\n\ttransitions\n\t\t.withExternal()\n\t\t\t.source(States.SI).target(States.S1)\n\t\t\t.event(Events.E1)\n\t\t\t.guard(guard1())\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S1).target(States.S2)\n\t\t\t.event(Events.E1)\n\t\t\t.guard(guard2())\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S2).target(States.S3)\n\t\t\t.event(Events.E2)\n\t\t\t.guardExpression(\"extendedState.variables.get('myvar')\");\n}\n")])])]),a("p",[t._v("你可以直接将"),a("code",[t._v("Guard")]),t._v("实现为匿名函数,或者创建自己的实现,并将适当的实现定义为 Bean。在前面的示例中,"),a("code",[t._v("guardExpression")]),t._v("检查名为"),a("code",[t._v("myvar")]),t._v("的扩展状态变量是否计算为"),a("code",[t._v("TRUE")]),t._v("。下面的示例实现了一些示例保护:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Bean\npublic Guard<States, Events> guard1() {\n\treturn new Guard<States, Events>() {\n\n\t\t@Override\n\t\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\t\treturn true;\n\t\t}\n\t};\n}\n\n@Bean\npublic BaseGuard guard2() {\n\treturn new BaseGuard();\n}\n\npublic class BaseGuard implements Guard<States, Events> {\n\n\t@Override\n\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\treturn false;\n\t}\n}\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[a("code",[t._v("StateContext")]),t._v("在[使用"),a("code",[t._v("StateContext")]),t._v("]节中进行了描述。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"带守卫的-spel-表达式"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#带守卫的-spel-表达式"}},[t._v("#")]),t._v(" 带守卫的 spel 表达式")]),t._v(" "),a("p",[t._v("你也可以使用 SPEL 表达式作为完全保护实现的替代。唯一的要求是表达式需要返回一个"),a("code",[t._v("Boolean")]),t._v("值来满足"),a("code",[t._v("Guard")]),t._v("实现。这可以用一个"),a("code",[t._v("guardExpression()")]),t._v("函数来演示,该函数将一个表达式作为参数。")]),t._v(" "),a("h3",{attrs:{id:"反应防护"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#反应防护"}},[t._v("#")]),t._v(" 反应防护")]),t._v(" "),a("p",[t._v("正常的"),a("code",[t._v("Guard")]),t._v("接口是一种简单的函数方法,它取"),a("code",[t._v("StateContext")]),t._v("并返回"),a("em",[t._v("布尔值")]),t._v("。在你阻塞方法本身之前,这里没有任何阻塞,这是一个问题,因为 Framework 无法知道它内部到底发生了什么。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public interface Guard<S, E> {\n\tboolean evaluate(StateContext<S, E> context);\n}\n")])])]),a("p",[t._v("为了克服这个问题,我们在内部更改了"),a("code",[t._v("Guard")]),t._v("处理,以处理普通 Java 的"),a("code",[t._v("Function")]),t._v(",并返回"),a("code",[t._v("StateContext")]),t._v("。通过这种方式,我们可以调用 Guard,并以一种反应式的方式完全评估它,仅当它被订阅时,并且以一种非阻塞的方式等待完成,并具有一个返回值。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public interface ReactiveGuard<S, E> extends Function<StateContext<S, E>, Mono<Boolean>> {\n}\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("内部旧的"),a("code",[t._v("Guard")]),t._v("接口是用反应堆单声道函数包装的。我们没有"),a("br"),t._v("控制你在那个方法中做什么!")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"使用扩展状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用扩展状态"}},[t._v("#")]),t._v(" 使用扩展状态")]),t._v(" "),a("p",[t._v("假设你需要创建一个状态机,该状态机跟踪用户在键盘上按下一个键的次数,然后在按键被按下 1000 次时终止。一个可能但非常幼稚的解决方案是为每 1000 次按键创建一个新的状态。你可能会突然有一个天文数字的状态,这自然是不太实际的。")]),t._v(" "),a("p",[t._v("这就是扩展的状态变量通过不需要添加更多的状态来驱动状态机更改而获得帮助的地方。相反,你可以在转换期间执行一个简单的变量更改。")]),t._v(" "),a("p",[a("code",[t._v("StateMachine")]),t._v("有一个名为"),a("code",[t._v("getExtendedState()")]),t._v("的方法。它返回一个名为"),a("code",[t._v("ExtendedState")]),t._v("的接口,该接口允许访问扩展的状态变量。你可以通过状态机直接访问这些变量,或者在操作或转换的回调期间通过"),a("code",[t._v("StateContext")]),t._v("访问这些变量。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('public Action<String, String> myVariableAction() {\n\treturn new Action<String, String>() {\n\n\t\t@Override\n\t\tpublic void execute(StateContext<String, String> context) {\n\t\t\tcontext.getExtendedState()\n\t\t\t\t.getVariables().put("mykey", "myvalue");\n\t\t}\n\t};\n}\n')])])]),a("p",[t._v("如果需要获得扩展状态变量更改的通知,则有两个选项:使用"),a("code",[t._v("StateMachineListener")]),t._v("或侦听"),a("code",[t._v("extendedStateChanged(key, value)")]),t._v("回调。下面的示例使用"),a("code",[t._v("extendedStateChanged")]),t._v("方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class ExtendedStateVariableListener\n\t\textends StateMachineListenerAdapter<String, String> {\n\n\t@Override\n\tpublic void extendedStateChanged(Object key, Object value) {\n\t\t// do something with changed variable\n\t}\n}\n")])])]),a("p",[t._v("或者,你可以为"),a("code",[t._v("OnExtendedStateChanged")]),t._v("实现 Spring 应用程序上下文侦听器。正如"),a("a",{attrs:{href:"#sm-listeners"}},[t._v("监听状态机事件")]),t._v("中提到的,你还可以侦听所有"),a("code",[t._v("StateMachineEvent")]),t._v("事件。下面的示例使用"),a("code",[t._v("onApplicationEvent")]),t._v("侦听状态更改:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class ExtendedStateVariableEventListener\n\t\timplements ApplicationListener<OnExtendedStateChanged> {\n\n\t@Override\n\tpublic void onApplicationEvent(OnExtendedStateChanged event) {\n\t\t// do something with changed variable\n\t}\n}\n")])])]),a("h2",{attrs:{id:"使用statecontext"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statecontext"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateContext")])]),t._v(" "),a("p",[t._v("["),a("code",[t._v("StateContext")]),t._v("](https://DOCS. Spring.io/ Spring-StateMachine/DOCS/3.0.1/api/org/SpringFramework/StateMachine/StateContext.html)是使用状态机时最重要的对象之一,因为它被传递到各种方法和回调中,以给出状态机的当前状态以及它可能的走向。你可以将其视为当前状态机级的快照,当"),a("code",[t._v("StateContext")]),t._v("被恢复时。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("在 Spring StateMachine1.0.x 中,"),a("code",[t._v("StateContext")]),t._v("的用法相对幼稚,"),a("br"),t._v("它是如何被用来作为简单的“pojo”传递信息的。"),a("br"),t._v("从 Spring StateMachine1.1.x 开始,通过使其成为状态机中的第一类公民,它的作用得到了极大的改进。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("你可以使用"),a("code",[t._v("StateContext")]),t._v("访问以下内容:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("当前的"),a("code",[t._v("Message")]),t._v(""),a("code",[t._v("Event")]),t._v("(或它们的"),a("code",[t._v("MessageHeaders")]),t._v(",如果已知的话)。")])]),t._v(" "),a("li",[a("p",[t._v("状态机的"),a("code",[t._v("Extended State")]),t._v("")])]),t._v(" "),a("li",[a("p",[a("code",[t._v("StateMachine")]),t._v("本身。")])]),t._v(" "),a("li",[a("p",[t._v("可能的状态机错误。")])]),t._v(" "),a("li",[a("p",[t._v("到当前"),a("code",[t._v("Transition")]),t._v(",如果适用的话。")])]),t._v(" "),a("li",[a("p",[t._v("状态机的源状态。")])]),t._v(" "),a("li",[a("p",[t._v("状态机的目标状态。")])]),t._v(" "),a("li",[a("p",[t._v("当前的"),a("code",[t._v("Stage")]),t._v(",如"),a("a",{attrs:{href:"#sm-statecontext-stage"}},[t._v("Stages")]),t._v("中所述。")])])]),t._v(" "),a("p",[a("code",[t._v("StateContext")]),t._v("被传递到各种组件中,例如"),a("code",[t._v("Action")]),t._v(""),a("code",[t._v("Guard")]),t._v("")]),t._v(" "),a("h3",{attrs:{id:"阶段"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#阶段"}},[t._v("#")]),t._v(" 阶段")]),t._v(" "),a("p",[t._v("["),a("code",[t._v("Stage")]),t._v("](https://DOCS. Spring.io/ Spring-stateMachine/DOCS/3.0.1/api/org/springframework/stateMachine/stateContext.stage.html)是一个"),a("code",[t._v("stage")]),t._v("状态机当前正在与用户交互的表示。当前可用的阶段是"),a("code",[t._v("EVENT_NOT_ACCEPTED")]),t._v(""),a("code",[t._v("EXTENDED_STATE_CHANGED")]),t._v(""),a("code",[t._v("STATE_CHANGED")]),t._v(""),a("code",[t._v("STATE_ENTRY")]),t._v(""),a("code",[t._v("STATE_EXIT")]),t._v(""),a("code",[t._v("STATEMACHINE_ERROR")]),t._v(""),a("code",[t._v("STATEMACHINE_START")]),t._v(""),a("code",[t._v("STATEMACHINE_STOP")]),t._v(""),a("code",[t._v("TRANSITION")]),t._v(",和"),a("code",[t._v("TRANSITION_END")]),t._v("。这些状态可能看起来很熟悉,因为它们与你可以与侦听器交互的方式相匹配(如"),a("a",{attrs:{href:"#sm-listeners"}},[t._v("监听状态机事件")]),t._v("中所述)。")]),t._v(" "),a("h2",{attrs:{id:"触发转换"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#触发转换"}},[t._v("#")]),t._v(" 触发转换")]),t._v(" "),a("p",[t._v("通过使用由触发器触发的转换来驱动状态机。当前支持的触发器是"),a("code",[t._v("EventTrigger")]),t._v(""),a("code",[t._v("TimerTrigger")]),t._v("")]),t._v(" "),a("h3",{attrs:{id:"使用eventtrigger"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用eventtrigger"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("EventTrigger")])]),t._v(" "),a("p",[a("code",[t._v("EventTrigger")]),t._v("是最有用的触发器,因为它允许你通过向状态机发送事件来直接与其交互。这些事件也被称为信号。你可以在配置期间将状态与转换关联,从而将触发器添加到转换中。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateMachine<String, String> stateMachine;\n\nvoid signalMachine() {\n\tstateMachine\n\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t.withPayload("E1").build()))\n\t\t.subscribe();\n\n\tMessage<String> message = MessageBuilder\n\t\t\t.withPayload("E2")\n\t\t\t.setHeader("foo", "bar")\n\t\t\t.build();\n\tstateMachine.sendEvent(Mono.just(message)).subscribe();\n}\n')])])]),a("p",[t._v("无论你发送一个事件还是多个事件,结果总是一个结果序列。这是因为在存在多个区域的情况下,结果将从这些区域的多台机器返回。这是用方法"),a("code",[t._v("sendEventCollect")]),t._v("表示的,该方法给出了结果列表。方法本身只是一个收集"),a("code",[t._v("Flux")]),t._v("as 列表的语法糖类。如果只有一个区域,则此列表包含一个结果。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('Message<String> message1 = MessageBuilder\n\t.withPayload("E1")\n\t.build();\n\nMono<List<StateMachineEventResult<String, String>>> results =\n\tstateMachine.sendEventCollect(Mono.just(message1));\n\nresults.subscribe();\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("在订阅了返回的 Flux 之前,什么都不会发生。有关它的更多信息,请参见"),a("a",{attrs:{href:"#sm-triggers-statemachineeventresult"}},[t._v("Statemachineeversult")]),t._v("")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("前面的示例通过构造"),a("code",[t._v("Mono")]),t._v("包装"),a("code",[t._v("Message")]),t._v("并订阅返回的"),a("code",[t._v("Flux")]),t._v("结果来发送事件。"),a("code",[t._v("Message")]),t._v("让我们向事件添加任意的额外信息,然后当(例如)实现操作时,事件对"),a("code",[t._v("StateContext")]),t._v("可见。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("消息头通常被传递,直到机器运行到"),a("br"),t._v("特定事件的完成。例如,如果一个事件正在导致"),a("br"),t._v("转换为具有匿名转换为"),a("br"),t._v("状态的"),a("code",[t._v("A")]),t._v("状态,则原始事件可用于处于"),a("code",[t._v("B")]),t._v("状态的动作或保护。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("也可以发送"),a("code",[t._v("Flux")]),t._v("的消息,而不是只发送一个带有"),a("code",[t._v("Mono")]),t._v("的消息。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('Message<String> message1 = MessageBuilder\n\t.withPayload("E1")\n\t.build();\nMessage<String> message2 = MessageBuilder\n\t.withPayload("E2")\n\t.build();\n\nFlux<StateMachineEventResult<String, String>> results =\n\tstateMachine.sendEvents(Flux.just(message1, message2));\n\nresults.subscribe();\n')])])]),a("h4",{attrs:{id:"statemachineeventresult"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#statemachineeventresult"}},[t._v("#")]),t._v(" statemachineeventresult")]),t._v(" "),a("p",[a("code",[t._v("StateMachineEventResult")]),t._v("包含有关事件发送结果的更详细信息。由此,你可以得到一个"),a("code",[t._v("Region")]),t._v(",它处理了一个事件,"),a("code",[t._v("Message")]),t._v("本身以及一个实际的"),a("code",[t._v("ResultType")]),t._v("。从"),a("code",[t._v("ResultType")]),t._v("中,你可以查看消息是否被接受、拒绝或推迟。一般来说,当下标完成时,事件被传递到机器中。")]),t._v(" "),a("h3",{attrs:{id:"使用timertrigger"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用timertrigger"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("TimerTrigger")])]),t._v(" "),a("p",[a("code",[t._v("TimerTrigger")]),t._v("当需要在没有任何用户交互的情况下自动触发某些内容时,是很有用的。"),a("code",[t._v("Trigger")]),t._v("通过在配置期间将计时器与转换关联,将其添加到转换中。")]),t._v(" "),a("p",[t._v("目前,有两种类型的支持定时器,一种是连续地触发定时器,另一种是在进入源状态后触发定时器。下面的示例展示了如何使用触发器:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config2 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.state("S2")\n\t\t\t\t.state("S3");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("S1").target("S2").event("E1")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("S1").target("S3").event("E2")\n\t\t\t\t.and()\n\t\t\t.withInternal()\n\t\t\t\t.source("S2")\n\t\t\t\t.action(timerAction())\n\t\t\t\t.timer(1000)\n\t\t\t\t.and()\n\t\t\t.withInternal()\n\t\t\t\t.source("S3")\n\t\t\t\t.action(timerAction())\n\t\t\t\t.timerOnce(1000);\n\t}\n\n\t@Bean\n\tpublic TimerAction timerAction() {\n\t\treturn new TimerAction();\n\t}\n}\n\npublic class TimerAction implements Action<String, String> {\n\n\t@Override\n\tpublic void execute(StateContext<String, String> context) {\n\t\t// do something in every 1 sec\n\t}\n}\n')])])]),a("p",[t._v("前面的示例有三个状态:"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v(""),a("code",[t._v("S3")]),t._v("。我们有一个正常的外部转换,分别是从"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v("和从"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S3")]),t._v("的事件"),a("code",[t._v("E1")]),t._v(""),a("code",[t._v("E2")]),t._v("。使用"),a("code",[t._v("TimerTrigger")]),t._v("的有趣部分是当我们定义源状态"),a("code",[t._v("S2")]),t._v(""),a("code",[t._v("S3")]),t._v("的内部转换时。")]),t._v(" "),a("p",[t._v("对于这两个转换,我们调用"),a("code",[t._v("Action")]),t._v(" Bean("),a("code",[t._v("timerAction")]),t._v("),其中源状态"),a("code",[t._v("S2")]),t._v("使用"),a("code",[t._v("timer")]),t._v(""),a("code",[t._v("S3")]),t._v("使用"),a("code",[t._v("timerOnce")]),t._v("。给出的值以毫秒为单位("),a("code",[t._v("1000")]),t._v("毫秒,在两种情况下都是一秒)。")]),t._v(" "),a("p",[t._v("一旦状态机接收到事件"),a("code",[t._v("E1")]),t._v(",它就会执行从"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v("的转换,计时器就会启动。当状态是"),a("code",[t._v("S2")]),t._v("时,"),a("code",[t._v("TimerTrigger")]),t._v("运行并导致与该状态相关的转换——在这种情况下,定义了"),a("code",[t._v("timerAction")]),t._v("的内部转换。")]),t._v(" "),a("p",[t._v("一旦状态机接收到"),a("code",[t._v("E2")]),t._v(",它就会执行从"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S3")]),t._v("的转换,计时器就会启动。此计时器仅在输入状态后执行一次(在计时器中定义的延迟之后)。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("在幕后,计时器是可能导致"),a("br"),t._v("转换发生的简单触发器。使用"),a("code",[t._v("timer()")]),t._v("定义转换保持"),a("br"),t._v("触发,并且仅当源状态处于活动状态时才会导致转换。"),a("br"),t._v("使用"),a("code",[t._v("timerOnce()")]),t._v("的转换有一点不同,因为它"),a("br"),t._v("仅在实际进入源状态时的延迟后才触发。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("如果你希望在输入状态时发生一次延迟"),a("br"),t._v("之后发生某些事情,请使用"),a("code",[t._v("timerOnce()")]),t._v("")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"监听状态机事件"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#监听状态机事件"}},[t._v("#")]),t._v(" 监听状态机事件")]),t._v(" "),a("p",[t._v("在某些用例中,你希望了解状态机正在发生什么,对某些事情做出反应,或者获取日志详细信息以用于调试目的。 Spring StateMachine 提供了用于添加侦听器的接口。然后,这些侦听器给出一个选项,在发生各种状态更改、操作等时获得回调。")]),t._v(" "),a("p",[t._v("你基本上有两种选择:监听 Spring 应用程序上下文事件或直接将监听器附加到状态机。这两者基本上提供了相同的信息。一个生成事件作为事件类,另一个通过侦听器接口生成回调。这两点都有优点和缺点,我们将在后面进行讨论。")]),t._v(" "),a("h3",{attrs:{id:"应用程序上下文事件"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#应用程序上下文事件"}},[t._v("#")]),t._v(" 应用程序上下文事件")]),t._v(" "),a("p",[t._v("应用程序上下文事件类是"),a("code",[t._v("OnTransitionStartEvent")]),t._v(""),a("code",[t._v("OnTransitionEvent")]),t._v(""),a("code",[t._v("OnTransitionEndEvent")]),t._v(""),a("code",[t._v("OnStateExitEvent")]),t._v(""),a("code",[t._v("OnStateEntryEvent")]),t._v(""),a("code",[t._v("OnStateChangedEvent")]),t._v(""),a("code",[t._v("OnStateMachineStart")]),t._v(""),a("code",[t._v("OnStateMachineStop")]),t._v(",以及扩展基本事件类的其他类,"),a("code",[t._v("StateMachineEvent")]),t._v("。这些可以与 Spring "),a("code",[t._v("ApplicationListener")]),t._v("一起使用。")]),t._v(" "),a("p",[a("code",[t._v("StateMachine")]),t._v("通过"),a("code",[t._v("StateMachineEventPublisher")]),t._v("发送上下文事件。如果用"),a("code",[t._v("@EnableStateMachine")]),t._v("注释了"),a("code",[t._v("@Configuration")]),t._v("类,则会自动创建默认实现。下面的示例从在"),a("code",[t._v("@Configuration")]),t._v("类中定义的 Bean 中获取"),a("code",[t._v("StateMachineApplicationEventListener")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class StateMachineApplicationEventListener\n\t\timplements ApplicationListener<StateMachineEvent> {\n\n\t@Override\n\tpublic void onApplicationEvent(StateMachineEvent event) {\n\t}\n}\n\n@Configuration\npublic class ListenerConfig {\n\n\t@Bean\n\tpublic StateMachineApplicationEventListener contextListener() {\n\t\treturn new StateMachineApplicationEventListener();\n\t}\n}\n")])])]),a("p",[t._v("上下文事件也可以通过使用"),a("code",[t._v("@EnableStateMachine")]),t._v("自动启用,"),a("code",[t._v("StateMachine")]),t._v("用于构建机器并注册为 Bean,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class ManualBuilderConfig {\n\n\t@Bean\n\tpublic StateMachine<String, String> stateMachine() throws Exception {\n\n\t\tBuilder<String, String> builder = StateMachineBuilder.builder();\n\t\tbuilder.configureStates()\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.state("S2");\n\t\tbuilder.configureTransitions()\n\t\t\t.withExternal()\n\t\t\t\t.source("S1")\n\t\t\t\t.target("S2")\n\t\t\t\t.event("E1");\n\t\treturn builder.build();\n\t}\n}\n')])])]),a("h3",{attrs:{id:"使用statemachinelistener"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statemachinelistener"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateMachineListener")])]),t._v(" "),a("p",[t._v("通过使用"),a("code",[t._v("StateMachineListener")]),t._v(",你可以扩展它并实现所有回调方法,或者使用"),a("code",[t._v("StateMachineListenerAdapter")]),t._v("类,它包含存根方法实现,并选择要覆盖的那些。下面的示例使用后一种方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class StateMachineEventListener\n\t\textends StateMachineListenerAdapter<States, Events> {\n\n\t@Override\n\tpublic void stateChanged(State<States, Events> from, State<States, Events> to) {\n\t}\n\n\t@Override\n\tpublic void stateEntered(State<States, Events> state) {\n\t}\n\n\t@Override\n\tpublic void stateExited(State<States, Events> state) {\n\t}\n\n\t@Override\n\tpublic void transition(Transition<States, Events> transition) {\n\t}\n\n\t@Override\n\tpublic void transitionStarted(Transition<States, Events> transition) {\n\t}\n\n\t@Override\n\tpublic void transitionEnded(Transition<States, Events> transition) {\n\t}\n\n\t@Override\n\tpublic void stateMachineStarted(StateMachine<States, Events> stateMachine) {\n\t}\n\n\t@Override\n\tpublic void stateMachineStopped(StateMachine<States, Events> stateMachine) {\n\t}\n\n\t@Override\n\tpublic void eventNotAccepted(Message<Events> event) {\n\t}\n\n\t@Override\n\tpublic void extendedStateChanged(Object key, Object value) {\n\t}\n\n\t@Override\n\tpublic void stateMachineError(StateMachine<States, Events> stateMachine, Exception exception) {\n\t}\n\n\t@Override\n\tpublic void stateContext(StateContext<States, Events> stateContext) {\n\t}\n}\n")])])]),a("p",[t._v("在前面的示例中,我们创建了自己的 Listener 类("),a("code",[t._v("StateMachineEventListener")]),t._v("),它扩展了"),a("code",[t._v("StateMachineListenerAdapter")]),t._v("")]),t._v(" "),a("p",[a("code",[t._v("stateContext")]),t._v("侦听器方法允许访问不同阶段上的各种"),a("code",[t._v("StateContext")]),t._v("更改。你可以在[using"),a("code",[t._v("StateContext")]),t._v("]中找到有关它的更多信息。")]),t._v(" "),a("p",[t._v("一旦定义了自己的侦听器,就可以使用"),a("code",[t._v("addStateListener")]),t._v("方法将其注册到状态机中。是在 Spring 配置中连接它,还是在应用程序生命周期的任何时候手动连接它,这是一个风格问题。下面的示例展示了如何附加监听器:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class Config7 {\n\n\t@Autowired\n\tStateMachine<States, Events> stateMachine;\n\n\t@Bean\n\tpublic StateMachineEventListener stateMachineEventListener() {\n\t\tStateMachineEventListener listener = new StateMachineEventListener();\n\t\tstateMachine.addStateListener(listener);\n\t\treturn listener;\n\t}\n\n}\n")])])]),a("h3",{attrs:{id:"限制和问题"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#限制和问题"}},[t._v("#")]),t._v(" 限制和问题")]),t._v(" "),a("p",[t._v("Spring 应用程序上下文不是那里最快的事件总线,因此我们建议对状态机发送的事件的速率给予一些考虑。为了获得更好的性能,使用"),a("code",[t._v("StateMachineListener")]),t._v("接口可能会更好。出于这个特定的原因,你可以使用带有"),a("code",[t._v("@EnableStateMachine")]),t._v(""),a("code",[t._v("@EnableStateMachineFactory")]),t._v(""),a("code",[t._v("contextEvents")]),t._v("标志来禁用 Spring 应用程序上下文事件,如上一节所示。下面的示例展示了如何禁用 Spring 应用程序上下文事件:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine(contextEvents = false)\npublic class Config8\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n}\n\n@Configuration\n@EnableStateMachineFactory(contextEvents = false)\npublic class Config9\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n}\n")])])]),a("h2",{attrs:{id:"上下文集成"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#上下文集成"}},[t._v("#")]),t._v(" 上下文集成")]),t._v(" "),a("p",[t._v("通过监听状态机的事件或使用带有状态和转换的操作来与状态机进行交互是有点限制的。这种方法有时会过于有限和冗长,无法与状态机所使用的应用程序创建交互。对于这个特定的用例,我们进行了 Spring 风格的上下文集成,可以轻松地将状态机功能插入到 bean 中。")]),t._v(" "),a("p",[t._v("已对可用的注释进行了协调,以允许访问与"),a("a",{attrs:{href:"#sm-listeners"}},[t._v("监听状态机事件")]),t._v("相同的状态机执行点。")]),t._v(" "),a("p",[t._v("你可以使用"),a("code",[t._v("@WithStateMachine")]),t._v("注释将状态机与现有的 Bean 关联起来。然后,你可以开始向 Bean 的方法添加受支持的注释。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@WithStateMachine\npublic class Bean1 {\n\n\t@OnTransition\n\tpublic void anyTransition() {\n\t}\n}\n")])])]),a("p",[t._v("你还可以使用注释"),a("code",[t._v("name")]),t._v("字段从应用程序上下文中附加任何其他状态机。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@WithStateMachine(name = "myMachineBeanName")\npublic class Bean2 {\n\n\t@OnTransition\n\tpublic void anyTransition() {\n\t}\n}\n')])])]),a("p",[t._v("有时,使用"),a("code",[t._v("machine id")]),t._v("会更方便,你可以将其设置为更好地识别多个实例。此 ID 映射到"),a("code",[t._v("StateMachine")]),t._v("接口中的"),a("code",[t._v("getId()")]),t._v("方法。下面的示例展示了如何使用它:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@WithStateMachine(id = "myMachineId")\npublic class Bean16 {\n\n\t@OnTransition\n\tpublic void anyTransition() {\n\t}\n}\n')])])]),a("p",[t._v("你也可以使用"),a("code",[t._v("@WithStateMachine")]),t._v("作为元注释,如前面的示例所示。在这种情况下,你可以用"),a("code",[t._v("WithMyBean")]),t._v("注释你的 Bean。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Target(ElementType.TYPE)\n@Retention(RetentionPolicy.RUNTIME)\n@WithStateMachine(name = "myMachineBeanName")\npublic @interface WithMyBean {\n}\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("这些方法的返回类型并不重要,并且有效地"),a("br"),t._v("被丢弃。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"启用集成"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#启用集成"}},[t._v("#")]),t._v(" 启用集成")]),t._v(" "),a("p",[t._v("你可以通过使用"),a("code",[t._v("@EnableWithStateMachine")]),t._v("注释启用"),a("code",[t._v("@WithStateMachine")]),t._v("的所有特性,该注释将所需的配置导入 Spring 应用程序上下文。"),a("code",[t._v("@EnableStateMachine")]),t._v(""),a("code",[t._v("@EnableStateMachineFactory")]),t._v("都已经使用此注释进行了注释,因此没有必要再次添加它。但是,如果一台机器是在没有配置适配器的情况下构建和配置的,则必须使用"),a("code",[t._v("@EnableWithStateMachine")]),t._v("才能使用"),a("code",[t._v("@WithStateMachine")]),t._v("的这些功能。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('public static StateMachine<String, String> buildMachine(BeanFactory beanFactory) throws Exception {\n\tBuilder<String, String> builder = StateMachineBuilder.builder();\n\n\tbuilder.configureConfiguration()\n\t\t.withConfiguration()\n\t\t\t.machineId("myMachineId")\n\t\t\t.beanFactory(beanFactory);\n\n\tbuilder.configureStates()\n\t\t.withStates()\n\t\t\t.initial("S1")\n\t\t\t.state("S2");\n\n\tbuilder.configureTransitions()\n\t\t.withExternal()\n\t\t\t.source("S1")\n\t\t\t.target("S2")\n\t\t\t.event("E1");\n\n\treturn builder.build();\n}\n\n@WithStateMachine(id = "myMachineId")\nstatic class Bean17 {\n\n\t@OnStateChanged\n\tpublic void onStateChanged() {\n\t}\n}\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("如果机器不是作为 Bean 创建的,则需要为机器设置"),a("code",[t._v("BeanFactory")]),t._v(",如 prededing 示例中所示。否则,TGE 机器"),a("br"),t._v("不知道调用"),a("code",[t._v("@WithStateMachine")]),t._v("方法的处理程序。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"方法参数"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#方法参数"}},[t._v("#")]),t._v(" 方法参数")]),t._v(" "),a("p",[t._v("每个注释都支持完全相同的一组可能的方法参数,但是运行时行为是不同的,这取决于注释本身和调用注释方法的阶段。要更好地理解上下文是如何工作的,请参见[使用"),a("code",[t._v("StateContext")]),t._v("]。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("有关方法参数之间的差异,请参阅本文后面的章节,其中描述了"),a("br"),t._v("个别注释。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("实际上,所有带注释的方法都是通过使用 Spring SPEL 表达式来调用的,这些表达式是在处理过程中动态构建的。为了实现这一点,这些表达式需要有一个根对象(它们对其进行求值)。这个根对象是"),a("code",[t._v("StateContext")]),t._v("。我们还在内部进行了一些调整,这样就可以直接访问"),a("code",[t._v("StateContext")]),t._v("方法,而无需通过上下文句柄。")]),t._v(" "),a("p",[t._v("最简单的方法参数是"),a("code",[t._v("StateContext")]),t._v("本身。下面的示例展示了如何使用它:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@WithStateMachine\npublic class Bean3 {\n\n\t@OnTransition\n\tpublic void anyTransition(StateContext<String, String> stateContext) {\n\t}\n}\n")])])]),a("p",[t._v("你可以访问"),a("code",[t._v("StateContext")]),t._v("内容的其余部分。参数的数量和顺序并不重要。下面的示例展示了如何访问"),a("code",[t._v("StateContext")]),t._v("内容的各个部分:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@WithStateMachine\npublic class Bean4 {\n\n\t@OnTransition\n\tpublic void anyTransition(\n\t\t\t@EventHeaders Map<String, Object> headers,\n\t\t\t@EventHeader("myheader1") Object myheader1,\n\t\t\t@EventHeader(name = "myheader2", required = false) String myheader2,\n\t\t\tExtendedState extendedState,\n\t\t\tStateMachine<String, String> stateMachine,\n\t\t\tMessage<String> message,\n\t\t\tException e) {\n\t}\n}\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("你可以使用"),a("code",[t._v("@EventHeader")]),t._v(",而不是使用所有带有"),a("code",[t._v("@EventHeaders")]),t._v("的事件头,它可以绑定到一个单独的头。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"转换注释"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#转换注释"}},[t._v("#")]),t._v(" 转换注释")]),t._v(" "),a("p",[t._v("转换的注释是"),a("code",[t._v("@OnTransition")]),t._v(""),a("code",[t._v("@OnTransitionStart")]),t._v(""),a("code",[t._v("@OnTransitionEnd")]),t._v("")]),t._v(" "),a("p",[t._v("这些注释的行为完全相同。为了说明它们是如何工作的,我们展示了"),a("code",[t._v("@OnTransition")]),t._v("是如何使用的。在这个注释中,你可以使用"),a("code",[t._v("source")]),t._v(""),a("code",[t._v("target")]),t._v("属性来限定转换。如果"),a("code",[t._v("source")]),t._v(""),a("code",[t._v("target")]),t._v("为空,则任何转换都是匹配的。下面的示例展示了如何使用"),a("code",[t._v("@OnTransition")]),t._v("注释(请记住"),a("code",[t._v("@OnTransitionStart")]),t._v(""),a("code",[t._v("@OnTransitionEnd")]),t._v("的工作方式相同):")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@WithStateMachine\npublic class Bean5 {\n\n\t@OnTransition(source = "S1", target = "S2")\n\tpublic void fromS1ToS2() {\n\t}\n\n\t@OnTransition\n\tpublic void anyTransition() {\n\t}\n}\n')])])]),a("p",[t._v("默认情况下,由于 Java 语言的限制,你无法使用"),a("code",[t._v("@OnTransition")]),t._v("注释来创建状态和事件枚举。出于这个原因,你需要使用字符串表示。")]),t._v(" "),a("p",[t._v("此外,你可以通过向方法添加所需的参数来访问"),a("code",[t._v("Event Headers")]),t._v(""),a("code",[t._v("ExtendedState")]),t._v("。然后使用这些参数自动调用该方法。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@WithStateMachine\npublic class Bean6 {\n\n\t@StatesOnTransition(source = States.S1, target = States.S2)\n\tpublic void fromS1ToS2(@EventHeaders Map<String, Object> headers, ExtendedState extendedState) {\n\t}\n}\n")])])]),a("p",[t._v("但是,如果你想要一个类型安全的注释,你可以创建一个新的注释,并使用"),a("code",[t._v("@OnTransition")]),t._v("作为元注释。这种用户级别的注释可以对实际的状态和事件枚举进行引用,框架尝试以相同的方式匹配这些状态和事件。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Target(ElementType.METHOD)\n@Retention(RetentionPolicy.RUNTIME)\n@OnTransition\npublic @interface StatesOnTransition {\n\n\tStates[] source() default {};\n\n\tStates[] target() default {};\n}\n")])])]),a("p",[t._v("在前面的示例中,我们以类型安全的方式创建了"),a("code",[t._v("@StatesOnTransition")]),t._v("注释,该注释定义了"),a("code",[t._v("source")]),t._v(""),a("code",[t._v("target")]),t._v("。下面的示例在 Bean 中使用了该注释:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@WithStateMachine\npublic class Bean7 {\n\n\t@StatesOnTransition(source = States.S1, target = States.S2)\n\tpublic void fromS1ToS2() {\n\t}\n}\n")])])]),a("h3",{attrs:{id:"状态注释"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态注释"}},[t._v("#")]),t._v(" 状态注释")]),t._v(" "),a("p",[t._v("下面的状态注释是可用的:"),a("code",[t._v("@OnStateChanged")]),t._v(""),a("code",[t._v("@OnStateEntry")]),t._v(",和"),a("code",[t._v("@OnStateExit")]),t._v("。下面的示例展示了如何使用"),a("code",[t._v("OnStateChanged")]),t._v("注释(其他两种方法的工作方式相同):")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@WithStateMachine\npublic class Bean8 {\n\n\t@OnStateChanged\n\tpublic void anyStateChange() {\n\t}\n}\n")])])]),a("p",[t._v(""),a("a",{attrs:{href:"#state-machine-transition-annotations"}},[t._v("过渡注释")]),t._v("一样,你可以定义目标状态和源状态。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@WithStateMachine\npublic class Bean9 {\n\n\t@OnStateChanged(source = "S1", target = "S2")\n\tpublic void stateChangeFromS1toS2() {\n\t}\n}\n')])])]),a("p",[t._v("为了类型安全,需要使用"),a("code",[t._v("@OnStateChanged")]),t._v("作为元注释来为枚举创建新的注释。下面的例子说明了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Target(ElementType.METHOD)\n@Retention(RetentionPolicy.RUNTIME)\n@OnStateChanged\npublic @interface StatesOnStates {\n\n\tStates[] source() default {};\n\n\tStates[] target() default {};\n}\n")])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@WithStateMachine\npublic class Bean10 {\n\n\t@StatesOnStates(source = States.S1, target = States.S2)\n\tpublic void fromS1ToS2() {\n\t}\n}\n")])])]),a("p",[t._v("状态进入和状态退出的方法的行为方式相同,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@WithStateMachine\npublic class Bean11 {\n\n\t@OnStateEntry\n\tpublic void anyStateEntry() {\n\t}\n\n\t@OnStateExit\n\tpublic void anyStateExit() {\n\t}\n}\n")])])]),a("h3",{attrs:{id:"事件注释"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#事件注释"}},[t._v("#")]),t._v(" 事件注释")]),t._v(" "),a("p",[t._v("有一个与事件相关的注释。它被命名为"),a("code",[t._v("@OnEventNotAccepted")]),t._v("。如果指定"),a("code",[t._v("event")]),t._v("属性,则可以侦听未被接受的特定事件。如果你没有指定一个事件,你可以列出未被接受的任何事件。下面的示例展示了使用"),a("code",[t._v("@OnEventNotAccepted")]),t._v("注释的两种方式:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@WithStateMachine\npublic class Bean12 {\n\n\t@OnEventNotAccepted\n\tpublic void anyEventNotAccepted() {\n\t}\n\n\t@OnEventNotAccepted(event = "E1")\n\tpublic void e1EventNotAccepted() {\n\t}\n}\n')])])]),a("h3",{attrs:{id:"状态机注释"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态机注释"}},[t._v("#")]),t._v(" 状态机注释")]),t._v(" "),a("p",[t._v("以下注释可用于状态机:"),a("code",[t._v("@OnStateMachineStart")]),t._v(""),a("code",[t._v("@OnStateMachineStop")]),t._v(""),a("code",[t._v("@OnStateMachineError")]),t._v("")]),t._v(" "),a("p",[t._v("在状态机的启动和停止过程中,会调用生命周期方法。下面的示例展示了如何使用"),a("code",[t._v("@OnStateMachineStart")]),t._v(""),a("code",[t._v("@OnStateMachineStop")]),t._v("来侦听这些事件:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@WithStateMachine\npublic class Bean13 {\n\n\t@OnStateMachineStart\n\tpublic void onStateMachineStart() {\n\t}\n\n\t@OnStateMachineStop\n\tpublic void onStateMachineStop() {\n\t}\n}\n")])])]),a("p",[t._v("如果状态机出现异常错误,则调用"),a("code",[t._v("@OnStateMachineStop")]),t._v("注释。下面的示例展示了如何使用它:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@WithStateMachine\npublic class Bean14 {\n\n\t@OnStateMachineError\n\tpublic void onStateMachineError() {\n\t}\n}\n")])])]),a("h3",{attrs:{id:"扩展状态注释"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#扩展状态注释"}},[t._v("#")]),t._v(" 扩展状态注释")]),t._v(" "),a("p",[t._v("有一个扩展的状态相关注释。它被命名为"),a("a",{attrs:{href:"#statemachine-examples-persist"}},[t._v("Persist")]),t._v("。你还可以只针对特定的"),a("code",[t._v("key")]),t._v("更改监听更改。下面的示例展示了如何使用"),a("code",[t._v("@OnExtendedStateChanged")]),t._v(",包括带"),a("code",[t._v("key")]),t._v("属性和不带"),a("code",[t._v("key")]),t._v("属性:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@WithStateMachine\npublic class Bean15 {\n\n\t@OnExtendedStateChanged\n\tpublic void anyStateChange() {\n\t}\n\n\t@OnExtendedStateChanged(key = "key1")\n\tpublic void key1Changed() {\n\t}\n}\n')])])]),a("h2",{attrs:{id:"使用statemachineaccessor"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statemachineaccessor"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateMachineAccessor")])]),t._v(" "),a("p",[a("code",[t._v("StateMachine")]),t._v("是与状态机通信的主要接口。有时,你可能需要对状态机及其嵌套的机器和区域的内部结构进行更动态和程序化的访问。对于这些用例,"),a("code",[t._v("StateMachine")]),t._v("公开了一个名为"),a("code",[t._v("StateMachineAccessor")]),t._v("的功能接口,该接口提供了一个接口来访问单个"),a("code",[t._v("StateMachine")]),t._v(""),a("code",[t._v("Region")]),t._v("实例。")]),t._v(" "),a("p",[a("code",[t._v("StateMachineFunction")]),t._v("是一个简单的函数接口,它允许你将"),a("code",[t._v("StateMachineAccess")]),t._v("接口应用到状态机。在 JDK7 中,这些代码创建的代码是一个很小的详细代码。然而,对于 JDK8Lambdas,DOCE 相对来说并不详细。")]),t._v(" "),a("p",[a("code",[t._v("doWithAllRegions")]),t._v("方法允许访问状态机中的所有"),a("code",[t._v("Region")]),t._v("实例。下面的示例展示了如何使用它:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("stateMachine.getStateMachineAccessor().doWithAllRegions(function -> function.setRelay(stateMachine));\n\nstateMachine.getStateMachineAccessor()\n\t.doWithAllRegions(access -> access.setRelay(stateMachine));\n")])])]),a("p",[a("code",[t._v("doWithRegion")]),t._v("方法允许访问状态机中的单个"),a("code",[t._v("Region")]),t._v("实例。下面的示例展示了如何使用它:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("stateMachine.getStateMachineAccessor().doWithRegion(function -> function.setRelay(stateMachine));\n\nstateMachine.getStateMachineAccessor()\n\t.doWithRegion(access -> access.setRelay(stateMachine));\n")])])]),a("p",[a("code",[t._v("withAllRegions")]),t._v("方法允许访问状态机中的所有"),a("code",[t._v("Region")]),t._v("实例。下面的示例展示了如何使用它:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("for (StateMachineAccess<String, String> access : stateMachine.getStateMachineAccessor().withAllRegions()) {\n\taccess.setRelay(stateMachine);\n}\n\nstateMachine.getStateMachineAccessor().withAllRegions()\n\t.stream().forEach(access -> access.setRelay(stateMachine));\n")])])]),a("p",[a("code",[t._v("withRegion")]),t._v("方法允许访问状态机中的单个"),a("code",[t._v("Region")]),t._v("实例。下面的示例展示了如何使用它:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("stateMachine.getStateMachineAccessor()\n\t.withRegion().setRelay(stateMachine);\n")])])]),a("h2",{attrs:{id:"使用statemachineinterceptor"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statemachineinterceptor"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateMachineInterceptor")])]),t._v(" "),a("p",[t._v("你可以使用"),a("code",[t._v("StateMachineInterceptor")]),t._v("接口,而不是使用"),a("code",[t._v("StateMachineListener")]),t._v("接口。一个概念上的区别是,你可以使用拦截器来拦截和停止当前状态更改或更改其转换逻辑。你可以使用一个名为"),a("code",[t._v("StateMachineInterceptorAdapter")]),t._v("的适配器类来覆盖缺省的 no-op 方法,而不是实现一个完整的接口。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("一个配方("),a("a",{attrs:{href:"#statemachine-recipes-persist"}},[t._v("Persist")]),t._v(")和一个样本"),a("br"),t._v(""),a("a",{attrs:{href:"#statemachine-examples-persist"}},[t._v("Persist")]),t._v(")与使用"),a("br"),t._v("拦截器有关。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("你可以通过"),a("code",[t._v("StateMachineAccessor")]),t._v("注册拦截器。拦截器的概念是一个相对较深的内部特性,因此不会直接通过"),a("code",[t._v("StateMachine")]),t._v("接口公开。")]),t._v(" "),a("p",[t._v("下面的示例展示了如何添加"),a("code",[t._v("StateMachineInterceptor")]),t._v("并覆盖选定的方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("stateMachine.getStateMachineAccessor()\n\t.withRegion().addStateMachineInterceptor(new StateMachineInterceptor<String, String>() {\n\n\t\t@Override\n\t\tpublic Message<String> preEvent(Message<String> message, StateMachine<String, String> stateMachine) {\n\t\t\treturn message;\n\t\t}\n\n\t\t@Override\n\t\tpublic StateContext<String, String> preTransition(StateContext<String, String> stateContext) {\n\t\t\treturn stateContext;\n\t\t}\n\n\t\t@Override\n\t\tpublic void preStateChange(State<String, String> state, Message<String> message,\n\t\t\t\tTransition<String, String> transition, StateMachine<String, String> stateMachine,\n\t\t\t\tStateMachine<String, String> rootStateMachine) {\n\t\t}\n\n\t\t@Override\n\t\tpublic StateContext<String, String> postTransition(StateContext<String, String> stateContext) {\n\t\t\treturn stateContext;\n\t\t}\n\n\t\t@Override\n\t\tpublic void postStateChange(State<String, String> state, Message<String> message,\n\t\t\t\tTransition<String, String> transition, StateMachine<String, String> stateMachine,\n\t\t\t\tStateMachine<String, String> rootStateMachine) {\n\t\t}\n\n\t\t@Override\n\t\tpublic Exception stateMachineError(StateMachine<String, String> stateMachine,\n\t\t\t\tException exception) {\n\t\t\treturn exception;\n\t\t}\n\t});\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("有关前面示例中显示的错误处理的更多信息,请参见"),a("a",{attrs:{href:"#sm-error-handling"}},[t._v("状态机错误处理")]),t._v("")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"状态机安全"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态机安全"}},[t._v("#")]),t._v(" 状态机安全")]),t._v(" "),a("p",[t._v("安全功能是建立在"),a("a",{attrs:{href:"https://projects.spring.io/spring-security",target:"_blank",rel:"noopener noreferrer"}},[t._v("Spring Security"),a("OutboundLink")],1),t._v("的功能之上的。当需要保护状态机执行的一部分以及与状态机的交互时,安全性功能非常方便。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("我们希望你对 Spring 安全性相当熟悉,这意味着"),a("br"),t._v("我们不会详细介绍整个安全框架是如何工作的。对于"),a("br"),t._v("此信息,你应该阅读 Spring 安全参考文档"),a("br"),t._v("(可用"),a("a",{attrs:{href:"https://spring.io/projects/spring-security#learn",target:"_blank",rel:"noopener noreferrer"}},[t._v("here"),a("OutboundLink")],1),t._v(")。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("安全防御的第一个层次自然是保护事件,这些事件真正驱动了状态机中将要发生的事情。然后,你可以为转换和操作定义更细粒度的安全设置。这类似于让员工进入一栋建筑,然后进入建筑内的特定房间,甚至可以打开和关闭特定房间的灯。如果你信任你的用户,那么事件安全性可能就是你所需要的。如果没有,则需要应用更详细的安全性。")]),t._v(" "),a("p",[t._v("你可以在"),a("a",{attrs:{href:"#sm-security-details"}},[t._v("理解安全")]),t._v("中找到更详细的信息。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("有关完整的示例,请参见"),a("a",{attrs:{href:"#statemachine-examples-security"}},[t._v("Security")]),t._v("示例。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"配置安全性"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#配置安全性"}},[t._v("#")]),t._v(" 配置安全性")]),t._v(" "),a("p",[t._v("用于安全性的所有通用配置都在"),a("code",[t._v("SecurityConfigurer")]),t._v("中完成,它是从"),a("code",[t._v("StateMachineConfigurationConfigurer")]),t._v("中获得的。默认情况下,安全性是禁用的,即使存在 Spring 安全性类。下面的示例展示了如何启用安全性:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\nstatic class Config4 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<String, String> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withSecurity()\n\t\t\t\t.enabled(true)\n\t\t\t\t.transitionAccessDecisionManager(null)\n\t\t\t\t.eventAccessDecisionManager(null);\n\t}\n}\n")])])]),a("p",[t._v("如果绝对需要,可以为事件和转换自定义"),a("code",[t._v("AccessDecisionManager")]),t._v("。如果不定义决策管理器或将其设置为"),a("code",[t._v("null")]),t._v(",则默认管理器将在内部创建。")]),t._v(" "),a("h3",{attrs:{id:"确保事件安全"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#确保事件安全"}},[t._v("#")]),t._v(" 确保事件安全")]),t._v(" "),a("p",[t._v("事件安全性在全局级别上由"),a("code",[t._v("RoleVoter")]),t._v("定义。下面的示例展示了如何启用事件安全性:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class Config1 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<String, String> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withSecurity()\n\t\t\t\t.enabled(true)\n\t\t\t\t.event("true")\n\t\t\t\t.event("ROLE_ANONYMOUS", ComparisonType.ANY);\n\t}\n}\n')])])]),a("p",[t._v("在前面的配置示例中,我们使用"),a("code",[t._v("true")]),t._v("的表达式,该表达式的求值总是"),a("code",[t._v("TRUE")]),t._v("。在实际应用程序中,使用始终计算为"),a("code",[t._v("TRUE")]),t._v("的表达式是没有意义的,但是它显示了表达式需要返回"),a("code",[t._v("TRUE")]),t._v(""),a("code",[t._v("FALSE")]),t._v("的一点。我们还定义了"),a("code",[t._v("ROLE_ANONYMOUS")]),t._v("的一个属性和"),a("code",[t._v("ComparisonType")]),t._v("的一个"),a("code",[t._v("ANY")]),t._v("。有关使用属性和表达式的更多信息,请参见"),a("a",{attrs:{href:"#sm-security-attributes-expressions"}},[t._v("使用安全属性和表达式")]),t._v("")]),t._v(" "),a("h3",{attrs:{id:"确保过渡"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#确保过渡"}},[t._v("#")]),t._v(" 确保过渡")]),t._v(" "),a("p",[t._v("你可以在全局范围内定义转换安全性,如下例所示。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class Config6 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<String, String> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withSecurity()\n\t\t\t\t.enabled(true)\n\t\t\t\t.transition("true")\n\t\t\t\t.transition("ROLE_ANONYMOUS", ComparisonType.ANY);\n\t}\n}\n')])])]),a("p",[t._v("如果安全性是在转换本身中定义的,那么它将覆盖任何全局设置的安全性。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class Config2 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("S0")\n\t\t\t\t.target("S1")\n\t\t\t\t.event("A")\n\t\t\t\t.secured("ROLE_ANONYMOUS", ComparisonType.ANY)\n\t\t\t\t.secured("hasTarget(\'S1\')");\n\t}\n}\n')])])]),a("p",[t._v("有关使用属性和表达式的更多信息,请参见"),a("a",{attrs:{href:"#sm-security-attributes-expressions"}},[t._v("使用安全属性和表达式")]),t._v("")]),t._v(" "),a("h3",{attrs:{id:"确保操作安全"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#确保操作安全"}},[t._v("#")]),t._v(" 确保操作安全")]),t._v(" "),a("p",[t._v("对于状态机中的动作没有专门的安全定义,但是你可以使用 Spring Security 中的全局方法安全性来保护动作。这要求将"),a("code",[t._v("Action")]),t._v("定义为 proxied"),a("code",[t._v("@Bean")]),t._v(",并将其"),a("code",[t._v("execute")]),t._v("方法注释为"),a("code",[t._v("@Secured")]),t._v("。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class Config3 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<String, String> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withSecurity()\n\t\t\t\t.enabled(true);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("S0")\n\t\t\t\t.state("S1");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("S0")\n\t\t\t\t.target("S1")\n\t\t\t\t.action(securedAction())\n\t\t\t\t.event("A");\n\t}\n\n\t@Scope(proxyMode = ScopedProxyMode.TARGET_CLASS)\n\t@Bean\n\tpublic Action<String, String> securedAction() {\n\t\treturn new Action<String, String>() {\n\n\t\t\t@Secured("ROLE_ANONYMOUS")\n\t\t\t@Override\n\t\t\tpublic void execute(StateContext<String, String> context) {\n\t\t\t}\n\t\t};\n\t}\n\n}\n')])])]),a("p",[t._v("Spring 安全性需要启用全局方法安全性。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableGlobalMethodSecurity(securedEnabled = true)\npublic static class Config5 extends WebSecurityConfigurerAdapter {\n\n\t@Autowired\n\tpublic void configureGlobal(AuthenticationManagerBuilder auth) throws Exception {\n\t\tauth\n\t\t\t.inMemoryAuthentication()\n\t\t\t\t.withUser("user").password("password").roles("USER");\n\t}\n}\n')])])]),a("p",[t._v("有关更多详细信息,请参见 Spring 安全参考指南(可用"),a("a",{attrs:{href:"https://spring.io/projects/spring-security#learn",target:"_blank",rel:"noopener noreferrer"}},[t._v("here"),a("OutboundLink")],1),t._v(")。")]),t._v(" "),a("h3",{attrs:{id:"使用安全属性和表达式"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用安全属性和表达式"}},[t._v("#")]),t._v(" 使用安全属性和表达式")]),t._v(" "),a("p",[t._v("通常,你可以通过两种方式定义安全属性:通过使用安全属性和通过使用安全表达式。属性更容易使用,但在功能方面相对有限。表达式提供了更多的功能,但使用起来有点困难。")]),t._v(" "),a("h4",{attrs:{id:"泛型属性用法"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#泛型属性用法"}},[t._v("#")]),t._v(" 泛型属性用法")]),t._v(" "),a("p",[t._v("默认情况下,用于事件和转换的"),a("code",[t._v("AccessDecisionManager")]),t._v("实例都使用"),a("code",[t._v("RoleVoter")]),t._v(",这意味着你可以使用 Spring Security 中的角色属性。")]),t._v(" "),a("p",[t._v("对于属性,我们有三种不同的比较类型:"),a("code",[t._v("ANY")]),t._v(""),a("code",[t._v("ALL")]),t._v(""),a("code",[t._v("MAJORITY")]),t._v("。这些比较类型映射到默认访问决策管理器(分别为"),a("code",[t._v("AffirmativeBased")]),t._v(""),a("code",[t._v("UnanimousBased")]),t._v(""),a("code",[t._v("ConsensusBased")]),t._v(")。如果你定义了一个自定义"),a("code",[t._v("S2")]),t._v(",那么比较类型将被有效地丢弃,因为它仅用于创建默认管理器。")]),t._v(" "),a("h4",{attrs:{id:"通用表达式用法"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#通用表达式用法"}},[t._v("#")]),t._v(" 通用表达式用法")]),t._v(" "),a("p",[t._v("安全表达式必须返回"),a("code",[t._v("TRUE")]),t._v(""),a("code",[t._v("FALSE")]),t._v("")]),t._v(" "),a("p",[t._v("表达式根对象的基类是"),a("code",[t._v("SecurityExpressionRoot")]),t._v("。它提供了一些常见的表达式,这些表达式在转换和事件安全性方面都可用。下表描述了最常用的内置表达式:")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th",[t._v("Expression")]),t._v(" "),a("th",[t._v("说明")])])]),t._v(" "),a("tbody",[a("tr",[a("td",[a("code",[t._v("hasRole([role])")])]),t._v(" "),a("td",[t._v("如果当前主体具有指定的角色,则返回"),a("code",[t._v("true")]),t._v("。通过"),a("br"),t._v("默认设置,如果提供的角色不以"),a("code",[t._v("ROLE_")]),t._v("开头,则添加"),a("br"),t._v("。你可以通过修改"),a("code",[t._v("DefaultWebSecurityExpressionHandler")]),t._v("上的"),a("code",[t._v("ANY")]),t._v("来自定义此项。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("hasAnyRole([role1,role2])")])]),t._v(" "),a("td",[t._v("返回"),a("code",[t._v("true")]),t._v("如果当前主体有任何提供的"),a("br"),t._v("角色(以逗号分隔的字符串列表给出)。默认情况下,如果每个"),a("br"),t._v("提供的角色不以"),a("code",[t._v("ROLE_")]),t._v("开头,则添加该角色。你可以通过修改"),a("code",[t._v("DefaultWebSecurityExpressionHandler")]),t._v("上的"),a("code",[t._v("defaultRolePrefix")]),t._v("来自定义此"),a("br"),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("hasAuthority([authority])")])]),t._v(" "),a("td",[t._v("如果当前主体具有指定的权限,则返回"),a("code",[t._v("true")]),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("hasAnyAuthority([authority1,authority2])")])]),t._v(" "),a("td",[t._v("返回"),a("code",[t._v("true")]),t._v(",如果当前主体有任何提供的"),a("br"),t._v("角色(以逗号分隔的字符串列表给出)。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("principal")])]),t._v(" "),a("td",[t._v("允许直接访问表示"),a("br"),t._v("当前用户的主体对象。")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("authentication")])]),t._v(" "),a("td",[t._v("允许直接访问当前"),a("code",[t._v("Authentication")]),t._v("对象从"),a("br"),t._v("中获得的"),a("br"),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("permitAll")])]),t._v(" "),a("td",[t._v("总是计算为"),a("code",[t._v("true")]),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("denyAll")])]),t._v(" "),a("td",[t._v("总是计算为"),a("code",[t._v("false")]),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("isAnonymous()")])]),t._v(" "),a("td",[t._v("如果当前主体是匿名用户,则返回"),a("code",[t._v("true")]),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("isRememberMe()")])]),t._v(" "),a("td",[t._v("如果当前主体是 rememe-me 用户,则返回"),a("code",[t._v("true")]),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("isAuthenticated()")])]),t._v(" "),a("td",[t._v("如果用户不是匿名的,则返回"),a("code",[t._v("true")]),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("isFullyAuthenticated()")])]),t._v(" "),a("td",[t._v("如果用户不是匿名者或 Remember-Me 用户,则返回"),a("code",[t._v("ROLE_")]),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("hasPermission(Object target, Object permission)")])]),t._v(" "),a("td",[t._v("如果用户能够访问"),a("br"),t._v("给定权限所提供的目标——例如,"),a("code",[t._v("hasPermission(domainObject, 'read')")]),t._v(",则返回"),a("code",[t._v("hasPermission(domainObject, 'read')")]),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("hasPermission(Object targetId, String targetType, Object<br/>permission)")])]),t._v(" "),a("td",[t._v("如果用户能够访问"),a("br"),t._v("给定权限所提供的目标——例如,"),a("code",[t._v("hasPermission(1,<br/>'com.example.domain.Message', 'read')")]),t._v(",则返回。")])])])]),t._v(" "),a("h4",{attrs:{id:"事件属性"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#事件属性"}},[t._v("#")]),t._v(" 事件属性")]),t._v(" "),a("p",[t._v("你可以使用"),a("code",[t._v("EVENT_")]),t._v("的前缀来匹配事件 ID。例如,匹配事件"),a("code",[t._v("A")]),t._v("将匹配属性"),a("code",[t._v("EVENT_A")]),t._v("")]),t._v(" "),a("h4",{attrs:{id:"事件表达式"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#事件表达式"}},[t._v("#")]),t._v(" 事件表达式")]),t._v(" "),a("p",[t._v("事件的表达式根对象的基类是"),a("code",[t._v("EventSecurityExpressionRoot")]),t._v("。它提供对"),a("code",[t._v("Message")]),t._v("对象的访问,该对象通过事件传递。"),a("code",[t._v("EventSecurityExpressionRoot")]),t._v("只有一种方法,下表对此进行了描述:")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th",[t._v("Expression")]),t._v(" "),a("th",[t._v("说明")])])]),t._v(" "),a("tbody",[a("tr",[a("td",[a("code",[t._v("hasEvent(Object event)")])]),t._v(" "),a("td",[t._v("如果事件与给定事件匹配,则返回"),a("code",[t._v("true")]),t._v("")])])])]),t._v(" "),a("h4",{attrs:{id:"转换属性"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#转换属性"}},[t._v("#")]),t._v(" 转换属性")]),t._v(" "),a("p",[t._v("当匹配转换源和目标时,可以分别使用"),a("code",[t._v("TRANSITION_SOURCE_")]),t._v(""),a("code",[t._v("TRANSITION_TARGET_")]),t._v("前缀。")]),t._v(" "),a("h4",{attrs:{id:"转换表达式"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#转换表达式"}},[t._v("#")]),t._v(" 转换表达式")]),t._v(" "),a("p",[t._v("用于转换的表达式根对象的基类是"),a("code",[t._v("DefaultWebSecurityExpressionHandler")]),t._v("。它提供了对"),a("code",[t._v("Transition")]),t._v("对象的访问,该对象被传递以进行转换更改。"),a("code",[t._v("TransitionSecurityExpressionRoot")]),t._v("有两个方法,下表对此进行了描述:")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th",[t._v("Expression")]),t._v(" "),a("th",[t._v("说明")])])]),t._v(" "),a("tbody",[a("tr",[a("td",[a("code",[t._v("hasSource(Object source)")])]),t._v(" "),a("td",[t._v("如果转换源匹配给定源,则返回"),a("code",[t._v("true")]),t._v("")])]),t._v(" "),a("tr",[a("td",[a("code",[t._v("hasTarget(Object target)")])]),t._v(" "),a("td",[t._v("如果转换目标与给定目标匹配,则返回"),a("code",[t._v("true")]),t._v("")])])])]),t._v(" "),a("h3",{attrs:{id:"理解安全"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#理解安全"}},[t._v("#")]),t._v(" 理解安全")]),t._v(" "),a("p",[t._v("本节提供了有关状态机中安全性如何工作的更详细信息。你可能真的不需要知道,但最好是透明的,而不是隐藏幕后发生的所有神奇的事情。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("只有当 Spring statemachine 运行在一个有围墙的"),a("br"),t._v("花园中时,安全性才有意义,该花园中的用户没有直接访问应用程序的权限,因此可以"),a("br"),t._v("修改 Spring 安全性的"),a("code",[t._v("SecurityContext")]),t._v("在本地线程中保持。"),a("br"),t._v("如果用户控制了 JVM,那么实际上根本就没有安全性"),a("br"),t._v("")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("安全性的集成点是使用["),a("code",[t._v("StateMachineInterceptor")]),t._v("](#sm-interceptor)创建的,如果启用了安全性,则会自动将其添加到状态机中。特定的类是"),a("code",[t._v("StateMachineSecurityInterceptor")]),t._v(",它拦截事件和转换。然后,该拦截器将查询 Spring Security 的"),a("code",[t._v("AccessDecisionManager")]),t._v(",以确定是否可以发送事件或是否可以执行转换。实际上,如果带有"),a("code",[t._v("AccessDecisionManager")]),t._v("的决定或投票导致异常,则该事件或转换将被拒绝。")]),t._v(" "),a("p",[t._v("由于 Spring Security 中"),a("code",[t._v("AccessDecisionManager")]),t._v("的工作方式,我们需要每个安全对象都有一个实例。这就是为什么对于事件和过渡有不同的管理人员的原因之一。在这种情况下,事件和转换是我们保护的不同的类对象。")]),t._v(" "),a("p",[t._v("默认情况下,对于事件,投票人("),a("code",[t._v("EventExpressionVoter")]),t._v(""),a("code",[t._v("EventVoter")]),t._v(",和"),a("code",[t._v("RoleVoter")]),t._v(")被添加到"),a("code",[t._v("AccessDecisionManager")]),t._v("中。")]),t._v(" "),a("p",[t._v("默认情况下,对于转换,投票人("),a("code",[t._v("TransitionExpressionVoter")]),t._v(""),a("code",[t._v("TransitionVoter")]),t._v(",和"),a("code",[t._v("RoleVoter")]),t._v(")被添加到"),a("br"),t._v("中。")]),t._v(" "),a("h2",{attrs:{id:"状态机错误处理"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态机错误处理"}},[t._v("#")]),t._v(" 状态机错误处理")]),t._v(" "),a("p",[t._v("如果状态机在状态转换逻辑期间检测到内部错误,则可能抛出异常。在内部处理此异常之前,你有机会进行拦截。")]),t._v(" "),a("p",[t._v("通常,你可以使用"),a("code",[t._v("StateMachineInterceptor")]),t._v("来拦截错误,下面的清单展示了一个例子:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("StateMachine<String, String> stateMachine;\n\nvoid addInterceptor() {\n\tstateMachine.getStateMachineAccessor()\n\t\t\t.doWithRegion(function ->\n\t\t\t\tfunction.addStateMachineInterceptor(new StateMachineInterceptorAdapter<String, String>() {\n\t\t\t\t\t@Override\n\t\t\t\t\tpublic Exception stateMachineError(StateMachine<String, String> stateMachine,\n\t\t\t\t\t\t\t\t\t\t\t\t\t   Exception exception) {\n\t\t\t\t\t\treturn exception;\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t\t);\n\n}\n")])])]),a("p",[t._v("当检测到错误时,将执行正常事件通知机制。这允许你使用"),a("code",[t._v("StateMachineListener")]),t._v("或 Spring 应用程序上下文事件侦听器。有关这些的更多信息,请参见"),a("a",{attrs:{href:"#sm-listeners"}},[t._v("监听状态机事件")]),t._v("")]),t._v(" "),a("p",[t._v("话虽如此,下面的示例展示了一个简单的侦听器:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class ErrorStateMachineListener\n\t\textends StateMachineListenerAdapter<String, String> {\n\n\t@Override\n\tpublic void stateMachineError(StateMachine<String, String> stateMachine, Exception exception) {\n\t\t// do something with error\n\t}\n}\n")])])]),a("p",[t._v("下面的示例显示了一个通用的"),a("code",[t._v("ApplicationListener")]),t._v("检查"),a("code",[t._v("StateMachineEvent")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class GenericApplicationEventListener\n\t\timplements ApplicationListener<StateMachineEvent> {\n\n\t@Override\n\tpublic void onApplicationEvent(StateMachineEvent event) {\n\t\tif (event instanceof OnStateMachineError) {\n\t\t\t// do something with error\n\t\t}\n\t}\n}\n")])])]),a("p",[t._v("你还可以直接定义"),a("code",[t._v("ApplicationListener")]),t._v("来只识别"),a("code",[t._v("StateMachineEvent")]),t._v("实例,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class ErrorApplicationEventListener\n\t\timplements ApplicationListener<OnStateMachineError> {\n\n\t@Override\n\tpublic void onApplicationEvent(OnStateMachineError event) {\n\t\t// do something with error\n\t}\n}\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("为转换定义的操作也有自己的错误处理"),a("br"),t._v("逻辑。见"),a("code",[t._v("StateMachineEvent")]),t._v("")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("有了反应性 API,就有可能从"),a("em",[t._v("Statemachineeversult")]),t._v("返回"),a("em",[t._v("行动")]),t._v("执行错误。具有简单的机器,其动作中的错误转换为状态"),a("code",[t._v("S1")]),t._v("")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class Config1 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states) throws Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("SI")\n\t\t\t\t.stateEntry("S1", (context) -> {\n\t\t\t\t\tthrow new RuntimeException("example error");\n\t\t\t\t});\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions) throws Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("SI")\n\t\t\t\t.target("S1")\n\t\t\t\t.event("E1");\n\t}\n}\n')])])]),a("p",[t._v("下面的测试概念显示了如何从"),a("em",[t._v("Statemachineeversult")]),t._v("消耗可能的错误。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nprivate StateMachine<String, String> machine;\n\n@Test\npublic void testActionEntryErrorWithEvent() throws Exception {\n\tStepVerifier.create(machine.startReactively()).verifyComplete();\n\tassertThat(machine.getState().getIds()).containsExactlyInAnyOrder("SI");\n\n\tStepVerifier.create(machine.sendEvent(Mono.just(MessageBuilder.withPayload("E1").build())))\n\t\t.consumeNextWith(result -> {\n\t\t\tStepVerifier.create(result.complete()).consumeErrorWith(e -> {\n\t\t\t\tassertThat(e).isInstanceOf(StateMachineException.class).hasMessageContaining("example error");\n\t\t\t}).verify();\n\t\t})\n\t\t.verifyComplete();\n\n\tassertThat(machine.getState().getIds()).containsExactlyInAnyOrder("S1");\n}\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("进入/退出操作中的错误不会阻止转换发生。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"状态机服务"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态机服务"}},[t._v("#")]),t._v(" 状态机服务")]),t._v(" "),a("p",[t._v("StateMachine 服务是更高级别的实现,旨在提供更多的用户级功能,以简化正常的运行时操作。目前,只存在一个服务接口("),a("code",[t._v("StateMachineEvent")]),t._v(")。")]),t._v(" "),a("h3",{attrs:{id:"使用statemachineservice"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statemachineservice"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateMachineService")])]),t._v(" "),a("p",[a("code",[t._v("StateMachineService")]),t._v("是一种接口,用于处理运行中的机器,并具有“获取”和“释放”机器的简单方法。它有一个默认的实现,名为"),a("code",[t._v("DefaultStateMachineService")]),t._v("")]),t._v(" "),a("h2",{attrs:{id:"any持久化状态机"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#any持久化状态机"}},[t._v("#")]),t._v(" "),a("code",[t._v("ANY")]),t._v("持久化状态机")]),t._v(" "),a("p",[t._v("传统上,状态机的一个实例是在一个运行中的程序中使用的。你可以通过使用动态构建器和工厂来实现更动态的行为,这允许按需进行状态机实例化。构建状态机的实例是一项相对繁重的操作。因此,如果你需要(例如)通过使用状态机来处理数据库中的任意状态更改,那么你需要找到一种更好、更快的方法来实现这一点。")]),t._v(" "),a("p",[t._v("持久化特性允许你将状态机的状态保存到外部存储库中,然后根据序列化状态重置状态机。例如,如果你有一个保存订单的数据库表,那么如果每次更改都需要构建一个新的实例,那么用状态机更新订单状态将是非常昂贵的。持久化特性允许你重置状态机状态,而无需实例化新的状态机实例。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("有一个配方(参见"),a("a",{attrs:{href:"#statemachine-recipes-persist"}},[t._v("Persist")]),t._v(")和一个样本"),a("br"),t._v("(参见"),a("a",{attrs:{href:"#statemachine-examples-persist"}},[t._v("Persist")]),t._v(")提供了有关"),a("br"),t._v("持久性状态的更多信息。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("虽然可以通过使用"),a("code",[t._v("StateMachineListener")]),t._v("构建自定义持久性特性,但它存在一个概念问题。当监听器通知状态更改时,状态更改已经发生。如果侦听器中的自定义持久方法无法更新外部存储库中的序列化状态,则状态机中的状态和外部存储库中的状态将处于不一致状态。")]),t._v(" "),a("p",[t._v("在状态机中的状态更改期间,你可以使用状态机拦截器尝试将序列化状态保存到外部存储中。如果此拦截器回调失败,你可以停止状态更改尝试,然后可以手动处理此错误,而不是以不一致的状态结束。有关如何使用拦截器,请参见[using"),a("code",[t._v("StateMachineInterceptor")]),t._v("]。")]),t._v(" "),a("h3",{attrs:{id:"使用statemachinecontext"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statemachinecontext"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateMachineContext")])]),t._v(" "),a("p",[t._v("使用普通的 Java 序列化不能持久保存"),a("code",[t._v("StateMachine")]),t._v(",因为对象图太丰富,并且包含太多对其他 Spring 上下文类的依赖关系。"),a("code",[t._v("StateMachineContext")]),t._v("是状态机的运行时表示形式,你可以使用它将现有机器恢复到由特定的"),a("code",[t._v("StateMachineContext")]),t._v("对象表示的状态。")]),t._v(" "),a("p",[a("code",[t._v("StateMachineContext")]),t._v("包含两种不同的方式来包含子上下文的信息。当机器包含正交区域时,通常使用这些方法。首先,上下文可以有一个子上下文列表,如果它们存在,则可以按原样使用。其次,你可以包括一个引用列表,如果未设置原始上下文子目录,则会使用这些引用。这些子引用实际上是在多个并行区域独立运行的机器中持久化的唯一方法。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[a("a",{attrs:{href:"#statemachine-examples-datajpamultipersist"}},[t._v("多数据持久化")]),t._v("示例显示了"),a("br"),t._v("如何持久化并行区域。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"使用statemachinepersister"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statemachinepersister"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateMachinePersister")])]),t._v(" "),a("p",[t._v("构建"),a("code",[t._v("StateMachineContext")]),t._v("然后从它恢复状态机,如果是手动完成的,这一直是有点“黑魔法”。"),a("code",[t._v("StateMachinePersister")]),t._v("接口旨在通过提供"),a("code",[t._v("persist")]),t._v(""),a("code",[t._v("restore")]),t._v("方法来简化这些操作。这个接口的默认实现是"),a("code",[t._v("DefaultStateMachinePersister")]),t._v("")]),t._v(" "),a("p",[t._v("我们可以通过跟踪测试中的片段来演示如何使用"),a("code",[t._v("StateMachinePersister")]),t._v("。我们首先为状态机创建两个类似的配置("),a("code",[t._v("machine1")]),t._v(""),a("code",[t._v("machine2")]),t._v(")。请注意,我们可以用其他方式为这个演示构建不同的机器,但这种方式适用于这种情况。下面的示例配置了两个状态机:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine(name = "machine1")\nstatic class Config1 extends Config {\n}\n\n@Configuration\n@EnableStateMachine(name = "machine2")\nstatic class Config2 extends Config {\n}\n\nstatic class Config extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states) throws Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.state("S1")\n\t\t\t\t.state("S2");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions) throws Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("S1")\n\t\t\t\t.target("S2")\n\t\t\t\t.event("E1");\n\t}\n}\n')])])]),a("p",[t._v("当我们使用"),a("code",[t._v("StateMachinePersist")]),t._v("对象时,我们可以创建一个内存中的实现。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("这个内存中的示例仅用于演示目的。对于真正的"),a("br"),t._v("应用程序,你应该使用真正的持久存储实现。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("下面的清单展示了如何使用内存中的示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("static class InMemoryStateMachinePersist implements StateMachinePersist<String, String, String> {\n\n\tprivate final HashMap<String, StateMachineContext<String, String>> contexts = new HashMap<>();\n\n\t@Override\n\tpublic void write(StateMachineContext<String, String> context, String contextObj) throws Exception {\n\t\tcontexts.put(contextObj, context);\n\t}\n\n\t@Override\n\tpublic StateMachineContext<String, String> read(String contextObj) throws Exception {\n\t\treturn contexts.get(contextObj);\n\t}\n}\n")])])]),a("p",[t._v("在实例化了这两台不同的机器之后,我们可以通过事件"),a("code",[t._v("E1")]),t._v(""),a("code",[t._v("S2")]),t._v("转换为状态"),a("code",[t._v("S2")]),t._v("。然后我们可以将其持久化并恢复"),a("code",[t._v("machine2")]),t._v("。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('InMemoryStateMachinePersist stateMachinePersist = new InMemoryStateMachinePersist();\nStateMachinePersister<String, String, String> persister = new DefaultStateMachinePersister<>(stateMachinePersist);\n\nStateMachine<String, String> stateMachine1 = context.getBean("machine1", StateMachine.class);\nStateMachine<String, String> stateMachine2 = context.getBean("machine2", StateMachine.class);\nstateMachine1.startReactively().block();\n\nstateMachine1\n\t.sendEvent(Mono.just(MessageBuilder\n\t\t.withPayload("E1").build()))\n\t.blockLast();\nassertThat(stateMachine1.getState().getIds()).containsExactly("S2");\n\npersister.persist(stateMachine1, "myid");\npersister.restore(stateMachine2, "myid");\nassertThat(stateMachine2.getState().getIds()).containsExactly("S2");\n')])])]),a("h3",{attrs:{id:"使用-redis"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用-redis"}},[t._v("#")]),t._v(" 使用 redis")]),t._v(" "),a("p",[a("code",[t._v("RepositoryStateMachinePersist")]),t._v("(实现"),a("code",[t._v("StateMachinePersist")]),t._v(")为将状态机持久化到 Redis 提供了支持。具体的实现是"),a("code",[t._v("RedisStateMachineContextRepository")]),t._v(",它使用"),a("code",[t._v("kryo")]),t._v("序列化将"),a("code",[t._v("StateMachineContext")]),t._v("持久化到"),a("code",[t._v("Redis")]),t._v("")]),t._v(" "),a("p",[t._v("对于"),a("code",[t._v("StateMachinePersister")]),t._v(",我们有一个与 Redis 相关的"),a("code",[t._v("RedisStateMachinePersister")]),t._v("实现,它接受一个"),a("code",[t._v("StateMachinePersist")]),t._v("的实例,并使用"),a("code",[t._v("String")]),t._v("作为其上下文对象。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("有关详细用法,请参见"),a("a",{attrs:{href:"#statemachine-examples-eventservice"}},[t._v("活动服务")]),t._v("示例。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[a("code",[t._v("RedisStateMachineContextRepository")]),t._v("需要一个"),a("code",[t._v("RedisConnectionFactory")]),t._v("才能使其工作。我们建议对它使用"),a("code",[t._v("JedisConnectionFactory")]),t._v(",如前面的示例所示。")]),t._v(" "),a("h3",{attrs:{id:"使用statemachineruntimepersister"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statemachineruntimepersister"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateMachineRuntimePersister")])]),t._v(" "),a("p",[a("code",[t._v("StateMachineRuntimePersister")]),t._v("是对"),a("code",[t._v("StateMachinePersist")]),t._v("的一个简单扩展,它添加了一个接口级方法来获得与其关联的"),a("code",[t._v("StateMachineInterceptor")]),t._v("。然后,需要此拦截器在状态更改期间持久化机器,而无需停止和启动机器。")]),t._v(" "),a("p",[t._v("目前,已有针对受支持的 Spring 数据存储库的该接口的实现。这些实现方式是"),a("code",[t._v("JpaPersistingStateMachineInterceptor")]),t._v(""),a("code",[t._v("MongoDbPersistingStateMachineInterceptor")]),t._v(""),a("code",[t._v("RedisPersistingStateMachineInterceptor")]),t._v("")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("有关详细用法,请参见"),a("a",{attrs:{href:"#statemachine-examples-datapersist"}},[t._v("数据持续存在")]),t._v("示例。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"spring-引导支持"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#spring-引导支持"}},[t._v("#")]),t._v(" Spring 引导支持")]),t._v(" "),a("p",[t._v("自动配置模块("),a("code",[t._v("spring-statemachine-autoconfigure")]),t._v(")包含与 Spring 引导集成的所有逻辑,该引导为自动配置和执行器提供了功能。你所需要的只是将这个 Spring StateMachine 库作为引导应用程序的一部分。")]),t._v(" "),a("h3",{attrs:{id:"监视和跟踪"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#监视和跟踪"}},[t._v("#")]),t._v(" 监视和跟踪")]),t._v(" "),a("p",[a("code",[t._v("BootStateMachineMonitor")]),t._v("将自动创建并与状态机关联。"),a("code",[t._v("BootStateMachineMonitor")]),t._v("是一个自定义"),a("code",[t._v("StateMachineMonitor")]),t._v("实现,它与 Spring boot 的"),a("code",[t._v("MeterRegistry")]),t._v("和通过自定义"),a("code",[t._v("StateMachineTraceRepository")]),t._v("的端点集成。你可以通过将"),a("code",[t._v("spring.statemachine.monitor.enabled")]),t._v("键设置为"),a("code",[t._v("StateMachineTraceRepository")]),t._v("来禁用此自动配置。"),a("a",{attrs:{href:"#statemachine-examples-monitoring"}},[t._v("Monitoring")]),t._v("示例展示了如何使用这种自动配置。")]),t._v(" "),a("h3",{attrs:{id:"存储库配置"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#存储库配置"}},[t._v("#")]),t._v(" 存储库配置")]),t._v(" "),a("p",[t._v("如果从 Classpath 中找到了所需的类,则 Spring 数据存储库和实体类扫描将自动为"),a("code",[t._v("StateMachineTraceRepository")]),t._v("自动配置。")]),t._v(" "),a("p",[t._v("当前支持的配置是"),a("code",[t._v("JPA")]),t._v(""),a("code",[t._v("Redis")]),t._v(""),a("code",[t._v("StateMachineTraceRepository")]),t._v("。可以分别使用"),a("code",[t._v("spring.statemachine.data.jpa.repositories.enabled")]),t._v(""),a("code",[t._v("spring.statemachine.data.redis.repositories.enabled")]),t._v(""),a("code",[t._v("spring.statemachine.data.mongo.repositories.enabled")]),t._v("属性禁用存储库自动配置。")]),t._v(" "),a("h2",{attrs:{id:"监视状态机"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#监视状态机"}},[t._v("#")]),t._v(" 监视状态机")]),t._v(" "),a("p",[t._v("你可以使用"),a("code",[t._v("StateMachineMonitor")]),t._v("来获取更多有关转换和执行操作所需的持续时间的信息。下面的清单展示了这个接口是如何实现的。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class TestStateMachineMonitor extends AbstractStateMachineMonitor<String, String> {\n\n\t@Override\n\tpublic void transition(StateMachine<String, String> stateMachine, Transition<String, String> transition,\n\t\t\tlong duration) {\n\t}\n\n\t@Override\n\tpublic void action(StateMachine<String, String> stateMachine,\n\t\t\tFunction<StateContext<String, String>, Mono<Void>> action, long duration) {\n\t}\n}\n")])])]),a("p",[t._v("有了"),a("code",[t._v("StateMachineMonitor")]),t._v("实现后,就可以通过配置将其添加到状态机中,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config1 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<String, String> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withMonitoring()\n\t\t\t\t.monitor(stateMachineMonitor());\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states) throws Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.state("S2");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions) throws Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("S1")\n\t\t\t\t.target("S2")\n\t\t\t\t.event("E1");\n\t}\n\n\t@Bean\n\tpublic StateMachineMonitor<String, String> stateMachineMonitor() {\n\t\treturn new TestStateMachineMonitor();\n\t}\n}\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("有关详细用法,请参见"),a("a",{attrs:{href:"#statemachine-examples-monitoring"}},[t._v("Monitoring")]),t._v("示例。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"使用分布状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用分布状态"}},[t._v("#")]),t._v(" 使用分布状态")]),t._v(" "),a("p",[t._v("分布式状态可能是 Spring 状态机中最复杂的概念之一。什么是分布式状态?单个状态机中的状态很容易理解,但是,当需要通过状态机引入共享的分布式状态时,事情就变得有点复杂了。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("分布式状态功能仍然是预览功能,并不"),a("br"),t._v(",但在此特定版本中被认为是稳定的。我们预计这一"),a("br"),t._v("功能将在其首次正式发布时成熟。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("有关通用配置支持的信息,请参见"),a("a",{attrs:{href:"#statemachine-config-commonsettings"}},[t._v("配置公共设置")]),t._v("。有关实际使用示例,请参见"),a("a",{attrs:{href:"#statemachine-examples-zookeeper"}},[t._v("Zookeeper")]),t._v("示例。")]),t._v(" "),a("p",[t._v("分布式状态机是通过"),a("code",[t._v("DistributedStateMachine")]),t._v("类实现的,该类封装了"),a("code",[t._v("StateMachine")]),t._v("的实际实例。"),a("code",[t._v("DistributedStateMachine")]),t._v("拦截与"),a("code",[t._v("StateMachine")]),t._v("实例的通信,并通过"),a("code",[t._v("StateMachineEnsemble")]),t._v("接口处理分布式状态抽象。根据实际的实现,还可以使用"),a("code",[t._v("StateMachinePersist")]),t._v("接口序列化"),a("code",[t._v("StateMachineContext")]),t._v(",其中包含足够的信息来重置"),a("code",[t._v("StateMachine")]),t._v("")]),t._v(" "),a("p",[t._v("虽然分布式状态机是通过抽象实现的,但目前只存在一个实现。它是以 ZooKeeper 为基础的。")]),t._v(" "),a("p",[t._v("下面的示例展示了如何配置基于 ZooKeeper 的分布式状态机 `:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic class Config\n\t\textends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<String, String> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withDistributed()\n\t\t\t\t.ensemble(stateMachineEnsemble())\n\t\t\t\t.and()\n\t\t\t.withConfiguration()\n\t\t\t\t.autoStartup(true);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\t// config states\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\t// config transitions\n\t}\n\n\t@Bean\n\tpublic StateMachineEnsemble<String, String> stateMachineEnsemble()\n\t\t\tthrows Exception {\n\t\treturn new ZookeeperStateMachineEnsemble<String, String>(curatorClient(), "/zkpath");\n\t}\n\n\t@Bean\n\tpublic CuratorFramework curatorClient()\n\t\t\tthrows Exception {\n\t\tCuratorFramework client = CuratorFrameworkFactory\n\t\t\t\t.builder()\n\t\t\t\t.defaultData(new byte[0])\n\t\t\t\t.connectString("localhost:2181").build();\n\t\tclient.start();\n\t\treturn client;\n\t}\n\n}\n')])])]),a("p",[t._v("你可以找到基于 Zookeeker 的分布式状态机"),a("a",{attrs:{href:"#appendices-zookeeper"}},[t._v("在附录中")]),t._v("的当前技术文档。")]),t._v(" "),a("h3",{attrs:{id:"statemachine使用zookeeperstatemachineensemble"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#statemachine使用zookeeperstatemachineensemble"}},[t._v("#")]),t._v(" "),a("code",[t._v("StateMachine")]),t._v("使用"),a("code",[t._v("ZookeeperStateMachineEnsemble")])]),t._v(" "),a("p",[a("code",[t._v("ZookeeperStateMachineEnsemble")]),t._v("本身需要两个强制设置,一个"),a("code",[t._v("curatorClient")]),t._v("的实例和一个"),a("code",[t._v("basePath")]),t._v("的实例。客户端是"),a("code",[t._v("CuratorFramework")]),t._v(",路径是"),a("code",[t._v("Zookeeper")]),t._v("实例中树的根。")]),t._v(" "),a("p",[t._v("可选地,你可以设置"),a("code",[t._v("cleanState")]),t._v(",它默认设置为"),a("code",[t._v("TRUE")]),t._v(",如果集成中没有成员,则清除现有数据。如果希望在应用程序重新启动期间保持分布式状态,可以将其设置为"),a("code",[t._v("FALSE")]),t._v("")]),t._v(" "),a("p",[t._v("可选地,你可以将"),a("code",[t._v("logSize")]),t._v("(默认为"),a("code",[t._v("32")]),t._v(")的大小设置为状态更改的 keep 历史记录。这个设置的值必须是 2 的幂。"),a("code",[t._v("curatorClient")]),t._v("通常是一个很好的默认值。如果某个特定状态机的滞后时间超过了日志的大小,那么它就会进入错误状态,并与集合断开连接,这表明它已经失去了历史记录,也失去了完全重建同步状态的能力。")]),t._v(" "),a("h2",{attrs:{id:"测试支持"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#测试支持"}},[t._v("#")]),t._v(" 测试支持")]),t._v(" "),a("p",[t._v("我们还添加了一组实用程序类,以方便对状态机实例的测试。这些在框架中使用,但对最终用户也非常有用。")]),t._v(" "),a("p",[a("code",[t._v("StateMachineTestPlanBuilder")]),t._v("构建了一个"),a("code",[t._v("StateMachineTestPlan")]),t._v(",它有一个方法(称为"),a("code",[t._v("test()")]),t._v(")。该方法运行一个计划。"),a("code",[t._v("StateMachineTestPlanBuilder")]),t._v("包含一个 Fluent Builder API,可以让你为计划添加步骤。在这些步骤中,你可以发送事件并检查各种条件,例如状态更改、转换和扩展的状态变量。")]),t._v(" "),a("p",[t._v("下面的示例使用"),a("code",[t._v("StateMachineBuilder")]),t._v("构建状态机:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('private StateMachine<String, String> buildMachine() throws Exception {\n\tStateMachineBuilder.Builder<String, String> builder = StateMachineBuilder.builder();\n\n\tbuilder.configureConfiguration()\n\t\t.withConfiguration()\n\t\t\t.autoStartup(true);\n\n\tbuilder.configureStates()\n\t\t\t.withStates()\n\t\t\t\t.initial("SI")\n\t\t\t\t.state("S1");\n\n\tbuilder.configureTransitions()\n\t\t\t.withExternal()\n\t\t\t\t.source("SI").target("S1")\n\t\t\t\t.event("E1")\n\t\t\t\t.action(c -> {\n\t\t\t\t\tc.getExtendedState().getVariables().put("key1", "value1");\n\t\t\t\t});\n\n\treturn builder.build();\n}\n')])])]),a("p",[t._v("在下面的测试计划中,我们有两个步骤。首先,我们检查初始状态("),a("code",[t._v("SI")]),t._v(")是否已设置。其次,我们发送一个事件("),a("code",[t._v("E1")]),t._v("),并期望发生一个状态更改,并期望机器以"),a("code",[t._v("S1")]),t._v("的状态结束。下面的清单显示了测试计划:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('StateMachine<String, String> machine = buildMachine();\nStateMachineTestPlan<String, String> plan =\n\t\tStateMachineTestPlanBuilder.<String, String>builder()\n\t\t\t.defaultAwaitTime(2)\n\t\t\t.stateMachine(machine)\n\t\t\t.step()\n\t\t\t\t.expectStates("SI")\n\t\t\t\t.and()\n\t\t\t.step()\n\t\t\t\t.sendEvent("E1")\n\t\t\t\t.expectStateChanged(1)\n\t\t\t\t.expectStates("S1")\n\t\t\t\t.expectVariable("key1")\n\t\t\t\t.expectVariable("key1", "value1")\n\t\t\t\t.expectVariableWith(hasKey("key1"))\n\t\t\t\t.expectVariableWith(hasValue("value1"))\n\t\t\t\t.expectVariableWith(hasEntry("key1", "value1"))\n\t\t\t\t.expectVariableWith(not(hasKey("key2")))\n\t\t\t\t.and()\n\t\t\t.build();\nplan.test();\n')])])]),a("p",[t._v("这些实用程序还在一个框架中用于测试分布式状态机的特性。请注意,你可以在一个计划中添加多台机器。如果你添加了多台机器,Yuo 还可以选择发送特定机器、随机机器或所有机器的事件。")]),t._v(" "),a("p",[t._v("前面的测试示例使用了以下 Hamcrest 导入:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("import static org.hamcrest.CoreMatchers.not;\nimport static org.hamcrest.collection.IsMapContaining.hasKey;\nimport static org.hamcrest.collection.IsMapContaining.hasValue;\n\nimport org.junit.jupiter.api.Test;\n\nimport static org.hamcrest.collection.IsMapContaining.hasEntry;\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("预期结果的所有可能选项都记录在 Javadoc 中["),a("code",[t._v("StateMachineTestPlanStepBuilder")]),t._v("](https://DOCS. Spring.io/ Spring-statemachine/DOCS/3.0.1/api/org/springframework/statemachine/test/statemachinetplanbuilder.statemachinetestplanstepbuilder.html)。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"eclipse-建模支持"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#eclipse-建模支持"}},[t._v("#")]),t._v(" Eclipse 建模支持")]),t._v(" "),a("p",[t._v("Eclipse Papyrus 框架支持使用 UI 建模定义状态机配置。")]),t._v(" "),a("p",[t._v("在 Eclipse 向导中,你可以使用 UML 图语言创建一个新的 Papyrus 模型。在这个示例中,它被命名为"),a("code",[t._v("simple-machine")]),t._v("。然后,你可以从各种类型的图中进行选择,并且你必须选择"),a("code",[t._v("StateMachine Diagram")]),t._v("")]),t._v(" "),a("p",[t._v("我们希望创建一个具有两个状态("),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v(")的机器,其中"),a("code",[t._v("S1")]),t._v("是初始状态。然后,我们需要创建事件"),a("code",[t._v("DefaultStateMachinePersister")]),t._v("来执行从"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v("的转换。在纸莎草纸中,机器看起来就像是下面的例子:")]),t._v(" "),a("img",{attrs:{alt:"simple machine",src:"images/simple-machine.png",width:"500"}}),t._v(" "),a("p",[t._v("在幕后,一个原始的 UML 文件将看起来像以下示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('<?xml version="1.0" encoding="UTF-8"?>\n<uml:Model xmi:version="20131001" xmlns:xmi="http://www.omg.org/spec/XMI/20131001" xmlns:uml="http://www.eclipse.org/uml2/5.0.0/UML" xmi:id="_AMP3IP8fEeW45bORGB4c_A" name="RootElement">\n  <packagedElement xmi:type="uml:StateMachine" xmi:id="_AMRFQP8fEeW45bORGB4c_A" name="StateMachine">\n    <region xmi:type="uml:Region" xmi:id="_AMRsUP8fEeW45bORGB4c_A" name="Region1">\n      <transition xmi:type="uml:Transition" xmi:id="_chgcgP8fEeW45bORGB4c_A" source="_EZrg4P8fEeW45bORGB4c_A" target="_FAvg4P8fEeW45bORGB4c_A">\n        <trigger xmi:type="uml:Trigger" xmi:id="_hs5jUP8fEeW45bORGB4c_A" event="_NeH84P8fEeW45bORGB4c_A"/>\n      </transition>\n      <transition xmi:type="uml:Transition" xmi:id="_egLIoP8fEeW45bORGB4c_A" source="_Fg0IEP8fEeW45bORGB4c_A" target="_EZrg4P8fEeW45bORGB4c_A"/>\n      <subvertex xmi:type="uml:State" xmi:id="_EZrg4P8fEeW45bORGB4c_A" name="S1"/>\n      <subvertex xmi:type="uml:State" xmi:id="_FAvg4P8fEeW45bORGB4c_A" name="S2"/>\n      <subvertex xmi:type="uml:Pseudostate" xmi:id="_Fg0IEP8fEeW45bORGB4c_A"/>\n    </region>\n  </packagedElement>\n  <packagedElement xmi:type="uml:Signal" xmi:id="_L01D0P8fEeW45bORGB4c_A" name="E1"/>\n  <packagedElement xmi:type="uml:SignalEvent" xmi:id="_NeH84P8fEeW45bORGB4c_A" name="SignalEventE1" signal="_L01D0P8fEeW45bORGB4c_A"/>\n</uml:Model>\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("在打开已定义为 UML 的现有模型时,你有三个"),a("br"),t._v("文件:"),a("code",[t._v(".di")]),t._v(""),a("code",[t._v(".notation")]),t._v(""),a("code",[t._v(".uml")]),t._v("。如果模型不是在你的"),a("br"),t._v("Eclipse 的会话中创建的,则它不了解如何打开实际状态"),a("br"),t._v("图表。这是 Papyrus 插件中的一个已知问题,并且有一个简单的"),a("br"),t._v("解决方法。在 Papyrus 透视图中,你可以看到"),a("br"),t._v("你的模型的模型资源管理器。双击 Diagram Statemachine Diagram,其中"),a("br"),t._v("指示 Eclipse 在其正确的 Papyrus"),a("br"),t._v("建模插件中打开此特定模型。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"使用umlstatemachinemodelfactory"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用umlstatemachinemodelfactory"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("UmlStateMachineModelFactory")])]),t._v(" "),a("p",[t._v("在项目中安装好 UML 文件之后,你可以使用"),a("code",[t._v("StateMachineModelConfigurer")]),t._v("将其导入到配置中,其中"),a("code",[t._v("StateMachineModelFactory")]),t._v("与模型相关联。"),a("code",[t._v("UmlStateMachineModelFactory")]),t._v("是一个特殊的工厂,它知道如何处理 Eclipse Papyrus_ 生成的 UML 结构。源 UML 文件既可以作为 Spring "),a("code",[t._v("Resource")]),t._v("给出,也可以作为普通的位置字符串给出。下面的示例展示了如何创建"),a("code",[t._v("UmlStateMachineModelFactory")]),t._v("的实例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic static class Config1 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineModelConfigurer<String, String> model) throws Exception {\n\t\tmodel\n\t\t\t.withModel()\n\t\t\t\t.factory(modelFactory());\n\t}\n\n\t@Bean\n\tpublic StateMachineModelFactory<String, String> modelFactory() {\n\t\treturn new UmlStateMachineModelFactory("classpath:org/springframework/statemachine/uml/docs/simple-machine.uml");\n\t}\n}\n')])])]),a("p",[t._v("像往常一样, Spring Statemachine 与守卫和动作一起工作,而守卫和动作被定义为 bean。这些都需要通过 UML 的内部建模结构来连接到 UML 中。下面的小节将展示如何在 UML 定义中定义定制的 Bean 引用。请注意,也可以手动注册特定的方法,而无需将这些方法定义为 bean。")]),t._v(" "),a("p",[t._v("如果"),a("code",[t._v("UmlStateMachineModelFactory")]),t._v("被创建为 Bean,则其"),a("code",[t._v("ResourceLoader")]),t._v("将自动连接以查找已注册的动作和保护。你还可以手动定义"),a("code",[t._v("StateMachineComponentResolver")]),t._v(",然后用它来查找这些组件。工厂还有"),a("em",[t._v("注册动作")]),t._v(""),a("em",[t._v("register守卫")]),t._v(" 方法,你可以使用它们来注册这些组件。有关此的更多信息,请参见[using"),a("code",[t._v("StateMachineComponentResolver")]),t._v("]。")]),t._v(" "),a("p",[t._v("当涉及到 Spring StateMachine 本身这样的实现时,UML 模型是相对松散的。 Spring StateMachine 将如何实现大量的特性和功能留给实际的实现。下面的部分将介绍 Spring StateMachine 如何基于 Eclipse Papyrus 插件实现 UML 模型。")]),t._v(" "),a("h4",{attrs:{id:"使用statemachinecomponentresolver"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用statemachinecomponentresolver"}},[t._v("#")]),t._v(" 使用"),a("code",[t._v("StateMachineComponentResolver")])]),t._v(" "),a("p",[t._v("下一个示例显示如何用"),a("code",[t._v("UmlStateMachineModelFactory")]),t._v("定义"),a("code",[t._v("StateMachineComponentResolver")]),t._v(",该函数分别注册"),a("code",[t._v("my行动")]),t._v(""),a("code",[t._v("myGuard")]),t._v("函数。请注意,这些组件不是作为 bean 创建的。下面的清单展示了这个示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic static class Config2 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineModelConfigurer<String, String> model) throws Exception {\n\t\tmodel\n\t\t\t.withModel()\n\t\t\t\t.factory(modelFactory());\n\t}\n\n\t@Bean\n\tpublic StateMachineModelFactory<String, String> modelFactory() {\n\t\tUmlStateMachineModelFactory factory = new UmlStateMachineModelFactory(\n\t\t\t\t"classpath:org/springframework/statemachine/uml/docs/simple-machine.uml");\n\t\tfactory.setStateMachineComponentResolver(stateMachineComponentResolver());\n\t\treturn factory;\n\t}\n\n\t@Bean\n\tpublic StateMachineComponentResolver<String, String> stateMachineComponentResolver() {\n\t\tDefaultStateMachineComponentResolver<String, String> resolver = new DefaultStateMachineComponentResolver<>();\n\t\tresolver.registerAction("myAction", myAction());\n\t\tresolver.registerGuard("myGuard", myGuard());\n\t\treturn resolver;\n\t}\n\n\tpublic Action<String, String> myAction() {\n\t\treturn new Action<String, String>() {\n\n\t\t\t@Override\n\t\t\tpublic void execute(StateContext<String, String> context) {\n\t\t\t}\n\t\t};\n\t}\n\n\tpublic Guard<String, String> myGuard() {\n\t\treturn new Guard<String, String>() {\n\n\t\t\t@Override\n\t\t\tpublic boolean evaluate(StateContext<String, String> context) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t};\n\t}\n}\n')])])]),a("h3",{attrs:{id:"创建一个模型"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#创建一个模型"}},[t._v("#")]),t._v(" 创建一个模型")]),t._v(" "),a("p",[t._v("我们首先创建一个空的状态机模型,如下图所示:")]),t._v(" "),a("img",{attrs:{alt:"papyrus gs 1",src:"images/papyrus-gs-1.png",width:"300"}}),t._v(" "),a("p",[t._v("你可以从创建一个新模型并为其命名开始,如下图所示:")]),t._v(" "),a("img",{attrs:{alt:"papyrus gs 2",src:"images/papyrus-gs-2.png",width:"300"}}),t._v(" "),a("p",[t._v("然后,你需要选择静态机械图,如下所示:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-3.png",alt:"莎草纸 GS3"}})]),t._v(" "),a("p",[t._v("你最终得到的是一个空的状态机。")]),t._v(" "),a("p",[t._v("在前面的图片中,你应该创建了一个名为"),a("code",[t._v("model")]),t._v("的示例。你应该已经结束了三个文件:"),a("code",[t._v("model.di")]),t._v(""),a("code",[t._v("model.notation")]),t._v(""),a("code",[t._v("model.uml")]),t._v("。然后,你可以在任何其他 Eclipse 实例中使用这些文件。此外,还可以将"),a("code",[t._v("model.uml")]),t._v("导入 Spring Statemachine。")]),t._v(" "),a("h3",{attrs:{id:"定义状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义状态"}},[t._v("#")]),t._v(" 定义状态")]),t._v(" "),a("p",[t._v("状态标识符来自图中的组件名称。你的机器中必须有一个初始状态,你可以通过添加一个根元素,然后绘制到你自己的初始状态的转换,如下图所示:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-4.png",alt:"莎草纸 GS4"}})]),t._v(" "),a("p",[t._v("在前面的图像中,我们添加了一个根元素和一个初始状态("),a("img",{attrs:{src:"images/papyrus-gs-4.png",alt:"莎草纸 GS4"}}),t._v(")。然后我们在这两者之间画了一个转换,以表明"),a("code",[t._v("S1")]),t._v("是一个初始状态。")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-5.png",alt:"莎草纸 GS5"}})]),t._v(" "),a("p",[t._v("在前面的图像中,我们添加了第二个状态("),a("code",[t._v("S2")]),t._v("),并添加了 S1 和 S2 之间的转换(表明我们有两个状态)。")]),t._v(" "),a("h3",{attrs:{id:"定义事件"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义事件"}},[t._v("#")]),t._v(" 定义事件")]),t._v(" "),a("p",[t._v("要将事件与转换关联起来,需要创建一个信号(在本例中为"),a("code",[t._v("E1")]),t._v(")。要这样做,请选择 rootelement new child signal。下图显示了结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-6.png",alt:"莎草纸 GS6"}})]),t._v(" "),a("p",[t._v("然后,你需要用新的信号"),a("code",[t._v("E1")]),t._v("来控制一个信号事件。为此,请选择 rootelement New Child SignalEvent。下图显示了结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-7.png",alt:"莎草纸 GS7"}})]),t._v(" "),a("p",[t._v("既然已经定义了"),a("code",[t._v("SignalEvent")]),t._v(",就可以使用它将触发器与转换关联起来。更多信息请参见"),a("a",{attrs:{href:"#sm-papyrus-transitions"}},[t._v("定义转换")]),t._v("")]),t._v(" "),a("h4",{attrs:{id:"推迟一项活动"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#推迟一项活动"}},[t._v("#")]),t._v(" 推迟一项活动")]),t._v(" "),a("p",[t._v("你可以将事件推迟到更合适的时间来处理它们。在 UML 中,这是从一个状态本身完成的。选择任何状态,在"),a("img",{attrs:{src:"images/papyrus-gs-10.png",alt:"莎草纸 GS10"}}),t._v("下创建一个新的触发器,并选择与要延迟的信号匹配的 SignalEvent。")]),t._v(" "),a("h3",{attrs:{id:"定义转换"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义转换"}},[t._v("#")]),t._v(" 定义转换")]),t._v(" "),a("p",[t._v("你可以通过在源状态和目标状态之间画一条转换线来创建转换。在前面的图像中,我们有状态"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v("以及两者之间的匿名转换。我们希望将事件"),a("code",[t._v("E1")]),t._v("与该转换关联起来。我们选择一个转换,创建一个新的触发器,并为此定义 SignalEvente1,如下图所示:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-8.png",alt:"莎草纸 GS8"}})]),t._v(" "),a("p",[t._v("这给出了类似于下图所示的安排:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-9.png",alt:"莎草纸 GS9"}})]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("如果在转换中省略了 SignalEvent,那么它将变成"),a("br"),t._v("匿名转换。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"定时器"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定时器"}},[t._v("#")]),t._v(" 定时器")]),t._v(" "),a("p",[t._v("转换也可以基于定时事件发生。 Spring Statemachine 支持两种类型的计时器,一种是在背景上连续触发的计时器,另一种是在进入状态时延迟触发一次的计时器。")]),t._v(" "),a("p",[t._v("若要向 Model Explorer 添加一个新的 TimeEvent 子程序,请在将表达式定义为 literalInteger 时进行修改。它的值(以毫秒为单位)成为计时器。离开是相对错误的,以使计时器连续射击。")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-10.png",alt:"莎草纸 GS10"}})]),t._v(" "),a("p",[t._v("要定义一个基于时间的事件,在进入状态时触发,该过程与前面描述的完全相同,但 Leave 相对设置为 true。下图显示了结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-11.png",alt:"莎草纸 GS11"}})]),t._v(" "),a("p",[t._v("然后,用户可以选择这些定时事件中的一个,而不是用于特定转换的信号事件。")]),t._v(" "),a("h3",{attrs:{id:"定义一个选择"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义一个选择"}},[t._v("#")]),t._v(" 定义一个选择")]),t._v(" "),a("p",[t._v("选择是通过将一个传入的转换绘制到一个选择状态,并将多个传出的转换从它绘制到目标状态来定义的。我们"),a("code",[t._v("StateConfigurer")]),t._v("中的配置模型允许你定义一个 if/elseif/else 结构。然而,对于 UML,我们需要使用单独的保护来进行传出转换。")]),t._v(" "),a("p",[t._v("你必须确保为转换定义的保护不会重叠,因此,无论发生什么情况,在任何给定的时间都只有一个保护的值为 true。这为选择分支评估提供了精确且可预测的结果。我们还建议在没有保护的情况下保留一次转换,以便保证至少有一个转换路径。下图显示了用三个分支进行选择的结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-16.png",alt:"莎草纸 GS16"}})]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("Junction 的工作原理类似,只是它允许多次传入"),a("br"),t._v("转换。因此,其行为与选择相比纯粹是"),a("br"),t._v("学术性的。选择传出转换的实际逻辑完全相同。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"定义交点"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义交点"}},[t._v("#")]),t._v(" 定义交点")]),t._v(" "),a("p",[t._v(""),a("a",{attrs:{href:"#sm-papyrus-choice"}},[t._v("定义一个选择")]),t._v("")]),t._v(" "),a("h3",{attrs:{id:"定义进场点和出场点"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义进场点和出场点"}},[t._v("#")]),t._v(" 定义进场点和出场点")]),t._v(" "),a("p",[t._v("你可以使用 EntryPoint 和 ExitPoint 来创建带有子状态的状态的受控入口和出口。在下面的状态图中,事件"),a("code",[t._v("E1")]),t._v(""),a("code",[t._v("E2")]),t._v("通过进入和退出状态"),a("code",[t._v("S2")]),t._v("而具有正常状态行为,其中正常状态行为是通过进入初始状态"),a("code",[t._v("S21")]),t._v("而发生的。")]),t._v(" "),a("p",[t._v("使用事件"),a("code",[t._v("E3")]),t._v("将机器带到"),a("code",[t._v("ENTRY")]),t._v("入口点,然后在任何时候都不激活初始状态"),a("code",[t._v("S22")]),t._v("。类似地,带有事件"),a("code",[t._v("StateMachineBuilder")]),t._v("的 exitPoint 控制特定的 exit into state"),a("code",[t._v("S4")]),t._v(",而来自"),a("code",[t._v("S2")]),t._v("的正常 exit 行为将使机器进入状态"),a("code",[t._v("S3")]),t._v("。在状态"),a("code",[t._v("S22")]),t._v("时,你可以从事件"),a("code",[t._v("E4")]),t._v(""),a("code",[t._v("E2")]),t._v("中进行选择,以使机器分别进入状态"),a("code",[t._v("S3")]),t._v(""),a("code",[t._v("S4")]),t._v("。下图显示了结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-17.png",alt:"莎草纸 GS17"}})]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("如果状态被定义为一个子机引用,并且需要使用入口和出口点,"),a("br"),t._v("你必须在外部定义一个 ConnectionPointReference,其入口和出口引用设置为指向一个子机引用内的正确的入口或出口点"),a("br"),t._v("。只有在此之后,才有可能实现"),a("br"),t._v("目标转换,从而正确地将"),a("br"),t._v("子机器引用从外部链接到内部。使用 ConnectionPointReference,你可能需要"),a("br"),t._v("才能从 Properties Advanced UML"),a("br"),t._v("Enter/Exit 中找到这些设置。UML 规范允许你定义多个条目和出口。但是,"),a("br"),t._v("使用状态机时,只允许使用一个状态机。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h3",{attrs:{id:"定义历史状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义历史状态"}},[t._v("#")]),t._v(" 定义历史状态")]),t._v(" "),a("p",[t._v("在研究历史国家时,有三个不同的概念在起作用。UML 定义了一段深刻的历史和一段浅薄的历史。当历史状态未知时,默认的历史状态开始起作用。以下各节对此作了说明。")]),t._v(" "),a("h4",{attrs:{id:"浅历史"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#浅历史"}},[t._v("#")]),t._v(" 浅历史")]),t._v(" "),a("p",[t._v("在下面的图像中,选择了浅历史,并定义了一个过渡到它:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-18.png",alt:"莎草纸 GS18"}})]),t._v(" "),a("h4",{attrs:{id:"深历史"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#深历史"}},[t._v("#")]),t._v(" 深历史")]),t._v(" "),a("p",[t._v("深度历史被用来表示具有其他深度嵌套状态的状态,从而为保存整个嵌套状态结构提供了机会。下图显示了一个使用了深层历史的定义:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-19.png",alt:"Papyrus GS19"}})]),t._v(" "),a("h4",{attrs:{id:"默认历史记录"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#默认历史记录"}},[t._v("#")]),t._v(" 默认历史记录")]),t._v(" "),a("p",[t._v("如果在某个历史记录上的转换在达到其最终状态之前没有被输入,则该历史记录上的转换就会终止,那么可以使用默认的历史记录机制强制转换到特定子状态。要做到这一点,你必须定义一个转换到这个默认状态。这是从"),a("code",[t._v("SH")]),t._v(""),a("code",[t._v("S22")]),t._v("的转换。")]),t._v(" "),a("p",[t._v("在下面的图像中,如果状态"),a("code",[t._v("S22")]),t._v("从未被激活,则输入状态"),a("code",[t._v("S2")]),t._v(",因为其历史从未被记录。如果状态"),a("code",[t._v("S2")]),t._v("一直处于活动状态,则选择"),a("code",[t._v("S20")]),t._v(""),a("code",[t._v("S21")]),t._v("")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-20.png",alt:"莎草纸 GS20"}})]),t._v(" "),a("h3",{attrs:{id:"定义-fork-和-join"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义-fork-和-join"}},[t._v("#")]),t._v(" 定义 fork 和 join")]),t._v(" "),a("p",[t._v("在纸莎草纸中,fork 和 join 都表示为 bar。如下一张图片所示,需要绘制一个从"),a("code",[t._v("FORK")]),t._v(""),a("code",[t._v("S2")]),t._v("状态的输出转换,以具有正交区域。"),a("code",[t._v("JOIN")]),t._v("则是相反的,在这种情况下,从传入的转换中收集连接的状态。")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-21.png",alt:"Papyrus GS21"}})]),t._v(" "),a("h3",{attrs:{id:"定义动作"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义动作"}},[t._v("#")]),t._v(" 定义动作")]),t._v(" "),a("p",[t._v("你可以通过使用一个行为来处理 SWTATE 进入和退出操作。有关此的更多信息,请参见"),a("a",{attrs:{href:"#sm-papyrus-beanref"}},[t._v("Defining a Bean Reference")]),t._v("")]),t._v(" "),a("h4",{attrs:{id:"使用初始操作"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用初始操作"}},[t._v("#")]),t._v(" 使用初始操作")]),t._v(" "),a("p",[t._v("在 UML 中定义了一个初始动作(如"),a("a",{attrs:{href:"#statemachine-config-actions"}},[t._v("配置动作")]),t._v("中所示),在转换过程中添加一个动作,该动作将从初始状态标记引导到实际状态。然后在状态机启动时运行该操作。")]),t._v(" "),a("h3",{attrs:{id:"定义守卫"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义守卫"}},[t._v("#")]),t._v(" 定义守卫")]),t._v(" "),a("p",[t._v("你可以通过首先添加一个约束,然后将其规范定义为 opaqueExpression 来定义一个 Guard,其工作方式与"),a("a",{attrs:{href:"#sm-papyrus-beanref"}},[t._v("Defining a Bean Reference")]),t._v("相同。")]),t._v(" "),a("h3",{attrs:{id:"定义-bean-引用"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义-bean-引用"}},[t._v("#")]),t._v(" 定义 Bean 引用")]),t._v(" "),a("p",[t._v("当需要在任何 UML 效果、操作或保护中进行 Bean 引用时,可以使用"),a("code",[t._v("FunctionBehavior")]),t._v(""),a("code",[t._v("OpaqueBehavior")]),t._v("进行引用,其中定义的语言需要是"),a("code",[t._v("bean")]),t._v(",并且语言体 msut 具有 Bean 引用 ID。")]),t._v(" "),a("h3",{attrs:{id:"定义-spel-引用"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#定义-spel-引用"}},[t._v("#")]),t._v(" 定义 spel 引用")]),t._v(" "),a("p",[t._v("当你需要在任何 UML 效果、操作或保护中使用 spel 表达式而不是 Bean 引用时,你可以通过使用"),a("code",[t._v("FunctionBehavior")]),t._v(""),a("code",[t._v("OpaqueBehavior")]),t._v("来实现,其中定义的语言需要是"),a("code",[t._v("spel")]),t._v(",并且语言主体必须是 spel 表达式。")]),t._v(" "),a("h3",{attrs:{id:"使用子机器引用"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用子机器引用"}},[t._v("#")]),t._v(" 使用子机器引用")]),t._v(" "),a("p",[t._v("通常,当你使用子状态时,你会将这些状态绘制到状态图中。图表可能会变得太复杂、太大而无法遵循,因此我们也支持将子状态定义为状态机引用。")]),t._v(" "),a("p",[t._v("要创建子机器引用,你必须首先创建一个新的图,并给它起一个名称(例如,substatemachine 图)。下图显示了要使用的菜单选择:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-12.png",alt:"莎草纸 GS12"}})]),t._v(" "),a("p",[t._v("给这张新的图表你所需要的图案。下面的图片显示了一个简单的设计作为示例:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-13.png",alt:"莎草纸 GS13"}})]),t._v(" "),a("p",[t._v("从要链接的状态(在本例中,M 状态"),a("code",[t._v("S2")]),t._v(")中,单击"),a("code",[t._v("Submachine")]),t._v("字段并选择链接的机器(在我们的示例中,"),a("code",[t._v("SubStateMachine")]),t._v(")。")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-14.png",alt:"莎草纸 GS14"}})]),t._v(" "),a("p",[t._v("最后,在下面的图像中,可以看到状态"),a("code",[t._v("S2")]),t._v("作为子状态链接到"),a("code",[t._v("SubStateMachine")]),t._v("")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-15.png",alt:"莎草纸 GS15"}})]),t._v(" "),a("h3",{attrs:{id:"使用机器导入"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#使用机器导入"}},[t._v("#")]),t._v(" 使用机器导入")]),t._v(" "),a("p",[t._v("在 UML 文件可以引用其他模型的情况下,也可以使用导入功能。")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/papyrus-gs-22.png",alt:"莎草纸 GS22"}})]),t._v(" "),a("p",[t._v(""),a("code",[t._v("UmlStateMachineModelFactory")]),t._v("中,可以使用额外的资源或位置来定义引用的模型文件。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic static class Config3 extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineModelConfigurer<String, String> model) throws Exception {\n\t\tmodel\n\t\t\t.withModel()\n\t\t\t\t.factory(modelFactory());\n\t}\n\n\t@Bean\n\tpublic StateMachineModelFactory<String, String> modelFactory() {\n\t\treturn new UmlStateMachineModelFactory(\n\t\t\t"classpath:org/springframework/statemachine/uml/import-main/import-main.uml",\n\t\t\tnew String[] { "classpath:org/springframework/statemachine/uml/import-sub/import-sub.uml" });\n\t}\n}\n')])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("UML 模型中的文件之间的链接需要是相对的"),a("br"),t._v(",否则,当模型文件从"),a("br"),t._v(" Classpath 复制到临时目录时,事情就会中断,这样 Eclipse 解析类就可以"),a("br"),t._v("读取这些文件。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"存储库支持"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#存储库支持"}},[t._v("#")]),t._v(" 存储库支持")]),t._v(" "),a("p",[t._v("本节包含与在 Spring statemachine 中使用“ Spring data repositories”有关的文档。")]),t._v(" "),a("h3",{attrs:{id:"存储库配置-2"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#存储库配置-2"}},[t._v("#")]),t._v(" 存储库配置")]),t._v(" "),a("p",[t._v("你可以将机器配置保留在外部存储中,可以根据需要从该存储中加载它,而不是通过使用 Java 配置或基于 UML 的配置来创建静态配置。这种集成通过 Spring 数据存储库抽象工作。")]),t._v(" "),a("p",[t._v("我们创建了一个特殊的"),a("code",[t._v("StateMachineModelFactory")]),t._v("实现,称为"),a("code",[t._v("RepositoryStateMachineModelFactory")]),t._v("。它可以使用基本存储库接口("),a("code",[t._v("StateRepository")]),t._v(""),a("code",[t._v("TransitionRepository")]),t._v(""),a("code",[t._v("ActionRepository")]),t._v(""),a("code",[t._v("GuardRepository")]),t._v(")和基本实体接口("),a("code",[t._v("RepositoryState")]),t._v(""),a("code",[t._v("RepositoryTransition")]),t._v(""),a("code",[t._v("RepositoryAction")]),t._v(",和"),a("code",[t._v("RepositoryGuard")]),t._v(")。")]),t._v(" "),a("p",[t._v("由于实体和存储库在 Spring 数据中的工作方式,从用户的角度来看,读访问可以像在"),a("code",[t._v("RepositoryStateMachineModelFactory")]),t._v("中所做的那样完全抽象。不需要知道存储库所使用的实际映射实体类。写入存储库总是依赖于使用真正的存储库特定的实体类。从机器配置的角度来看,我们不需要知道这些,这意味着我们不需要知道实际实现是 JPA、Redis 还是 Spring 数据支持的任何其他方式。当你手动尝试将新的状态或转换写入支持的存储库时,使用实际的与存储库相关的实体类将起作用。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[a("code",[t._v("RepositoryState")]),t._v(""),a("code",[t._v("RepositoryTransition")]),t._v("的实体类有一个"),a("code",[t._v("machineId")]),t._v("字段,该字段由你支配,可以用于"),a("br"),t._v("不同配置之间的区分——例如,如果机器是通过"),a("br"),t._v("构建的。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("实际的实现在后面的小节中有详细的说明。下面的图像是存储库配置的 UML 等效状态图。")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-repository-simplemachine.png",alt:"SM 存储库 SimpleMachine"}})]),t._v(" "),a("p",[t._v("图 1。SimpleMachine")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-repository-simplesubmachine.png",alt:"SM 存储库 SimpleSubmachine"}})]),t._v(" "),a("p",[t._v("图 2。SimpleSubmachine")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-repository-showcasemachine.png",alt:"SM 存储库展示了机器"}})]),t._v(" "),a("p",[t._v("图 3。ShowcaseMachine")]),t._v(" "),a("h4",{attrs:{id:"jpa"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#jpa"}},[t._v("#")]),t._v(" JPA")]),t._v(" "),a("p",[t._v("JPA 的实际存储库实现是"),a("code",[t._v("JpaStateRepository")]),t._v(""),a("code",[t._v("JpaTransitionRepository")]),t._v(""),a("code",[t._v("JpaActionRepository")]),t._v(""),a("code",[t._v("JpaGuardRepository")]),t._v(",它们分别由实体类"),a("code",[t._v("JpaRepositoryState")]),t._v(""),a("code",[t._v("JpaRepositoryTransition")]),t._v(""),a("code",[t._v("JpaRepositoryAction")]),t._v(""),a("code",[t._v("JpaRepositoryGuard")]),t._v("支持。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("不幸的是,版本“1.2.8”必须对 JPA 的实体"),a("br"),t._v("模型中使用的表名进行更改。以前,生成的表名"),a("br"),t._v("总是有"),a("code",[t._v("JPA_REPOSITORY_")]),t._v("的前缀,派生自实体类"),a("br"),t._v("名称。由于这导致了数据库对数据库对象长度施加"),a("br"),t._v("限制的问题,因此所有实体类都有"),a("br"),t._v("专门的定义来强制表名。例如,"),a("code",[t._v("JPA_REPOSITORY_STATE")]),t._v("现在是“state”——以其他"),a("br"),t._v("ntity 类为例。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("JPA 手动更新状态和转换的通用方法在下面的示例中显示(与"),a("a",{attrs:{href:"#image-sm-repository-simplemachine"}},[t._v("SimpleMachine")]),t._v("中所示的机器等效):")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateRepository<JpaRepositoryState> stateRepository;\n\n@Autowired\nTransitionRepository<JpaRepositoryTransition> transitionRepository;\n\nvoid addConfig() {\n\tJpaRepositoryState stateS1 = new JpaRepositoryState("S1", true);\n\tJpaRepositoryState stateS2 = new JpaRepositoryState("S2");\n\tJpaRepositoryState stateS3 = new JpaRepositoryState("S3");\n\n\tstateRepository.save(stateS1);\n\tstateRepository.save(stateS2);\n\tstateRepository.save(stateS3);\n\n\tJpaRepositoryTransition transitionS1ToS2 = new JpaRepositoryTransition(stateS1, stateS2, "E1");\n\tJpaRepositoryTransition transitionS2ToS3 = new JpaRepositoryTransition(stateS2, stateS3, "E2");\n\n\ttransitionRepository.save(transitionS1ToS2);\n\ttransitionRepository.save(transitionS2ToS3);\n}\n')])])]),a("p",[t._v("下面的示例也相当于"),a("a",{attrs:{href:"#image-sm-repository-simplesubmachine"}},[t._v("SimpleSubmachine")]),t._v("中所示的机器。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateRepository<JpaRepositoryState> stateRepository;\n\n@Autowired\nTransitionRepository<JpaRepositoryTransition> transitionRepository;\n\nvoid addConfig() {\n\tJpaRepositoryState stateS1 = new JpaRepositoryState("S1", true);\n\tJpaRepositoryState stateS2 = new JpaRepositoryState("S2");\n\tJpaRepositoryState stateS3 = new JpaRepositoryState("S3");\n\n\tJpaRepositoryState stateS21 = new JpaRepositoryState("S21", true);\n\tstateS21.setParentState(stateS2);\n\tJpaRepositoryState stateS22 = new JpaRepositoryState("S22");\n\tstateS22.setParentState(stateS2);\n\n\tstateRepository.save(stateS1);\n\tstateRepository.save(stateS2);\n\tstateRepository.save(stateS3);\n\tstateRepository.save(stateS21);\n\tstateRepository.save(stateS22);\n\n\tJpaRepositoryTransition transitionS1ToS2 = new JpaRepositoryTransition(stateS1, stateS2, "E1");\n\tJpaRepositoryTransition transitionS2ToS3 = new JpaRepositoryTransition(stateS21, stateS22, "E2");\n\tJpaRepositoryTransition transitionS21ToS22 = new JpaRepositoryTransition(stateS2, stateS3, "E3");\n\n\ttransitionRepository.save(transitionS1ToS2);\n\ttransitionRepository.save(transitionS2ToS3);\n\ttransitionRepository.save(transitionS21ToS22);\n}\n')])])]),a("p",[t._v("首先,你必须访问所有存储库。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Autowired\nStateRepository<JpaRepositoryState> stateRepository;\n\n@Autowired\nTransitionRepository<JpaRepositoryTransition> transitionRepository;\n\n@Autowired\nActionRepository<JpaRepositoryAction> actionRepository;\n\n@Autowired\nGuardRepository<JpaRepositoryGuard> guardRepository;\n")])])]),a("p",[t._v("其次,你要创造行动和守卫。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('JpaRepositoryGuard foo0Guard = new JpaRepositoryGuard();\nfoo0Guard.setName("foo0Guard");\n\nJpaRepositoryGuard foo1Guard = new JpaRepositoryGuard();\nfoo1Guard.setName("foo1Guard");\n\nJpaRepositoryAction fooAction = new JpaRepositoryAction();\nfooAction.setName("fooAction");\n\nguardRepository.save(foo0Guard);\nguardRepository.save(foo1Guard);\nactionRepository.save(fooAction);\n')])])]),a("p",[t._v("第三,你必须创造国家。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('JpaRepositoryState stateS0 = new JpaRepositoryState("S0", true);\nstateS0.setInitialAction(fooAction);\nJpaRepositoryState stateS1 = new JpaRepositoryState("S1", true);\nstateS1.setParentState(stateS0);\nJpaRepositoryState stateS11 = new JpaRepositoryState("S11", true);\nstateS11.setParentState(stateS1);\nJpaRepositoryState stateS12 = new JpaRepositoryState("S12");\nstateS12.setParentState(stateS1);\nJpaRepositoryState stateS2 = new JpaRepositoryState("S2");\nstateS2.setParentState(stateS0);\nJpaRepositoryState stateS21 = new JpaRepositoryState("S21", true);\nstateS21.setParentState(stateS2);\nJpaRepositoryState stateS211 = new JpaRepositoryState("S211", true);\nstateS211.setParentState(stateS21);\nJpaRepositoryState stateS212 = new JpaRepositoryState("S212");\nstateS212.setParentState(stateS21);\n\nstateRepository.save(stateS0);\nstateRepository.save(stateS1);\nstateRepository.save(stateS11);\nstateRepository.save(stateS12);\nstateRepository.save(stateS2);\nstateRepository.save(stateS21);\nstateRepository.save(stateS211);\nstateRepository.save(stateS212);\n')])])]),a("p",[t._v("第四,也是最后一点,你必须创建转换。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('JpaRepositoryTransition transitionS1ToS1 = new JpaRepositoryTransition(stateS1, stateS1, "A");\ntransitionS1ToS1.setGuard(foo1Guard);\n\nJpaRepositoryTransition transitionS1ToS11 = new JpaRepositoryTransition(stateS1, stateS11, "B");\nJpaRepositoryTransition transitionS21ToS211 = new JpaRepositoryTransition(stateS21, stateS211, "B");\nJpaRepositoryTransition transitionS1ToS2 = new JpaRepositoryTransition(stateS1, stateS2, "C");\nJpaRepositoryTransition transitionS1ToS0 = new JpaRepositoryTransition(stateS1, stateS0, "D");\nJpaRepositoryTransition transitionS211ToS21 = new JpaRepositoryTransition(stateS211, stateS21, "D");\nJpaRepositoryTransition transitionS0ToS211 = new JpaRepositoryTransition(stateS0, stateS211, "E");\nJpaRepositoryTransition transitionS1ToS211 = new JpaRepositoryTransition(stateS1, stateS211, "F");\nJpaRepositoryTransition transitionS2ToS21 = new JpaRepositoryTransition(stateS2, stateS21, "F");\nJpaRepositoryTransition transitionS11ToS211 = new JpaRepositoryTransition(stateS11, stateS211, "G");\n\nJpaRepositoryTransition transitionS0 = new JpaRepositoryTransition(stateS0, stateS0, "H");\ntransitionS0.setKind(TransitionKind.INTERNAL);\ntransitionS0.setGuard(foo0Guard);\ntransitionS0.setActions(new HashSet<>(Arrays.asList(fooAction)));\n\nJpaRepositoryTransition transitionS1 = new JpaRepositoryTransition(stateS1, stateS1, "H");\ntransitionS1.setKind(TransitionKind.INTERNAL);\n\nJpaRepositoryTransition transitionS2 = new JpaRepositoryTransition(stateS2, stateS2, "H");\ntransitionS2.setKind(TransitionKind.INTERNAL);\ntransitionS2.setGuard(foo1Guard);\ntransitionS2.setActions(new HashSet<>(Arrays.asList(fooAction)));\n\nJpaRepositoryTransition transitionS11ToS12 = new JpaRepositoryTransition(stateS11, stateS12, "I");\nJpaRepositoryTransition transitionS12ToS212 = new JpaRepositoryTransition(stateS12, stateS212, "I");\nJpaRepositoryTransition transitionS211ToS12 = new JpaRepositoryTransition(stateS211, stateS12, "I");\n\nJpaRepositoryTransition transitionS11 = new JpaRepositoryTransition(stateS11, stateS11, "J");\nJpaRepositoryTransition transitionS2ToS1 = new JpaRepositoryTransition(stateS2, stateS1, "K");\n\ntransitionRepository.save(transitionS1ToS1);\ntransitionRepository.save(transitionS1ToS11);\ntransitionRepository.save(transitionS21ToS211);\ntransitionRepository.save(transitionS1ToS2);\ntransitionRepository.save(transitionS1ToS0);\ntransitionRepository.save(transitionS211ToS21);\ntransitionRepository.save(transitionS0ToS211);\ntransitionRepository.save(transitionS1ToS211);\ntransitionRepository.save(transitionS2ToS21);\ntransitionRepository.save(transitionS11ToS211);\ntransitionRepository.save(transitionS0);\ntransitionRepository.save(transitionS1);\ntransitionRepository.save(transitionS2);\ntransitionRepository.save(transitionS11ToS12);\ntransitionRepository.save(transitionS12ToS212);\ntransitionRepository.save(transitionS211ToS12);\ntransitionRepository.save(transitionS11);\ntransitionRepository.save(transitionS2ToS1);\n')])])]),a("p",[t._v("你可以找到一个完整的示例"),a("a",{attrs:{href:"#statemachine-examples-datajpa"}},[t._v("here")]),t._v("。这个示例还展示了如何从具有实体类定义的现有 JSON 文件中预填充存储库。")]),t._v(" "),a("h4",{attrs:{id:"redis"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#redis"}},[t._v("#")]),t._v(" Redis")]),t._v(" "),a("p",[t._v("Redis 实例的实际存储库实现是"),a("code",[t._v("RedisStateRepository")]),t._v(""),a("code",[t._v("RedisTransitionRepository")]),t._v(""),a("code",[t._v("RedisActionRepository")]),t._v(""),a("code",[t._v("RedisGuardRepository")]),t._v(",它们分别由实体类"),a("code",[t._v("RedisRepositoryState")]),t._v(""),a("code",[t._v("RedisRepositoryTransition")]),t._v(""),a("code",[t._v("RedisRepositoryAction")]),t._v(""),a("code",[t._v("RedisRepositoryGuard")]),t._v("支持。")]),t._v(" "),a("p",[t._v("下一个示例展示了手动更新 REDI 状态和转换的通用方法。这相当于"),a("a",{attrs:{href:"#image-sm-repository-simplemachine"}},[t._v("SimpleMachine")]),t._v("中所示的机器。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateRepository<RedisRepositoryState> stateRepository;\n\n@Autowired\nTransitionRepository<RedisRepositoryTransition> transitionRepository;\n\nvoid addConfig() {\n\tRedisRepositoryState stateS1 = new RedisRepositoryState("S1", true);\n\tRedisRepositoryState stateS2 = new RedisRepositoryState("S2");\n\tRedisRepositoryState stateS3 = new RedisRepositoryState("S3");\n\n\tstateRepository.save(stateS1);\n\tstateRepository.save(stateS2);\n\tstateRepository.save(stateS3);\n\n\tRedisRepositoryTransition transitionS1ToS2 = new RedisRepositoryTransition(stateS1, stateS2, "E1");\n\tRedisRepositoryTransition transitionS2ToS3 = new RedisRepositoryTransition(stateS2, stateS3, "E2");\n\n\ttransitionRepository.save(transitionS1ToS2);\n\ttransitionRepository.save(transitionS2ToS3);\n}\n')])])]),a("p",[t._v("下面的示例相当于"),a("a",{attrs:{href:"#image-sm-repository-simplesubmachine"}},[t._v("SimpleSubmachine")]),t._v("中所示的机器:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateRepository<RedisRepositoryState> stateRepository;\n\n@Autowired\nTransitionRepository<RedisRepositoryTransition> transitionRepository;\n\nvoid addConfig() {\n\tRedisRepositoryState stateS1 = new RedisRepositoryState("S1", true);\n\tRedisRepositoryState stateS2 = new RedisRepositoryState("S2");\n\tRedisRepositoryState stateS3 = new RedisRepositoryState("S3");\n\n\tstateRepository.save(stateS1);\n\tstateRepository.save(stateS2);\n\tstateRepository.save(stateS3);\n\n\tRedisRepositoryTransition transitionS1ToS2 = new RedisRepositoryTransition(stateS1, stateS2, "E1");\n\tRedisRepositoryTransition transitionS2ToS3 = new RedisRepositoryTransition(stateS2, stateS3, "E2");\n\n\ttransitionRepository.save(transitionS1ToS2);\n\ttransitionRepository.save(transitionS2ToS3);\n}\n')])])]),a("h4",{attrs:{id:"mongodb"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#mongodb"}},[t._v("#")]),t._v(" MongoDB")]),t._v(" "),a("p",[t._v("MongoDB 实例的实际存储库实现是"),a("code",[t._v("MongoDbStateRepository")]),t._v(""),a("code",[t._v("MongoDbTransitionRepository")]),t._v(""),a("code",[t._v("MongoDbActionRepository")]),t._v(""),a("code",[t._v("MongoDbGuardRepository")]),t._v(",它们分别由实体类"),a("code",[t._v("MongoDbRepositoryState")]),t._v(""),a("code",[t._v("MongoDbRepositoryTransition")]),t._v(""),a("code",[t._v("MongoDbRepositoryAction")]),t._v(""),a("code",[t._v("MongoDbRepositoryGuard")]),t._v("支持。")]),t._v(" "),a("p",[t._v("下一个示例展示了手动更新 MongoDB 的状态和转换的通用方法。这相当于"),a("a",{attrs:{href:"#image-sm-repository-simplemachine"}},[t._v("SimpleMachine")]),t._v("中所示的机器。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateRepository<MongoDbRepositoryState> stateRepository;\n\n@Autowired\nTransitionRepository<MongoDbRepositoryTransition> transitionRepository;\n\nvoid addConfig() {\n\tMongoDbRepositoryState stateS1 = new MongoDbRepositoryState("S1", true);\n\tMongoDbRepositoryState stateS2 = new MongoDbRepositoryState("S2");\n\tMongoDbRepositoryState stateS3 = new MongoDbRepositoryState("S3");\n\n\tstateRepository.save(stateS1);\n\tstateRepository.save(stateS2);\n\tstateRepository.save(stateS3);\n\n\tMongoDbRepositoryTransition transitionS1ToS2 = new MongoDbRepositoryTransition(stateS1, stateS2, "E1");\n\tMongoDbRepositoryTransition transitionS2ToS3 = new MongoDbRepositoryTransition(stateS2, stateS3, "E2");\n\n\ttransitionRepository.save(transitionS1ToS2);\n\ttransitionRepository.save(transitionS2ToS3);\n}\n')])])]),a("p",[t._v("下面的示例与"),a("a",{attrs:{href:"#image-sm-repository-simplesubmachine"}},[t._v("SimpleSubmachine")]),t._v("中所示的机器等价。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateRepository<MongoDbRepositoryState> stateRepository;\n\n@Autowired\nTransitionRepository<MongoDbRepositoryTransition> transitionRepository;\n\nvoid addConfig() {\n\tMongoDbRepositoryState stateS1 = new MongoDbRepositoryState("S1", true);\n\tMongoDbRepositoryState stateS2 = new MongoDbRepositoryState("S2");\n\tMongoDbRepositoryState stateS3 = new MongoDbRepositoryState("S3");\n\n\tMongoDbRepositoryState stateS21 = new MongoDbRepositoryState("S21", true);\n\tstateS21.setParentState(stateS2);\n\tMongoDbRepositoryState stateS22 = new MongoDbRepositoryState("S22");\n\tstateS22.setParentState(stateS2);\n\n\tstateRepository.save(stateS1);\n\tstateRepository.save(stateS2);\n\tstateRepository.save(stateS3);\n\tstateRepository.save(stateS21);\n\tstateRepository.save(stateS22);\n\n\tMongoDbRepositoryTransition transitionS1ToS2 = new MongoDbRepositoryTransition(stateS1, stateS2, "E1");\n\tMongoDbRepositoryTransition transitionS2ToS3 = new MongoDbRepositoryTransition(stateS21, stateS22, "E2");\n\tMongoDbRepositoryTransition transitionS21ToS22 = new MongoDbRepositoryTransition(stateS2, stateS3, "E3");\n\n\ttransitionRepository.save(transitionS1ToS2);\n\ttransitionRepository.save(transitionS2ToS3);\n\ttransitionRepository.save(transitionS21ToS22);\n}\n')])])]),a("h3",{attrs:{id:"存储库持久性"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#存储库持久性"}},[t._v("#")]),t._v(" 存储库持久性")]),t._v(" "),a("p",[t._v("除了存储机器配置(如"),a("a",{attrs:{href:"#sm-repository-config"}},[t._v("存储库配置")]),t._v("所示),在外部存储库中,还可以将机器持久化到存储库中。")]),t._v(" "),a("p",[a("code",[t._v("StateMachineRepository")]),t._v("接口是与机器持久性交互的中心接入点,并由实体类"),a("code",[t._v("RepositoryStateMachine")]),t._v("支持。")]),t._v(" "),a("h4",{attrs:{id:"jpa-2"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#jpa-2"}},[t._v("#")]),t._v(" JPA")]),t._v(" "),a("p",[t._v("JPA 的实际存储库实现是"),a("code",[t._v("JpaStateMachineRepository")]),t._v(",它由实体类"),a("code",[t._v("JpaRepositoryStateMachine")]),t._v("支持。")]),t._v(" "),a("p",[t._v("下面的示例展示了在 JPA 中持久化机器的通用方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateMachineRepository<JpaRepositoryStateMachine> stateMachineRepository;\n\nvoid persist() {\n\n\tJpaRepositoryStateMachine machine = new JpaRepositoryStateMachine();\n\tmachine.setMachineId("machine");\n\tmachine.setState("S1");\n\t// raw byte[] representation of a context\n\tmachine.setStateMachineContext(new byte[] { 0 });\n\n\tstateMachineRepository.save(machine);\n}\n')])])]),a("h4",{attrs:{id:"redis-2"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#redis-2"}},[t._v("#")]),t._v(" Redis")]),t._v(" "),a("p",[t._v("Redis 的实际存储库实现是"),a("code",[t._v("RedisStateMachineRepository")]),t._v(",它由实体类"),a("code",[t._v("RedisRepositoryStateMachine")]),t._v("支持。")]),t._v(" "),a("p",[t._v("下面的示例展示了为 Redis 持久化机器的通用方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateMachineRepository<RedisRepositoryStateMachine> stateMachineRepository;\n\nvoid persist() {\n\n\tRedisRepositoryStateMachine machine = new RedisRepositoryStateMachine();\n\tmachine.setMachineId("machine");\n\tmachine.setState("S1");\n\t// raw byte[] representation of a context\n\tmachine.setStateMachineContext(new byte[] { 0 });\n\n\tstateMachineRepository.save(machine);\n}\n')])])]),a("h4",{attrs:{id:"mongodb-2"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#mongodb-2"}},[t._v("#")]),t._v(" MongoDB")]),t._v(" "),a("p",[t._v("MongoDB 的实际存储库实现是"),a("code",[t._v("MongoDbStateMachineRepository")]),t._v(",它由实体类"),a("code",[t._v("MongoDbRepositoryStateMachine")]),t._v("支持。")]),t._v(" "),a("p",[t._v("下面的示例展示了为 MongoDB 持久化机器的通用方法:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Autowired\nStateMachineRepository<MongoDbRepositoryStateMachine> stateMachineRepository;\n\nvoid persist() {\n\n\tMongoDbRepositoryStateMachine machine = new MongoDbRepositoryStateMachine();\n\tmachine.setMachineId("machine");\n\tmachine.setState("S1");\n\t// raw byte[] representation of a context\n\tmachine.setStateMachineContext(new byte[] { 0 });\n\n\tstateMachineRepository.save(machine);\n}\n')])])]),a("h1",{attrs:{id:"食谱"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#食谱"}},[t._v("#")]),t._v(" 食谱")]),t._v(" "),a("p",[t._v("本章包含现有内置状态机配方的文档。")]),t._v(" "),a("p",[t._v("Spring 机械是一个基础框架。也就是说,除了 Spring 框架之外,它没有更高级别的功能或许多依赖关系。因此,正确使用状态机可能很困难。为了提供帮助,我们创建了一组解决常见用例的配方模块。")]),t._v(" "),a("p",[t._v("食谱到底是什么?状态机配方是一个解决常见用例的模块。从本质上讲,状态机配方既是一个示例,也是我们试图使你易于重用和扩展的一个示例。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("食谱是对 Spring "),a("br"),t._v("Statemachine 项目进行外部贡献的一种很好的方式。如果你还没有准备好为"),a("br"),t._v("框架核心本身做出贡献,那么一个自定义的通用配方是与其他用户共享"),a("br"),t._v("功能的一种很好的方法。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"坚持"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#坚持"}},[t._v("#")]),t._v(" 坚持")]),t._v(" "),a("p",[t._v("持久化配方是一个简单的实用程序,它允许你使用单个状态机实例来持久化和更新存储库中任意项的状态。")]),t._v(" "),a("p",[t._v("这个配方的主要类别是"),a("code",[t._v("PersistStateMachineHandler")]),t._v(",它做了三个假设:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("一个"),a("code",[t._v("StateMachine<String, String>")]),t._v("的实例需要与"),a("code",[t._v("PersistStateMachineHandler")]),t._v("一起使用。请注意,状态和事件必须是"),a("code",[t._v("String")]),t._v("的类型。")])]),t._v(" "),a("li",[a("p",[a("code",[t._v("PersistStateChangeListener")]),t._v("需要向处理程序注册才能对持久请求做出反应。")])]),t._v(" "),a("li",[a("p",[a("code",[t._v("handleEventWithState")]),t._v("方法用于编排状态更改。")])])]),t._v(" "),a("p",[t._v("你可以在"),a("a",{attrs:{href:"#statemachine-examples-persist"}},[t._v("Persist")]),t._v("处找到一个示例,该示例展示了如何使用此食谱。")]),t._v(" "),a("h2",{attrs:{id:"任务"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#任务"}},[t._v("#")]),t._v(" 任务")]),t._v(" "),a("p",[t._v("任务配方是运行使用状态机的"),a("code",[t._v("Runnable")]),t._v("实例的 DAG(有向丙烯酸图)的概念。这个配方是根据"),a("a",{attrs:{href:"#statemachine-examples-tasks"}},[t._v("Tasks")]),t._v("样本中介绍的想法开发的。")]),t._v(" "),a("p",[t._v("下一张图片展示了状态机的一般概念。在此状态图中,"),a("code",[t._v("TASKS")]),t._v("下的所有内容都显示了单个任务如何执行的通用概念。因为这个配方允许你注册一个任务的深层次 dag(这意味着实际状态图将是一个深嵌套子状态和区域的集合),所以我们没有必要更精确。")]),t._v(" "),a("p",[t._v("例如,如果只有两个已注册的任务,当"),a("code",[t._v("TASK_id")]),t._v("被替换为"),a("code",[t._v("TASK_1")]),t._v(""),a("code",[t._v("TASK_2")]),t._v("(假设已注册的任务 ID 为"),a("code",[t._v("1")]),t._v(""),a("code",[t._v("2")]),t._v(")时,下面的状态图将是正确的。")]),t._v(" "),a("img",{attrs:{alt:"statechart9",src:"images/statechart9.png",width:"500"}}),t._v(" "),a("p",[t._v("执行"),a("code",[t._v("Runnable")]),t._v("可能会导致错误。特别是如果涉及复杂的 DAG 任务,你希望有一种方法来处理任务执行错误,然后有一种方法来继续执行,而不执行已经成功执行的任务。另外,如果可以自动处理一些执行错误,那将是很好的。作为最后的后备措施,如果无法自动处理错误,则将状态机置入用户可以手动处理错误的状态。")]),t._v(" "),a("p",[a("code",[t._v("TasksHandler")]),t._v("包含用于配置处理程序实例的构建器方法,并遵循一个简单的构建器模式。你可以使用这个构建器来注册"),a("code",[t._v("Runnable")]),t._v("任务和"),a("code",[t._v("TasksListener")]),t._v("实例并定义"),a("code",[t._v("StateMachinePersist")]),t._v("钩子。")]),t._v(" "),a("p",[t._v("现在,我们可以使用一个运行简单睡眠的简单"),a("code",[t._v("Runnable")]),t._v(",如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("private Runnable sleepRunnable() {\n\treturn new Runnable() {\n\n\t\t@Override\n\t\tpublic void run() {\n\t\t\ttry {\n\t\t\t\tThread.sleep(2000);\n\t\t\t} catch (InterruptedException e) {\n\t\t\t}\n\t\t}\n\t};\n}\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("前面的示例是本章所有示例的基础。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("要执行多个"),a("code",[t._v("sleepRunnable")]),t._v("任务,你可以从"),a("code",[t._v("TasksHandler")]),t._v("注册任务并执行"),a("code",[t._v("runTasks()")]),t._v("方法,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('TasksHandler handler = TasksHandler.builder()\n\t\t.task("1", sleepRunnable())\n\t\t.task("2", sleepRunnable())\n\t\t.task("3", sleepRunnable())\n\t\t.build();\n\nhandler.runTasks();\n')])])]),a("p",[t._v("要侦听任务执行中发生的情况,你可以用"),a("code",[t._v("TasksHandler")]),t._v("注册"),a("code",[t._v("TasksListener")]),t._v("的实例。如果你不想实现完整的接口,则此食谱提供一个适配器"),a("code",[t._v("TasksListenerAdapter")]),t._v("。侦听器提供了各种钩子来侦听任务执行事件。下面的示例显示了"),a("code",[t._v("MyTasksListener")]),t._v("类的定义:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("private class MyTasksListener extends TasksListenerAdapter {\n\n\t@Override\n\tpublic void onTasksStarted() {\n\t}\n\n\t@Override\n\tpublic void onTasksContinue() {\n\t}\n\n\t@Override\n\tpublic void onTaskPreExecute(Object id) {\n\t}\n\n\t@Override\n\tpublic void onTaskPostExecute(Object id) {\n\t}\n\n\t@Override\n\tpublic void onTaskFailed(Object id, Exception exception) {\n\t}\n\n\t@Override\n\tpublic void onTaskSuccess(Object id) {\n\t}\n\n\t@Override\n\tpublic void onTasksSuccess() {\n\t}\n\n\t@Override\n\tpublic void onTasksError() {\n\t}\n\n\t@Override\n\tpublic void onTasksAutomaticFix(TasksHandler handler, StateContext<String, String> context) {\n\t}\n}\n")])])]),a("p",[t._v("你可以使用构建器注册侦听器,也可以直接使用"),a("code",[t._v("TasksHandler")]),t._v("注册侦听器,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('MyTasksListener listener1 = new MyTasksListener();\nMyTasksListener listener2 = new MyTasksListener();\n\nTasksHandler handler = TasksHandler.builder()\n\t\t.task("1", sleepRunnable())\n\t\t.task("2", sleepRunnable())\n\t\t.task("3", sleepRunnable())\n\t\t.listener(listener1)\n\t\t.build();\n\nhandler.addTasksListener(listener2);\nhandler.removeTasksListener(listener2);\n\nhandler.runTasks();\n')])])]),a("p",[t._v("每个任务都需要具有唯一的标识符,并且(可选地)可以将任务定义为子任务。实际上,这会产生大量的任务。下面的示例展示了如何创建一个由任务组成的深度嵌套 DAG:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('TasksHandler handler = TasksHandler.builder()\n\t\t.task("1", sleepRunnable())\n\t\t.task("1", "12", sleepRunnable())\n\t\t.task("1", "13", sleepRunnable())\n\t\t.task("2", sleepRunnable())\n\t\t.task("2", "22", sleepRunnable())\n\t\t.task("2", "23", sleepRunnable())\n\t\t.task("3", sleepRunnable())\n\t\t.task("3", "32", sleepRunnable())\n\t\t.task("3", "33", sleepRunnable())\n\t\t.build();\n\nhandler.runTasks();\n')])])]),a("p",[t._v("当发生错误并且运行这些任务的状态机进入"),a("code",[t._v("ERROR")]),t._v("状态时,你可以调用"),a("code",[t._v("fixCurrentProblems")]),t._v("处理程序方法来重置保留在状态机的扩展状态变量中的任务的当前状态。然后,你可以使用"),a("code",[t._v("continueFromError")]),t._v("处理程序方法来指示状态机从"),a("code",[t._v("ERROR")]),t._v("状态转换回"),a("code",[t._v("READY")]),t._v("状态,在那里你可以再次运行任务。下面的示例展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('TasksHandler handler = TasksHandler.builder()\n\t\t.task("1", sleepRunnable())\n\t\t.task("2", sleepRunnable())\n\t\t.task("3", sleepRunnable())\n\t\t.build();\n\n\t\thandler.runTasks();\n\t\thandler.fixCurrentProblems();\n\t\thandler.continueFromError();\n')])])]),a("h1",{attrs:{id:"状态机示例"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态机示例"}},[t._v("#")]),t._v(" 状态机示例")]),t._v(" "),a("p",[t._v("引用文档的这一部分解释了状态机以及示例代码和 UML 状态图的使用。在表示状态图、 Spring StateMachine 配置和应用程序使用状态机所做的工作之间的关系时,我们使用了一些快捷方式。对于完整的示例,你应该研究样例存储库。")]),t._v(" "),a("p",[t._v("在正常的构建周期中,样本直接从主源分布构建。本章包括以下示例:")]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-turnstile"}},[t._v("Turnstile")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-turnstilereactive"}},[t._v("旋转门反应式")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-showcase"}},[t._v("Showcase")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-cdplayer"}},[t._v("CD Player")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-tasks"}},[t._v("Tasks")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-washer"}},[t._v("Washer")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-persist"}},[t._v("Persist")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-zookeeper"}},[t._v("Zookeeper")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-web"}},[t._v("Web")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-scope"}},[t._v("Scope")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-security"}},[t._v("Security")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-eventservice"}},[t._v("活动服务")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-deploy"}},[t._v("Deploy")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-ordershipping"}},[t._v("订单运输")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-datajpa"}},[t._v("JPA 配置")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-datapersist"}},[t._v("数据持续存在")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-datajpapersist"}},[t._v("Data JPA Persist")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-datajpamultipersist"}},[t._v("多数据持久化")])]),t._v(" "),a("p",[a("a",{attrs:{href:"#statemachine-examples-monitoring"}},[t._v("Monitoring")])]),t._v(" "),a("p",[t._v("下面的清单展示了如何构建示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("./gradlew clean build -x test\n")])])]),a("p",[t._v("每个示例都位于"),a("code",[t._v("spring-statemachine-samples")]),t._v("下的自己的目录中。这些示例基于 Spring boot 和 Spring shell,你可以在每个示例项目的"),a("code",[t._v("build/libs")]),t._v("目录下找到常用的引导 fat jar。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("我们在本节中引用的 JAR 的文件名是在构建这个文档的"),a("br"),t._v("过程中填充的,这意味着,如果你从"),a("br"),t._v("Master 构建示例,那么你的文件带有"),a("code",[t._v("BUILD-SNAPSHOT")]),t._v("后缀。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("h2",{attrs:{id:"转门"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#转门"}},[t._v("#")]),t._v(" 转门")]),t._v(" "),a("p",[t._v("旋转门是一种简单的设备,如果付款,它就可以让你访问。这是一个很容易用状态机建模的概念。在最简单的形式中,只有两种状态:"),a("code",[t._v("LOCKED")]),t._v(""),a("code",[t._v("UNLOCKED")]),t._v("。两个事件,"),a("code",[t._v("COIN")]),t._v(""),a("code",[t._v("PUSH")]),t._v("可能发生,这取决于是否有人付款或试图通过旋转栅门。下图显示了状态机:")]),t._v(" "),a("img",{attrs:{alt:"statechart1",src:"images/statechart1.png",width:"500"}}),t._v(" "),a("p",[t._v("下面的清单显示了定义可能状态的枚举:")]),t._v(" "),a("p",[t._v("国家")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States {\n    LOCKED, UNLOCKED\n}\n")])])]),a("p",[t._v("下面的清单显示了定义事件的枚举:")]),t._v(" "),a("p",[t._v("事件")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum Events {\n    COIN, PUSH\n}\n")])])]),a("p",[t._v("下面的清单显示了配置状态机的代码:")]),t._v(" "),a("p",[t._v("Configuration")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\nstatic class StateMachineConfig\n\t\textends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.LOCKED)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source(States.LOCKED)\n\t\t\t\t.target(States.UNLOCKED)\n\t\t\t\t.event(Events.COIN)\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source(States.UNLOCKED)\n\t\t\t\t.target(States.LOCKED)\n\t\t\t\t.event(Events.PUSH);\n\t}\n\n}\n")])])]),a("p",[t._v("通过运行"),a("code",[t._v("turnstile")]),t._v("样例,你可以看到这个样例状态机如何与事件交互。下面的清单显示了如何执行此操作,并显示了该命令的输出:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("$ java -jar spring-statemachine-samples-turnstile-3.0.1.jar\n\nsm>sm print\n+----------------------------------------------------------------+\n|                              SM                                |\n+----------------------------------------------------------------+\n|                                                                |\n|         +----------------+          +----------------+         |\n|     *--\x3e|     LOCKED     |          |    UNLOCKED    |         |\n|         +----------------+          +----------------+         |\n|     +---| entry/         |          | entry/         |---+     |\n|     |   | exit/          |          | exit/          |   |     |\n|     |   |                |          |                |   |     |\n| PUSH|   |                |---COIN--\x3e|                |   |COIN |\n|     |   |                |          |                |   |     |\n|     |   |                |          |                |   |     |\n|     |   |                |<--PUSH---|                |   |     |\n|     +--\x3e|                |          |                |<--+     |\n|         |                |          |                |         |\n|         +----------------+          +----------------+         |\n|                                                                |\n+----------------------------------------------------------------+\n\nsm>sm start\nState changed to LOCKED\nState machine started\n\nsm>sm event COIN\nState changed to UNLOCKED\nEvent COIN send\n\nsm>sm event PUSH\nState changed to LOCKED\nEvent PUSH send\n")])])]),a("h2",{attrs:{id:"转门反应式"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#转门反应式"}},[t._v("#")]),t._v(" 转门反应式")]),t._v(" "),a("p",[t._v("旋转门反应是对"),a("a",{attrs:{href:"#statemachine-examples-turnstile"}},[t._v("Turnstile")]),t._v("样本的增强,使用相同的"),a("em",[t._v("机械")]),t._v("概念,并添加一个与"),a("em",[t._v("机械")]),t._v("反应界面进行反应通信的反应 Web 层。")]),t._v(" "),a("p",[a("code",[t._v("StateMachineController")]),t._v("是一个简单的"),a("code",[t._v("@RestController")]),t._v(",其中我们自动连接"),a("code",[t._v("StateMachine")]),t._v("")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Autowired\nprivate StateMachine<States, Events> stateMachine;\n")])])]),a("p",[t._v("我们创建第一个映射来返回机器状态。由于状态不是从机器反作用地产生的,因此我们可以"),a("em",[t._v("推迟")]),t._v("它,这样当订阅返回的"),a("code",[t._v("Mono")]),t._v("时,就会请求实际的状态。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@GetMapping("/state")\npublic Mono<States> state() {\n\treturn Mono.defer(() -> Mono.justOrEmpty(stateMachine.getState().getId()));\n}\n')])])]),a("p",[t._v("要将单个事件或多个事件发送到机器,我们可以在传入和传出层中使用"),a("code",[t._v("Flux")]),t._v(""),a("code",[t._v("EventResult")]),t._v("这里仅针对此示例,并简单地包装"),a("code",[t._v("ResultType")]),t._v("和事件。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@PostMapping("/events")\npublic Flux<EventResult> events(@RequestBody Flux<EventData> eventData) {\n\treturn eventData\n\t\t.filter(ed -> ed.getEvent() != null)\n\t\t.map(ed -> MessageBuilder.withPayload(ed.getEvent()).build())\n\t\t.flatMap(m -> stateMachine.sendEvent(Mono.just(m)))\n\t\t.map(EventResult::new);\n}\n')])])]),a("p",[t._v("你可以使用以下命令来运行示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("$ java -jar spring-statemachine-samples-turnstilereactive-3.0.1.jar\n")])])]),a("p",[t._v("获得状态的示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("GET http://localhost:8080/state\n")])])]),a("p",[t._v("然后会回应:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('"LOCKED"\n')])])]),a("p",[t._v("发送事件的示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('POST http://localhost:8080/events\ncontent-type: application/json\n\n{\n    "event": "COIN"\n}\n')])])]),a("p",[t._v("然后会回应:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('[\n  {\n    "event": "COIN",\n    "resultType": "ACCEPTED"\n  }\n]\n')])])]),a("p",[t._v("你可以发布多个事件:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('POST http://localhost:8080/events\ncontent-type: application/json\n\n[\n    {\n        "event": "COIN"\n    },\n    {\n        "event": "PUSH"\n    }\n]\n')])])]),a("p",[t._v("然后,Response 包含两个事件的结果:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('[\n  {\n    "event": "COIN",\n    "resultType": "ACCEPTED"\n  },\n  {\n    "event": "PUSH",\n    "resultType": "ACCEPTED"\n  }\n]\n')])])]),a("h2",{attrs:{id:"展示"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#展示"}},[t._v("#")]),t._v(" 展示")]),t._v(" "),a("p",[t._v("Showcase 是一个复杂的状态机,它显示了所有可能的转换拓扑,最多可达四个状态嵌套级别。下图显示了状态机:")]),t._v(" "),a("img",{attrs:{alt:"statechart2",src:"images/statechart2.png",width:"500"}}),t._v(" "),a("p",[t._v("下面的清单显示了定义可能状态的枚举:")]),t._v(" "),a("p",[t._v("国家")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States {\n    S0, S1, S11, S12, S2, S21, S211, S212\n}\n")])])]),a("p",[t._v("下面的清单显示了定义事件的枚举:")]),t._v(" "),a("p",[t._v("事件")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum Events {\n    A, B, C, D, E, F, G, H, I\n}\n")])])]),a("p",[t._v("下面的清单显示了配置状态机的代码:")]),t._v(" "),a("p",[t._v("配置状态")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\tthrows Exception {\n\tstates\n\t\t.withStates()\n\t\t\t.initial(States.S0, fooAction())\n\t\t\t.state(States.S0)\n\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent(States.S0)\n\t\t\t\t.initial(States.S1)\n\t\t\t\t.state(States.S1)\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States.S1)\n\t\t\t\t\t.initial(States.S11)\n\t\t\t\t\t.state(States.S11)\n\t\t\t\t\t.state(States.S12)\n\t\t\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent(States.S0)\n\t\t\t\t.state(States.S2)\n\t\t\t\t.and()\n\t\t\t\t.withStates()\n\t\t\t\t\t.parent(States.S2)\n\t\t\t\t\t.initial(States.S21)\n\t\t\t\t\t.state(States.S21)\n\t\t\t\t\t.and()\n\t\t\t\t\t.withStates()\n\t\t\t\t\t\t.parent(States.S21)\n\t\t\t\t\t\t.initial(States.S211)\n\t\t\t\t\t\t.state(States.S211)\n\t\t\t\t\t\t.state(States.S212);\n}\n")])])]),a("p",[t._v("下面的清单显示了配置状态机转换的代码:")]),t._v(" "),a("p",[t._v("配置-转换")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\tthrows Exception {\n\ttransitions\n\t\t.withExternal()\n\t\t\t.source(States.S1).target(States.S1).event(Events.A)\n\t\t\t.guard(foo1Guard())\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S1).target(States.S11).event(Events.B)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S21).target(States.S211).event(Events.B)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S1).target(States.S2).event(Events.C)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S2).target(States.S1).event(Events.C)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S1).target(States.S0).event(Events.D)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S211).target(States.S21).event(Events.D)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S0).target(States.S211).event(Events.E)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S1).target(States.S211).event(Events.F)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S2).target(States.S11).event(Events.F)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S11).target(States.S211).event(Events.G)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S211).target(States.S0).event(Events.G)\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.S0).event(Events.H)\n\t\t\t.guard(foo0Guard())\n\t\t\t.action(fooAction())\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.S2).event(Events.H)\n\t\t\t.guard(foo1Guard())\n\t\t\t.action(fooAction())\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.S1).event(Events.H)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S11).target(States.S12).event(Events.I)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S211).target(States.S212).event(Events.I)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S12).target(States.S212).event(Events.I);\n\n}\n")])])]),a("p",[t._v("下面的清单显示了配置状态机操作和保护的代码:")]),t._v(" "),a("p",[t._v("配置-动作和保护")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Bean\npublic FooGuard foo0Guard() {\n\treturn new FooGuard(0);\n}\n\n@Bean\npublic FooGuard foo1Guard() {\n\treturn new FooGuard(1);\n}\n\n@Bean\npublic FooAction fooAction() {\n\treturn new FooAction();\n}\n")])])]),a("p",[t._v("下面的列表显示了如何定义单个操作:")]),t._v(" "),a("p",[t._v("Action")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('private static class FooAction implements Action<States, Events> {\n\n\t@Override\n\tpublic void execute(StateContext<States, Events> context) {\n\t\tMap<Object, Object> variables = context.getExtendedState().getVariables();\n\t\tInteger foo = context.getExtendedState().get("foo", Integer.class);\n\t\tif (foo == null) {\n\t\t\tlog.info("Init foo to 0");\n\t\t\tvariables.put("foo", 0);\n\t\t} else if (foo == 0) {\n\t\t\tlog.info("Switch foo to 1");\n\t\t\tvariables.put("foo", 1);\n\t\t} else if (foo == 1) {\n\t\t\tlog.info("Switch foo to 0");\n\t\t\tvariables.put("foo", 0);\n\t\t}\n\t}\n}\n')])])]),a("p",[t._v("下面的清单显示了如何定义单个保护:")]),t._v(" "),a("p",[t._v("Guard")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('private static class FooGuard implements Guard<States, Events> {\n\n\tprivate final int match;\n\n\tpublic FooGuard(int match) {\n\t\tthis.match = match;\n\t}\n\n\t@Override\n\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\tObject foo = context.getExtendedState().getVariables().get("foo");\n\t\treturn !(foo == null || !foo.equals(match));\n\t}\n}\n')])])]),a("p",[t._v("下面的清单显示了此状态机运行时产生的输出,以及向其发送的各种事件:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>sm start\nInit foo to 0\nEntry state S0\nEntry state S1\nEntry state S11\nState machine started\n\nsm>sm event A\nEvent A send\n\nsm>sm event C\nExit state S11\nExit state S1\nEntry state S2\nEntry state S21\nEntry state S211\nEvent C send\n\nsm>sm event H\nSwitch foo to 1\nInternal transition source=S0\nEvent H send\n\nsm>sm event C\nExit state S211\nExit state S21\nExit state S2\nEntry state S1\nEntry state S11\nEvent C send\n\nsm>sm event A\nExit state S11\nExit state S1\nEntry state S1\nEntry state S11\nEvent A send\n")])])]),a("p",[t._v("在前面的输出中,我们可以看到:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("启动状态机,通过超状态("),a("code",[t._v("S11")]),t._v(")和("),a("code",[t._v("S0")]),t._v(")将状态机带到其初始状态("),a("code",[t._v("S11")]),t._v(")。此外,扩展状态变量"),a("code",[t._v("foo")]),t._v("初始化为"),a("code",[t._v("0")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("我们尝试用事件"),a("code",[t._v("A")]),t._v("在状态"),a("code",[t._v("S1")]),t._v("中执行自转换,但没有发生任何事情,因为该转换由变量"),a("code",[t._v("foo")]),t._v("保护为"),a("code",[t._v("1")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("我们发送事件"),a("code",[t._v("C")]),t._v(",它将我们带到另一个状态机,在那里输入初始状态("),a("code",[t._v("S211")]),t._v(")及其超状态。在这里,我们可以使用事件"),a("code",[t._v("H")]),t._v(",它执行一个简单的内部转换来翻转"),a("code",[t._v("foo")]),t._v("变量。然后我们使用事件"),a("code",[t._v("C")]),t._v("返回。")])]),t._v(" "),a("li",[a("p",[t._v("事件"),a("code",[t._v("A")]),t._v("再次发送,现在"),a("code",[t._v("S1")]),t._v("执行自转换,因为该保护计算为"),a("code",[t._v("true")]),t._v("")])])]),t._v(" "),a("p",[t._v("下面的示例提供了对层次结构状态及其事件处理如何工作的更详细的了解:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>sm variables\nNo variables\n\nsm>sm start\nInit foo to 0\nEntry state S0\nEntry state S1\nEntry state S11\nState machine started\n\nsm>sm variables\nfoo=0\n\nsm>sm event H\nInternal transition source=S1\nEvent H send\n\nsm>sm variables\nfoo=0\n\nsm>sm event C\nExit state S11\nExit state S1\nEntry state S2\nEntry state S21\nEntry state S211\nEvent C send\n\nsm>sm variables\nfoo=0\n\nsm>sm event H\nSwitch foo to 1\nInternal transition source=S0\nEvent H send\n\nsm>sm variables\nfoo=1\n\nsm>sm event H\nSwitch foo to 0\nInternal transition source=S2\nEvent H send\n\nsm>sm variables\nfoo=0\n")])])]),a("p",[t._v("在前面的示例中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("我们在不同的阶段打印扩展的状态变量。")])]),t._v(" "),a("li",[a("p",[t._v("通过 event"),a("code",[t._v("H")]),t._v(",我们最终运行了一个内部转换,该转换是用它的源状态记录的。")])]),t._v(" "),a("li",[a("p",[t._v("注意事件"),a("code",[t._v("H")]),t._v("在不同的状态下是如何处理的("),a("code",[t._v("S0")]),t._v(""),a("code",[t._v("S1")]),t._v(",和"),a("code",[t._v("S2")]),t._v(")。这是一个很好的例子,说明层次结构状态及其事件处理是如何工作的。如果由于保护条件,State"),a("code",[t._v("S2")]),t._v("无法处理事件"),a("code",[t._v("H")]),t._v(",则下一步检查其父事件。这保证了,当机器处于"),a("code",[t._v("S2")]),t._v("状态时,"),a("code",[t._v("foo")]),t._v("标志始终是翻转的。然而,在"),a("code",[t._v("S1")]),t._v("状态下,事件"),a("code",[t._v("H")]),t._v("总是与其虚拟转换匹配,而没有保护或操作,因此它永远不会发生。")])])]),t._v(" "),a("h2",{attrs:{id:"cd-播放机"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#cd-播放机"}},[t._v("#")]),t._v(" CD 播放机")]),t._v(" "),a("p",[t._v("CD 播放机是一个示例,它类似于许多人在现实世界中使用的一个用例。CD 播放机本身是一个非常简单的实体,允许用户打开一副牌,插入或更改一个磁盘,然后通过按下各种按钮来驱动播放机的功能("),a("code",[t._v("eject")]),t._v(""),a("code",[t._v("play")]),t._v(""),a("code",[t._v("stop")]),t._v(""),a("code",[t._v("pause")]),t._v(""),a("code",[t._v("rewind")]),t._v(",以及"),a("code",[t._v("backward")]),t._v(")。")]),t._v(" "),a("p",[t._v("我们当中有多少人真正考虑过,要制造出能与硬件交互的代码来驱动 CD 播放机,需要做些什么。是的,球员的概念很简单,但是,如果你看看幕后,事情实际上变得有点复杂。")]),t._v(" "),a("p",[t._v("你可能已经注意到,如果你的甲板是打开的,而你按下播放,甲板关闭和一首歌曲开始播放(如果 CD 被插入)。从某种意义上说,当套牌打开时,你首先需要关闭它,然后尝试开始播放(如果实际插入了 CD,则再次尝试)。希望你现在已经意识到,一个简单的 CD 播放机是如此简单。当然,你可以用一个简单的类来包装所有这些内容,这个类有几个布尔变量,可能还有几个嵌套的 if-else 子句。这样就可以了,但是如果你需要让所有这些行为变得更加复杂,那又如何呢?你真的想继续添加更多的标志和 if-else 子句吗?")]),t._v(" "),a("p",[t._v("下图显示了我们的简单 CD 播放机的状态机:")]),t._v(" "),a("img",{attrs:{alt:"statechart3",src:"images/statechart3.png",width:"500"}}),t._v(" "),a("p",[t._v("本节的其余部分将介绍这个示例及其状态机是如何设计的,以及这两个状态机是如何相互交互的。以下三个配置部分在"),a("code",[t._v("EnumStateMachineConfigurerAdapter")]),t._v("中使用。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\tthrows Exception {\n\tstates\n\t\t.withStates()\n\t\t\t.initial(States.IDLE)\n\t\t\t.state(States.IDLE)\n\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent(States.IDLE)\n\t\t\t\t.initial(States.CLOSED)\n\t\t\t\t.state(States.CLOSED, closedEntryAction(), null)\n\t\t\t\t.state(States.OPEN)\n\t\t\t\t.and()\n\t\t.withStates()\n\t\t\t.state(States.BUSY)\n\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent(States.BUSY)\n\t\t\t\t.initial(States.PLAYING)\n\t\t\t\t.state(States.PLAYING)\n\t\t\t\t.state(States.PAUSED);\n\n}\n")])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\tthrows Exception {\n\ttransitions\n\t\t.withExternal()\n\t\t\t.source(States.CLOSED).target(States.OPEN).event(Events.EJECT)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.OPEN).target(States.CLOSED).event(Events.EJECT)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.OPEN).target(States.CLOSED).event(Events.PLAY)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.PLAYING).target(States.PAUSED).event(Events.PAUSE)\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.PLAYING)\n\t\t\t.action(playingAction())\n\t\t\t.timer(1000)\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.PLAYING).event(Events.BACK)\n\t\t\t.action(trackAction())\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.PLAYING).event(Events.FORWARD)\n\t\t\t.action(trackAction())\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.PAUSED).target(States.PLAYING).event(Events.PAUSE)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.BUSY).target(States.IDLE).event(Events.STOP)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.IDLE).target(States.BUSY).event(Events.PLAY)\n\t\t\t.action(playAction())\n\t\t\t.guard(playGuard())\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.OPEN).event(Events.LOAD).action(loadAction());\n}\n")])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Bean\npublic ClosedEntryAction closedEntryAction() {\n\treturn new ClosedEntryAction();\n}\n\n@Bean\npublic LoadAction loadAction() {\n\treturn new LoadAction();\n}\n\n@Bean\npublic TrackAction trackAction() {\n\treturn new TrackAction();\n}\n\n@Bean\npublic PlayAction playAction() {\n\treturn new PlayAction();\n}\n\n@Bean\npublic PlayingAction playingAction() {\n\treturn new PlayingAction();\n}\n\n@Bean\npublic PlayGuard playGuard() {\n\treturn new PlayGuard();\n}\n")])])]),a("p",[t._v("在前面的配置中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("我们使用"),a("code",[t._v("EnumStateMachineConfigurerAdapter")]),t._v("来配置状态和转换。")])]),t._v(" "),a("li",[a("p",[a("code",[t._v("CLOSED")]),t._v(""),a("code",[t._v("OPEN")]),t._v("状态被定义为"),a("code",[t._v("IDLE")]),t._v("的子状态,"),a("code",[t._v("PLAYING")]),t._v(""),a("code",[t._v("PAUSED")]),t._v("状态被定义为"),a("code",[t._v("BUSY")]),t._v("的子状态。")])]),t._v(" "),a("li",[a("p",[t._v(""),a("code",[t._v("CLOSED")]),t._v("状态下,我们添加了一个名为"),a("code",[t._v("closedEntryAction")]),t._v("的 Bean 条目操作。")])]),t._v(" "),a("li",[a("p",[t._v("在转换过程中,我们主要将事件映射到预期的状态转换,例如"),a("code",[t._v("EJECT")]),t._v("关闭和打开一副牌,"),a("code",[t._v("PLAY")]),t._v(""),a("code",[t._v("STOP")]),t._v(""),a("code",[t._v("PAUSE")]),t._v("进行它们的自然转换。对于其他转换,我们进行了以下操作:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("对于 Source State"),a("code",[t._v("PLAYING")]),t._v(",我们添加了一个定时器触发器,它可以自动跟踪播放音轨中的经过时间,并具有决定何时切换到下一音轨的功能。")])]),t._v(" "),a("li",[a("p",[t._v("对于"),a("code",[t._v("PLAY")]),t._v("事件,如果源状态是"),a("code",[t._v("IDLE")]),t._v(",而目标状态是"),a("code",[t._v("BUSY")]),t._v(",我们定义了一个名为"),a("code",[t._v("playAction")]),t._v("的操作和一个名为"),a("code",[t._v("playGuard")]),t._v("的保护。")])]),t._v(" "),a("li",[a("p",[t._v("对于"),a("code",[t._v("LOAD")]),t._v("事件和"),a("code",[t._v("OPEN")]),t._v("状态,我们定义了一个名为"),a("code",[t._v("loadAction")]),t._v("的动作的内部转换,该动作跟踪插入带有扩展状态变量的磁盘。")])]),t._v(" "),a("li",[a("p",[a("code",[t._v("PLAYING")]),t._v("状态定义了三个内部转换。一种是由一个计时器触发的,该计时器运行一个名为"),a("code",[t._v("playingAction")]),t._v("的操作,该操作会更新扩展的状态变量。另外两个转换使用带有不同事件的"),a("code",[t._v("trackAction")]),t._v("(分别为"),a("code",[t._v("BACK")]),t._v(""),a("code",[t._v("FORWARD")]),t._v(")来处理用户想要在轨道中返回或前进时的情况。")])])])])]),t._v(" "),a("p",[t._v("这台机器只有六种状态,这些状态由以下枚举定义:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States {\n\t// super state of PLAYING and PAUSED\n    BUSY,\n    PLAYING,\n    PAUSED,\n\t// super state of CLOSED and OPEN\n    IDLE,\n    CLOSED,\n    OPEN\n}\n")])])]),a("p",[t._v("事件表示用户可以按下的按钮,以及用户是否将光盘加载到播放机中。下面的枚举定义了事件:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum Events {\n    PLAY, STOP, PAUSE, EJECT, LOAD, FORWARD, BACK\n}\n")])])]),a("p",[a("code",[t._v("cdPlayer")]),t._v(""),a("code",[t._v("library")]),t._v("bean 用于驱动应用程序。下面的清单显示了这两个 bean 的定义:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Bean\npublic CdPlayer cdPlayer() {\n\treturn new CdPlayer();\n}\n\n@Bean\npublic Library library() {\n\treturn Library.buildSampleLibrary();\n}\n")])])]),a("p",[t._v("我们将扩展的状态变量键定义为简单的枚举,如下所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum Variables {\n\tCD, TRACK, ELAPSEDTIME\n}\n\npublic enum Headers {\n\tTRACKSHIFT\n}\n")])])]),a("p",[t._v("我们希望使这个示例类型是安全的,因此我们定义了自己的注释("),a("code",[t._v("@国家OnTransition")]),t._v("),它有一个强制的元注释("),a("code",[t._v("@OnTransition")]),t._v(")。下面的清单定义了"),a("code",[t._v("@国家OnTransition")]),t._v("注释:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Target(ElementType.METHOD)\n@Retention(RetentionPolicy.RUNTIME)\n@OnTransition\npublic @interface StatesOnTransition {\n\n\tStates[] source() default {};\n\n\tStates[] target() default {};\n\n}\n")])])]),a("p",[a("code",[t._v("ClosedEntryAction")]),t._v(""),a("code",[t._v("CLOSED")]),t._v("状态的一个条目操作,如果存在磁盘,则向状态机发送"),a("code",[t._v("PLAY")]),t._v("事件。下面的列表定义了"),a("code",[t._v("ClosedEntryAction")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public static class ClosedEntryAction implements Action<States, Events> {\n\n\t@Override\n\tpublic void execute(StateContext<States, Events> context) {\n\t\tif (context.getTransition() != null\n\t\t\t\t&& context.getEvent() == Events.PLAY\n\t\t\t\t&& context.getTransition().getTarget().getId() == States.CLOSED\n\t\t\t\t&& context.getExtendedState().getVariables().get(Variables.CD) != null) {\n\t\t\tcontext.getStateMachine()\n\t\t\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t\t\t.withPayload(Events.PLAY).build()))\n\t\t\t\t.subscribe();\n\t\t}\n\t}\n}\n")])])]),a("p",[a("code",[t._v("LoadAction")]),t._v("如果事件头包含有关要加载的磁盘的信息,则更新扩展状态变量。下面的列表定义了"),a("code",[t._v("LoadAction")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public static class LoadAction implements Action<States, Events> {\n\n\t@Override\n\tpublic void execute(StateContext<States, Events> context) {\n\t\tObject cd = context.getMessageHeader(Variables.CD);\n\t\tcontext.getExtendedState().getVariables().put(Variables.CD, cd);\n\t}\n}\n")])])]),a("p",[a("code",[t._v("PlayAction")]),t._v("重置玩家的运行时间,该时间作为扩展状态变量保留。下面的列表定义了"),a("code",[t._v("PlayAction")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public static class PlayAction implements Action<States, Events> {\n\n\t@Override\n\tpublic void execute(StateContext<States, Events> context) {\n\t\tcontext.getExtendedState().getVariables().put(Variables.ELAPSEDTIME, 0l);\n\t\tcontext.getExtendedState().getVariables().put(Variables.TRACK, 0);\n\t}\n}\n")])])]),a("p",[a("code",[t._v("PlayGuard")]),t._v("如果"),a("code",[t._v("CD")]),t._v("扩展状态变量不表示磁盘已加载,则用"),a("code",[t._v("PLAY")]),t._v("事件保护从"),a("code",[t._v("IDLE")]),t._v(""),a("code",[t._v("BUSY")]),t._v("的转换。下面的列表定义了"),a("code",[t._v("PlayGuard")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public static class PlayGuard implements Guard<States, Events> {\n\n\t@Override\n\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\tExtendedState extendedState = context.getExtendedState();\n\t\treturn extendedState.getVariables().get(Variables.CD) != null;\n\t}\n}\n")])])]),a("p",[a("code",[t._v("PlayingAction")]),t._v("更新了一个名为"),a("code",[t._v("ELAPSEDTIME")]),t._v("的扩展状态变量,玩家可以使用该变量读取和更新其 LCD 状态显示。"),a("code",[t._v("PlayingAction")]),t._v("还可以在用户返回或前进轨道时处理轨道移动。下面的示例定义了"),a("code",[t._v("PlayingAction")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public static class PlayingAction implements Action<States, Events> {\n\n\t@Override\n\tpublic void execute(StateContext<States, Events> context) {\n\t\tMap<Object, Object> variables = context.getExtendedState().getVariables();\n\t\tObject elapsed = variables.get(Variables.ELAPSEDTIME);\n\t\tObject cd = variables.get(Variables.CD);\n\t\tObject track = variables.get(Variables.TRACK);\n\t\tif (elapsed instanceof Long) {\n\t\t\tlong e = ((Long)elapsed) + 1000l;\n\t\t\tif (e > ((Cd) cd).getTracks()[((Integer) track)].getLength()*1000) {\n\t\t\t\tcontext.getStateMachine()\n\t\t\t\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t\t\t\t.withPayload(Events.FORWARD)\n\t\t\t\t\t\t.setHeader(Headers.TRACKSHIFT.toString(), 1).build()))\n\t\t\t\t\t.subscribe();\n\t\t\t} else {\n\t\t\t\tvariables.put(Variables.ELAPSEDTIME, e);\n\t\t\t}\n\t\t}\n\t}\n}\n")])])]),a("p",[a("code",[t._v("TrackAction")]),t._v("当用户在音轨中返回或前进时,处理音轨转换操作。如果一个音轨是光盘上的最后一个,则停止播放,并将"),a("code",[t._v("STOP")]),t._v("事件发送到状态机。下面的示例定义了"),a("code",[t._v("TrackAction")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public static class TrackAction implements Action<States, Events> {\n\n\t@Override\n\tpublic void execute(StateContext<States, Events> context) {\n\t\tMap<Object, Object> variables = context.getExtendedState().getVariables();\n\t\tObject trackshift = context.getMessageHeader(Headers.TRACKSHIFT.toString());\n\t\tObject track = variables.get(Variables.TRACK);\n\t\tObject cd = variables.get(Variables.CD);\n\t\tif (trackshift instanceof Integer && track instanceof Integer && cd instanceof Cd) {\n\t\t\tint next = ((Integer)track) + ((Integer)trackshift);\n\t\t\tif (next >= 0 &&  ((Cd)cd).getTracks().length > next) {\n\t\t\t\tvariables.put(Variables.ELAPSEDTIME, 0l);\n\t\t\t\tvariables.put(Variables.TRACK, next);\n\t\t\t} else if (((Cd)cd).getTracks().length <= next) {\n\t\t\t\tcontext.getStateMachine()\n\t\t\t\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t\t\t\t.withPayload(Events.STOP).build()))\n\t\t\t\t\t.subscribe();\n\t\t\t}\n\t\t}\n\t}\n}\n")])])]),a("p",[t._v("状态机的另一个重要方面是它们有自己的职责(主要围绕处理状态),并且所有应用程序级别的逻辑都应该保持在外部。这意味着应用程序需要一种与状态机交互的方式。另外,请注意,我们用"),a("code",[t._v("CdPlayer")]),t._v("注释了"),a("code",[t._v("@WithStateMachine")]),t._v(",它指示状态机从你的 POJO 中查找方法,然后通过各种转换调用这些方法。下面的示例展示了它如何更新其 LCD 状态显示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@OnTransition(target = "BUSY")\npublic void busy(ExtendedState extendedState) {\n\tObject cd = extendedState.getVariables().get(Variables.CD);\n\tif (cd != null) {\n\t\tcdStatus = ((Cd)cd).getName();\n\t}\n}\n')])])]),a("p",[t._v("在前面的示例中,当转换发生在目标状态"),a("code",[t._v("BUSY")]),t._v("时,我们使用"),a("code",[t._v("@OnTransition")]),t._v("注释来钩住回调。")]),t._v(" "),a("p",[t._v("下面的清单显示了我们的状态机如何处理播放器是否关闭:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@StatesOnTransition(target = {States.CLOSED, States.IDLE})\npublic void closed(ExtendedState extendedState) {\n\tObject cd = extendedState.getVariables().get(Variables.CD);\n\tif (cd != null) {\n\t\tcdStatus = ((Cd)cd).getName();\n\t} else {\n\t\tcdStatus = "No CD";\n\t}\n\ttrackStatus = "";\n}\n')])])]),a("p",[a("code",[t._v("@OnTransition")]),t._v("(我们在前面的示例中使用了它)只能用于与枚举匹配的字符串。"),a("code",[t._v("@StatesOnTransition")]),t._v("允许你创建自己的使用实际枚举的类型安全注释。")]),t._v(" "),a("p",[t._v("下面的示例展示了这个状态机的实际工作方式。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>sm start\nEntry state IDLE\nEntry state CLOSED\nState machine started\n\nsm>cd lcd\nNo CD\n\nsm>cd library\n0: Greatest Hits\n  0: Bohemian Rhapsody  05:56\n  1: Another One Bites the Dust  03:36\n1: Greatest Hits II\n  0: A Kind of Magic  04:22\n  1: Under Pressure  04:08\n\nsm>cd eject\nExit state CLOSED\nEntry state OPEN\n\nsm>cd load 0\nLoading cd Greatest Hits\n\nsm>cd play\nExit state OPEN\nEntry state CLOSED\nExit state CLOSED\nExit state IDLE\nEntry state BUSY\nEntry state PLAYING\n\nsm>cd lcd\nGreatest Hits Bohemian Rhapsody 00:03\n\nsm>cd forward\n\nsm>cd lcd\nGreatest Hits Another One Bites the Dust 00:04\n\nsm>cd stop\nExit state PLAYING\nExit state BUSY\nEntry state IDLE\nEntry state CLOSED\n\nsm>cd lcd\nGreatest Hits\n")])])]),a("p",[t._v("在前一次运行中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("启动状态机,这将使机器被初始化。")])]),t._v(" "),a("li",[a("p",[t._v("CD 播放机的液晶屏状态是打印的。")])]),t._v(" "),a("li",[a("p",[t._v("光盘库已经印好了。")])]),t._v(" "),a("li",[a("p",[t._v("CD 播放机的甲板打开了。")])]),t._v(" "),a("li",[a("p",[t._v("索引为 0 的 CD 被加载到一个甲板中。")])]),t._v(" "),a("li",[a("p",[t._v("由于插入了一张光盘,播放会导致卡片组关闭并立即播放。")])]),t._v(" "),a("li",[a("p",[t._v("我们打印液晶状态,并要求下一个轨道.")])]),t._v(" "),a("li",[a("p",[t._v("我们不玩了。")])])]),t._v(" "),a("h2",{attrs:{id:"任务-2"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#任务-2"}},[t._v("#")]),t._v(" 任务")]),t._v(" "),a("p",[t._v("Tasks 示例演示了区域内的并行任务处理,并添加了错误处理来自动或手动修复任务问题,然后继续回到可以再次运行任务的状态。下图显示了任务状态机:")]),t._v(" "),a("img",{attrs:{alt:"statechart5",src:"images/statechart5.png",width:"500"}}),t._v(" "),a("p",[t._v("在此状态机中的高级级别上:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("我们总是尝试进入"),a("code",[t._v("READY")]),t._v("状态,这样我们就可以使用 Run 事件来执行任务。")])]),t._v(" "),a("li",[a("p",[t._v("由三个独立的区域组成的 TKHE状态被置于和状态的中间,这将导致这些区域进入它们的初始状态,并被它们的结束状态连接起来。")])]),t._v(" "),a("li",[a("p",[t._v(""),a("code",[t._v("JOIN")]),t._v("状态,我们自动进入"),a("code",[t._v("CHOICE")]),t._v("状态,该状态检查扩展状态变量中是否存在错误标志。任务可以设置这些标志,这样做使"),a("code",[t._v("CHOICE")]),t._v("状态能够进入"),a("code",[t._v("ERROR")]),t._v("状态,在该状态中,可以自动或手动处理错误。")])]),t._v(" "),a("li",[a("p",[a("code",[t._v("ERROR")]),t._v("中的"),a("code",[t._v("AUTOMATIC")]),t._v("状态可以尝试自动修复错误,如果成功,则返回"),a("code",[t._v("READY")]),t._v("。如果错误是无法自动处理的,则需要用户干预,并通过"),a("code",[t._v("FALLBACK")]),t._v("事件将机器放入"),a("code",[t._v("MANUAL")]),t._v("状态。")])])]),t._v(" "),a("p",[t._v("下面的清单显示了定义可能状态的枚举:")]),t._v(" "),a("p",[t._v("States")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States {\n    READY,\n    FORK, JOIN, CHOICE,\n    TASKS, T1, T1E, T2, T2E, T3, T3E,\n    ERROR, AUTOMATIC, MANUAL\n}\n")])])]),a("p",[t._v("下面的清单显示了定义事件的枚举:")]),t._v(" "),a("p",[t._v("事件")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum Events {\n    RUN, FALLBACK, CONTINUE, FIX;\n}\n")])])]),a("p",[t._v("下面的列表配置了可能的状态:")]),t._v(" "),a("p",[t._v("配置状态")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\tthrows Exception {\n\tstates\n\t\t.withStates()\n\t\t\t.initial(States.READY)\n\t\t\t.fork(States.FORK)\n\t\t\t.state(States.TASKS)\n\t\t\t.join(States.JOIN)\n\t\t\t.choice(States.CHOICE)\n\t\t\t.state(States.ERROR)\n\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent(States.TASKS)\n\t\t\t\t.initial(States.T1)\n\t\t\t\t.end(States.T1E)\n\t\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent(States.TASKS)\n\t\t\t\t.initial(States.T2)\n\t\t\t\t.end(States.T2E)\n\t\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent(States.TASKS)\n\t\t\t\t.initial(States.T3)\n\t\t\t\t.end(States.T3E)\n\t\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent(States.ERROR)\n\t\t\t\t.initial(States.AUTOMATIC)\n\t\t\t\t.state(States.AUTOMATIC, automaticAction(), null)\n\t\t\t\t.state(States.MANUAL);\n}\n")])])]),a("p",[t._v("以下清单配置了可能的转换:")]),t._v(" "),a("p",[t._v("配置-转换")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\tthrows Exception {\n\ttransitions\n\t\t.withExternal()\n\t\t\t.source(States.READY).target(States.FORK)\n\t\t\t.event(Events.RUN)\n\t\t\t.and()\n\t\t.withFork()\n\t\t\t.source(States.FORK).target(States.TASKS)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.T1).target(States.T1E)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.T2).target(States.T2E)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.T3).target(States.T3E)\n\t\t\t.and()\n\t\t.withJoin()\n\t\t\t.source(States.TASKS).target(States.JOIN)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.JOIN).target(States.CHOICE)\n\t\t\t.and()\n\t\t.withChoice()\n\t\t\t.source(States.CHOICE)\n\t\t\t.first(States.ERROR, tasksChoiceGuard())\n\t\t\t.last(States.READY)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.ERROR).target(States.READY)\n\t\t\t.event(Events.CONTINUE)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.AUTOMATIC).target(States.MANUAL)\n\t\t\t.event(Events.FALLBACK)\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.MANUAL)\n\t\t\t.action(fixAction())\n\t\t\t.event(Events.FIX);\n}\n")])])]),a("p",[t._v("下面的保护发送一个选择项进入"),a("code",[t._v("ERROR")]),t._v("状态,如果发生错误,则需要返回"),a("code",[t._v("TRUE")]),t._v("。此保护检查所有扩展状态变量("),a("code",[t._v("T1")]),t._v(""),a("code",[t._v("T2")]),t._v(",和"),a("code",[t._v("T3")]),t._v(")是否"),a("code",[t._v("TRUE")]),t._v("")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean\npublic Guard<States, Events> tasksChoiceGuard() {\n\treturn new Guard<States, Events>() {\n\n\t\t@Override\n\t\tpublic boolean evaluate(StateContext<States, Events> context) {\n\t\t\tMap<Object, Object> variables = context.getExtendedState().getVariables();\n\t\t\treturn !(ObjectUtils.nullSafeEquals(variables.get("T1"), true)\n\t\t\t\t\t&& ObjectUtils.nullSafeEquals(variables.get("T2"), true)\n\t\t\t\t\t&& ObjectUtils.nullSafeEquals(variables.get("T3"), true));\n\t\t}\n\t};\n}\n')])])]),a("p",[t._v("下面的操作将事件发送到状态机,以请求下一步,即返回或继续返回到 Ready。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean\npublic Action<States, Events> automaticAction() {\n\treturn new Action<States, Events>() {\n\n\t\t@Override\n\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\tMap<Object, Object> variables = context.getExtendedState().getVariables();\n\t\t\tif (ObjectUtils.nullSafeEquals(variables.get("T1"), true)\n\t\t\t\t\t&& ObjectUtils.nullSafeEquals(variables.get("T2"), true)\n\t\t\t\t\t&& ObjectUtils.nullSafeEquals(variables.get("T3"), true)) {\n\t\t\t\tcontext.getStateMachine()\n\t\t\t\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t\t\t\t.withPayload(Events.CONTINUE).build()))\n\t\t\t\t\t.subscribe();\n\t\t\t} else {\n\t\t\t\tcontext.getStateMachine()\n\t\t\t\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t\t\t\t.withPayload(Events.FALLBACK).build()))\n\t\t\t\t\t.subscribe();\n\t\t\t}\n\t\t}\n\t};\n}\n\n@Bean\npublic Action<States, Events> fixAction() {\n\treturn new Action<States, Events>() {\n\n\t\t@Override\n\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\tMap<Object, Object> variables = context.getExtendedState().getVariables();\n\t\t\tvariables.put("T1", true);\n\t\t\tvariables.put("T2", true);\n\t\t\tvariables.put("T3", true);\n\t\t\tcontext.getStateMachine()\n\t\t\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t\t\t.withPayload(Events.CONTINUE).build()))\n\t\t\t\t.subscribe();\n\t\t}\n\t};\n}\n')])])]),a("p",[t._v("缺省区域执行是同步的,这意味着区域将按顺序进行处理。在这个示例中,我们只是希望所有任务区域都能得到并行处理。这可以通过定义"),a("code",[t._v("RegionExecutionPolicy")]),t._v("来实现:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineConfigurationConfigurer<States, Events> config)\n\t\tthrows Exception {\n\tconfig\n\t\t.withConfiguration()\n\t\t\t.regionExecutionPolicy(RegionExecutionPolicy.PARALLEL);\n}\n")])])]),a("p",[t._v("下面的示例展示了这个状态机的实际工作方式:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>sm start\nState machine started\nEntry state READY\n\nsm>tasks run\nExit state READY\nEntry state TASKS\nrun task on T2\nrun task on T1\nrun task on T3\nrun task on T2 done\nrun task on T1 done\nrun task on T3 done\nEntry state T2\nEntry state T1\nEntry state T3\nExit state T2\nExit state T1\nExit state T3\nEntry state T3E\nEntry state T1E\nEntry state T2E\nExit state TASKS\nEntry state READY\n")])])]),a("p",[t._v("在前面的清单中,我们可以看到任务运行了多次。在下一个列表中,我们将介绍错误:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>tasks list\nTasks {T1=true, T3=true, T2=true}\n\nsm>tasks fail T1\n\nsm>tasks list\nTasks {T1=false, T3=true, T2=true}\n\nsm>tasks run\nEntry state TASKS\nrun task on T1\nrun task on T3\nrun task on T2\nrun task on T1 done\nrun task on T3 done\nrun task on T2 done\nEntry state T1\nEntry state T3\nEntry state T2\nEntry state T1E\nEntry state T2E\nEntry state T3E\nExit state TASKS\nEntry state JOIN\nExit state JOIN\nEntry state ERROR\nEntry state AUTOMATIC\nExit state AUTOMATIC\nExit state ERROR\nEntry state READY\n")])])]),a("p",[t._v("在前面的列表中,如果我们模拟 Task T1 的失败,它将自动修复。在下一个列表中,我们将介绍更多错误:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>tasks list\nTasks {T1=true, T3=true, T2=true}\n\nsm>tasks fail T2\n\nsm>tasks run\nEntry state TASKS\nrun task on T2\nrun task on T1\nrun task on T3\nrun task on T2 done\nrun task on T1 done\nrun task on T3 done\nEntry state T2\nEntry state T1\nEntry state T3\nEntry state T1E\nEntry state T2E\nEntry state T3E\nExit state TASKS\nEntry state JOIN\nExit state JOIN\nEntry state ERROR\nEntry state AUTOMATIC\nExit state AUTOMATIC\nEntry state MANUAL\n\nsm>tasks fix\nExit state MANUAL\nExit state ERROR\nEntry state READY\n")])])]),a("p",[t._v("在 precding 示例中,如果我们模拟任务"),a("code",[t._v("T2")]),t._v(""),a("code",[t._v("T3")]),t._v("中的任一项失败,状态机将进入"),a("code",[t._v("MANUAL")]),t._v("状态,在此状态下,需要手动解决问题,然后才能返回"),a("code",[t._v("READY")]),t._v("状态。")]),t._v(" "),a("h2",{attrs:{id:"洗衣机"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#洗衣机"}},[t._v("#")]),t._v(" 洗衣机")]),t._v(" "),a("p",[t._v("洗衣机示例演示了如何使用历史状态来恢复模拟断电情况下的运行状态配置。")]),t._v(" "),a("p",[t._v("任何使用过洗衣机的人都知道,如果你以某种方式暂停程序,它就会在不暂停的情况下从相同的状态继续运行。你可以通过使用历史伪状态在状态机中实现这种行为。下图显示了我们的洗衣机状态机:")]),t._v(" "),a("img",{attrs:{alt:"statechart6",src:"images/statechart6.png",width:"500"}}),t._v(" "),a("p",[t._v("下面的清单显示了定义可能状态的枚举:")]),t._v(" "),a("p",[t._v("States")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States {\n    RUNNING, HISTORY, END,\n    WASHING, RINSING, DRYING,\n    POWEROFF\n}\n")])])]),a("p",[t._v("下面的清单显示了定义事件的枚举:")]),t._v(" "),a("p",[t._v("事件")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum Events {\n    RINSE, DRY, STOP,\n    RESTOREPOWER, CUTPOWER\n}\n")])])]),a("p",[t._v("下面的列表配置了可能的状态:")]),t._v(" "),a("p",[t._v("配置状态")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\tthrows Exception {\n\tstates\n\t\t.withStates()\n\t\t\t.initial(States.RUNNING)\n\t\t\t.state(States.POWEROFF)\n\t\t\t.end(States.END)\n\t\t\t.and()\n\t\t\t.withStates()\n\t\t\t\t.parent(States.RUNNING)\n\t\t\t\t.initial(States.WASHING)\n\t\t\t\t.state(States.RINSING)\n\t\t\t\t.state(States.DRYING)\n\t\t\t\t.history(States.HISTORY, History.SHALLOW);\n}\n")])])]),a("p",[t._v("以下清单配置了可能的转换:")]),t._v(" "),a("p",[t._v("配置-转换")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\tthrows Exception {\n\ttransitions\n\t\t.withExternal()\n\t\t\t.source(States.WASHING).target(States.RINSING)\n\t\t\t.event(Events.RINSE)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.RINSING).target(States.DRYING)\n\t\t\t.event(Events.DRY)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.RUNNING).target(States.POWEROFF)\n\t\t\t.event(Events.CUTPOWER)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.POWEROFF).target(States.HISTORY)\n\t\t\t.event(Events.RESTOREPOWER)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.RUNNING).target(States.END)\n\t\t\t.event(Events.STOP);\n}\n")])])]),a("p",[t._v("下面的示例展示了这个状态机的实际工作方式:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>sm start\nEntry state RUNNING\nEntry state WASHING\nState machine started\n\nsm>sm event RINSE\nExit state WASHING\nEntry state RINSING\nEvent RINSE send\n\nsm>sm event DRY\nExit state RINSING\nEntry state DRYING\nEvent DRY send\n\nsm>sm event CUTPOWER\nExit state DRYING\nExit state RUNNING\nEntry state POWEROFF\nEvent CUTPOWER send\n\nsm>sm event RESTOREPOWER\nExit state POWEROFF\nEntry state RUNNING\nEntry state WASHING\nEntry state DRYING\nEvent RESTOREPOWER send\n")])])]),a("p",[t._v("在前一次运行中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("启动状态机,这将使机器被初始化。")])]),t._v(" "),a("li",[a("p",[t._v("状态机进入冲洗状态。")])]),t._v(" "),a("li",[a("p",[t._v("状态机进入干燥状态。")])]),t._v(" "),a("li",[a("p",[t._v("状态机切断电源,进入断电状态。")])]),t._v(" "),a("li",[a("p",[t._v("状态从历史状态恢复,这将使状态机恢复到其先前已知的状态。")])])]),t._v(" "),a("h2",{attrs:{id:"坚持-2"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#坚持-2"}},[t._v("#")]),t._v(" 坚持")]),t._v(" "),a("p",[t._v("持久化是一个示例,它使用"),a("a",{attrs:{href:"#statemachine-recipes-persist"}},[t._v("Persist")]),t._v("配方来演示如何通过状态机控制数据库条目更新逻辑。")]),t._v(" "),a("p",[t._v("下图显示了状态机逻辑和配置:")]),t._v(" "),a("img",{attrs:{alt:"statechart10",src:"images/statechart10.png",width:"500"}}),t._v(" "),a("p",[t._v("下面的清单显示了状态机配置:")]),t._v(" "),a("p",[t._v("Statemachine 配置")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\nstatic class StateMachineConfig\n\t\textends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("PLACED")\n\t\t\t\t.state("PROCESSING")\n\t\t\t\t.state("SENT")\n\t\t\t\t.state("DELIVERED");\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("PLACED").target("PROCESSING")\n\t\t\t\t.event("PROCESS")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("PROCESSING").target("SENT")\n\t\t\t\t.event("SEND")\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("SENT").target("DELIVERED")\n\t\t\t\t.event("DELIVER");\n\t}\n\n}\n')])])]),a("p",[t._v("以下配置创建"),a("code",[t._v("PersistStateMachineHandler")]),t._v(":")]),t._v(" "),a("p",[t._v("处理程序配置")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\nstatic class PersistHandlerConfig {\n\n\t@Autowired\n\tprivate StateMachine<String, String> stateMachine;\n\n\t@Bean\n\tpublic Persist persist() {\n\t\treturn new Persist(persistStateMachineHandler());\n\t}\n\n\t@Bean\n\tpublic PersistStateMachineHandler persistStateMachineHandler() {\n\t\treturn new PersistStateMachineHandler(stateMachine);\n\t}\n\n}\n")])])]),a("p",[t._v("下面的清单显示了与此示例一起使用的"),a("code",[t._v("Order")]),t._v("类:")]),t._v(" "),a("p",[t._v("订单类")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('public static class Order {\n\tint id;\n\tString state;\n\n\tpublic Order(int id, String state) {\n\t\tthis.id = id;\n\t\tthis.state = state;\n\t}\n\n\t@Override\n\tpublic String toString() {\n\t\treturn "Order [id=" + id + ", state=" + state + "]";\n\t}\n\n}\n')])])]),a("p",[t._v("下面的示例显示了状态机的输出:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>persist db\nOrder [id=1, state=PLACED]\nOrder [id=2, state=PROCESSING]\nOrder [id=3, state=SENT]\nOrder [id=4, state=DELIVERED]\n\nsm>persist process 1\nExit state PLACED\nEntry state PROCESSING\n\nsm>persist db\nOrder [id=2, state=PROCESSING]\nOrder [id=3, state=SENT]\nOrder [id=4, state=DELIVERED]\nOrder [id=1, state=PROCESSING]\n\nsm>persist deliver 3\nExit state SENT\nEntry state DELIVERED\n\nsm>persist db\nOrder [id=2, state=PROCESSING]\nOrder [id=4, state=DELIVERED]\nOrder [id=1, state=PROCESSING]\nOrder [id=3, state=DELIVERED]\n")])])]),a("p",[t._v("在前面的运行中,状态机:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("列出了现有嵌入式数据库中的行,该数据库已经填充了示例数据。")])]),t._v(" "),a("li",[a("p",[t._v("请求将命令"),a("code",[t._v("1")]),t._v("更新为"),a("code",[t._v("PROCESSING")]),t._v("状态。")])]),t._v(" "),a("li",[a("p",[t._v("再次列出数据库条目,并查看状态已从"),a("code",[t._v("PLACED")]),t._v("更改为"),a("code",[t._v("PROCESSING")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("更新命令"),a("code",[t._v("3")]),t._v("将其状态从"),a("code",[t._v("SENT")]),t._v("更新为"),a("code",[t._v("DELIVERED")]),t._v("")])])]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("你可能想知道数据库在哪里,因为在示例代码中不存在它的"),a("br"),t._v("符号。这个示例是基于 Spring boot 和,"),a("br"),t._v("因为必要的类在 Classpath 中,所以嵌入式的"),a("code",[t._v("HSQL")]),t._v("实例"),a("br"),t._v("是自动创建的。,"),a("br"),a("br"),t._v(" Spring boot 甚至创建了"),a("code",[t._v("JdbcTemplate")]),t._v("的实例,你可以将其"),a("br"),t._v("自动连接,就像我们在"),a("code",[t._v("Persist.java")]),t._v("中所做的那样,如下面的清单所示:"),a("br"),a("br"),a("code",[t._v("<br/>@Autowired<br/>private JdbcTemplate jdbcTemplate;<br/>")])])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("接下来,我们需要处理状态更改。下面的清单展示了我们是如何做到这一点的:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('public void change(int order, String event) {\n\tOrder o = jdbcTemplate.queryForObject("select id, state from orders where id = ?",\n\t\t\tnew RowMapper<Order>() {\n\t\t\t\tpublic Order mapRow(ResultSet rs, int rowNum) throws SQLException {\n\t\t\t\t\treturn new Order(rs.getInt("id"), rs.getString("state"));\n\t\t\t\t}\n\t\t\t}, new Object[] { order });\n\thandler.handleEventWithStateReactively(MessageBuilder\n\t\t\t.withPayload(event).setHeader("order", order).build(), o.state)\n\t\t.subscribe();\n}\n')])])]),a("p",[t._v("最后,我们使用"),a("code",[t._v("PersistStateChangeListener")]),t._v("来更新数据库,如下面的清单所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('private class LocalPersistStateChangeListener implements PersistStateChangeListener {\n\n\t@Override\n\tpublic void onPersist(State<String, String> state, Message<String> message,\n\t\t\tTransition<String, String> transition, StateMachine<String, String> stateMachine) {\n\t\tif (message != null && message.getHeaders().containsKey("order")) {\n\t\t\tInteger order = message.getHeaders().get("order", Integer.class);\n\t\t\tjdbcTemplate.update("update orders set state = ? where id = ?", state.getId(), order);\n\t\t}\n\t}\n}\n')])])]),a("h2",{attrs:{id:"动物园管理员"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#动物园管理员"}},[t._v("#")]),t._v(" 动物园管理员")]),t._v(" "),a("p",[t._v("ZooKeeper 是来自"),a("a",{attrs:{href:"#statemachine-examples-turnstile"}},[t._v("Turnstile")]),t._v("示例的分布式版本。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("此示例需要一个外部"),a("code",[t._v("Zookeeper")]),t._v("实例,该实例可从"),a("code",[t._v("localhost")]),t._v("访问,并具有默认端口和设置。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("此示例的配置几乎与"),a("code",[t._v("turnstile")]),t._v("示例相同。我们只为配置"),a("code",[t._v("StateMachineEnsemble")]),t._v("的分布式状态机添加配置,如下面的清单所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineConfigurationConfigurer<String, String> config) throws Exception {\n\tconfig\n\t\t.withDistributed()\n\t\t\t.ensemble(stateMachineEnsemble());\n}\n")])])]),a("p",[t._v("实际的"),a("code",[t._v("StateMachineEnsemble")]),t._v("需要与"),a("code",[t._v("CuratorFramework")]),t._v("客户端一起创建为 Bean,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean\npublic StateMachineEnsemble<String, String> stateMachineEnsemble() throws Exception {\n\treturn new ZookeeperStateMachineEnsemble<String, String>(curatorClient(), "/foo");\n}\n\n@Bean\npublic CuratorFramework curatorClient() throws Exception {\n\tCuratorFramework client = CuratorFrameworkFactory.builder().defaultData(new byte[0])\n\t\t\t.retryPolicy(new ExponentialBackoffRetry(1000, 3))\n\t\t\t.connectString("localhost:2181").build();\n\tclient.start();\n\treturn client;\n}\n')])])]),a("p",[t._v("对于下一个示例,我们需要创建两个不同的 shell 实例。我们需要创建一个实例,看看会发生什么,然后创建第二个实例。下面的命令启动 shell 实例(请记住现在只启动一个实例):")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@n1:~# java -jar spring-statemachine-samples-zookeeper-3.0.1.jar\n")])])]),a("p",[t._v("启动状态机时,其初始状态为"),a("code",[t._v("LOCKED")]),t._v("。然后,它发送一个"),a("code",[t._v("COIN")]),t._v("事件以转换为"),a("code",[t._v("UNLOCKED")]),t._v("状态。下面的示例显示了发生的情况:")]),t._v(" "),a("p",[t._v("贝壳 1 号")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>sm start\nEntry state LOCKED\nState machine started\n\nsm>sm event COIN\nExit state LOCKED\nEntry state UNLOCKED\nEvent COIN send\n\nsm>sm state\nUNLOCKED\n")])])]),a("p",[t._v("现在,你可以使用与启动第一个状态机相同的命令,打开第二个 shell 实例并启动状态机。你应该看到输入的是分布式状态("),a("code",[t._v("UNLOCKED")]),t._v("),而不是默认的初始状态("),a("code",[t._v("LOCKED")]),t._v(")。")]),t._v(" "),a("p",[t._v("下面的示例展示了状态机及其输出:")]),t._v(" "),a("p",[t._v("贝壳 2 号")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>sm start\nState machine started\n\nsm>sm state\nUNLOCKED\n")])])]),a("p",[t._v("然后从任一个 shell(我们在下一个示例中使用第二个实例)发送一个"),a("code",[t._v("PUSH")]),t._v("事件,将其从"),a("code",[t._v("UNLOCKED")]),t._v("转换到"),a("code",[t._v("LOCKED")]),t._v("状态。下面的示例显示了状态机命令及其输出:")]),t._v(" "),a("p",[t._v("贝壳 2 号")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>sm event PUSH\nExit state UNLOCKED\nEntry state LOCKED\nEvent PUSH send\n")])])]),a("p",[t._v("在另一个 shell(如果在第二个 shell 中运行前面的命令,则为第一个 shell)中,你应该看到状态会自动更改,这是基于在 ZooKeeper 中保留的分布式状态。下面的示例显示了状态机命令及其输出:")]),t._v(" "),a("p",[t._v("贝壳 1 号")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("sm>Exit state UNLOCKED\nEntry state LOCKED\n")])])]),a("h2",{attrs:{id:"web"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#web"}},[t._v("#")]),t._v(" web")]),t._v(" "),a("p",[t._v("Web 是一个分布式状态机示例,它使用 ZooKeeper 状态机来处理分布式状态。见"),a("a",{attrs:{href:"#statemachine-examples-zookeeper"}},[t._v("Zookeeper")]),t._v("")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("这个示例旨在针对多个不同的主机在多个"),a("br"),t._v("浏览器会话上运行。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("此示例使用来自"),a("a",{attrs:{href:"#statemachine-examples-showcase"}},[t._v("Showcase")]),t._v("的修改过的状态机结构来处理分布式状态机。下图显示了状态机逻辑:")]),t._v(" "),a("img",{attrs:{alt:"statechart11",src:"images/statechart11.png",width:"500"}}),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("由于这个示例的性质,"),a("code",[t._v("Zookeeper")]),t._v("状态机的一个实例预计将从 localhost 中为每个单独的示例实例提供"),a("br"),t._v("")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("这个演示使用了一个启动三个不同示例实例的示例。如果在同一台主机上运行不同的实例,则需要通过在命令中添加"),a("code",[t._v("--server.port=<myport>")]),t._v("来区分每个实例所使用的端口。否则,每个主机的默认端口是"),a("code",[t._v("8080")]),t._v("")]),t._v(" "),a("p",[t._v("在此示例运行中,我们有三个主机:"),a("code",[t._v("n1")]),t._v(""),a("code",[t._v("n2")]),t._v(""),a("code",[t._v("n3")]),t._v("。每个都有一个正在运行的本地 ZooKeeper 实例和一个在端口"),a("code",[t._v("8080")]),t._v("上运行的状态机示例。")]),t._v(" "),a("p",[t._v("在不同的终端中,通过运行以下命令启动三个不同的状态机:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-web-3.0.1.jar\n")])])]),a("p",[t._v("当所有实例都在运行时,当你使用浏览器访问它们时,你应该会看到所有实例都显示了类似的信息。状态应该是"),a("code",[t._v("S0")]),t._v(""),a("code",[t._v("S1")]),t._v(",和"),a("code",[t._v("S11")]),t._v("。名为"),a("code",[t._v("foo")]),t._v("的扩展状态变量的值应该为"),a("code",[t._v("0")]),t._v("。主要的状态是"),a("code",[t._v("S11")]),t._v("")]),t._v(" "),a("img",{attrs:{alt:"sm dist n1 1",src:"images/sm-dist-n1-1.png",width:"500"}}),t._v(" "),a("p",[t._v("当你在任何浏览器窗口中按下"),a("code",[t._v("Event C")]),t._v("按钮时,分布状态将更改为"),a("code",[t._v("S211,")]),t._v(",这是与类型"),a("code",[t._v("C")]),t._v("的事件相关联的转换所表示的目标状态。下图显示了这一变化:")]),t._v(" "),a("img",{attrs:{alt:"sm dist n2 2",src:"images/sm-dist-n2-2.png",width:"500"}}),t._v(" "),a("p",[t._v("现在我们可以按下"),a("code",[t._v("Event H")]),t._v("按钮,看到在所有状态机上运行的内部转换将名为"),a("code",[t._v("foo")]),t._v("的扩展状态变量的值从"),a("code",[t._v("0")]),t._v("更改为"),a("code",[t._v("1")]),t._v("。此更改首先在接收该事件的状态机上完成,然后传播到其他状态机。你应该只看到名为"),a("code",[t._v("foo")]),t._v("的变量从"),a("code",[t._v("0")]),t._v("更改为"),a("code",[t._v("1")]),t._v("")]),t._v(" "),a("img",{attrs:{alt:"sm dist n3 3",src:"images/sm-dist-n3-3.png",width:"500"}}),t._v(" "),a("p",[t._v("最后,我们可以发送"),a("code",[t._v("Event K")]),t._v(",它将状态机状态带回到状态"),a("code",[t._v("S11")]),t._v("。你应该会在所有浏览器中看到这种情况。下图显示了在一个浏览器中的结果:")]),t._v(" "),a("img",{attrs:{alt:"sm dist n1 4",src:"images/sm-dist-n1-4.png",width:"500"}}),t._v(" "),a("h2",{attrs:{id:"范围"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#范围"}},[t._v("#")]),t._v(" 范围")]),t._v(" "),a("p",[t._v("范围是一个状态机示例,它使用会话范围为每个用户提供一个单独的实例。下图显示了作用域状态机中的状态和事件:")]),t._v(" "),a("img",{attrs:{alt:"statechart12",src:"images/statechart12.png",width:"500"}}),t._v(" "),a("p",[t._v("这个简单的状态机有三种状态:"),a("code",[t._v("S0")]),t._v(""),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v("。它们之间的转换由三个事件控制:"),a("code",[t._v("A")]),t._v(""),a("code",[t._v("B")]),t._v(""),a("code",[t._v("C")]),t._v("")]),t._v(" "),a("p",[t._v("要启动状态机,请在终端中运行以下命令:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-scope-3.0.1.jar\n")])])]),a("p",[t._v("当实例运行时,你可以打开浏览器并使用状态机。如果你在不同的浏览器中打开相同的页面(例如,一个在 Chrome 中,一个在 Firefox 中),那么你应该为每个用户获得一个新的状态机实例会话。下图显示了浏览器中的状态机:")]),t._v(" "),a("img",{attrs:{alt:"sm scope 1",src:"images/sm-scope-1.png",width:"500"}}),t._v(" "),a("h2",{attrs:{id:"安全性"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#安全性"}},[t._v("#")]),t._v(" 安全性")]),t._v(" "),a("p",[t._v("安全性是一个状态机示例,它使用了保护状态机的大多数可能组合。它确保发送事件、转换和操作的安全性。下图显示了状态机的状态和事件:")]),t._v(" "),a("img",{attrs:{alt:"statechart13",src:"images/statechart13.png",width:"500"}}),t._v(" "),a("p",[t._v("要启动状态机,请运行以下命令:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-secure-3.0.1.jar\n")])])]),a("p",[t._v("我们通过要求用户具有"),a("code",[t._v("USER")]),t._v("的角色来保护事件发送。 Spring 安全性确保没有其他用户可以将事件发送到此状态机。下面的列表保证了事件发送的安全性:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Override\npublic void configure(StateMachineConfigurationConfigurer<States, Events> config)\n\t\tthrows Exception {\n\tconfig\n\t\t.withConfiguration()\n\t\t\t.autoStartup(true)\n\t\t\t.and()\n\t\t.withSecurity()\n\t\t\t.enabled(true)\n\t\t\t.event(\"hasRole('USER')\");\n}\n")])])]),a("p",[t._v("在这个示例中,我们定义了两个用户:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("一个名为"),a("code",[t._v("user")]),t._v("的用户,他的角色是"),a("code",[t._v("USER")])])]),t._v(" "),a("li",[a("p",[t._v("一个名为"),a("code",[t._v("admin")]),t._v("的用户,他有两个角色:"),a("code",[t._v("USER")]),t._v(""),a("code",[t._v("ADMIN")])])])]),t._v(" "),a("p",[t._v("这两个用户的密码都是"),a("code",[t._v("password")]),t._v("。下面的清单配置了这两个用户:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@EnableWebSecurity\n@EnableGlobalMethodSecurity(securedEnabled = true)\nstatic class SecurityConfig extends WebSecurityConfigurerAdapter {\n\n\t@Autowired\n\tpublic void configureGlobal(AuthenticationManagerBuilder auth) throws Exception {\n\t\tauth\n\t\t\t.inMemoryAuthentication()\n\t\t\t\t.withUser("user")\n\t\t\t\t\t.password("password")\n\t\t\t\t\t.roles("USER")\n\t\t\t\t\t.and()\n\t\t\t\t.withUser("admin")\n\t\t\t\t\t.password("password")\n\t\t\t\t\t.roles("USER", "ADMIN");\n\t}\n}\n')])])]),a("p",[t._v("根据示例开头所示的状态图,我们定义了状态之间的各种转换。只有具有活动"),a("code",[t._v("ADMIN")]),t._v("角色的用户才能运行"),a("code",[t._v("S2")]),t._v(""),a("code",[t._v("S3")]),t._v("之间的外部转换。类似地,只有"),a("code",[t._v("ADMIN")]),t._v("才能运行"),a("code",[t._v("S1")]),t._v("状态的内部转换。下面的清单定义了这些转换,包括它们的安全性:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Override\npublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\tthrows Exception {\n\ttransitions\n\t\t.withExternal()\n\t\t\t.source(States.S0).target(States.S1).event(Events.A)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S1).target(States.S2).event(Events.B)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S2).target(States.S0).event(Events.C)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S2).target(States.S3).event(Events.E)\n\t\t\t.secured("ROLE_ADMIN", ComparisonType.ANY)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.S3).target(States.S0).event(Events.C)\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.S0).event(Events.D)\n\t\t\t.action(adminAction())\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.S1).event(Events.F)\n\t\t\t.action(transitionAction())\n\t\t\t.secured("ROLE_ADMIN", ComparisonType.ANY);\n}\n')])])]),a("p",[t._v("下面的列表使用了一个名为"),a("code",[t._v("adminAction")]),t._v("的方法,其返回类型是"),a("code",[t._v("Action")]),t._v(",以指定该操作是使用"),a("code",[t._v("ADMIN")]),t._v("角色进行安全保护的:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Scope(proxyMode = ScopedProxyMode.TARGET_CLASS)\n@Bean\npublic Action<States, Events> adminAction() {\n\treturn new Action<States, Events>() {\n\n\t\t@Secured("ROLE_ADMIN")\n\t\t@Override\n\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\tlog.info("Executed only for admin role");\n\t\t}\n\t};\n}\n')])])]),a("p",[t._v("当发送事件"),a("code",[t._v("F")]),t._v("时,下面的"),a("code",[t._v("Action")]),t._v("运行状态"),a("code",[t._v("S")]),t._v("的内部转换。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean\npublic Action<States, Events> transitionAction() {\n\treturn new Action<States, Events>() {\n\n\t\t@Override\n\t\tpublic void execute(StateContext<States, Events> context) {\n\t\t\tlog.info("Executed only for admin role");\n\t\t}\n\t};\n}\n')])])]),a("p",[t._v("转换本身使用"),a("code",[t._v("ADMIN")]),t._v("角色进行保护,因此,如果当前用户不讨厌该角色,则此转换不会运行。")]),t._v(" "),a("h2",{attrs:{id:"活动服务"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#活动服务"}},[t._v("#")]),t._v(" 活动服务")]),t._v(" "),a("p",[t._v("事件服务示例展示了如何使用状态机概念作为事件的处理引擎。这个样本是从一个问题演变而来的:")]),t._v(" "),a("p",[t._v("我可以使用 Spring StateMachine 作为一种微服务,将事件馈送到不同的状态机实例吗?事实上, Spring StateMachine 可以将事件馈送给潜在的数百万个不同的状态机实例。")]),t._v(" "),a("p",[t._v("这个示例使用"),a("code",[t._v("Redis")]),t._v("实例来持久化状态机实例。")]),t._v(" "),a("p",[t._v("显然,由于内存限制,在 JVM 中使用 100 万个状态机实例是个坏主意。这导致了 Spring StateMachine 的其他特性,这些特性允许你持久化"),a("code",[t._v("StateMachineContext")]),t._v("并重用现有实例。")]),t._v(" "),a("p",[t._v("对于这个示例,我们假设一个购物应用程序将不同类型的"),a("code",[t._v("PageView")]),t._v("事件发送到一个单独的微服务,然后该服务使用状态机跟踪用户的行为。下图显示了状态模型,它具有几个状态,这些状态表示用户导航产品项目列表,从购物车中添加和删除项目,进入付款页面并启动付款操作:")]),t._v(" "),a("img",{attrs:{alt:"statechart14",src:"images/statechart14.png",width:"500"}}),t._v(" "),a("p",[t._v("实际的购物应用程序将通过(例如)使用 REST 调用将这些事件发送到此服务中。稍后会有更多关于这方面的内容。")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("请记住,这里的重点是要有一个公开"),a("code",[t._v("REST")]),t._v("API 的应用程序,用户可以使用该 API 发送事件,对于每个请求,"),a("br"),t._v("状态机都可以处理这些事件。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("下面的状态机配置对状态图中的内容进行建模。各种动作会更新状态机的"),a("code",[t._v("Extended State")]),t._v(",以跟踪进入各种状态的条目的数量,以及"),a("code",[t._v("ADD")]),t._v(""),a("code",[t._v("DEL")]),t._v("的内部转换被调用了多少次,以及"),a("code",[t._v("PAY")]),t._v("是否已执行:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean(name = "stateMachineTarget")\n@Scope(scopeName="prototype")\npublic StateMachine<States, Events> stateMachineTarget() throws Exception {\n\tBuilder<States, Events> builder = StateMachineBuilder.<States, Events>builder();\n\n\tbuilder.configureConfiguration()\n\t\t.withConfiguration()\n\t\t\t.autoStartup(true);\n\n\tbuilder.configureStates()\n\t\t.withStates()\n\t\t\t.initial(States.HOME)\n\t\t\t.states(EnumSet.allOf(States.class));\n\n\tbuilder.configureTransitions()\n\t\t.withInternal()\n\t\t\t.source(States.ITEMS).event(Events.ADD)\n\t\t\t.action(addAction())\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.CART).event(Events.DEL)\n\t\t\t.action(delAction())\n\t\t\t.and()\n\t\t.withInternal()\n\t\t\t.source(States.PAYMENT).event(Events.PAY)\n\t\t\t.action(payAction())\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.HOME).target(States.ITEMS)\n\t\t\t.action(pageviewAction())\n\t\t\t.event(Events.VIEW_I)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.CART).target(States.ITEMS)\n\t\t\t.action(pageviewAction())\n\t\t\t.event(Events.VIEW_I)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.ITEMS).target(States.CART)\n\t\t\t.action(pageviewAction())\n\t\t\t.event(Events.VIEW_C)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.PAYMENT).target(States.CART)\n\t\t\t.action(pageviewAction())\n\t\t\t.event(Events.VIEW_C)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.CART).target(States.PAYMENT)\n\t\t\t.action(pageviewAction())\n\t\t\t.event(Events.VIEW_P)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.ITEMS).target(States.HOME)\n\t\t\t.action(resetAction())\n\t\t\t.event(Events.RESET)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.CART).target(States.HOME)\n\t\t\t.action(resetAction())\n\t\t\t.event(Events.RESET)\n\t\t\t.and()\n\t\t.withExternal()\n\t\t\t.source(States.PAYMENT).target(States.HOME)\n\t\t\t.action(resetAction())\n\t\t\t.event(Events.RESET);\n\n\treturn builder.build();\n}\n')])])]),a("p",[t._v("暂时不要关注"),a("code",[t._v("stateMachineTarget")]),t._v(""),a("code",[t._v("@Scope")]),t._v(",正如我们在本节后面解释的那样。")]),t._v(" "),a("p",[t._v("我们设置了一个"),a("code",[t._v("RedisConnectionFactory")]),t._v(",它默认为本地主机和默认端口。我们使用"),a("code",[t._v("StateMachinePersist")]),t._v(""),a("code",[t._v("RepositoryStateMachinePersist")]),t._v("实现。最后,我们创建一个"),a("code",[t._v("RedisStateMachinePersister")]),t._v(",它使用了以前创建的"),a("code",[t._v("StateMachinePersist")]),t._v(" Bean。")]),t._v(" "),a("p",[t._v("然后在处理"),a("code",[t._v("Controller")]),t._v("调用的"),a("code",[t._v("REST")]),t._v("调用中使用这些函数,如下面的清单所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Bean\npublic RedisConnectionFactory redisConnectionFactory() {\n\treturn new JedisConnectionFactory();\n}\n\n@Bean\npublic StateMachinePersist<States, Events, String> stateMachinePersist(RedisConnectionFactory connectionFactory) {\n\tRedisStateMachineContextRepository<States, Events> repository =\n\t\t\tnew RedisStateMachineContextRepository<States, Events>(connectionFactory);\n\treturn new RepositoryStateMachinePersist<States, Events>(repository);\n}\n\n@Bean\npublic RedisStateMachinePersister<States, Events> redisStateMachinePersister(\n\t\tStateMachinePersist<States, Events, String> stateMachinePersist) {\n\treturn new RedisStateMachinePersister<States, Events>(stateMachinePersist);\n}\n")])])]),a("p",[t._v("我们创建一个名为"),a("code",[t._v("stateMachineTarget")]),t._v("的 Bean。状态机实例化是一种相对昂贵的操作,因此最好尝试池实例,而不是为每个请求实例化一个新实例。要做到这一点,我们首先创建一个"),a("code",[t._v("poolTargetSource")]),t._v(",它将"),a("code",[t._v("stateMachineTarget")]),t._v("封装在一起,并将它的最大大小设置为 3。然后通过使用"),a("code",[t._v("request")]),t._v("作用域来代理此"),a("code",[t._v("poolTargetSource")]),t._v(""),a("code",[t._v("ProxyFactoryBean")]),t._v("。实际上,这意味着每个"),a("code",[t._v("REST")]),t._v("请求都会从 Bean 工厂获得一个池状态机实例。稍后,我们将展示如何使用这些实例。下面的清单显示了我们如何创建"),a("code",[t._v("ProxyFactoryBean")]),t._v("并设置目标源:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean\n@Scope(value = "request", proxyMode = ScopedProxyMode.TARGET_CLASS)\npublic ProxyFactoryBean stateMachine() {\n\tProxyFactoryBean pfb = new ProxyFactoryBean();\n\tpfb.setTargetSource(poolTargetSource());\n\treturn pfb;\n}\n')])])]),a("p",[t._v("下面的清单显示了我们设置最大大小和设置目标 Bean Name:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean\npublic CommonsPool2TargetSource poolTargetSource() {\n\tCommonsPool2TargetSource pool = new CommonsPool2TargetSource();\n\tpool.setMaxSize(3);\n\tpool.setTargetBeanName("stateMachineTarget");\n\treturn pool;\n}\n')])])]),a("p",[t._v("现在我们可以进入实际的演示了。你需要在具有默认设置的 localhost 上运行一个 Redis 服务器。然后,你需要通过运行以下命令来运行基于引导的示例应用程序:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-eventservice-3.0.1.jar\n")])])]),a("p",[t._v("在浏览器中,你会看到如下内容:")]),t._v(" "),a("img",{attrs:{alt:"sm eventservice 1",src:"images/sm-eventservice-1.png",width:"500"}}),t._v(" "),a("p",[t._v("在这个 UI 中,可以使用三个用户:"),a("code",[t._v("joe")]),t._v(""),a("code",[t._v("bob")]),t._v(""),a("code",[t._v("dave")]),t._v("。单击按钮会显示当前状态和扩展状态。在单击按钮之前启用单选按钮将为该用户发送特定事件。这种安排可以让你使用 UI。")]),t._v(" "),a("p",[t._v("在我们的"),a("code",[t._v("StateMachineController")]),t._v("中,我们 autowire"),a("code",[t._v("StateMachine")]),t._v(""),a("code",[t._v("StateMachinePersister")]),t._v(""),a("code",[t._v("StateMachine")]),t._v(""),a("code",[t._v("request")]),t._v("的作用域,因此你可以为每个请求获得一个新的实例,而"),a("code",[t._v("StateMachinePersist")]),t._v("是一个正常的单例 Bean。下面列出了 AutoWires"),a("code",[t._v("StateMachine")]),t._v(""),a("code",[t._v("StateMachinePersist")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Autowired\nprivate StateMachine<States, Events> stateMachine;\n\n@Autowired\nprivate StateMachinePersister<States, Events, String> stateMachinePersister;\n")])])]),a("p",[t._v("在下面的清单中,"),a("code",[t._v("feedAndGetState")]),t._v("与 UI 一起使用,以执行与实际"),a("code",[t._v("REST")]),t._v("API 可能执行的相同的操作:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@RequestMapping("/state")\npublic String feedAndGetState(@RequestParam(value = "user", required = false) String user,\n\t\t@RequestParam(value = "id", required = false) Events id, Model model) throws Exception {\n\tmodel.addAttribute("user", user);\n\tmodel.addAttribute("allTypes", Events.values());\n\tmodel.addAttribute("stateChartModel", stateChartModel);\n\t// we may get into this page without a user so\n\t// do nothing with a state machine\n\tif (StringUtils.hasText(user)) {\n\t\tresetStateMachineFromStore(user);\n\t\tif (id != null) {\n\t\t\tfeedMachine(user, id);\n\t\t}\n\t\tmodel.addAttribute("states", stateMachine.getState().getIds());\n\t\tmodel.addAttribute("extendedState", stateMachine.getExtendedState().getVariables());\n\t}\n\treturn "states";\n}\n')])])]),a("p",[t._v("在下面的清单中,"),a("code",[t._v("feedPageview")]),t._v("是一个"),a("code",[t._v("REST")]),t._v("方法,它接受带有 JSON 内容的 POST。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@RequestMapping(value = "/feed",method= RequestMethod.POST)\n@ResponseStatus(HttpStatus.OK)\npublic void feedPageview(@RequestBody(required = true) Pageview event) throws Exception {\n\tAssert.notNull(event.getUser(), "User must be set");\n\tAssert.notNull(event.getId(), "Id must be set");\n\tresetStateMachineFromStore(event.getUser());\n\tfeedMachine(event.getUser(), event.getId());\n}\n')])])]),a("p",[t._v("在下面的清单中,"),a("code",[t._v("feedMachine")]),t._v("将一个事件发送到"),a("code",[t._v("StateMachine")]),t._v("中,并通过使用"),a("code",[t._v("StateMachinePersister")]),t._v("来保持其状态:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('private void feedMachine(String user, Events id) throws Exception {\n\tstateMachine\n\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t.withPayload(id).build()))\n\t\t.blockLast();\n\tstateMachinePersister.persist(stateMachine, "testprefix:" + user);\n}\n')])])]),a("p",[t._v("下面的清单显示了用于为特定用户恢复状态机的"),a("code",[t._v("resetStateMachineFromStore")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('private StateMachine<States, Events> resetStateMachineFromStore(String user) throws Exception {\n\treturn stateMachinePersister.restore(stateMachine, "testprefix:" + user);\n}\n')])])]),a("p",[t._v("正如通常使用 UI 发送事件一样,你也可以使用"),a("code",[t._v("REST")]),t._v("调用来执行相同的操作,如下 curl 命令所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('# curl http://localhost:8080/feed -H "Content-Type: application/json" --data \'{"user":"joe","id":"VIEW_I"}\'\n')])])]),a("p",[t._v("在这一点上,你应该在 Redis 中使用"),a("code",[t._v("testprefix:joe")]),t._v("键的内容,如下例所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('$ ./redis-cli\n127.0.0.1:6379> KEYS *\n1) "testprefix:joe"\n')])])]),a("p",[t._v("接下来的三张图片显示了"),a("code",[t._v("joe")]),t._v("的状态何时从"),a("code",[t._v("HOME")]),t._v("更改为"),a("code",[t._v("ITEMS")]),t._v(",以及"),a("code",[t._v("ADD")]),t._v("动作何时执行。")]),t._v(" "),a("p",[t._v("下面这张"),a("code",[t._v("ADD")]),t._v("事件正在发送的图片:")]),t._v(" "),a("img",{attrs:{alt:"sm eventservice 2",src:"images/sm-eventservice-2.png",width:"500"}}),t._v(" "),a("p",[t._v("现在你仍然处于"),a("code",[t._v("ITEMS")]),t._v("状态,并且内部转换导致"),a("code",[t._v("COUNT")]),t._v("扩展状态变量增加到"),a("code",[t._v("1")]),t._v(",如下图所示:")]),t._v(" "),a("img",{attrs:{alt:"sm eventservice 3",src:"images/sm-eventservice-3.png",width:"500"}}),t._v(" "),a("p",[t._v("现在,你可以运行下面的"),a("code",[t._v("curl")]),t._v("REST 调用几次(或者通过 UI 执行),并在每次调用时看到"),a("code",[t._v("COUNT")]),t._v("变量的增加:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('# curl http://localhost:8080/feed -H "Content-Type: application/json" # --data \'{"user":"joe","id":"ADD"}\'\n')])])]),a("p",[t._v("下图显示了这些操作的结果:")]),t._v(" "),a("img",{attrs:{alt:"sm eventservice 4",src:"images/sm-eventservice-4.png",width:"500"}}),t._v(" "),a("h2",{attrs:{id:"部署"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#部署"}},[t._v("#")]),t._v(" 部署")]),t._v(" "),a("p",[t._v("部署示例展示了如何使用状态机概念和 UML 建模来提供一个通用的错误处理状态。这个状态机是一个相对复杂的示例,说明了如何使用各种特性来提供一个集中的错误处理概念。下图显示了部署状态机:")]),t._v(" "),a("img",{attrs:{alt:"model deployer",src:"images/model-deployer.png",width:"500"}}),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("前面的状态图是使用 Eclipse Papyrus 插件"),a("br"),t._v("(参见"),a("a",{attrs:{href:"#sm-papyrus"}},[t._v("Eclipse 建模支持")]),t._v(")设计的,并通过生成的 UML"),a("br"),t._v("模型文件导入 Spring Statemachine。在模型中定义的动作和保护是从 Spring 应用程序上下文中解析"),a("br"),t._v("的。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("在这个状态机场景中,我们有两个用户试图执行的不同行为("),a("code",[t._v("DEPLOY")]),t._v(""),a("code",[t._v("UNDEPLOY")]),t._v(")。")]),t._v(" "),a("p",[t._v("在前面的状态图中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v(""),a("code",[t._v("DEPLOY")]),t._v("状态下,"),a("code",[t._v("INSTALL")]),t._v(""),a("code",[t._v("START")]),t._v("状态是有条件地输入的。如果已经安装了产品,则直接输入"),a("code",[t._v("START")]),t._v(",如果安装失败,则无需尝试"),a("code",[t._v("START")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v(""),a("code",[t._v("UNDEPLOY")]),t._v("状态下,如果应用程序已经在运行,则有条件地输入"),a("code",[t._v("STOP")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("对于"),a("code",[t._v("DEPLOY")]),t._v(""),a("code",[t._v("UNDEPLOY")]),t._v("的条件选择是通过这些状态中的选择伪态完成的,这些选择是由守卫选择的。")])]),t._v(" "),a("li",[a("p",[t._v("我们使用出口点伪状态来更好地控制"),a("code",[t._v("DEPLOY")]),t._v(""),a("code",[t._v("UNDEPLOY")]),t._v("状态的出口。")])]),t._v(" "),a("li",[a("p",[t._v("在从"),a("code",[t._v("DEPLOY")]),t._v(""),a("code",[t._v("UNDEPLOY")]),t._v("退出后,我们通过一个结伪状态来选择是否通过"),a("code",[t._v("ERROR")]),t._v("状态(如果将一个错误添加到扩展状态中)。")])]),t._v(" "),a("li",[a("p",[t._v("最后,我们回到"),a("code",[t._v("READY")]),t._v("状态来处理新的请求。")])])]),t._v(" "),a("p",[t._v("现在我们可以开始实际的演示了。通过运行以下命令来运行基于引导的示例应用程序:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-deploy-3.0.1.jar\n")])])]),a("p",[t._v("在浏览器中,你可以看到如下图片:")]),t._v(" "),a("img",{attrs:{alt:"sm deploy 1",src:"images/sm-deploy-1.png",width:"500"}}),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("由于我们没有真正的安装、启动或停止功能,我们"),a("br"),t._v("通过检查特定消息头的存在来模拟失败。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("现在,你可以开始向机器发送事件,并选择各种消息头来驱动功能。")]),t._v(" "),a("h2",{attrs:{id:"订单运输"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#订单运输"}},[t._v("#")]),t._v(" 订单运输")]),t._v(" "),a("p",[t._v("订单运输示例展示了如何使用状态机概念来构建一个简单的订单处理系统。")]),t._v(" "),a("p",[t._v("下图显示了驱动此订单运输示例的状态图。")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-ordershipping-1.png",alt:"SM OrderShipping1"}})]),t._v(" "),a("p",[t._v("在前面的状态图中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("状态机进入"),a("code",[t._v("WAIT_NEW_ORDER")]),t._v("(默认)状态。")])]),t._v(" "),a("li",[a("p",[t._v("事件"),a("code",[t._v("PLACE_ORDER")]),t._v("转换为"),a("code",[t._v("RECEIVE_ORDER")]),t._v("状态,执行条目操作("),a("code",[t._v("entryReceiveOrder")]),t._v(")。")])]),t._v(" "),a("li",[a("p",[t._v("如果订单是"),a("code",[t._v("OK")]),t._v(",则状态机进入两个区域,一个处理订单生产,另一个处理用户级支付。否则,状态机进入"),a("code",[t._v("CUSTOMER_ERROR")]),t._v(",这是一个最终状态。")])]),t._v(" "),a("li",[a("p",[t._v("状态机在较低的区域中循环以提醒用户进行支付,直到"),a("code",[t._v("RECEIVE_PAYMENT")]),t._v("被成功发送以指示正确的支付。")])]),t._v(" "),a("li",[a("p",[t._v("这两个区域进入等待状态("),a("code",[t._v("WAIT_PRODUCT")]),t._v(""),a("code",[t._v("WAIT_ORDER")]),t._v("),在退出父正交态("),a("code",[t._v("HANDLE_ORDER")]),t._v(")之前,它们被连接在一起。")])]),t._v(" "),a("li",[a("p",[t._v("最后,状态机通过"),a("code",[t._v("SHIP_ORDER")]),t._v("到达其最终状态("),a("code",[t._v("ORDER_SHIPPED")]),t._v(")。")])])]),t._v(" "),a("p",[t._v("下面的命令运行示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-ordershipping-3.0.1.jar\n")])])]),a("p",[t._v("在浏览器中,你可以看到类似于以下图像的内容。你可以从选择一个客户和一个订单开始创建状态机。")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-ordershipping-2.png",alt:"SM OrderShipping2"}})]),t._v(" "),a("p",[t._v("现在已经创建了特定订单的状态机,你可以开始下订单并发送付款。其他设置(例如"),a("code",[t._v("makeProdPlan")]),t._v(""),a("code",[t._v("produce")]),t._v(""),a("code",[t._v("payment")]),t._v(")允许你控制状态机的工作方式。下图显示了等待订单的状态机:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-ordershipping-3.png",alt:"SM OrderShipping3"}})]),t._v(" "),a("p",[t._v("最后,你可以通过刷新页面来查看机器的功能,如下图所示:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-ordershipping-4.png",alt:"SM OrderShipping4"}})]),t._v(" "),a("h2",{attrs:{id:"jpa-配置"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#jpa-配置"}},[t._v("#")]),t._v(" JPA 配置")]),t._v(" "),a("p",[t._v("JPA 配置示例展示了如何在数据库中保存机器配置的情况下使用状态机概念。这个示例使用带有 H2 控制台的嵌入式 H2 数据库(以方便使用数据库)。")]),t._v(" "),a("p",[t._v("这个示例使用"),a("code",[t._v("spring-statemachine-autoconfigure")]),t._v("(默认情况下,自动配置 JPA 所需的存储库和实体类)。因此,你只需要"),a("code",[t._v("@SpringBootApplication")]),t._v("。下面的示例显示了带有"),a("code",[t._v("Application")]),t._v("注释的"),a("code",[t._v("@SpringBootApplication")]),t._v("类:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@SpringBootApplication\npublic class Application {\n\n\tpublic static void main(String[] args) {\n\t\tSpringApplication.run(Application.class, args);\n\t}\n}\n")])])]),a("p",[t._v("下面的示例展示了如何创建"),a("code",[t._v("RepositoryStateMachineModelFactory")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachineFactory\npublic static class Config extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Autowired\n\tprivate StateRepository<? extends RepositoryState> stateRepository;\n\n\t@Autowired\n\tprivate TransitionRepository<? extends RepositoryTransition> transitionRepository;\n\n\t@Override\n\tpublic void configure(StateMachineModelConfigurer<String, String> model) throws Exception {\n\t\tmodel\n\t\t\t.withModel()\n\t\t\t\t.factory(modelFactory());\n\t}\n\n\t@Bean\n\tpublic StateMachineModelFactory<String, String> modelFactory() {\n\t\treturn new RepositoryStateMachineModelFactory(stateRepository, transitionRepository);\n\t}\n}\n")])])]),a("p",[t._v("你可以使用以下命令来运行示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-datajpa-3.0.1.jar\n")])])]),a("p",[t._v(""),a("code",[t._v("[http://localhost:8080](http://localhost:8080)")]),t._v("上访问应用程序会为每个请求创建一个新构造的机器。然后,你可以选择将事件发送到机器。可能的事件和机器配置都会随每个请求从数据库中更新。下图显示了此状态机启动时创建的 UI 和初始事件:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpa-1.png",alt:"SM Datajpa1"}})]),t._v(" "),a("p",[t._v("要访问嵌入式控制台,可以使用 JDBC URL(如果尚未设置,它是"),a("code",[t._v("jdbc:h2:mem:testdb")]),t._v(")。下图显示了 H2 控制台:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpa-2.png",alt:"SM Datajpa2"}})]),t._v(" "),a("p",[t._v("在控制台上,你可以看到数据库表并根据需要对它们进行修改。下图显示了 UI 中一个简单查询的结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpa-3.png",alt:"SM Datajpa3"}})]),t._v(" "),a("p",[t._v("既然已经完成了这一步,你可能想知道这些默认状态和转换是如何被填充到数据库中的。 Spring 数据有一个很好的技巧来自动填充存储库,我们通过"),a("code",[t._v("Jackson2RepositoryPopulatorFactoryBean")]),t._v("使用了这个功能。下面的示例展示了我们如何创建这样的 Bean:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean\npublic StateMachineJackson2RepositoryPopulatorFactoryBean jackson2RepositoryPopulatorFactoryBean() {\n\tStateMachineJackson2RepositoryPopulatorFactoryBean factoryBean = new StateMachineJackson2RepositoryPopulatorFactoryBean();\n\tfactoryBean.setResources(new Resource[]{new ClassPathResource("data.json")});\n\treturn factoryBean;\n}\n')])])]),a("p",[t._v("下面的清单显示了我们用来填充数据库的数据源:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('[\n\t{\n\t\t"@id": "10",\n\t\t"_class": "org.springframework.statemachine.data.jpa.JpaRepositoryAction",\n\t\t"spel": "T(System).out.println(\'hello exit S1\')"\n\t},\n\t{\n\t\t"@id": "11",\n\t\t"_class": "org.springframework.statemachine.data.jpa.JpaRepositoryAction",\n\t\t"spel": "T(System).out.println(\'hello entry S2\')"\n\t},\n\t{\n\t\t"@id": "12",\n\t\t"_class": "org.springframework.statemachine.data.jpa.JpaRepositoryAction",\n\t\t"spel": "T(System).out.println(\'hello state S3\')"\n\t},\n\t{\n\t\t"@id": "13",\n\t\t"_class": "org.springframework.statemachine.data.jpa.JpaRepositoryAction",\n\t\t"spel": "T(System).out.println(\'hello\')"\n\t},\n\t{\n\t\t"@id": "1",\n\t\t"_class": "org.springframework.statemachine.data.jpa.JpaRepositoryState",\n\t\t"initial": true,\n\t\t"state": "S1",\n\t\t"exitActions": ["10"]\n\t},\n\t{\n\t\t"@id": "2",\n\t\t"_class": "org.springframework.statemachine.data.jpa.JpaRepositoryState",\n\t\t"initial": false,\n\t\t"state": "S2",\n\t\t"entryActions": ["11"]\n\t},\n\t{\n\t\t"@id": "3",\n\t\t"_class": "org.springframework.statemachine.data.jpa.JpaRepositoryState",\n\t\t"initial": false,\n\t\t"state": "S3",\n\t\t"stateActions": ["12"]\n\t},\n\t{\n\t\t"_class": "org.springframework.statemachine.data.jpa.JpaRepositoryTransition",\n\t\t"source": "1",\n\t\t"target": "2",\n\t\t"event": "E1",\n\t\t"kind": "EXTERNAL"\n\t},\n\t{\n\t\t"_class": "org.springframework.statemachine.data.jpa.JpaRepositoryTransition",\n\t\t"source": "2",\n\t\t"target": "3",\n\t\t"event": "E2",\n\t\t"actions": ["13"]\n\t}\n]\n')])])]),a("h2",{attrs:{id:"数据持久化"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#数据持久化"}},[t._v("#")]),t._v(" 数据持久化")]),t._v(" "),a("p",[t._v("数据持久化示例展示了如何使用外部存储库中的持久化机器来实现状态机概念。这个示例使用带有 H2 控制台的嵌入式 H2 数据库(以方便使用数据库)。你还可以选择启用 Redis 或 MongoDB。")]),t._v(" "),a("p",[t._v("这个示例使用"),a("code",[t._v("spring-statemachine-autoconfigure")]),t._v("(默认情况下,自动配置 JPA 所需的存储库和实体类)。因此,你只需要"),a("code",[t._v("@SpringBootApplication")]),t._v("。下面的示例显示了带有"),a("code",[t._v("@SpringBootApplication")]),t._v("注释的"),a("code",[t._v("Application")]),t._v("类:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@SpringBootApplication\npublic class Application {\n\n\tpublic static void main(String[] args) {\n\t\tSpringApplication.run(Application.class, args);\n\t}\n}\n")])])]),a("p",[a("code",[t._v("StateMachineRuntimePersister")]),t._v("接口在"),a("code",[t._v("StateMachine")]),t._v("的运行时级别上工作。它的实现"),a("code",[t._v("JpaPersistingStateMachineInterceptor")]),t._v("意在与 JPA 一起使用。下面的列表创建了"),a("code",[t._v("StateMachineRuntimePersister")]),t._v(" Bean:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@Profile("jpa")\npublic static class JpaPersisterConfig {\n\n\t@Bean\n\tpublic StateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister(\n\t\t\tJpaStateMachineRepository jpaStateMachineRepository) {\n\t\treturn new JpaPersistingStateMachineInterceptor<>(jpaStateMachineRepository);\n\t}\n}\n')])])]),a("p",[t._v("下面的示例展示了如何使用非常相似的配置为 MongoDB 创建 Bean:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@Profile("mongo")\npublic static class MongoPersisterConfig {\n\n\t@Bean\n\tpublic StateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister(\n\t\t\tMongoDbStateMachineRepository jpaStateMachineRepository) {\n\t\treturn new MongoDbPersistingStateMachineInterceptor<>(jpaStateMachineRepository);\n\t}\n}\n')])])]),a("p",[t._v("下面的示例展示了如何使用非常相似的配置为 Redis 创建 Bean:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@Profile("redis")\npublic static class RedisPersisterConfig {\n\n\t@Bean\n\tpublic StateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister(\n\t\t\tRedisStateMachineRepository jpaStateMachineRepository) {\n\t\treturn new RedisPersistingStateMachineInterceptor<>(jpaStateMachineRepository);\n\t}\n}\n')])])]),a("p",[t._v("可以使用"),a("code",[t._v("withPersistence")]),t._v("配置方法将"),a("code",[t._v("StateMachine")]),t._v("配置为使用运行时持久性。下面的清单展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Autowired\nprivate StateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister;\n\n@Override\npublic void configure(StateMachineConfigurationConfigurer<States, Events> config)\n\t\tthrows Exception {\n\tconfig\n\t\t.withPersistence()\n\t\t\t.runtimePersister(stateMachineRuntimePersister);\n}\n")])])]),a("p",[t._v("这个示例还使用"),a("code",[t._v("DefaultStateMachineService")]),t._v(",这使得使用多台机器更容易。下面的清单显示了如何创建"),a("code",[t._v("DefaultStateMachineService")]),t._v("的实例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Bean\npublic StateMachineService<States, Events> stateMachineService(\n\t\tStateMachineFactory<States, Events> stateMachineFactory,\n\t\tStateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister) {\n\treturn new DefaultStateMachineService<States, Events>(stateMachineFactory, stateMachineRuntimePersister);\n}\n")])])]),a("p",[t._v("下面的清单显示了驱动此示例中"),a("code",[t._v("StateMachineService")]),t._v("的逻辑:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("private synchronized StateMachine<States, Events> getStateMachine(String machineId) throws Exception {\n\tlistener.resetMessages();\n\tif (currentStateMachine == null) {\n\t\tcurrentStateMachine = stateMachineService.acquireStateMachine(machineId);\n\t\tcurrentStateMachine.addStateListener(listener);\n\t\tcurrentStateMachine.startReactively().block();\n\t} else if (!ObjectUtils.nullSafeEquals(currentStateMachine.getId(), machineId)) {\n\t\tstateMachineService.releaseStateMachine(currentStateMachine.getId());\n\t\tcurrentStateMachine.stopReactively().block();\n\t\tcurrentStateMachine = stateMachineService.acquireStateMachine(machineId);\n\t\tcurrentStateMachine.addStateListener(listener);\n\t\tcurrentStateMachine.startReactively().block();\n\t}\n\treturn currentStateMachine;\n}\n")])])]),a("p",[t._v("你可以使用以下命令来运行示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-datapersist-3.0.1.jar\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("默认情况下,"),a("code",[t._v("jpa")]),t._v("配置文件在"),a("code",[t._v("application.yml")]),t._v("中启用。如果你想尝试"),a("br"),t._v("其他后端,请启用"),a("code",[t._v("mongo")]),t._v("配置文件或"),a("code",[t._v("redis")]),t._v("配置文件。"),a("br"),t._v("以下命令指定使用哪个配置文件("),a("code",[t._v("jpa")]),t._v("是默认配置文件,"),a("br"),t._v("但我们为了完整起见将其包括在内):"),a("br"),a("br"),a("code",[t._v("<br/># java -jar spring-statemachine-samples-datapersist-3.0.1.jar --spring.profiles.active=jpa<br/># java -jar spring-statemachine-samples-datapersist-3.0.1.jar --spring.profiles.active=mongo<br/># java -jar spring-statemachine-samples-datapersist-3.0.1.jar --spring.profiles.active=redis<br/>")])])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("通过"),a("a",{attrs:{href:"http://localhost:8080",target:"_blank",rel:"noopener noreferrer"}},[t._v("http://localhost:8080"),a("OutboundLink")],1),t._v("访问应用程序,会为每个请求创建一个新构造的状态机,你可以选择将事件发送到机器。可能的事件和机器配置都会随每个请求从数据库中更新。")]),t._v(" "),a("p",[t._v("这个示例中的状态机有一个简单的配置,其中状态的 1’到’6’,事件的’E1’到’E6’,以在这些状态之间转换状态机。可以使用两个状态机标识符("),a("code",[t._v("datajpapersist1")]),t._v(""),a("code",[t._v("datajpapersist2")]),t._v(")来请求特定的状态机。下图显示了允许你选择机器和事件的 UI,并显示了这样做时会发生什么:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpapersist-1.png",alt:"Sm DatajpaPainstive1"}})]),t._v(" "),a("p",[t._v("该示例默认使用机器“DataJpaPainstive1”,并转到其初始状态“1”。下图显示了使用这些默认值的结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpapersist-2.png",alt:"Sm DatajpaPastive2"}})]),t._v(" "),a("p",[t._v("如果将事件"),a("code",[t._v("E1")]),t._v(""),a("code",[t._v("E2")]),t._v("发送到"),a("code",[t._v("datajpapersist1")]),t._v("状态机,则其状态保持为’s3’。下图显示了这样做的结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpapersist-3.png",alt:"SM DatajpaPainsive3"}})]),t._v(" "),a("p",[t._v("如果你随后请求状态机"),a("code",[t._v("datajpapersist1")]),t._v("但没有发送任何事件,则状态机将被恢复到其持久状态"),a("code",[t._v("S3")]),t._v("")]),t._v(" "),a("h2",{attrs:{id:"数据多持久化"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#数据多持久化"}},[t._v("#")]),t._v(" 数据多持久化")]),t._v(" "),a("p",[t._v("多 Ersist 样本是另外两个样本的扩展:"),a("a",{attrs:{href:"#statemachine-examples-datajpa"}},[t._v("JPA Configuration")]),t._v(""),a("a",{attrs:{href:"#statemachine-examples-datapersist"}},[t._v("数据持续存在")]),t._v("。我们仍然将机器配置保存在数据库中,并将其持久化到数据库中。然而,这一次,我们也有了一个包含两个正交区域的机器,以显示这些区域是如何独立地持久化的。这个示例还使用了带有 H2 控制台的嵌入式 H2 数据库(以方便使用数据库)。")]),t._v(" "),a("p",[t._v("这个示例使用"),a("code",[t._v("spring-statemachine-autoconfigure")]),t._v("(默认情况下,自动配置 JPA 所需的存储库和实体类)。因此,你只需要"),a("code",[t._v("@SpringBootApplication")]),t._v("。下面的示例显示了带有"),a("code",[t._v("@SpringBootApplication")]),t._v("注释的"),a("code",[t._v("Application")]),t._v("类:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@SpringBootApplication\npublic class Application {\n\n\tpublic static void main(String[] args) {\n\t\tSpringApplication.run(Application.class, args);\n\t}\n}\n")])])]),a("p",[t._v("与其他数据驱动的示例一样,我们再次创建"),a("code",[t._v("StateMachineRuntimePersister")]),t._v(",如下面的清单所示:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Bean\npublic StateMachineRuntimePersister<String, String, String> stateMachineRuntimePersister(\n\t\tJpaStateMachineRepository jpaStateMachineRepository) {\n\treturn new JpaPersistingStateMachineInterceptor<>(jpaStateMachineRepository);\n}\n")])])]),a("p",[t._v("a"),a("code",[t._v("StateMachineService")]),t._v(" Bean 使得使用机器更容易。下面的清单展示了如何创建这样的 Bean:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Bean\npublic StateMachineService<String, String> stateMachineService(\n\t\tStateMachineFactory<String, String> stateMachineFactory,\n\t\tStateMachineRuntimePersister<String, String, String> stateMachineRuntimePersister) {\n\treturn new DefaultStateMachineService<String, String>(stateMachineFactory, stateMachineRuntimePersister);\n}\n")])])]),a("p",[t._v("我们使用 JSON 数据导入配置。下面的示例创建一个 Bean 来执行此操作:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Bean\npublic StateMachineJackson2RepositoryPopulatorFactoryBean jackson2RepositoryPopulatorFactoryBean() {\n\tStateMachineJackson2RepositoryPopulatorFactoryBean factoryBean = new StateMachineJackson2RepositoryPopulatorFactoryBean();\n\tfactoryBean.setResources(new Resource[] { new ClassPathResource("datajpamultipersist.json") });\n\treturn factoryBean;\n}\n')])])]),a("p",[t._v("下面的列表显示了我们如何得到"),a("code",[t._v("RepositoryStateMachineModelFactory")]),t._v(":")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachineFactory\npublic static class Config extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Autowired\n\tprivate StateRepository<? extends RepositoryState> stateRepository;\n\n\t@Autowired\n\tprivate TransitionRepository<? extends RepositoryTransition> transitionRepository;\n\n\t@Autowired\n\tprivate StateMachineRuntimePersister<String, String, String> stateMachineRuntimePersister;\n\n\t@Override\n\tpublic void configure(StateMachineConfigurationConfigurer<String, String> config)\n\t\t\tthrows Exception {\n\t\tconfig\n\t\t\t.withPersistence()\n\t\t\t\t.runtimePersister(stateMachineRuntimePersister);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineModelConfigurer<String, String> model)\n\t\t\tthrows Exception {\n\t\tmodel\n\t\t\t.withModel()\n\t\t\t\t.factory(modelFactory());\n\t}\n\n\t@Bean\n\tpublic StateMachineModelFactory<String, String> modelFactory() {\n\t\treturn new RepositoryStateMachineModelFactory(stateRepository, transitionRepository);\n\t}\n}\n")])])]),a("p",[t._v("你可以使用以下命令运行示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-datajpamultipersist-3.0.1.jar\n")])])]),a("p",[t._v("访问"),a("code",[t._v("[http://localhost:8080](http://localhost:8080)")]),t._v("上的应用程序会为每个请求创建一个新构造的机器,并允许你将事件发送到机器。可能的事件和状态机配置将从数据库中针对每个请求进行更新。我们还打印出所有状态机上下文和当前根机,如下图所示:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpamultipersist-1.png",alt:"SM DatajpamultiPersistent1"}})]),t._v(" "),a("p",[t._v("名为"),a("code",[t._v("datajpamultipersist1")]),t._v("的状态机是一个简单的“平面”机器,其中状态"),a("code",[t._v("S1")]),t._v(""),a("code",[t._v("S2")]),t._v(""),a("code",[t._v("S3")]),t._v("分别由事件"),a("code",[t._v("E1")]),t._v(""),a("code",[t._v("E2")]),t._v(""),a("code",[t._v("E3")]),t._v("来转换。然而,名为"),a("code",[t._v("datajpamultipersist2")]),t._v("的状态机在根级别下直接包含两个区域("),a("code",[t._v("R1")]),t._v(""),a("code",[t._v("R2")]),t._v(")。这就是为什么这个根级机器实际上没有状态。我们需要根级别的机器来承载这些区域。")]),t._v(" "),a("p",[t._v("区域"),a("code",[t._v("R1")]),t._v(""),a("code",[t._v("R2")]),t._v(""),a("code",[t._v("datajpamultipersist2")]),t._v("状态机中包含状态"),a("code",[t._v("S10")]),t._v(""),a("code",[t._v("S11")]),t._v(",以及"),a("code",[t._v("S12")]),t._v(""),a("code",[t._v("S20")]),t._v(""),a("code",[t._v("S21")]),t._v(""),a("code",[t._v("S22")]),t._v("(分别)。事件"),a("code",[t._v("E10")]),t._v(""),a("code",[t._v("E11")]),t._v(",和"),a("code",[t._v("E12")]),t._v("用于表示区域"),a("code",[t._v("R1")]),t._v(",事件"),a("code",[t._v("E20")]),t._v(""),a("code",[t._v("E21")]),t._v(",事件"),a("code",[t._v("E22")]),t._v("用于表示区域"),a("code",[t._v("R2")]),t._v("。下面的图像显示了当我们将事件"),a("code",[t._v("E10")]),t._v(""),a("code",[t._v("E20")]),t._v("发送到"),a("code",[t._v("datajpamultipersist2")]),t._v("状态机时会发生什么:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpamultipersist-2.png",alt:"SM DatajpamultiPersistent2"}})]),t._v(" "),a("p",[t._v("区域有它们自己的上下文和它们自己的 ID,而实际的 ID 是用"),a("code",[t._v("#")]),t._v("和区域 ID 后置的。如下图所示,数据库中的不同区域具有不同的上下文:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpamultipersist-3.png",alt:"SM DatajpamultiPersistent3"}})]),t._v(" "),a("h2",{attrs:{id:"数据-jpa-持久化"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#数据-jpa-持久化"}},[t._v("#")]),t._v(" 数据 JPA 持久化")]),t._v(" "),a("p",[t._v("数据持久化示例展示了如何使用外部存储库中的持久化机器来实现状态机概念。这个示例使用带有 H2 控制台的嵌入式 H2 数据库(以方便使用数据库)。你还可以选择启用 Redis 或 MongoDB。")]),t._v(" "),a("p",[t._v("这个示例使用"),a("code",[t._v("spring-statemachine-autoconfigure")]),t._v("(默认情况下,自动配置 JPA 所需的存储库和实体类)。因此,你只需要"),a("code",[t._v("@SpringBootApplication")]),t._v("。下面的示例显示了带有"),a("code",[t._v("@SpringBootApplication")]),t._v("注释的"),a("code",[t._v("Application")]),t._v("类:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@SpringBootApplication\npublic class Application {\n\n\tpublic static void main(String[] args) {\n\t\tSpringApplication.run(Application.class, args);\n\t}\n}\n")])])]),a("p",[a("code",[t._v("StateMachineRuntimePersister")]),t._v("接口在"),a("code",[t._v("StateMachine")]),t._v("的运行时级别上工作。其实现"),a("code",[t._v("JpaPersistingStateMachineInterceptor")]),t._v("意在与 JPA 一起使用。下面的列表创建了"),a("code",[t._v("StateMachineRuntimePersister")]),t._v(" Bean:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@Profile("jpa")\npublic static class JpaPersisterConfig {\n\n\t@Bean\n\tpublic StateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister(\n\t\t\tJpaStateMachineRepository jpaStateMachineRepository) {\n\t\treturn new JpaPersistingStateMachineInterceptor<>(jpaStateMachineRepository);\n\t}\n}\n')])])]),a("p",[t._v("下面的示例展示了如何使用非常相似的配置为 MongoDB 创建 Bean:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@Profile("mongo")\npublic static class MongoPersisterConfig {\n\n\t@Bean\n\tpublic StateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister(\n\t\t\tMongoDbStateMachineRepository jpaStateMachineRepository) {\n\t\treturn new MongoDbPersistingStateMachineInterceptor<>(jpaStateMachineRepository);\n\t}\n}\n')])])]),a("p",[t._v("下面的示例展示了如何使用非常相似的配置为 Redis 创建 Bean:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@Profile("redis")\npublic static class RedisPersisterConfig {\n\n\t@Bean\n\tpublic StateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister(\n\t\t\tRedisStateMachineRepository jpaStateMachineRepository) {\n\t\treturn new RedisPersistingStateMachineInterceptor<>(jpaStateMachineRepository);\n\t}\n}\n')])])]),a("p",[t._v("你可以使用"),a("code",[t._v("withPersistence")]),t._v("配置方法将"),a("code",[t._v("StateMachine")]),t._v("配置为使用运行时持久性。下面的清单展示了如何做到这一点:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Autowired\nprivate StateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister;\n\n@Override\npublic void configure(StateMachineConfigurationConfigurer<States, Events> config)\n\t\tthrows Exception {\n\tconfig\n\t\t.withPersistence()\n\t\t\t.runtimePersister(stateMachineRuntimePersister);\n}\n")])])]),a("p",[t._v("这个示例还使用"),a("code",[t._v("DefaultStateMachineService")]),t._v(",这使得使用多台机器更容易。下面的清单显示了如何创建"),a("code",[t._v("DefaultStateMachineService")]),t._v("的实例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Bean\npublic StateMachineService<States, Events> stateMachineService(\n\t\tStateMachineFactory<States, Events> stateMachineFactory,\n\t\tStateMachineRuntimePersister<States, Events, String> stateMachineRuntimePersister) {\n\treturn new DefaultStateMachineService<States, Events>(stateMachineFactory, stateMachineRuntimePersister);\n}\n")])])]),a("p",[t._v("下面的清单显示了驱动此示例中"),a("code",[t._v("StateMachineService")]),t._v("的逻辑:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("private synchronized StateMachine<States, Events> getStateMachine(String machineId) throws Exception {\n\tlistener.resetMessages();\n\tif (currentStateMachine == null) {\n\t\tcurrentStateMachine = stateMachineService.acquireStateMachine(machineId);\n\t\tcurrentStateMachine.addStateListener(listener);\n\t\tcurrentStateMachine.startReactively().block();\n\t} else if (!ObjectUtils.nullSafeEquals(currentStateMachine.getId(), machineId)) {\n\t\tstateMachineService.releaseStateMachine(currentStateMachine.getId());\n\t\tcurrentStateMachine.stopReactively().block();\n\t\tcurrentStateMachine = stateMachineService.acquireStateMachine(machineId);\n\t\tcurrentStateMachine.addStateListener(listener);\n\t\tcurrentStateMachine.startReactively().block();\n\t}\n\treturn currentStateMachine;\n}\n")])])]),a("p",[t._v("你可以使用以下命令来运行示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-datapersist-3.0.1.jar\n")])])]),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("默认情况下,"),a("code",[t._v("jpa")]),t._v("配置文件在"),a("code",[t._v("application.yml")]),t._v("中启用。如果你想尝试"),a("br"),t._v("其他后端,请启用"),a("code",[t._v("mongo")]),t._v("配置文件或"),a("code",[t._v("redis")]),t._v("配置文件。"),a("br"),t._v("下面的命令指定使用哪个配置文件("),a("code",[t._v("jpa")]),t._v("是缺省的,"),a("br"),t._v("但为了完整起见我们将其包括在内):"),a("br"),a("br"),a("code",[t._v("<br/># java -jar spring-statemachine-samples-datapersist-3.0.1.jar --spring.profiles.active=jpa<br/># java -jar spring-statemachine-samples-datapersist-3.0.1.jar --spring.profiles.active=mongo<br/># java -jar spring-statemachine-samples-datapersist-3.0.1.jar --spring.profiles.active=redis<br/>")])])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v(""),a("a",{attrs:{href:"http://localhost:8080",target:"_blank",rel:"noopener noreferrer"}},[t._v("http://localhost:8080"),a("OutboundLink")],1),t._v("上访问应用程序会为每个请求创建一个新构造的状态机,你可以选择将事件发送到机器。可能的事件和机器配置都会随每个请求从数据库中更新。")]),t._v(" "),a("p",[t._v("这个示例中的状态机有一个简单的配置,其中状态的 1’到’6’,事件的’E1’到’E6’,以在这些状态之间转换状态机。可以使用两个状态机标识符("),a("code",[t._v("datajpapersist1")]),t._v(""),a("code",[t._v("datajpapersist2")]),t._v(")来请求特定的状态机。下图显示了允许你选择机器和事件的 UI,并显示了这样做时会发生什么:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpapersist-1.png",alt:"Sm DatajpaPainstive1"}})]),t._v(" "),a("p",[t._v("该示例默认使用机器“DataJpaPainstive1”,并转到其初始状态“1”。下图显示了使用这些默认值的结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpapersist-2.png",alt:"Sm DatajpaPastive2"}})]),t._v(" "),a("p",[t._v("如果将事件"),a("code",[t._v("E1")]),t._v(""),a("code",[t._v("E2")]),t._v("发送到"),a("code",[t._v("datajpapersist1")]),t._v("状态机,则其状态保持为’S3’。下图显示了这样做的结果:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-datajpapersist-3.png",alt:"SM DatajpaPainsive3"}})]),t._v(" "),a("p",[t._v("如果你随后请求状态机"),a("code",[t._v("datajpapersist1")]),t._v("但没有发送任何事件,则状态机将被恢复到其持久状态"),a("code",[t._v("S3")]),t._v("")]),t._v(" "),a("h2",{attrs:{id:"监测"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#监测"}},[t._v("#")]),t._v(" 监测")]),t._v(" "),a("p",[t._v("监视示例展示了如何使用状态机概念来监视状态机转换和操作。下面的清单配置了我们在此示例中使用的状态机:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@Configuration\n@EnableStateMachine\npublic static class Config extends StateMachineConfigurerAdapter<String, String> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<String, String> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial("S1")\n\t\t\t\t.state("S2", null, (c) -> {System.out.println("hello");})\n\t\t\t\t.state("S3", (c) -> {System.out.println("hello");}, null);\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<String, String> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source("S1").target("S2").event("E1")\n\t\t\t\t.action((c) -> {System.out.println("hello");})\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source("S2").target("S3").event("E2");\n\t}\n}\n')])])]),a("p",[t._v("你可以使用以下命令来运行示例:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("# java -jar spring-statemachine-samples-monitoring-3.0.1.jar\n")])])]),a("p",[t._v("下图显示了状态机的初始状态:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-monitoring-1.png",alt:"SM 监控 1"}})]),t._v(" "),a("p",[t._v("下图显示了在我们执行了一些操作之后状态机的状态:")]),t._v(" "),a("p",[a("img",{attrs:{src:"images/sm-monitoring-2.png",alt:"SM 监控 2"}})]),t._v(" "),a("p",[t._v("通过运行以下两个"),a("code",[t._v("curl")]),t._v("命令(如其输出所示),你可以从 Spring 启动中查看指标:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('# curl http://localhost:8080/actuator/metrics/ssm.transition.duration\n\n{\n  "name":"ssm.transition.duration",\n  "measurements":[\n    {\n      "statistic":"COUNT",\n      "value":3.0\n    },\n    {\n      "statistic":"TOTAL_TIME",\n      "value":0.007\n    },\n    {\n      "statistic":"MAX",\n      "value":0.004\n    }\n  ],\n  "availableTags":[\n    {\n      "tag":"transitionName",\n      "values":[\n        "INITIAL_S1",\n        "EXTERNAL_S1_S2"\n      ]\n    }\n  ]\n}\n')])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('# curl http://localhost:8080/actuator/metrics/ssm.transition.transit\n\n{\n  "name":"ssm.transition.transit",\n  "measurements":[\n    {\n      "statistic":"COUNT",\n      "value":3.0\n    }\n  ],\n  "availableTags":[\n    {\n      "tag":"transitionName",\n      "values":[\n        "EXTERNAL_S1_S2",\n        "INITIAL_S1"\n      ]\n    }\n  ]\n}\n')])])]),a("p",[t._v("你还可以通过运行以下"),a("code",[t._v("curl")]),t._v("命令(如其输出所示)来查看从 Spring 启动的跟踪:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('# curl http://localhost:8080/actuator/statemachinetrace\n\n[\n  {\n    "timestamp":"2018-02-11T06:44:12.723+0000",\n    "info":{\n      "duration":2,\n      "machine":null,\n      "transition":"EXTERNAL_S1_S2"\n    }\n  },\n  {\n    "timestamp":"2018-02-11T06:44:12.720+0000",\n    "info":{\n      "duration":0,\n      "machine":null,\n      "action":"demo.monitoring.StateMachineConfig$Config$$Lambda$576/[email protected]"\n    }\n  },\n  {\n    "timestamp":"2018-02-11T06:44:12.714+0000",\n    "info":{\n      "duration":1,\n      "machine":null,\n      "transition":"INITIAL_S1"\n    }\n  },\n  {\n    "timestamp":"2018-02-11T06:44:09.689+0000",\n    "info":{\n      "duration":4,\n      "machine":null,\n      "transition":"INITIAL_S1"\n    }\n  }\n]\n')])])]),a("h1",{attrs:{id:"faq"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#faq"}},[t._v("#")]),t._v(" FAQ")]),t._v(" "),a("p",[t._v("本章回答了 Spring 机器用户最常问的问题。")]),t._v(" "),a("h2",{attrs:{id:"状态变化"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态变化"}},[t._v("#")]),t._v(" 状态变化")]),t._v(" "),a("p",[t._v("我怎样才能自动转到下一个州呢?")]),t._v(" "),a("p",[t._v("你可以从三种方法中进行选择:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("实现一个操作,并向状态机发送一个适当的事件,以触发转换到适当的目标状态。")])]),t._v(" "),a("li",[a("p",[t._v("在一个状态中定义一个延迟事件,并在发送一个事件之前,发送另一个延迟的事件。这样做会在处理该事件更方便时导致下一个适当的状态转换。")])]),t._v(" "),a("li",[a("p",[t._v("实现无触发转换,当进入状态并完成其操作时,它会自动导致状态转换到下一个状态。")])])]),t._v(" "),a("h2",{attrs:{id:"扩展状态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#扩展状态"}},[t._v("#")]),t._v(" 扩展状态")]),t._v(" "),a("p",[t._v("如何在状态机启动时初始化变量?")]),t._v(" "),a("p",[t._v("状态机中的一个重要概念是,除非触发器导致可以触发动作的状态转换,否则不会真正发生任何事情。然而,话虽如此, Spring Statemachine 总是在状态机启动时具有初始转换。通过这个初始转换,你可以运行一个简单的操作,在"),a("code",[t._v("StateContext")]),t._v("中,它可以对扩展的状态变量执行任意操作。")]),t._v(" "),a("h1",{attrs:{id:"附录"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#附录"}},[t._v("#")]),t._v(" 附录")]),t._v(" "),a("h2",{attrs:{id:"附录-a-支持内容"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#附录-a-支持内容"}},[t._v("#")]),t._v(" 附录 A:支持内容")]),t._v(" "),a("p",[t._v("本附录提供了有关在此参考文档中使用的类和材料的通用信息。")]),t._v(" "),a("h3",{attrs:{id:"本文档中使用的类"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#本文档中使用的类"}},[t._v("#")]),t._v(" 本文档中使用的类")]),t._v(" "),a("p",[t._v("下面的列表显示了在整个参考指南中使用的类:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States {\n    SI,S1,S2,S3,S4,SF\n}\n")])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States2 {\n\tS1,S2,S3,S4,S5,SF,\n\tS2I,S21,S22,S2F,\n\tS3I,S31,S32,S3F\n}\n")])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States3 {\n    S1,S2,SH,\n    S2I,S21,S22,S2F\n}\n")])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum Events {\n    E1,E2,E3,E4,EF\n}\n")])])]),a("h2",{attrs:{id:"附录-b-状态机概念"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#附录-b-状态机概念"}},[t._v("#")]),t._v(" 附录 B:状态机概念")]),t._v(" "),a("p",[t._v("本附录提供了有关状态机的一般信息。")]),t._v(" "),a("h3",{attrs:{id:"快速示例"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#快速示例"}},[t._v("#")]),t._v(" 快速示例")]),t._v(" "),a("p",[t._v("假设我们有名为"),a("code",[t._v("STATE1")]),t._v(""),a("code",[t._v("STATE2")]),t._v("的状态,以及名为"),a("code",[t._v("EVENT1")]),t._v(""),a("code",[t._v("EVENT2")]),t._v("的事件,你可以定义状态机的逻辑,如下图所示:")]),t._v(" "),a("img",{attrs:{alt:"statechart0",src:"images/statechart0.png",width:"500"}}),t._v(" "),a("p",[t._v("下面的清单在前面的图像中定义了状态机:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public enum States {\n\tSTATE1, STATE2\n}\n\npublic enum Events {\n\tEVENT1, EVENT2\n}\n")])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("@Configuration\n@EnableStateMachine\npublic class Config1 extends EnumStateMachineConfigurerAdapter<States, Events> {\n\n\t@Override\n\tpublic void configure(StateMachineStateConfigurer<States, Events> states)\n\t\t\tthrows Exception {\n\t\tstates\n\t\t\t.withStates()\n\t\t\t\t.initial(States.STATE1)\n\t\t\t\t.states(EnumSet.allOf(States.class));\n\t}\n\n\t@Override\n\tpublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)\n\t\t\tthrows Exception {\n\t\ttransitions\n\t\t\t.withExternal()\n\t\t\t\t.source(States.STATE1).target(States.STATE2)\n\t\t\t\t.event(Events.EVENT1)\n\t\t\t\t.and()\n\t\t\t.withExternal()\n\t\t\t\t.source(States.STATE2).target(States.STATE1)\n\t\t\t\t.event(Events.EVENT2);\n\t}\n}\n")])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('@WithStateMachine\npublic class MyBean {\n\n\t@OnTransition(target = "STATE1")\n\tvoid toState1() {\n\t}\n\n\t@OnTransition(target = "STATE2")\n\tvoid toState2() {\n\t}\n}\n')])])]),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("public class MyApp {\n\n\t@Autowired\n\tStateMachine<States, Events> stateMachine;\n\n\tvoid doSignals() {\n\t\tstateMachine\n\t\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t\t.withPayload(Events.EVENT1).build()))\n\t\t\t.subscribe();\n\t\tstateMachine\n\t\t\t.sendEvent(Mono.just(MessageBuilder\n\t\t\t\t.withPayload(Events.EVENT2).build()))\n\t\t\t.subscribe();\n\t}\n}\n")])])]),a("h3",{attrs:{id:"术语表"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#术语表"}},[t._v("#")]),t._v(" 术语表")]),t._v(" "),a("p",[a("strong",[t._v("状态机")])]),t._v(" "),a("p",[t._v("驱动一组状态以及区域、转换和事件的主要实体。")]),t._v(" "),a("p",[a("strong",[t._v("国家")])]),t._v(" "),a("p",[t._v("状态模拟一种情况,在这种情况下,一定的不变条件成立。状态是状态机的主要实体,在状态机中,状态变化是由事件驱动的。")]),t._v(" "),a("p",[a("strong",[t._v("扩展状态")])]),t._v(" "),a("p",[t._v("扩展状态是保存在状态机中的一组特殊变量,以减少所需状态的数量。")]),t._v(" "),a("p",[a("strong",[t._v("过渡")])]),t._v(" "),a("p",[t._v("转换是源状态和目标状态之间的关系。它可能是复合转换的一部分,它将状态机从一个状态配置转换到另一个状态配置,表示状态机对特定类型事件发生的完整响应。")]),t._v(" "),a("p",[a("strong",[t._v("事件")])]),t._v(" "),a("p",[t._v("一种实体,它被发送到状态机,然后驱动各种状态变化。")]),t._v(" "),a("p",[a("strong",[t._v("初始状态")])]),t._v(" "),a("p",[t._v("状态机启动的一种特殊状态。初始状态总是绑定到特定的状态机或区域。具有多个区域的状态机可以具有多个初始状态。")]),t._v(" "),a("p",[a("strong",[t._v("结束状态")])]),t._v(" "),a("p",[t._v("(也称为最终状态)一种特殊的状态,表示封闭区域已完成。如果封闭区域直接包含在状态机中,并且状态机中的所有其他区域也完成了,则整个状态机就完成了。")]),t._v(" "),a("p",[a("strong",[t._v("历史状态")])]),t._v(" "),a("p",[t._v("一种伪状态,使状态机记住其上一次活动状态。存在两种类型的历史状态:"),a("em",[t._v("浅层")]),t._v("(只记住顶层状态)和"),a("em",[t._v("")]),t._v("(记住子机器中的活动状态)。")]),t._v(" "),a("p",[a("strong",[t._v("选择状态")])]),t._v(" "),a("p",[t._v("允许基于(例如)事件头或扩展状态变量进行转换选择的一种伪状态。")]),t._v(" "),a("p",[a("strong",[t._v("连接状态")])]),t._v(" "),a("p",[t._v("一种伪状态,与选择状态相对相似,但允许多个传入转换,而选择只允许一个传入转换。")]),t._v(" "),a("p",[a("strong",[t._v("分叉状态")])]),t._v(" "),a("p",[t._v("一种可控制地进入某一区域的伪状态。")]),t._v(" "),a("p",[a("strong",[t._v("加入状态")])]),t._v(" "),a("p",[t._v("一种伪状态,它使人可以控制地离开某一区域。")]),t._v(" "),a("p",[a("strong",[t._v("入口点")])]),t._v(" "),a("p",[t._v("允许受控进入子机的一种伪状态。")]),t._v(" "),a("p",[a("strong",[t._v("出口点")])]),t._v(" "),a("p",[t._v("允许受控退出子机的一种伪状态。")]),t._v(" "),a("p",[a("strong",[t._v("地区")])]),t._v(" "),a("p",[t._v("区域是复合状态或状态机的正交部分。它包含状态和转换。")]),t._v(" "),a("p",[a("strong",[t._v("守卫")])]),t._v(" "),a("p",[t._v("一种基于扩展状态变量和事件参数的值动态评估的布尔表达式。保护条件影响状态机的行为,只在它们的估值为"),a("code",[t._v("TRUE")]),t._v("时启用操作或转换,并在它们的估值为"),a("code",[t._v("FALSE")]),t._v("时禁用它们。")]),t._v(" "),a("p",[a("strong",[t._v("行动")])]),t._v(" "),a("p",[t._v("动作是在触发转换过程中运行的行为。")]),t._v(" "),a("h3",{attrs:{id:"状态机速成课程"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#状态机速成课程"}},[t._v("#")]),t._v(" 状态机速成课程")]),t._v(" "),a("p",[t._v("本附录为状态机概念提供了一个通用的速成课程。")]),t._v(" "),a("h4",{attrs:{id:"国家"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#国家"}},[t._v("#")]),t._v(" 国家")]),t._v(" "),a("p",[t._v("状态是状态机可以使用的模型。将状态描述为一个真实世界的例子总是比试图在文档中使用抽象的概念更容易。为此,请考虑一个简单的键盘示例——我们大多数人每天都使用一个键盘。如果你有一个完整的键盘,左边有正常键,右边有数字键盘,你可能已经注意到,数字键盘可能处于两种不同的状态,这取决于 Numlock 是否被激活。如果它不是活动的,按数字垫键将导致导航使用箭头等。如果数字键盘是活动的,按下这些键将导致数字被键入。从本质上讲,键盘的数字键盘部分可以处于两种不同的状态。")]),t._v(" "),a("p",[t._v("要将状态概念与编程联系起来,这意味着你可以使用状态、状态变量或与状态机的另一种交互,而不是使用标志、嵌套的 if/else/break 子句或其他不切实际(有时是曲折的)逻辑。")]),t._v(" "),a("h4",{attrs:{id:"伪态"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#伪态"}},[t._v("#")]),t._v(" 伪态")]),t._v(" "),a("p",[t._v("伪状态是一种特殊类型的状态,通常通过赋予状态一个特殊的含义(例如初始状态),将更高级的逻辑引入状态机。然后,状态机可以通过执行 UML 状态机概念中可用的各种操作,在内部对这些状态做出反应。")]),t._v(" "),a("h5",{attrs:{id:"首字母"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#首字母"}},[t._v("#")]),t._v(" 首字母")]),t._v(" "),a("p",[t._v("对于每个状态机,总是需要"),a("strong",[t._v("初始伪态")]),t._v("状态,无论你是拥有简单的一级状态机还是由子机或区域组成的更复杂的状态机。初始状态定义了状态机启动时的运行位置。没有它,状态机就不会成型。")]),t._v(" "),a("h5",{attrs:{id:""}},[a("a",{staticClass:"header-anchor",attrs:{href:"#完"}},[t._v("#")]),t._v("")]),t._v(" "),a("p",[a("strong",[t._v("终止伪态")]),t._v("(也称为“结束状态”)表示特定状态机已达到其最终状态。实际上,这意味着状态机不再处理任何事件,也不会传输到任何其他状态。然而,在子机为区域的情况下,状态机可以从其终端状态重新启动。")]),t._v(" "),a("h5",{attrs:{id:"选择"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#选择"}},[t._v("#")]),t._v(" 选择")]),t._v(" "),a("p",[t._v("你可以使用"),a("strong",[t._v("选择伪态")]),t._v("从这个状态中选择一个动态转换的条件分支。通过保护对动态条件进行评估,从而选择一条支路。通常使用简单的 if/elseif/else 结构来确保选择了一个分支。否则,状态机可能会陷入死锁,并且配置格式不正确。")]),t._v(" "),a("h5",{attrs:{id:"交点"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#交点"}},[t._v("#")]),t._v(" 交点")]),t._v(" "),a("p",[a("strong",[t._v("连接伪态")]),t._v("在功能上与 choice 相似,因为两者都是用 if/elseif/else 结构实现的。唯一真正的区别是,连接允许多个传入转换,而选择只允许一个。因此,差异在很大程度上是学术性的,但确实有一些差异,例如,在设计状态机时使用的是真实的 UI 建模框架。")]),t._v(" "),a("h5",{attrs:{id:"历史"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#历史"}},[t._v("#")]),t._v(" 历史")]),t._v(" "),a("p",[t._v("你可以使用"),a("strong",[t._v("历史伪国家")]),t._v("来记住最近的活动状态配置。在状态机退出后,你可以使用历史状态来恢复先前已知的配置。有两种类型的历史状态可用:"),a("code",[t._v("SHALLOW")]),t._v("(只记住状态机本身的活动状态)和"),a("code",[t._v("DEEP")]),t._v("(也记住嵌套状态)。")]),t._v(" "),a("p",[t._v("历史状态可以通过监听状态机事件来在外部实现,但是这将很快导致非常困难的逻辑,尤其是在状态机包含复杂的嵌套结构的情况下。让状态机本身来处理历史状态的记录,会使事情变得简单得多。用户只需创建一个转换到历史状态,状态机就会处理所需的逻辑,以返回到上次已知的记录状态。")]),t._v(" "),a("p",[t._v("如果转换在历史状态上终止,而该状态以前没有被输入(换句话说,不存在先前的历史)或它已经到达其结束状态,则转换可以通过使用默认的历史机制,强制状态机到特定子状态。此转换起源于历史状态,并终止于包含历史状态的区域的特定顶点(默认历史状态)。只有当其执行导致历史状态并且该状态以前从未处于活动状态时,才会进行此转换。否则,将执行进入该区域的正常历史记录。如果未定义缺省历史转换,则执行该区域的标准缺省条目。")]),t._v(" "),a("h5",{attrs:{id:"叉子"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#叉子"}},[t._v("#")]),t._v(" 叉子")]),t._v(" "),a("p",[t._v("你可以使用"),a("strong",[t._v("Fork 伪态")]),t._v("对一个或多个区域进行显式输入。下图显示了叉子的工作原理:")]),t._v(" "),a("img",{attrs:{alt:"statechart7",src:"images/statechart7.png",width:"500"}}),t._v(" "),a("p",[t._v("目标状态可以是承载区域的父状态,这仅意味着区域通过输入其初始状态而被激活。你还可以直接将目标添加到区域中的任何状态,这允许更多的受控进入状态。")]),t._v(" "),a("h5",{attrs:{id:"加入"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#加入"}},[t._v("#")]),t._v(" 加入")]),t._v(" "),a("p",[a("strong",[t._v("加入伪状态")]),t._v("将源自不同区域的几个转换合并在一起。它通常用于等待和阻止参与区域进入其加入目标状态。下图显示了连接的工作原理:")]),t._v(" "),a("img",{attrs:{alt:"statechart8",src:"images/statechart8.png",width:"500"}}),t._v(" "),a("p",[t._v("源状态可以是承载区域的父状态,这意味着连接状态是参与区域的终端状态。你还可以将源状态定义为区域中的任何状态,这允许受控地退出区域。")]),t._v(" "),a("h5",{attrs:{id:"切入点"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#切入点"}},[t._v("#")]),t._v(" 切入点")]),t._v(" "),a("p",[a("strong",[t._v("入口点伪态")]),t._v("表示提供状态机或状态机内部封装的状态机或复合状态的入口点。在 OWNS 入口点的状态机或复合状态的每个区域中,在该区域内最多只有一个从入口点到顶点的转换。")]),t._v(" "),a("h5",{attrs:{id:"退出点"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#退出点"}},[t._v("#")]),t._v(" 退出点")]),t._v(" "),a("p",[a("strong",[t._v("出口点伪状态")]),t._v("是状态机或复合状态的出口点,它提供了状态机或状态机内部的封装。在复合状态的任何区域(或由子机状态引用的状态机)的出口点终止的转换意味着退出该复合状态或子机状态(执行其相关的退出行为)。")]),t._v(" "),a("h4",{attrs:{id:"防范条件"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#防范条件"}},[t._v("#")]),t._v(" 防范条件")]),t._v(" "),a("p",[t._v("保护条件是根据扩展的状态变量和事件参数计算为"),a("code",[t._v("TRUE")]),t._v(""),a("code",[t._v("FALSE")]),t._v("的表达式。保护与动作和转换一起使用,以动态地选择是否应该运行特定的动作或转换。各种防护措施、事件参数和扩展的状态变量的存在使状态机的设计更加简单。")]),t._v(" "),a("h4",{attrs:{id:"事件"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#事件"}},[t._v("#")]),t._v(" 事件")]),t._v(" "),a("p",[t._v("事件是用于驱动状态机的最常用的触发行为。还有其他方法可以触发状态机中的行为(例如计时器),但事件才是真正让用户与状态机交互的方法。事件也被称为“信号”。它们基本上表示可能改变状态机状态的东西。")]),t._v(" "),a("h4",{attrs:{id:"转换"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#转换"}},[t._v("#")]),t._v(" 转换")]),t._v(" "),a("p",[t._v("转换是源状态和目标状态之间的关系。从一种状态切换到另一种状态是由触发器引起的状态转换。")]),t._v(" "),a("h5",{attrs:{id:"内部转换"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#内部转换"}},[t._v("#")]),t._v(" 内部转换")]),t._v(" "),a("p",[t._v("当需要运行一个操作而不需要引起状态转换时,使用内部转换。在内部转换中,源状态和目标状态总是相同的,并且在没有状态进入和退出动作的情况下,它与自转换相同。")]),t._v(" "),a("h5",{attrs:{id:"外部与局部转换"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#外部与局部转换"}},[t._v("#")]),t._v(" 外部与局部转换")]),t._v(" "),a("p",[t._v("在大多数情况下,外部和局部转换在功能上是等价的,除非转换发生在超级状态和次状态之间。如果目标状态是源状态的子状态,则局部转换不会导致源状态的退出和进入。相反,如果目标是源状态的超状态,则局部转换不会导致目标状态的退出和进入。下面的图像显示了具有非常简单的超级和次状态的局部和外部转换之间的区别:")]),t._v(" "),a("img",{attrs:{alt:"statechart4",src:"images/statechart4.png",width:"500"}}),t._v(" "),a("h4",{attrs:{id:"触发器"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#触发器"}},[t._v("#")]),t._v(" 触发器")]),t._v(" "),a("p",[t._v("触发器开始转换。触发器可以由事件驱动,也可以由计时器驱动。")]),t._v(" "),a("h4",{attrs:{id:"动作"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#动作"}},[t._v("#")]),t._v(" 动作")]),t._v(" "),a("p",[t._v("动作实际上是将状态机状态更改粘附到用户自己的代码中。状态机可以对状态机中的各种更改和步骤(例如进入或退出状态)或进行状态转换运行操作。")]),t._v(" "),a("p",[t._v("动作通常具有对状态上下文的访问权限,这使运行中的代码可以选择以各种方式与状态机交互。状态上下文公开了整个状态机,因此用户可以访问扩展的状态变量、事件头(如果转换是基于事件的话),或者是一种实际的转换(在这种转换中,可以看到更详细的关于这种状态变化来自何处以及它将走向何处的信息)。")]),t._v(" "),a("h4",{attrs:{id:"分层状态机"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#分层状态机"}},[t._v("#")]),t._v(" 分层状态机")]),t._v(" "),a("p",[t._v("当特定的状态必须同时存在时,层次状态机的概念被用来简化状态设计。")]),t._v(" "),a("p",[t._v("分层状态实际上是 UML 状态机相对于传统状态机(如 Mealy 或 Moore)的一种创新。层次结构状态允许你定义某种级别的抽象(类似于 Java 开发人员如何使用抽象类定义类结构)。例如,使用嵌套状态机,你可以在多个状态级别上定义转换(可能具有不同的条件)。状态机总是尝试查看当前状态是否能够处理事件,以及转换保护条件。如果这些条件不求值到"),a("code",[t._v("TRUE")]),t._v(",状态机只看到超级状态可以处理什么。")]),t._v(" "),a("h4",{attrs:{id:"区域"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#区域"}},[t._v("#")]),t._v(" 区域")]),t._v(" "),a("p",[t._v("区域(也称为正交区域)通常被视为应用于状态的排他或(异或)操作。用状态机表示的区域的概念通常有点难以理解,但是通过一个简单的示例,事情会变得简单一些。")]),t._v(" "),a("p",[t._v("我们中的一些人拥有全尺寸键盘,主键在左侧,数字键在右侧。你可能已经注意到,双方都有自己的状态,如果你按下一个“Numlock”键(它只会改变数字键盘本身的行为),你就会看到这种状态。如果你没有一个全尺寸的键盘,你可以购买一个外部 USB 号码板。考虑到键盘的左侧和右侧可以彼此独立存在,它们必须具有完全不同的状态,这意味着它们在不同的状态机上运行。在状态机术语中,键盘的主要部分是一个区域,而数字键盘是另一个区域。")]),t._v(" "),a("p",[t._v("将两个不同的状态机作为完全独立的实体来处理会有点不方便,因为它们仍然以某种方式一起工作。这种独立性允许正交区域在状态机的单个状态中以多个同时状态合并在一起。")]),t._v(" "),a("h2",{attrs:{id:"附录-c-分布式状态机技术论文"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#附录-c-分布式状态机技术论文"}},[t._v("#")]),t._v(" 附录 C:分布式状态机技术论文")]),t._v(" "),a("p",[t._v("本附录提供了关于使用 Spring Statemachine 的 ZooKeeper 实例的更详细的技术文档。")]),t._v(" "),a("h3",{attrs:{id:"摘要"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#摘要"}},[t._v("#")]),t._v(" 摘要")]),t._v(" "),a("p",[t._v("在单个 JVM 上运行的单个状态机实例之上引入“分布式状态”是一个困难而复杂的主题。“分布式状态机”的概念在简单状态机的基础上引入了一些相对复杂的问题,这是由于它的运行到完成模型,以及更普遍的是由于它的单线程执行模型,尽管正交区域可以并行运行。另一个自然的问题是,状态机转换执行是由触发器驱动的,触发器基于"),a("code",[t._v("event")]),t._v(""),a("code",[t._v("timer")]),t._v("")]),t._v(" "),a("p",[t._v("Spring 状态机试图通过支持分布式状态机来解决通过 JVM 边界跨越通用“状态机”的问题。在这里,我们展示了你可以在多个 JVM 和 Spring 应用程序上下文中使用通用的“状态机”概念。")]),t._v(" "),a("p",[t._v("我们发现,如果"),a("code",[t._v("Distributed State Machine")]),t._v("抽象是精心选择的,并且支持分布式状态库保证"),a("code",[t._v("CP")]),t._v("就绪,那么就有可能创建一个一致的状态机,该状态机可以在集成中的其他状态机之间共享分布式状态。")]),t._v(" "),a("p",[t._v("我们的结果表明,如果支持存储库是“CP”(讨论"),a("a",{attrs:{href:"#state-machine-technical-paper-introduction"}},[t._v("later")]),t._v("),则分布式状态更改是一致的。我们期望我们的分布式状态机能够为需要处理共享分布式状态的应用程序提供一个基础。该模型旨在为云应用程序提供更好的方法,使其能够更容易地相互通信,而无需显式地构建这些分布式状态概念。")]),t._v(" "),a("h3",{attrs:{id:"引言-2"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#引言-2"}},[t._v("#")]),t._v(" 引言")]),t._v(" "),a("p",[t._v("Spring 状态机不强制使用单线程执行模型,因为,一旦使用了多个区域,如果应用了必要的配置,则可以并行执行区域。这是一个重要的主题,因为一旦用户想要执行并行状态机,它就会使独立区域的状态更改更快。")]),t._v(" "),a("p",[t._v("当状态更改不再由本地 JVM 或本地状态机实例中的触发器驱动时,转换逻辑需要在任意持久存储中进行外部控制。该存储需要有一种方法,在分布式状态更改时通知参与状态机。")]),t._v(" "),a("p",[a("a",{attrs:{href:"https://en.wikipedia.org/wiki/CAP_theorem",target:"_blank",rel:"noopener noreferrer"}},[t._v("上限定理"),a("OutboundLink")],1),t._v("指出,分布式计算机系统不可能同时提供以下三种保证:一致性、可用性和分区容限。")]),t._v(" "),a("p",[t._v("这意味着,无论选择什么作为支持持久性存储,最好是“CP”。在这种情况下,“CP”的意思是“一致性”和“分区容忍”。很自然,分布式服务器并不关心它的“上限”级别,但在现实中,“一致性”和“分区容忍度”比“可用性”更重要。这就是为什么(例如)ZooKeeper 使用“CP”存储的确切原因。")]),t._v(" "),a("p",[t._v("本文中介绍的所有测试都是通过在以下环境中运行定制的 Jepsen 测试来完成的:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("具有节点 n1、n2、n3、n4 和 n5 的簇。")])]),t._v(" "),a("li",[a("p",[t._v("每个节点都有一个"),a("code",[t._v("Zookeeper")]),t._v("实例,该实例与所有其他节点一起构造一个集成。")])]),t._v(" "),a("li",[a("p",[t._v("每个节点都安装了"),a("a",{attrs:{href:"#statemachine-examples-web"}},[t._v("Web")]),t._v("示例,以连接到本地的"),a("code",[t._v("Zookeeper")]),t._v("节点。")])]),t._v(" "),a("li",[a("p",[t._v("每个状态机实例仅与本地"),a("code",[t._v("Zookeeper")]),t._v("实例通信。虽然将一台机器连接到多个实例是可能的,但此处不使用它。")])]),t._v(" "),a("li",[a("p",[t._v("启动所有状态机实例时,使用 ZooKeeper 集成创建"),a("code",[t._v("StateMachineEnsemble")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("每个示例都包含一个自定义 REST API,Jepsen 使用该 API 发送事件并检查特定的状态机状态。")])])]),t._v(" "),a("p",[a("code",[t._v("Spring Distributed Statemachine")]),t._v("的所有 Jepsen 测试都可以从"),a("a",{attrs:{href:"https://github.com/spring-projects/spring-statemachine/tree/master/jepsen/spring-statemachine-jepsen",target:"_blank",rel:"noopener noreferrer"}},[t._v("杰普森测试。"),a("OutboundLink")],1),t._v("获得")]),t._v(" "),a("h3",{attrs:{id:"通用概念"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#通用概念"}},[t._v("#")]),t._v(" 通用概念")]),t._v(" "),a("p",[a("code",[t._v("Distributed State Machine")]),t._v("的一个设计决策是,不要让每个单独的状态机实例意识到它是“分布式集成”的一部分。因为"),a("code",[t._v("StateMachine")]),t._v("的主要功能和特性可以通过其接口访问,所以将此实例包装在"),a("code",[t._v("DistributedStateMachine")]),t._v("中是有意义的,该实例拦截所有状态机通信,并与一个集成协作来协调分布式状态更改。")]),t._v(" "),a("p",[t._v("另一个重要的概念是能够保存来自状态机的足够信息,以便将状态机状态从任意状态重置为新的反序列化状态。当一个新的状态机实例与一个集成连接并需要将其内部状态与分布式状态同步时,这是自然需要的。与使用分布式状态和状态持久化的概念一起,创建分布式状态机是可能的。目前,"),a("code",[t._v("Distributed State Machine")]),t._v("的唯一支持存储库是通过使用 ZooKeeper 实现的。")]),t._v(" "),a("p",[t._v("正如"),a("a",{attrs:{href:"#sm-distributed"}},[t._v("使用分布状态")]),t._v("中提到的,通过将"),a("code",[t._v("StateMachine")]),t._v("的实例包装在"),a("code",[t._v("DistributedStateMachine")]),t._v("中来启用分布式状态。具体的"),a("code",[t._v("StateMachineEnsemble")]),t._v("实现是"),a("code",[t._v("ZookeeperStateMachineEnsemble")]),t._v("提供与 ZooKeeper 的集成。")]),t._v(" "),a("h3",{attrs:{id:"zookeeperstatemachinepersist的作用"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#zookeeperstatemachinepersist的作用"}},[t._v("#")]),t._v(" "),a("code",[t._v("ZookeeperStateMachinePersist")]),t._v("的作用")]),t._v(" "),a("p",[t._v("我们希望有一个通用接口("),a("code",[t._v("StateMachinePersist")]),t._v("),它可以将"),a("code",[t._v("StateMachineContext")]),t._v("持久化到任意存储中,并且"),a("code",[t._v("ZookeeperStateMachinePersist")]),t._v(""),a("code",[t._v("Zookeeper")]),t._v("实现这个接口。")]),t._v(" "),a("h3",{attrs:{id:"zookeeperstatemachineensemble的作用"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#zookeeperstatemachineensemble的作用"}},[t._v("#")]),t._v(" "),a("code",[t._v("ZookeeperStateMachineEnsemble")]),t._v("的作用")]),t._v(" "),a("p",[t._v("虽然分布式状态机使用一组序列化的上下文来更新其自身的状态,但在使用 ZooKeeper 时,我们遇到了一个关于如何侦听这些上下文更改的概念性问题。我们可以将上下文序列化到 ZooKeeper"),a("code",[t._v("znode")]),t._v("中,并最终在"),a("code",[t._v("znode")]),t._v("数据被修改时进行监听。但是,"),a("code",[t._v("Zookeeper")]),t._v("并不能保证每次数据更改都会收到通知,因为对于"),a("code",[t._v("znode")]),t._v("已注册的"),a("code",[t._v("watcher")]),t._v("一旦触发就会被禁用,并且用户需要重新注册"),a("code",[t._v("watcher")]),t._v("。在此短时间内,"),a("code",[t._v("znode")]),t._v("数据可以被更改,从而导致事件丢失。通过以并发方式更改来自多个线程的数据,实际上很容易错过这些事件。")]),t._v(" "),a("p",[t._v("为了克服这个问题,我们将单个上下文的更改保持在多个"),a("code",[t._v("znodes")]),t._v("中,并使用一个简单的整数计数器来标记"),a("code",[t._v("znode")]),t._v("是当前活动的。这样做可以让我们重播错过的事件。我们不希望创建越来越多的 Znode,然后稍后删除旧的 Znode。相反,我们使用了一个简单的 Znode 循环集的概念。这使我们能够使用预定义的 Znodes 集,其中当前节点可以通过一个简单的整数计数器来确定。通过跟踪主"),a("code",[t._v("znode")]),t._v("数据版本(在"),a("code",[t._v("Zookeeper")]),t._v("中,它是一个整数),我们已经拥有了这个计数器。")]),t._v(" "),a("p",[t._v("循环缓冲区的大小被要求为 2 的幂,以避免整数溢出时出现麻烦。因此,我们不需要处理任何具体的案件。")]),t._v(" "),a("h3",{attrs:{id:"分布式公差"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#分布式公差"}},[t._v("#")]),t._v(" 分布式公差")]),t._v(" "),a("p",[t._v("为了展示在实际生活中针对状态机的各种分布式操作是如何工作的,我们使用一组 Jepsen 测试来模拟在实际分布式集群中可能发生的各种情况。其中包括网络层面上的“大脑分裂”,多个“分布式状态机”的并行事件,以及“扩展状态变量”的变化。Jepsen 测试基于"),a("a",{attrs:{href:"#statemachine-examples-web"}},[t._v("Web")]),t._v("示例,其中此示例实例在多个主机上运行,在运行状态机的每个节点上都有一个 ZooKeeper 实例。从本质上讲,每个状态机样本都连接到一个本地 ZooKeeper 实例,这使我们能够通过使用 Jepsen 来模拟网络条件。")]),t._v(" "),a("p",[t._v("本章后面所示的图形包含直接映射到状态图的状态和事件,可以在"),a("a",{attrs:{href:"#statemachine-examples-web"}},[t._v("Web")]),t._v("中找到。")]),t._v(" "),a("h4",{attrs:{id:"孤立事件"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#孤立事件"}},[t._v("#")]),t._v(" 孤立事件")]),t._v(" "),a("p",[t._v("将一个孤立的单个事件发送到集成中的一个状态机中是最简单的测试场景,并且演示了一个状态机中的状态更改在集成中被正确地传播到其他状态机中。")]),t._v(" "),a("p",[t._v("在此测试中,我们演示了一台机器中的状态更改最终会导致其他机器中的状态更改一致。下图显示了测试状态机的事件和状态更改:")]),t._v(" "),a("img",{attrs:{alt:"sm tech isolated events",src:"images/sm-tech-isolated-events.png",width:"500"}}),t._v(" "),a("p",[t._v("在前面的图片中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("所有机器都报告状态"),a("code",[t._v("S21")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("事件"),a("code",[t._v("I")]),t._v("被发送到节点"),a("code",[t._v("n1")]),t._v(",并且所有节点报告状态从"),a("code",[t._v("S21")]),t._v("更改为"),a("code",[t._v("S22")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("事件"),a("code",[t._v("C")]),t._v("被发送到节点"),a("code",[t._v("n2")]),t._v(",并且所有节点报告状态从"),a("code",[t._v("S22")]),t._v("更改为"),a("code",[t._v("S211")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("事件"),a("code",[t._v("I")]),t._v("被发送到节点"),a("code",[t._v("n5")]),t._v(",并且所有节点报告状态从"),a("code",[t._v("S211")]),t._v("更改为"),a("code",[t._v("S212")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("事件"),a("code",[t._v("K")]),t._v("被发送到节点"),a("code",[t._v("n3")]),t._v(",并且所有节点报告状态从"),a("code",[t._v("S212")]),t._v("更改为"),a("code",[t._v("S21")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("我们循环事件"),a("code",[t._v("I")]),t._v(""),a("code",[t._v("C")]),t._v(""),a("code",[t._v("I")]),t._v(",和"),a("code",[t._v("K")]),t._v("一次,通过随机节点。")])])]),t._v(" "),a("h4",{attrs:{id:"并行事件"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#并行事件"}},[t._v("#")]),t._v(" 并行事件")]),t._v(" "),a("p",[t._v("多个分布式状态机的一个逻辑问题是,如果同一事件在完全相同的时间被发送到多个状态机,那么其中只有一个事件会导致分布式状态转换。这在某种程度上是一种预期的场景,因为能够更改分布式状态的第一个状态机(对于此事件)控制分布式转换逻辑。实际上,所有接收到相同事件的其他机器都会默默地丢弃该事件,因为分布式状态不再处于可以处理特定事件的状态。")]),t._v(" "),a("p",[t._v("在下图所示的测试中,我们证明了在整个集合中由并行事件引起的状态变化最终会在所有机器中引起一致的状态变化:")]),t._v(" "),a("img",{attrs:{alt:"sm tech parallel events",src:"images/sm-tech-parallel-events.png",width:"500"}}),t._v(" "),a("p",[t._v("在前面的图像中,我们使用了与前面的示例中使用的相同的事件流("),a("a",{attrs:{href:"#sm-tech-isolated-events"}},[t._v("孤立事件")]),t._v("),不同的是,事件总是被发送到所有节点。")]),t._v(" "),a("h4",{attrs:{id:"并发扩展状态变量更改"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#并发扩展状态变量更改"}},[t._v("#")]),t._v(" 并发扩展状态变量更改")]),t._v(" "),a("p",[t._v("扩展状态机变量不能保证在任何给定的时间都是原子的,但是,在分布式状态更改之后,集成中的所有状态机都应该具有同步的扩展状态。")]),t._v(" "),a("p",[t._v("在这个测试中,我们证明了在一个分布式状态机中扩展状态变量的变化最终会在所有分布式状态机中变得一致。下图显示了这个测试:")]),t._v(" "),a("img",{attrs:{alt:"sm tech isolated events with variable",src:"images/sm-tech-isolated-events-with-variable.png",width:"500"}}),t._v(" "),a("p",[t._v("在前面的图片中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("事件"),a("code",[t._v("J")]),t._v("是发送到节点"),a("code",[t._v("n5")]),t._v(",带有事件变量"),a("code",[t._v("testVariable")]),t._v("的值"),a("code",[t._v("v1")]),t._v("。然后,所有节点报告具有一个名为"),a("code",[t._v("testVariable")]),t._v("的变量,其值为"),a("code",[t._v("v1")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("事件"),a("code",[t._v("J")]),t._v("从变量"),a("code",[t._v("v2")]),t._v("重复到"),a("code",[t._v("v8")]),t._v(",执行相同的检查。")])])]),t._v(" "),a("h4",{attrs:{id:"分区公差"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#分区公差"}},[t._v("#")]),t._v(" 分区公差")]),t._v(" "),a("p",[t._v("我们需要始终假定,集群中的事情迟早会出问题,无论是 ZooKeeper 实例的崩溃、状态机的崩溃,还是“大脑分裂”之类的网络问题。(大脑分裂是指现有集群成员被隔离,因此只有部分宿主能够看到彼此的情况)。通常的情况是,大脑分裂会造成一个整体的少数和多数分割,这样在网络状态得到修复之前,少数群体中的主机无法参与一个整体。")]),t._v(" "),a("p",[t._v("在下面的测试中,我们证明了在集合中分裂的各种类型的大脑最终会导致所有分布状态机处于完全同步的状态。")]),t._v(" "),a("p",[t._v("在网络中,有两种情况是直脑分割的,其中"),a("code",[t._v("Zookeeper")]),t._v(""),a("code",[t._v("Statemachine")]),t._v("实例被分割为两半(假设每个"),a("code",[t._v("Statemachine")]),t._v("实例都连接到本地"),a("code",[t._v("Zookeeper")]),t._v("实例):")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("如果当前的动物园管理员领导保持在多数,那么所有与多数相关的客户都会保持正常运行。")])]),t._v(" "),a("li",[a("p",[t._v("如果当前的 ZooKeeper 领导者是少数人,所有客户都会与其断开连接,并尝试重新连接,直到以前的少数人成员成功地重新加入现有的多数人团队。")])])]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("在我们当前的 Jepsen 测试中,我们不能将 ZooKeeper Split-Brain"),a("br"),t._v("中的领导者处于多数或少数之间的情况分开,因此我们需要"),a("br"),t._v("多次运行测试来完成这种情况。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("在下面的图中,我们将状态机错误状态映射到"),a("code",[t._v("error")]),t._v(",以表示状态机处于错误状态,而不是"),a("br"),t._v("正常状态。在解释图表状态时,请记住这一点。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("在第一个测试中,我们展示了,当一个现有的动物园管理员领导者被保留在大多数情况下,五分之三的机器继续保持原状。下图显示了这个测试:")]),t._v(" "),a("img",{attrs:{alt:"sm tech partition half 1",src:"images/sm-tech-partition-half-1.png",width:"500"}}),t._v(" "),a("p",[t._v("在前面的图片中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("第一个事件"),a("code",[t._v("C")]),t._v("被发送到所有机器,导致状态更改为"),a("code",[t._v("S211")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("Jepsen Nemesis 导致大脑分裂,这会导致"),a("code",[t._v("n1/n2/n5")]),t._v(""),a("code",[t._v("n3/n4")]),t._v("的分裂。节点"),a("code",[t._v("n3/n4")]),t._v("留在少数,节点"),a("code",[t._v("n1/n2/n5")]),t._v("构造新的健康多数。大多数节点保持正常运行,没有问题,但少数节点进入错误状态。")])]),t._v(" "),a("li",[a("p",[t._v("Jepsen 修复了网络,一段时间后,节点"),a("code",[t._v("n3/n4")]),t._v("重新加入到集成中,并同步其分布式状态。")])]),t._v(" "),a("li",[a("p",[t._v("最后,将事件"),a("code",[t._v("K1")]),t._v("发送到所有状态机,以确保集成正常工作。此状态更改将返回到状态"),a("code",[t._v("S21")]),t._v("")])])]),t._v(" "),a("p",[t._v("在第二个测试中,我们表明,当现有的动物园管理员领导者保持在少数时,所有的机器都会出错。下图显示了第二个测试:")]),t._v(" "),a("img",{attrs:{alt:"sm tech partition half 2",src:"images/sm-tech-partition-half-2.png",width:"500"}}),t._v(" "),a("p",[t._v("在前面的图片中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("第一个事件"),a("code",[t._v("C")]),t._v("被发送到所有机器,导致状态更改为"),a("code",[t._v("S211")]),t._v("")])]),t._v(" "),a("li",[a("p",[t._v("Jepsen Nemesis 会导致大脑分裂,这会导致分割,使得现有的"),a("code",[t._v("Zookeeper")]),t._v("领导者保持在少数,并且所有实例都与集合断开连接。")])]),t._v(" "),a("li",[a("p",[t._v("Jepsen 修复了网络,一段时间后,所有节点都会重新加入到集成中,并同步其分布式状态。")])]),t._v(" "),a("li",[a("p",[t._v("最后,将事件"),a("code",[t._v("K1")]),t._v("发送到所有状态机,以确保 Ensemble 正常工作。此状态更改将返回到状态"),a("code",[t._v("S21")]),t._v("")])])]),t._v(" "),a("h4",{attrs:{id:"崩溃和连接公差"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#崩溃和连接公差"}},[t._v("#")]),t._v(" 崩溃和连接公差")]),t._v(" "),a("p",[t._v("在这个测试中,我们演示了杀死一个现有的状态机,然后将一个新实例重新加入到一个集成中,可以保持分布式状态的健康,并且新加入的状态机可以正确地同步它们的状态。下图显示了碰撞和连接公差测试:")]),t._v(" "),a("img",{attrs:{alt:"sm tech stop start",src:"images/sm-tech-stop-start.png",width:"500"}}),t._v(" "),a("table",[a("thead",[a("tr",[a("th"),t._v(" "),a("th",[t._v("在此测试中,不首先检查"),a("code",[t._v("X")]),t._v("与最后"),a("code",[t._v("X")]),t._v("之间的状态。"),a("br"),t._v("因此,该图显示了介于两者之间的一条平直线。检查状态"),a("br"),t._v(""),a("code",[t._v("S21")]),t._v("之间状态变化的确切位置。")])])]),t._v(" "),a("tbody")]),t._v(" "),a("p",[t._v("在前面的图片中:")]),t._v(" "),a("ul",[a("li",[a("p",[t._v("所有状态机都从初始状态("),a("code",[t._v("S21")]),t._v(")转换为状态"),a("code",[t._v("S211")]),t._v(",这样我们就可以在连接期间测试适当的状态同步。")])]),t._v(" "),a("li",[a("p",[a("code",[t._v("X")]),t._v("标记特定节点已崩溃并启动的时间。")])]),t._v(" "),a("li",[a("p",[t._v("同时,我们请求所有机器的状态并绘制结果。")])]),t._v(" "),a("li",[a("p",[t._v("最后,我们做一个简单的转换,从"),a("code",[t._v("S211")]),t._v("返回"),a("code",[t._v("S21")]),t._v(",以确保所有状态机仍然正常工作。")])])]),t._v(" "),a("h2",{attrs:{id:"开发人员文档"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#开发人员文档"}},[t._v("#")]),t._v(" 开发人员文档")]),t._v(" "),a("p",[t._v("本附录为可能想要参与的开发人员或其他想要理解状态机如何工作或理解其内部概念的人员提供了通用信息。")]),t._v(" "),a("h3",{attrs:{id:"statemachine-配置模型"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#statemachine-配置模型"}},[t._v("#")]),t._v(" statemachine 配置模型")]),t._v(" "),a("p",[a("code",[t._v("StateMachineModel")]),t._v("和其他相关的 SPI 类是各种配置和工厂类之间的抽象。这也允许其他人更容易地集成来构建状态机。")]),t._v(" "),a("p",[t._v("如下面的清单所示,你可以通过使用配置数据类构建一个模型,然后要求工厂构建一个状态机来实例化状态机:")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('// setup configuration data\nConfigurationData<String, String> configurationData = new ConfigurationData<>();\n\n// setup states data\nCollection<StateData<String, String>> stateData = new ArrayList<>();\nstateData.add(new StateData<String, String>("S1", true));\nstateData.add(new StateData<String, String>("S2"));\nStatesData<String, String> statesData = new StatesData<>(stateData);\n\n// setup transitions data\nCollection<TransitionData<String, String>> transitionData = new ArrayList<>();\ntransitionData.add(new TransitionData<String, String>("S1", "S2", "E1"));\nTransitionsData<String, String> transitionsData = new TransitionsData<>(transitionData);\n\n// setup model\nStateMachineModel<String, String> stateMachineModel = new DefaultStateMachineModel<>(configurationData, statesData,\n\t\ttransitionsData);\n\n// instantiate machine via factory\nObjectStateMachineFactory<String, String> factory = new ObjectStateMachineFactory<>(stateMachineModel);\nStateMachine<String, String> stateMachine = factory.getStateMachine();\n')])])]),a("h2",{attrs:{id:"附录-d-反应堆迁移指南"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#附录-d-反应堆迁移指南"}},[t._v("#")]),t._v(" 附录 D:反应堆迁移指南")]),t._v(" "),a("p",[a("code",[t._v("3.x")]),t._v("工作的主要任务是在内部和外部尽可能多地从命令式代码移动和更改到一个反应世界。这意味着一些主接口已经添加了新的 reative 方法,并且大多数内部执行 LOCIG(在适用的情况下)已经被转移到由反应堆处理。本质上,这意味着线程处理模型与"),a("code",[t._v("2.x")]),t._v("相比有很大的不同。下面几章将对所有这些变化进行回顾。")]),t._v(" "),a("h3",{attrs:{id:"与机器通信"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#与机器通信"}},[t._v("#")]),t._v(" 与机器通信")]),t._v(" "),a("p",[t._v("我们在"),a("code",[t._v("StateMachine")]),t._v("中添加了新的反应方法,同时仍然保留了旧的阻塞事件方法。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("Flux<StateMachineEventResult<S, E>> sendEvent(Mono<Message<E>> event);\n\nFlux<StateMachineEventResult<S, E>> sendEvents(Flux<Message<E>> events);\n\nMono<List<StateMachineEventResult<S, E>>> sendEventCollect(Mono<Message<E>> event);\n")])])]),a("p",[t._v("我们现在只研究 Spring "),a("code",[t._v("Message")]),t._v("和反应堆"),a("code",[t._v("Mono")]),t._v(""),a("code",[t._v("Flux")]),t._v("类。你可以发送一个"),a("code",[t._v("Mono")]),t._v(""),a("code",[t._v("Message")]),t._v(",并接收回一个"),a("code",[t._v("Flux")]),t._v(""),a("code",[t._v("StateMachineEventResult")]),t._v("。请记住,在订阅"),a("code",[t._v("Flux")]),t._v("之前,不会发生任何事情。有关此返回值的更多信息,请参见"),a("a",{attrs:{href:"#sm-triggers-statemachineeventresult"}},[t._v("Statemachineeversult")]),t._v("。方法"),a("code",[t._v("sendEventCollect")]),t._v("只是一个语法糖,用于传递"),a("code",[t._v("Mono")]),t._v("并获得"),a("code",[t._v("Mono")]),t._v(",将结果包装为列表。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('Message<String> message = MessageBuilder.withPayload("EVENT").build();\nmachine.sendEvent(Mono.just(message)).subscribe();\n')])])]),a("p",[t._v("你还可以发送"),a("code",[t._v("Flux")]),t._v("消息,而不是单个"),a("code",[t._v("Mono")]),t._v("消息。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v("machine.sendEvents(Flux.just(message)).subscribe();\n")])])]),a("p",[t._v("所有的反应堆方法都在你的处置中,例如,当事件处理完成时,不要阻塞并执行某些操作,你可以执行类似的操作。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('Mono<Message<String>> mono = Mono.just(MessageBuilder.withPayload("EVENT").build());\nmachine.sendEvent(mono)\n\t.doOnComplete(() -> {\n\t\tSystem.out.println("Event handling complete");\n\t})\n\t.subscribe();\n')])])]),a("p",[t._v("对于已接受的状态,返回"),a("code",[t._v("boolean")]),t._v("的旧 API 方法仍然存在,但不建议在将来的版本中删除。")]),t._v(" "),a("div",{staticClass:"language- extra-class"},[a("pre",{pre:!0,attrs:{class:"language-text"}},[a("code",[t._v('boolean accepted = machine.sendEvent("EVENT");\n')])])]),a("h3",{attrs:{id:"任务执行器和任务调度程序"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#任务执行器和任务调度程序"}},[t._v("#")]),t._v(" 任务执行器和任务调度程序")]),t._v(" "),a("p",[t._v("具有"),a("code",[t._v("TaskExecutor")]),t._v("的静态机器执行和具有"),a("code",[t._v("TaskScheduler")]),t._v("的状态动作调度已被完全取代,以利于反应堆的执行和调度。")]),t._v(" "),a("p",[t._v("从本质上讲,在主线程之外的执行需要在两个地方执行,首先是 "),a("em",[t._v("state actions")]),t._v(",它需要是可删除的,其次是"),a("em",[t._v("地区")]),t._v(",它应该总是独立执行的。目前,我们选择只使用"),a("em",[t._v("反应堆")]),a("code",[t._v("Schedulers.parallel()")]),t._v(",这应该会带来相对较好的结果,因为它试图自动使用系统中可用的 CPU 内核数量。")]),t._v(" "),a("h3",{attrs:{id:"反应实例"}},[a("a",{staticClass:"header-anchor",attrs:{href:"#反应实例"}},[t._v("#")]),t._v(" 反应实例")]),t._v(" "),a("p",[t._v("虽然大多数示例仍然是相同的,但我们已经对其中的一些示例进行了全面检查,并创建了一些新的示例:")]),t._v(" "),a("ul",[a("li",[t._v("tunrstile ractive"),a("a",{attrs:{href:"#statemachine-examples-turnstilereactive"}},[t._v("旋转门反应式")])])])])}),[],!1,null,null,null);e.default=s.exports}}]);