众所周知,现在全国都提倡生育,凡符合享受国家规定90天以上产假的生育女职工可享受生育营养补贴300元、围产保健补贴700元,高达几百元的生育津贴促进了大量妇女同志的生育意愿。此外,各地出台了很多补贴政策,像在深圳,现在只要父母一方有深圳户籍,生育第一个子女的,办理出生入户后发放一次性生育补贴3000元,另外每年发放1500元育儿补贴,直至该子女满3周岁。三年合计可以领到7500元的巨款,这样丰厚的生育补贴,让我们这些即便是居住在深圳这样花销巨大的城市,也完全有能力生育并抚养子女了!
尽管生育的资金已经没有任何的问题,唯一稍有不足的地方就是准生证的办理还是有些麻烦,如果双方都没有深圳户籍,虽然可以在深圳办理准生证,但仍然需要先准备材料并前往户籍所在地的公安分局出入境管理处办理预审,然后领取预审表,缴纳50元的申请费,再到深圳户政局来办理准生证,然后只需要15个工作日,就能办理下来了。
这个准生证的流程如此的不可动摇,想必有非常深层的考虑,应该是一套极其复杂的系统,堪比CAE
、EDA
、BIM
等类型的工业设计软件产品,我们的普通的开发人员想复现这样复杂的系统几乎是不可能的,不过我发现了一个蓝海赛道,就是准死证的系统设计。
可以想一想一个基本的问题,人有生就有死,有准生证必然就有准死证,随着老龄化程度加剧,死亡管制势在必行,应当提倡晚退休、晚死,少死、优死等具体内容。可能有人会说死亡不是有死亡证明么?不不,死亡证明是相当于出生证呀,一个是事前,一个是事后,这个完全是两套系统,不可混淆,那么准死证应当如何设计呢?参考一些类似的系统流程,我大约设计了如下几个步骤流程:
这样一算下来系统确实有够复杂的,那么流程有了,那么具体需要怎么设计呢,因为这个系统很复杂,所以设计起来是高度抽象的,所以要先选好语言,简单的封装继承多态不足以描述这样的系统,应该用scala
试一试。
为什么用scala
呢?其实看过我文章的朋友可能知道,我多年前就尝试过scala
的scalatra
框架,感觉这个语言本身还是相当可以的,虽然你能搜到JetBrains早在2011年就推出了kotlin
,但真正走进大众视野已经是2016年,到那时候才有1.0
这个稳定版,而scala
则是个实打实的老语言,早在2001年就启动项目了,现在大部分人用的2.x
版本早在2006年就推出了,可以说是个稳定了十几年没什么大问题。不过呢,虽然敲定了语言,但设计系统是没有那么快的,因为scala3
出来了,自然不肯再去用scala2
做新东西了,虽然已经出来一年多了,但相信很多人还没有尝试,毕竟这样的大版本总需要一年的稳定时间吧,如今一年之期已到,可以开始学习一下scala3
了,所以这篇文章主要是先用scala
进行简单的语法试练,先新建一个scala
项目:
没错我们选3.3.0
这个版本,它不同于老套的2.xx
,可以说已经达到了抽象的巅峰水平。紧接着新建一个类,类名为Sys
,然后是main()
,看看它的写法:
@main def hello() = println("Hello, World")
是的,曾经的2版本它还需要这样写:
object hello {
def main(args: Array[String]) = {
println("Hello, World!")
}
}
object Hello extends App {
println("Hello, world")
}
虽然上面写的都是object
,不过众所周知,写class
也是可以的,但如今它还不能写进class
里面了,比如说这样:
class Sys:
@main def hello() = println("Hello, World")
这就会报错,运行不了的,但如果是
object Sys:
@main def hello() = println("Hello, World")
就没有关系,这里请注意,其实不但class
和object
不需要{}
了,方法也是可以写:的,而match
这样的关键字则可以直接省去,:
都不需要,请看例子:
import java.util.UUID
object Sys:
@main def hello() = println(get("办事"))
private def get(input: String): String =
input match
case "摇号" => UUID.randomUUID().toString
case "有诉求" => "请找相关部门"
case "办事" => Thread.sleep(0xfffff); "请等待"
case "丑闻" => "已辟谣"
case "出现问题" => "一刀切"
case "恶性事件" => "临时工干的"
case "老鼠" => "鸭子"
case _ => throw UnsupportedOperationException()
其实scala3
并非只有这种语法上的精简了,他还有个新鲜玩意,交叉类型(近似于于scala2
中的复合类型with
关键字):
trait Defendant:
def defend(): Unit
trait Judge:
def ruling(): String
class Traitor extends Defendant, Judge:
override def defend(): Unit = null
override def ruling(): String = "无罪"
object Sys:
@main def hello() = println(prosecute(Traitor()))
private def prosecute(unit: Defendant & Judge): String =
unit.defend()
unit.ruling()
可以看到prosecute()
是交叉了两个类型的,并非使用Traitor
来做参数类型。说到这个类型,你可能发现了我没有写new
关键字,这不会和kotlin
搞混了吧?其实这正是scala3
很重要的一个新特性,不过他们对这个特性名的称呼有些普通,叫creator applications。
回到交叉类型这个话题,当然这个交叉类型并非只能由两个类型构成,也并不一定要与参数完全对应,看我多加几个类型试一下:
trait Defendant:
def defend(): Unit
trait Judge:
def ruling(): String
trait Murderer:
def murder() = null
trait Parasite:
def vampire() = null
class Traitor extends Defendant, Judge, Murderer, Parasite:
override def defend(): Unit = null
override def ruling(): String = "无罪"
object Sys:
@main def hello() = println(prosecute(Traitor()))
private def prosecute(unit: Defendant & Judge & Murderer): String =
unit.defend()
unit.ruling()
这样看就能看出它的方便之处了,可以不必定义一些只有一个方法用的接口了。当然啦,这些其实都符合我们的直觉,也只是更方便了一些,下面再来看一个新玩具,扩展方法:
class House:
def dwell(): Unit = null
object Sys:
@main def hello() = println(live())
extension (house: House)
private def future() = Long.MaxValue
private def live(): Long =
val house = House()
house.future()
这也算是个很有趣的特性,这个扩展当然也不止可以加一个方法了,之前的版本要么新做一个class
去继承,要么做个object
再把需要扩展的对象当参数传过去,现在能直接扩展自然是好看很多。
最后可以留意一下,不但scala 3
项目可以用之前的scala 2
的库,而且老scala 2
项目不用升级也能用scala 3
的库,这是非常有趣的,因为他有一个中间表示形式的东西,详细可以参考官网的文档A new compatibility era
那么后续会继续进行这个系统的设计,如何进行这样一个大系统的模块拆分和基本架构搭建呢?敬请期待。
参考
- docs.scala-lang.org/scala3/new-…
- docs.scala-lang.org/scala3/book…
- docs.scala-lang.org/scala3/refe…
- docs.scala-lang.org/tour/compou…
- docs.scala-lang.org/scala3/book…