react中的状态机_在基于状态图的状态机上使用React的模式

react中的状态机

by Shawn McKay

肖恩·麦凯(Shawn McKay)

在基于状态图的状态机上使用React的模式 (Patterns for using React with Statechart-based state machines)

Statecharts and state machines offer a promising path for designing and managing complex state in apps. For more on why statecharts rock, see the first article of this series.

状态图和状态机为设计和管理应用程序中的复杂状态提供了一条有希望的途径。 有关状态图为何摇摆的更多信息,请参阅本系列的第一篇文章 。

But if statecharts are such an excellent solution for managing UI & state in Javascript (JS), why isn’t there more momentum behind them?

但是,如果状态图是用Javascript(JS)管理UI和状态的出色解决方案,为什么它们背后没有更多的动力呢?

One of the main reasons statecharts have not grown in popularity within the front-end world is that best practices have yet to be established. It’s not abundantly clear how to use state machines with popular component-based UI libraries such as React, Vue, or Angular.

状态图在前端世界中尚未普及的主要原因之一是尚未建立最佳实践。 尚不清楚如何将状态机与流行的基于组件的UI库(如React,Vue或Angular)一起使用。

While it may be too early to declare best practices for statecharts in JS, we can explore some patterns used by existing state machine integration libraries.

虽然在JS中声明状态图的最佳实践为时过早,但我们可以探索现有状态机集成库所使用的一些模式。

状态图机 (Statechart machine)

Statecharts work both for visual design and as the underlying code for a graph-based state machine.

状态图既可以用于视觉设计,又可以用作基于图的状态机的基础代码。

Bear in mind that we’re in the early days of using statecharts with JS, and it may be worth experimenting with a variety of libraries or even developing your own. That being said, XState is currently leading the pack for statechart machine libraries in JS.

请记住,我们处于将状态图与JS一起使用的初期,可能值得尝试各种库甚至开发自己的库。 话虽这么说, XState目前在JS中领先于Statechart机器库。

The above state machine code can generate a much more readable statechart diagram when passed as JSON to the XState Visualizer.

上面的状态机代码作为JSON传递给XState Visualizer时,可以生成更具可读性的状态图。

You can even work the other way, starting by visually designing and then exporting to an XState configuration using sketch.systems. We don’t have all the pieces in one place yet, but there are no serious technical barriers to an open source solution.

您甚至可以以其他方式工作,从视觉设计开始,然后使用sketch.systems导出到XState配置。 我们还没有将所有内容都放在一个地方,但是开源解决方案没有严重的技术障碍。

Now that we have an idea of what XState does, let’s look at what it doesn’t do.

既然我们已经了解了XState的功能,那么让我们看看它没有做什么。

XState Tagline: “stateless finite state machines and statecharts”.
XState口号:“无状态的有限状态机和状态图”。

So what does it mean for a state machine to be stateless?

那么,状态机是状态的意味着什么呢?

无状态机器 (Stateless machines)

Stateless machines offer an unopinionated blueprint for state management — a kind of “roll your own” solution that doesn’t dictate where or how state in your application is stored.

无状态机器为状态管理提供了一个不受质疑的蓝图 -一种“滚动自己的”解决方案,它不指示应用程序中状态的存储位置或方式。

Much like a presentational component, a stateless machine is made of pure functions, is immutable, and maintains no state. It tracks no past, current, or future — but it can be used to help you calculate each.

无状态机器很像表示组件,是由纯功能组成的,是不变的,并且不保持任何状态。 它不会跟踪过去,现在或将来,但是可以用来帮助您计算每个。

Managing your state can be as easy as storing it in a local state variable.

管理状态就像将其存储在本地状态变量中一样容易。

Stateless machines don’t give you much out of the box. To trigger a transition, we must always pass in the current state node in to find the next. XState can let you know which actions should be fired on each state change, but you’ll have to find a way to manage the actions yourself.

无状态机器无法为您提供很多便利。 要触发转换,我们必须始终传入当前状态节点以查找下一个。 XState可以让您知道在每次状态更改时应触发哪些操作,但是您必须找到一种自行管理操作的方法。

If you’re interested in a more complete solution, consider making your state machine stateful.

