前端如何优雅建模?

广告位招租
扫码页面底部二维码联系

这篇文章放在“杂”栏目下面,说明会是一篇转载请注明出处:www.tangshuang.net【原创内容,转载请注明出处】不成体系的文章。我想谈一下如何在前端优雅本文作者:唐霜,转载请注明出处。著作权归作者所有,禁止商业用途转载。的建模。直入正题!

【作者:唐霜】【作者:唐霜】原创内容,盗版必究。未经授权,禁止复制转载。

前端建模包括两个层面的建模:业务领域建模【本文受版权保护】著作权归作者所有,禁止商业用途转载。和交互领域建模。这两者基本上没有本质联系【原创内容,转载请注明出处】【原创不易,请尊重版权】,但是在前端这个场景下,有的时候又有一些【原创内容,转载请注明出处】【本文首发于唐霜的博客】特殊的情况。我们来看看如何在这两个层面建转载请注明出处:www.tangshuang.net未经授权,禁止复制转载。模。

原创内容,盗版必究。【原创内容,转载请注明出处】【版权所有,侵权必究】著作权归作者所有,禁止商业用途转载。本文作者:唐霜,转载请注明出处。

业务领域建模著作权归作者所有,禁止商业用途转载。

【原创不易,请尊重版权】本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】【原创不易,请尊重版权】【未经授权禁止转载】

简单讲,业务领域建模,就是把业务实体与其【原创不易,请尊重版权】【本文首发于唐霜的博客】逻辑进行建模。在知乎有小伙伴留言说,前端本文版权归作者所有,未经授权不得转载。【作者:唐霜】不怎么适合DDD,因为前端是贫血模型。但本文作者:唐霜,转载请注明出处。【版权所有】唐霜 www.tangshuang.net在我实际实践中,我更多是充血模型。业务模著作权归作者所有,禁止商业用途转载。【原创不易,请尊重版权】型需要包含字段本身,以及复杂的业务逻辑。转载请注明出处:www.tangshuang.net【版权所有】唐霜 www.tangshuang.net你可能会讲,业务逻辑会被放在后端,但是实【本文首发于唐霜的博客】【未经授权禁止转载】际上前端也要这个业务逻辑,比如当一个订单【版权所有】唐霜 www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】的负责人是组织中的某个职位的角色时,需要著作权归作者所有,禁止商业用途转载。本文作者:唐霜,转载请注明出处。在订单推进过程中填写审核时间这个信息。那本文版权归作者所有,未经授权不得转载。本文作者:唐霜,转载请注明出处。么,在前端,必须去判断当前用户是否是该角未经授权,禁止复制转载。【版权所有】唐霜 www.tangshuang.net色,或者获得当前用户的某个权限。这个逻辑本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】是跑不掉的。

【原创内容,转载请注明出处】转载请注明出处:www.tangshuang.net本文版权归作者所有,未经授权不得转载。

最近,我升级了tyshemo,支持了装饰【原创内容,转载请注明出处】【未经授权禁止转载】器的方式进行meta的定义。现在,你可以【版权所有,侵权必究】【版权所有】唐霜 www.tangshuang.net这样定义自己的模型:

【未经授权禁止转载】著作权归作者所有,禁止商业用途转载。【未经授权禁止转载】未经授权,禁止复制转载。
import { Model, meta, state } from 'tyshemo'

class OrderModel extends Model {
  @meta({
    type: { 
      user_id: String,
    },
  })
  master = null

  @meta({
    type: Number,
  })
  total_price = 0

  @state()
  role = 'member'

  canFillDate() {
    return this.role === 'admin'
  }
}

通过@meta来装饰字段,通过@stat本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。e来装饰状态属性。这样撰写模型,会有更舒【访问 www.tangshuang.net 获取更多精彩内容】【访问 www.tangshuang.net 获取更多精彩内容】服的感觉,而且可以更好的兼容typesc【原创不易,请尊重版权】著作权归作者所有,禁止商业用途转载。ript,避免以前使用static属性定本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。义时,无法与typescript很好结合【关注微信公众号:wwwtangshuangnet】【本文首发于唐霜的博客】的问题。

本文作者:唐霜,转载请注明出处。【作者:唐霜】【版权所有】唐霜 www.tangshuang.net转载请注明出处:www.tangshuang.net【关注微信公众号:wwwtangshuangnet】

