译者注:这篇文章是Reflectable enums in Swift 3的中文翻译
如果你正在用siwft编写程序并且需要一个数据模型来表示联系人,比如说一个联系人可能是来自你的通讯录或者是FaceBoook的朋友。你可能像下面一样初始化一个结构体的Contact:
这种方法非常简单方便但是有一个显著地缺点。注意到结构体Contact有两个可选的属性:phoneNUmber和username.这两个属性是根据不同的联系人类型有条件的分配一个值。
除非开发者碰巧看到属性的注释,否则的话每个属性都不是非空的规则是不明显的。这样的设计很容易使开发者犯错。如果我们不能意外的访问到了一个属性空,显然会更好,因为你忘记检查联系人的类型。
一种更普遍的解决方式是创建像下面的枚举:
联系人枚举类型有两种,每一种类型都有自己关联的struct类型。当我们创建这样的枚举的时候不可能出现偶然访问错误的属性,比如说AddressBook 联系人的username属性或者是FaceBook联系人的phoneNumber属性,因为这时候它们对于每一种联系人已经在定义是独立分开。这些结构体包含了正确的values和他们关联的联系人类型。哈哈,现在不在需要可选的属性了!
然后这种方式还有一种缺点。大多数场景下,已一种统一的方式访问联系人枚举中共同的属性是便捷的,例如像下面这样:
为了解决这种问题,你可以在Contact的枚举中创建一个计算属性,计算属性根据枚举关联的struct类型在switch的block中读取。
这个简单的例子入上面的代码看起来似乎没有太大的问题。但是真正开发程序的时候,数据model可能会有许多许多的属性,这时候各种各样的model我们都需要去用这种代码去处理,这样这种代码会很快变得一个繁琐的编写并且难以维护。显然,无聊冗余的代码是bugs的源头。我们必须有一种更好的方式去解决。。。
这种问题可以通过Swift中的反射API去解决。反射是Swift一种内省代码的方式,换句话谁就是在运行时可以可以自己。和其他平台比较,比如.Net,Java,swift反射只提供了一些非常基本的功能,但在这里已经足够我们使用了。
让我们定义一个RefrectableEnum 协议:
这个协议唯一的方法在协议的扩展中提供了默认实现。它使用反射在enum内部关联struct并且找到其中的属性。该协议能够减轻我们之前代码的编码负担,例如:
重要的是添加到Contact的属性具有和关联的struct枚举cases完全相同的属性。使用Swift的#function 意味着调用value(of :)的属性的名称作为propertyName参数传入(译者注:实际上上述例子中你也完全可以使用属性对应的名称字符串,这里看起来高大上而已...)。
上面代码使用了向下强转,因为部分代码是缺少安全性的。它依赖于命名的约定。编译器不能保证在运行时编译成功。value(of:)的实现中使用了可选类型的强制解包,如果初始化不能匹配上可能会使代码crash。这里我的目的是遇到问题就crash(比如说关联的struct的属性没有一个确定的name和type),但这是必要的,因为编译器不能警告这种代码提出修改建议。如果你想编译器能够容错处理,那么我的解决方案可能不太适合你。
上面的代码仅仅是一种粗滤的草稿,他可能不太符合你的应用,但为未来的优化提供了一个很好的开始。上面的代码包含ReflectableEnum你可以在下面的链接中找到:
https://github.com/ijoshsmith/reflectable-enum
愉快的使用Swift吧!