如果您对更完整的解决方案感兴趣,请考虑使状态机变为有状态。

状态机 (Stateful machines)

A stateful machine tracks your node position on the state graph and manages the firing of actions. There is no need to pass in the current state on transitions — it tracks your current state node.

有状态机会跟踪您在状态图上的节点位置并管理动作的触发。 过渡时无需传递当前状态-它会跟踪您的当前状态节点。

As a summary, the instance of the stateful machine above:

作为总结,上述有状态机的实例:

  • determines the green state position at “Ringing”

    确定“ Ring”处的绿色状态位置
  • limits the possible purple’active transition events to CANCEL or SNOOZE

    将可能的“紫色”活动过渡事件限制为“ CANCEL或“ SNOOZE

  • fires the startRing action on entry

    在进入时触发startRing操作

  • fires the stopRing action on leaving the state

    离开状态时触发stopRing动作

Of course, there is more than one way to create a stateful machine. We’re back to the question of where to manage state:

当然,创建状态机的方法不止一种。 我们回到了在哪里管理状态的问题:

  • within the existing component state?

    在现有组件状态内?
  • in a connected state machine?

    在连接状态机中?

Let’s explore some design patterns with examples, starting with stateful components.

让我们从状态组件开始,通过示例探索一些设计模式。

有状态的组件 (Stateful components)

A stateful component, as you might imagine, manages state within the component, or within a wrapping higher-order component. In React, this would be as state. Storing state within a UI library ensures that changes won’t be missed and will trigger re-renders.

您可能会想到,有状态组件将管理该组件内或包装的高阶组件内的状态。 在React中,这将是state 。 将状态存储在UI库中可确保不会丢失更改,并会触发重新渲染。

This is the approach of a library called React-Automata that uses a higher-order component initiated by withStatechart.

这是称为React-Automata的库的方法,该库使用由withStatechart发起的高阶组件。

React-Automata offers several patterns for using statecharts with components:

React-Automata提供了几种使用状态图和组件的模式:

  • state from props

    道具状态
  • conditional rendering from a context

    上下文中的条件渲染
  • state from actions

    从行动状态

We’ll go over each pattern and consider the pros and cons.

我们将仔细研究每种模式,并考虑其优缺点。

道具状态 (State from Props)

Passing state directly into components seems like the most obvious solution.

将状态直接传递到组件似乎是最明显的解决方案。

In React-Automata, state can be passed by accessing it on the machineState prop — a reference to the actual state machine.

在React-Automata中,可以通过在machineState道具上访问状态来传递状态,该状态是对实际状态机的引用。

But be wary, this is by no means best practice. In the example above, the integration has coupled the statechart to the component, leading to a poor separation of concerns.

但是要小心, 这绝不是最佳实践 。 在上面的示例中,集成将状态图耦合到了组件,从而导致关注点分离不佳。

Consider that the statechart and components can allow for a clean divide as they solve different problems:

考虑到状态图和组件可以解决不同的问题,因此可以进行清晰的划分:

  • statecharts: when things happen, for example, enter state, actions fired

    状态图: 事情发生时,例如进入状态,触发动作

  • components: how and what happens, for example, the view, user interactions

    组件: 如何以及如何发生,例如视图,用户交互

Alternatively, you could decouple the component from the state machine by conditionally rendering with a default of no render.

或者,您可以通过默认不渲染的有条件渲染来将组件与状态机分离。

Certainly, there must be a more natural way to set up conditional rendering without having to turn all your renders into if/else and switch statements.

当然,必须有一种更自然的方法来设置条件渲染,而不必将所有渲染都转换为if/elseswitch语句。

从上下文进行条件渲染 (Conditional rendering from a context)

State accessed by a context doesn’t need to be passed directly.

上下文访问的状态不需要直接传递。

React-Automata provides a pattern for conditional rendering of child components using React’s context and a <State> component. Note that the value property can match on a string, array of strings, or even a glob-based pattern.

React-Automata提供了一种使用React上下文和<Sta te>组件有条件地渲染子组件的模式。 注意THA t the值属性可以匹配的字符串,字符串数组,或甚至基于水珠图案。