简单讲,通过业务建模,我们得到了一些模型本文作者:唐霜,转载请注明出处。本文版权归作者所有,未经授权不得转载。,这些模型是独立的,自治的,在不被使用的本文作者:唐霜,转载请注明出处。【作者:唐霜】时候,它独立描述了该业务对象的各种字段及转载请注明出处:www.tangshuang.net【本文首发于唐霜的博客】其逻辑,但由于不在具体的业务场景中被使用原创内容,盗版必究。本文版权归作者所有,未经授权不得转载。,因此也只能表现有限的业务信息,它只能告【作者:唐霜】未经授权,禁止复制转载。诉读代码的人“我有什么,能做什么”,而不【关注微信公众号:wwwtangshuangnet】本文作者:唐霜,转载请注明出处。能告诉“我做了什么”。只有使用这些模型的【转载请注明来源】原创内容,盗版必究。实例,放到具体的业务模块中,才能完成真正【原创不易,请尊重版权】【未经授权禁止转载】的模块编程。所以,单纯讲,业务建模虽然重【原创内容,转载请注明出处】原创内容,盗版必究。要且有用,但是如果不被适当的人使用,就会【作者:唐霜】原创内容,盗版必究。非常混乱,毫无头绪。

著作权归作者所有,禁止商业用途转载。著作权归作者所有,禁止商业用途转载。本文作者:唐霜,转载请注明出处。【版权所有】唐霜 www.tangshuang.net

建立业务模型,可以把有关需求文档中,有关【版权所有,侵权必究】著作权归作者所有,禁止商业用途转载。核心业务的东西分出一层。

【本文首发于唐霜的博客】【转载请注明来源】著作权归作者所有,禁止商业用途转载。【版权所有】唐霜 www.tangshuang.net

交互领域建模转载请注明出处:www.tangshuang.net

转载请注明出处:www.tangshuang.net【转载请注明来源】未经授权,禁止复制转载。原创内容,盗版必究。【原创不易,请尊重版权】

这是后端没有的东西。直白讲,交互领域建模原创内容,盗版必究。【版权所有,侵权必究】就是写类似Vue一样的View Mode【版权所有】唐霜 www.tangshuang.net本文作者:唐霜,转载请注明出处。l,但是没有template那块。简单说【访问 www.tangshuang.net 获取更多精彩内容】未经授权,禁止复制转载。,就是建立一个模型,考虑到将来在view著作权归作者所有,禁止商业用途转载。【作者:唐霜】层使用它,所以该模型的所有api,都是为【转载请注明来源】【未经授权禁止转载】view设计的,主要目标,是和需求文档中【版权所有,侵权必究】【原创不易,请尊重版权】有关交互相关的描述一一对应。在交互模型中【本文受版权保护】原创内容,盗版必究。,实例化业务模型,把业务模型变成交互模型【作者:唐霜】【未经授权禁止转载】内部的状态。当在view中实例化交互模型转载请注明出处:www.tangshuang.net【原创不易,请尊重版权】,就看不到业务模型了,view拿着交互模【版权所有,侵权必究】转载请注明出处:www.tangshuang.net型的接口进行渲染和事件回调。

【作者:唐霜】著作权归作者所有,禁止商业用途转载。【版权所有】唐霜 www.tangshuang.net【关注微信公众号:wwwtangshuangnet】【版权所有】唐霜 www.tangshuang.net

我在nautil中提供了可用于建立交互模【本文受版权保护】原创内容,盗版必究。型的一个体系。举个例子:

【未经授权禁止转载】转载请注明出处:www.tangshuang.net本文版权归作者所有,未经授权不得转载。
import { Component } from 'nautil'
import { SomeController } from './some.controller' // 写好的交互模型

class MyPage extends Component {
  controller = new SomeController() // 实例化交互模型

  SubmitButton = this.controller.turn((props) => {
    const { someModel } = this.controller // 读取交互模型内的某个业务模型
    const { total_price } = someModel // 读取业务字段值
    return (
      <button className={total_price > 100 ? 'sale-count' : undefined}>Submit</button>
    )
  })

  render() {
    const { SubmitButton } = this // 读取定义好的组件
  }
}

上面这段代码中,SomeControll【转载请注明来源】转载请注明出处:www.tangshuang.neter是一个交互模型,里面使用了另外一个S原创内容,盗版必究。本文版权归作者所有,未经授权不得转载。omeModel业务模型。但是,对于vi本文版权归作者所有,未经授权不得转载。转载请注明出处:www.tangshuang.netew层而言,你不需要知道它是一个业务模型【本文首发于唐霜的博客】原创内容,盗版必究。,你只需要调用它即可。

【未经授权禁止转载】著作权归作者所有,禁止商业用途转载。【版权所有,侵权必究】【版权所有】唐霜 www.tangshuang.net【未经授权禁止转载】

Nautil在controller中提供【关注微信公众号:wwwtangshuangnet】著作权归作者所有,禁止商业用途转载。了turn方法,用于把一个用到contr【未经授权禁止转载】【访问 www.tangshuang.net 获取更多精彩内容】oller的普通的组件转化为一个被con【本文受版权保护】【原创不易,请尊重版权】troller控制的组件,当contro原创内容,盗版必究。原创内容,盗版必究。ller中某些特定信息发生变化时,这些组【作者:唐霜】【原创不易,请尊重版权】件就会自动更新。