If the state value matches Ringing, the children inside of the State component will render. Otherwise, nothing.

如果状态值匹配Ringing ,则将渲染State组件内的子级。 否则,什么都没有。

State from context can help clarify the number of possible finite state view combinations. As in the case above, it’s clear there are only two possible configurations.

上下文中的状态可以帮助阐明可能的有限状态视图组合的数量。 与上面的情况一样,很明显只有两种可能的配置。

If view configurations start to get out of hand, React-Automata offers a render prop pattern that passes in a boolean based on the value.

如果视图配置开始失控,React-Automata将提供一个渲染道具模式,该模式根据该值传入布尔值。

Similarly, it’s possible to conditional render based on context actions.

同样,有可能根据上下文操作进行条件渲染。

Conditionally rendering based on state or actions maintains a coupling between the statechart and components, but less explicitly through context. How might you give components their isolated state apart from statecharts?

基于状态或动作的有条件渲染可维持状态图和组件之间的耦合,但通过上下文不太明确。 除了状态图,您如何赋予组件孤立的状态?

从行动状态 (State from actions)

It’s possible to use statecharts to update the internal state of a linked component using actions as triggers.

可以使用状态图以动作作为触发器来更新链接组件的内部状态。

React-automata checks the methods on a component and calls the functions if the names match the actions being fired.

React-automata检查组件上的方法,并在名称与要触发的动作匹配时调用函数。

As an example, the onEntry action startRing is fired as the state machine enters Ringing, causing the AlarmClock state to change to ringing. On leaving the Ringing state, stopRing is fired, and ringing is set to false.

例如,当状态机进入Ringing ,会触发onEntry操作startRing ,导致AlarmClock状态更改为ringing 。 退出Ringing状态时,将触发stopRing ,并将ringing设置为false

Note that, although of these methods are called with params, the methods already have access to whatever they need from machineState through props.

请注意,尽管这些方法是用params调用的,但这些方法已经可以通过props从machineState访问所需的machineState

Using internal component state managed through actions leads to a strong decoupling of components from state charts. However, it can also create a degree of clutter or confusion in components. It is not explicitly clear how or when methods will be called without examining the names of actions in the statechart. For this reason, I often call my actions and methods enterX or exitX in order to make it explicitly clear why and where they are being fired.

使用通过操作管理的内部组件状态会导致组件与状态图之间的强烈分离。 但是,它也会在组件中造成一定程度的混乱或混乱。 在不检查状态图中动作名称的情况下,尚不清楚如何或何时调用方法。 因此,我经常将我的操作和方法enterXexitX ,以便清楚地阐明其原因和触发位置。

外部状态机 (External state machines)

Another option worth considering is storing state outside of your UI framework. As with other state management libraries like Redux, components can be connected to an external state machine and updated with “on state change” and “on action” events.

另一个值得考虑的选择是将状态存储在UI框架之外。 与Redux等其他状态管理库一样,组件可以连接到外部状态机,并通过“状态更改”和“动作中”事件进行更新。

As an example, XStateful is a wrapper around XState that handles state, transitions, emitting events, triggering actions, and more.

例如, XStateful是XState的包装,用于处理状态,转换,发出事件,触发动作等。

XStateful works well with a React connector called XStateful-React.

XStateful与一个名为XStateful-React的React连接器很好地配合使用。

XStateful-React has much in common with React-Automata. But there is at least one signficant difference — the state machine instance is not managed within any component.

XStateful-React与React-Automata有很多共同点。 但是,至少有一个显着的区别-状态机实例不在任何组件中进行管理。

So how does external state from reducers work in XStateful?

那么,Reducer的外部状态如何在XStateful中工作?

状态和数据 (State and data)

Applications often require more than just the state node in a state graph— they require data as well. Often this data needs to be synced across components, in a way that can be frustrated if it must be passed from the uppermost shared parent.

应用程序通常不仅需要状态图中的状态节点,还需要数据。 通常,此数据需要跨组件同步,如果必须从最上层共享父级传递数据,可能会感到沮丧。

There are existing popular solutions for syncing data, such as Redux, or my state management wrapper for Redux. Unfortunately, these don’t play well with many state wrappers such as React-Automata due to an open issue with passing refs in React Redux (see this open issue with connect() and React.forwardRef).

有用于同步数据的现有流行解决方案,例如Redux或我的 Redux 状态管理包装器 。 不幸的是,由于在React Redux中传递引用存在开放性问题 (请参见connect()和React.forwardRef的此开放性问题) ,这些方法在许多状态包装器(例如React-Automata)中无法很好地发挥作用。

A complete state solution should manage both state and data.

完整的状态解决方案应同时管理状态和数据。

XStateful offers just such a state and data solution using a state reducer pattern, similar to Redux.

XStateful使用状态减少器模式提供了这种状态和数据解决方案,类似于Redux。

State machine subscribers listen and update changes based on actions emitted from the state machine. Note that XState refers to data as extended state, or extstate.

状态机订户根据状态机发出的操作来侦听和更新更改。 请注意,XState将数据称为扩展状态extstate

This particular Reducer pattern may seem unfamiliar, however, it’s heavily used in projects such as ReasonReact.

这种特殊的Reducer模式可能看起来并不熟悉,但是它在诸如ReasonReact之类的项目中大量使用。

Data can also be accessed in conditional renders on the property cond.

也可以在属性cond条件渲染中访问数据。

Be careful with using state to conditionally render components, as it creates a non-deterministic set of possible states. No longer are you limited to the number of states, but now to the number of state and data combinations. You lose out on deterministic features, discussed later in the testing section.

使用状态来有条件地渲染组件时要小心 ,因为它会创建一组不确定的可能状态。 您不再局限于状态数,而是现在仅限于状态数和数据组合数。 您将失去确定性功能,稍后在测试部分中进行讨论。

This data can be passed into your component using a render prop pattern.

可以使用渲染道具图案将此数据传递到您的组件中。

There is less of a need for state management tools like Redux if data can be stored within a complete state machine tool like XStateful.

如果可以将数据存储在完整的状态机工具(例如XStateful)中,则对Redux之类的状态管理工具的需求就更少了。

测试中 (Testing)

State machines also offer a better path for front-end testing.

状态机还为前端测试提供了更好的途径。

The deterministic nature of state machines creates the possibility of simplified front-end testing.

状态机的确定性性质提供了简化前端测试的可能性。

In React-Automata you can autogenerate snapshot tests using testStatechart, a method that takes the XState configuration and the component.

在React-Automata中,您可以使用testStatechart自动生成快照测试,该方法采用XState配置和组件。

testStatechart runs through the state graph and creates a Jest snapshot test for each possible configuration of the component. It will toggle on and off your various <State />, <Action /> components, leading to a recording of all possible conditional rendering combinations.

testStatechart贯穿状态图,并为组件的每个可能配置创建一个Jest快照测试 。 它将打开和关闭各种<State / >, < Action />组件,从而记录所有可能的条件渲染组合。

开发工具 (Devtools)

Devtools play an active role in what makes a library developer-friendly — debugging can be the hardest or most straightforward part of your job.

Devtools在使库对开发人员友好的过程中扮演着积极的角色-调试可能是您工作中最困难或最直接的部分。

In this respect, React-Automata offers a helpful integration via Redux Devtools. Each connected component becomes a named instance in the devtools, and each transition and action are displayed chronologically as actions are presented in Redux devtools.

在这方面,React-Automata通过Redux Devtools提供了有用的集成。 每个连接的组件在devtools中成为一个命名实例,并且每个动作和动作都按时间顺序显示在Redux devtools中。

XState offers an entirely new set of variables to track. Consider the following example by Erik Mogensen on the kinds of information an XState debugger may track.

XState提供了一组全新的变量来跟踪。 考虑一下Erik Mogensen的以下示例,该示例涉及XState调试器可能跟踪的信息。

This is not to say that state machine devtools need to look like our existing devtools. State machine devtools present an opportunity for a more visual debugging experience.

这并不是说状态机devtools必须看起来像我们现有的devtools。 状态机devtools提供了获得更直观的调试体验的机会。

结论 (Conclusion)

While we’re still in the early days of statecharts in JS, there are enough options available to start developing applications on top of XState. We can learn from these development patterns to both improve available libraries and to create tools to support the enormous potential of visual-based programming.