【未经授权禁止转载】【未经授权禁止转载】【访问 www.tangshuang.net 获取更多精彩内容】未经授权,禁止复制转载。【本文受版权保护】

分层【版权所有,侵权必究】

转载请注明出处:www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】著作权归作者所有,禁止商业用途转载。

前端代码分层管理,从代码量上,并不比铁板本文作者:唐霜,转载请注明出处。未经授权,禁止复制转载。一块的管理多多少,毕竟所有的代码,都来自【本文受版权保护】未经授权,禁止复制转载。产品的需求描述。但是,分层管理所带来的灵【原创内容,转载请注明出处】原创内容,盗版必究。活性、可维护性是不可估量的。

【原创不易,请尊重版权】原创内容,盗版必究。【版权所有】唐霜 www.tangshuang.net

如果你写一个有很多块的页面的vue组件,【版权所有】唐霜 www.tangshuang.net转载请注明出处:www.tangshuang.net你就会发现,你的这个组件会越来越多交织在本文作者:唐霜,转载请注明出处。原创内容,盗版必究。一起的代码,从一开始很容易理解,这几个状著作权归作者所有,禁止商业用途转载。【本文受版权保护】态和这几个方法是关于最顶上这一块的,但是【版权所有】唐霜 www.tangshuang.net【版权所有,侵权必究】,随着页面其他块的交互代码的增多,你就会转载请注明出处:www.tangshuang.net原创内容,盗版必究。发现,这个状态会在哪块用?这个方法会在什【本文首发于唐霜的博客】【原创内容,转载请注明出处】么情况下调?能不能删?可不可以改?都需要【作者:唐霜】本文作者:唐霜,转载请注明出处。上下反复读代码来确认。

原创内容,盗版必究。【原创内容,转载请注明出处】【原创不易,请尊重版权】

而如果你采取代码分层,你会先针对业务本身【版权所有,侵权必究】著作权归作者所有,禁止商业用途转载。的实体进行建模,然后对业务中的交互进行建本文作者:唐霜,转载请注明出处。本文版权归作者所有,未经授权不得转载。模,最后才是view层的编写。此时,vi【访问 www.tangshuang.net 获取更多精彩内容】【原创内容,转载请注明出处】ew层的代码会清晰很多,因为它不再去管理【原创内容,转载请注明出处】【版权所有】唐霜 www.tangshuang.net属于业务的逻辑,而更多的是用和回调。

【版权所有】唐霜 www.tangshuang.net原创内容,盗版必究。著作权归作者所有,禁止商业用途转载。

依赖【未经授权禁止转载】

【未经授权禁止转载】【转载请注明来源】著作权归作者所有,禁止商业用途转载。【作者:唐霜】转载请注明出处:www.tangshuang.net

在view层,我们用vue或react来原创内容,盗版必究。著作权归作者所有,禁止商业用途转载。写,我更多使用react,因为react【本文受版权保护】【未经授权禁止转载】没有使用Proxy或defineProp【未经授权禁止转载】【关注微信公众号:wwwtangshuangnet】erty,可以使得我们建模时,使用更多魔本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。法。但是,有些时候,由于view层的特殊原创内容,盗版必究。本文作者:唐霜,转载请注明出处。机制,导致我们如果完全脱离框架进行建模时【转载请注明来源】【作者:唐霜】,不得不提供一些多余的接口,来帮助和vi本文作者:唐霜,转载请注明出处。著作权归作者所有,禁止商业用途转载。ew进行依赖绑定。

未经授权,禁止复制转载。【版权所有,侵权必究】转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。

以我工作的项目为例,我们使用angula著作权归作者所有,禁止商业用途转载。【访问 www.tangshuang.net 获取更多精彩内容】rjs作为主体框架,如果我单纯使用ESM【原创内容,转载请注明出处】【未经授权禁止转载】odule的模块,就没有办法直接使用$r【访问 www.tangshuang.net 获取更多精彩内容】本文作者:唐霜,转载请注明出处。ootScope等这种angularjs【本文受版权保护】本文作者:唐霜,转载请注明出处。内置的服务,但是如果我提供一个angul【原创不易,请尊重版权】【本文首发于唐霜的博客】ar factory,就会牺牲模型的可移本文作者:唐霜,转载请注明出处。【未经授权禁止转载】植性,为之后跨平台复用带来问题。所以,n原创内容,盗版必究。【本文首发于唐霜的博客】autil中提供了controller.本文版权归作者所有,未经授权不得转载。【未经授权禁止转载】turn这个方法来实现模型层和视图层的连【转载请注明来源】【未经授权禁止转载】接,简单说就是在框架层面调用forceU【原创不易,请尊重版权】【本文受版权保护】pdate来实现重新渲染。

【访问 www.tangshuang.net 获取更多精彩内容】著作权归作者所有,禁止商业用途转载。原创内容,盗版必究。