虽然我们仍处于JS状态图的早期,但是有足够的可用选项来开始在XState之上开发应用程序。 我们可以从这些开发模式中学习,以改进可用的库并创建工具来支持基于视觉的编程的巨大潜力。

Having developed applications with statecharts over the past three months, I’ve personally found these new patterns to be a breath of fresh air. Collaboration has become much more comfortable, as team members can visually grasp the underlying logic of a significant and growing system.

在过去三个月中使用状态图开发了应用程序之后,我个人发现这些新模式令人耳目一新。 由于团队成员可以从视觉上掌握重要且不断发展的系统的基本逻辑,因此协作变得更加舒适。

My hope is that this article will help others find statechart-based development more approachable. If you found it helpful, give a clap and pass it on :)

我希望本文将帮助其他人发现基于状态图的开发更加平易近人。 如果您觉得有用,请鼓掌并继续进行:)

翻译自: https://www.freecodecamp.org/news/patterns-for-using-react-with-statechart-based-state-machines-33e6ab754605/

react中的状态机

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/393565.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

android scheme打开天猫,淘宝

直接上代码 Intent intent new Intent(); intent.setAction("android.intent.action.VIEW"); /*String url "taobao://shop.m.taobao.com/shop/shop_index.htm?shop_id131259851&spma230r.7195193.1997079397.8.Pp3ZMM&point" "%7B%22…

leetcode1337. 方阵中战斗力最弱的 K 行(优先队列)

给你一个大小为 m * n 的方阵 mat&#xff0c;方阵由若干军人和平民组成&#xff0c;分别用 1 和 0 表示。 请你返回方阵中战斗力最弱的 k 行的索引&#xff0c;按从最弱到最强排序。 如果第 i 行的军人数量少于第 j 行&#xff0c;或者两行军人数量相同但 i 小于 j&#xff…

dp 1.4协议_浅析关于HDMI接口与DP接口

显示器现在主流已经为HDMI接口与DP接口&#xff0c;那么这些接口都有什么区别&#xff0c;以下表格会大致做个区分&#xff0c;建议优先使用DP接口。&#xff08;HDMI2.1接口目前仅发布协议&#xff0c;尚未大规模商用在高清电视机上有部分应用&#xff0c;Mini DP接口版本为DP…

浅谈 JDBC 中 CreateStatement 和 PrepareStatement 的区别与优劣。

浅谈 JDBC 中 CreateStatement 和 PrepareStatement 的区别与优劣。

jsp 构建单页应用_如何使用服务器端Blazor构建单页应用程序

jsp 构建单页应用介绍 (Introduction) In this article, we will create a Single Page Application (SPA) using server-side Blazor. We will use an Entity Framework Core database. Single-Page Applications are web applications that load a single HTML page. They dy…

Apache的虚拟主机配置

虚拟主机配置一般可以分为&#xff1a;基于域名基于端口基于IP配置虚拟主机检查防火墙&#xff0c;端口是否打开apache的配置文件。service iptables status #查看防火墙netstat -anp | grep 8021 #端口是必须要考虑的问题locate httpd.confmkdir -p /usr/local/apache/conf/ex…

oracle 的使用

一. docker 模式下进入数据库 ubuntujiang:~$ sudo docker ps -a sudo: unable to resolve host jiang CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS …

sql number转varchar_MySQL 指南之 SQL 语句基础

个人所有文章整理在此篇&#xff0c;将陆续更新收录:知无涯&#xff0c;行者之路莫言终(我的编程之路)零、结构化查询语言&#xff1a;SQL(Structured Query Language)DDL 数据定义语言 管理库&#xff0c;表DML 数据操作语言 增删改查 DCL 数据控制语言 数据控制&#xff0c;权…

leetcode744. 寻找比目标字母大的最小字母(二分查找)

给你一个排序后的字符列表 letters &#xff0c;列表中只包含小写英文字母。另给出一个目标字母 target&#xff0c;请你寻找在这一有序列表里比目标字母大的最小字母。 在比较时&#xff0c;字母是依序循环出现的。举个例子&#xff1a; 如果目标字母 target ‘z’ 并且字符…

H3C交换机 汇聚接口上应用策略路由

QOS 方式的策略路由只能应用在接口或者Vlan接口上&#xff0c;无法应用在2层或者3层的汇聚接口上可以使用PBR的方式将策略路由应用到接口上&#xff0c;QOS策略路由可以和PBR策略路由共存&#xff0c;并且QOS的优先级更高&#xff0c;先匹配QOS&#xff0c;如果匹配不到再匹配P…

【Java NIO深入研究3】文件锁

1.1概述——文件锁 文件锁定初看起来可能让人迷惑。它 似乎 指的是防止程序或者用户访问特定文件。事实上&#xff0c;文件锁就像常规的 Java 对象锁 — 它们是 劝告式的&#xff08;advisory&#xff09; 锁。它们不阻止任何形式的数据访问&#xff0c;相反&#xff0c;它们通…

2018-2019-2 网络对抗技术 20165202 Exp9 Web安全基础

博客目录 一、实践内容 跨站脚本攻击XSS跨站请求伪造CSRFSQL注入攻击二、实验中遇到的问题及解决三、基础问题回答四、实验总结一、实践内容 本实践的目标理解常用网络攻击技术的基本原理。Webgoat实践下相关实验。具体过程&#xff1a; 跨站脚本攻击XSS跨站请求伪造CSRFSQL注入…

leetcode696. 计数二进制子串

给定一个字符串 s&#xff0c;计算具有相同数量0和1的非空(连续)子字符串的数量&#xff0c;并且这些子字符串中的所有0和所有1都是组合在一起的。 重复出现的子串要计算它们出现的次数。 示例 1 : 输入: “00110011” 输出: 6 解释: 有6个子串具有相同数量的连续1和0&#…

idea无法忍受_不要成为无法忍受的软件开发人员

idea无法忍受by Bruce Flow通过布鲁斯流 不要成为无法忍受的软件开发人员 (Don’t be the software developer you can’t stand working with) I have more than 10 years of industry experience as a software developer. I am working at one of the largest tech compani…

项目需求分析答辩总结

前言 组长成员★530 雨勤311 旭403 俊223 元437 海辉你一定也不想做的 本次作业没什么可看的 项目需求分析看完记得投币的 一分钟宣传视频本组对其他各组评审结果 编号团队名称项目名称报告格式/20演示内容/20答辩内容/20PPT制作/20演讲/20总分/1001天机组指尖加密15.013.613.6…

xml编辑 html_【c# .net】Xml读写操作(XmlTextReader 类)

Xml读写操作&#xff08;XmlTextReader 类&#xff09;1.XML简介1&#xff09;XML 和 HTML 的设计初衷XML 被设计用来传输和存储数据HTML 被设计用来显示数据2&#xff09;什么是 XML?XML 指可扩展标记语言&#xff08;EXtensible Markup Language&#xff09;XML 是一种标记语…

jQ层级选择器

<!DOCTYPE html><html> <head> <meta charset"utf-8" /> <title></title> </head> <body> </body></html> <h1>沁园春雪</h1> <h2>毛***</h2> <div> <span> <d…

spring 获取url参数

1. usl格式&#xff1a; http://localhost:8080/contact/delete/3 java代码 RequestMapping(value"/delete/{id}")public Map<String,Object> delete(PathVariable("id") Integer id){boolean result contactService.deleteContactById(id); 也可以…

leetcode475. 供暖器(二分查找)

冬季已经来临。 你的任务是设计一个有固定加热半径的供暖器向所有房屋供暖。 现在&#xff0c;给出位于一条水平线上的房屋和供暖器的位置&#xff0c;找到可以覆盖所有房屋的最小加热半径。 所以&#xff0c;你的输入将会是房屋和供暖器的位置。你将输出供暖器的最小加热半径…

如何在命令行中使用jq将JSON转换为CSV

by Knut Melvr通过纳特梅尔瓦 如何在命令行中使用jq将JSON转换为CSV (How to transform JSON to CSV using jq in the command line) The shell tool jq is awesome for dealing with JSON-data. It can also transform that data into handy CSV-files, ready for all your s…