我想看bug(typec科普)

有了Bug,先看看“Type”

译者 | 崔皓

1、开篇

不懂原始类型的程序员,往往由于急于求成,上手很快,最后却发现被各种Bug耽误进度。本文以经典的邮箱类型、货币类型、密码类型为例,利用好类型系统能够很好地改进编码方式,同时为技术人找回“打稳地基”的快乐。

2、字符串类型变身成为邮箱类型

笔者已经厌倦了使用原始类型,并试图通过使用原始类型,来为一个领域进行建模。

字符串值(String)类型不仅仅用来保存用户的电子邮件地址或国籍信息,还可以有更丰富的用途。我需要一个EmailAddress的类型,并定义它不能为空,同时希望有单一的入口来创建该类型的对象。在返回一个值之前,需要被验证和规范化。

同时,也希望该数据类型有一些方法,如.Domain()或.NonAliasValue(),在输入foo+bar@gmail.com时,会分别调用这两个方法并返回gmail.com和foo@gmail.com。

在类型设计中应该考虑这种有用的功能,该功能的引入有助于防止错误的发生,并提高了类型的可维护性。

3、设计良好、功能实用的类型

例如,一个EmailAddress可以提供两个方法来检查是否相等。

lEquals方法用来判断两个(规范化)的电子邮件地址是否相同,如果相同将返回true。

lEqualsInPrinciple1.

该方法对于foo@gmail.com和foo+bar@gmail.com的输入会判断相同,因此也会返回true。(这里假设两个邮箱都是同一个人注册的,因此相同需要判断两个邮箱“相等”)

特定类型的方法在不同的使用场景下都会发挥不同的作用。如果用户jane@gmail.com注册,但又用Jane@gmail.com登录,那么用户的登录不应该失败(仅仅存在首字母大小写的区别)。同样的,如果户用使用电子邮件地址(foo@gmail.com)和另一个注册账户(foo+svc@gmail.com)联系客户支持,相同就需要对这两个邮箱进行有效匹配。这些都是典型应用场景,如果没有散落在代码库中的业务逻辑,仅凭一个简单的字符串是无法满足的。

注意:根据Office RFC描述,电子邮件地址中@符号之前的部分可以区分大小写,但所有主要的电子邮件主机都将其视为不区分大小写,因此,域名类型也考虑这方面的问题。

4、好的类型可以防止Bug

顺着上面邮箱类型的例子,如果我们想走得更远,假如希望一个电子邮件地址可以被验证或未被验证。通常的做法是,通过向个人的收件箱发送一个独特的代码来验证电子邮件地址。这些 "商业 "上的互动也可以通过类型系统来表达。例如,创建一个叫做VerifiedEmailAddress的第二个类型。该类型可以继承自EmailAddress。并且确保代码中只有一个地方可以产生VerifiedEmailAddress的实例,即负责验证用户地址的服务。如此这般,应用程序的其他部分可以依靠这个新类型来防止Bug。

任何发送电子邮件的功能都可以依靠该类来验证的电子邮件地址的安全性。想象一下,如果电子邮件地址是通过简单的字符串来表达的,会是怎样的情况。

因此,要找到相关的用户账户,检查一些模糊的标志,如HasVerifiedEmail或IsActive,确保这些标志设置是正确的,而不会在默认构造函数中被错误地初始化为真。有太多的错误空间由于使用了原始字符串,导致有些检查不到位的情况,这种使用原始类型的表达方式被认为是懒惰和缺乏想象力的编程。

5、富类型免受错误的侵扰

另一个很好的例子是货币!我已经数不清有多少应用程序使用十进制来表达货币值。也已经数不清有多少应用程序使用十进制类型表达货币值。为什么呢?

这种类型有很多问题,甚至很难理解。每个与钱打交道的领域都应该有专门的货币类型。货币类型应该包括货币和运算符重载(或其他安全功能),以防止出现100美元与20英镑相乘这样的愚蠢错误。此外,并非每种货币在小数点后都只有两位数。有些货币,如巴林或科威特第纳尔有三位。如果你在致力处理投资或银行贷款,那么你最好确保你呈现的Unidad de Fomento有4个小数点。这些问题已经很重要了,足以保证有一个专门的Moneytype,但这还远远不够。

除非在系统内部完成所有功能,否则就不得不与第三方系统打交道。例如,大多数支付网关都是以整数值来请求和响应资金。由于整数值不能涵盖类似浮点数(双数类型)的四舍五入运算,因此比浮点数更受欢迎。唯一需要注意的是,数值必须以小单位(如美分、便士、迪拉姆、格罗兹、科佩克等)传输,这意味着如果你的程序处理小数点数值,在与外部API对话时,你将不得不不断地来回转换它们。如前所述,并不是每种货币都使用两个小数点,所以不是每次都是简单的乘/除以100。事情很快就会变得很困难,如果这些业务规则被封装成一个简洁的单一类型,事情就会被大大简化。

var x = Money.FromMinorUnit(100, "GBP"):£1var y = Money.FromUnit(100.50, "GBP"):£1.50Console.WriteLine(x.AsUnit()):1.5Console.WriteLine(x.AsMinorUnit()):1501.2.3.4.

如果这还不够复杂的话,各国也有不同的货币格式来表示货币。在英国,"一万英镑和五十便士 "将被表示为10,000.50,但在德国,"一万欧元和五十美分 "将被显示为10.000,50。试想一下,如果这些规则没有放到统一的货币类型中,那么在整个代码库中会有多少与货币相关的代码被分割开来。

此外,一个专门的货币类型可以包括许多功能,这将使货币价值的工作变得轻而易举。

var gbp = Currency.Parse("GBP");var loc = Locale.Parse("Europe/London");var mOney= Money.FromMinorUnit(1000050, gbp);money.Format(loc) // ==> £10,000.50money.FormatVerbose(loc) // ==> GBP 10,000.50money.FormatShort(loc) // ==> £10k1.2.3.4.5.6.

当然,建立这样一个Money类型在开始的时候会有点费劲,但是一旦它被实现并经过测试,那么代码库的其他部分就可以带来更大的安全性,并防止大多数的Bug的产生,否则这些Bug会随着时间的推移而慢慢出现。即使像Money.FromUnit(decimal v, Currency c)或Money.FromMinorUnit(int v, Currency c)这样的小功能看起来并不多,但它使参与连续开发的程序员能够意识到,用户输入或外部API收到的值是否包含在其中,这样可以在一开始就防止Bug的产生。

6、聪明的类型设计减少副作用

富类型的伟大之处在于,可以以任何的方式来塑造它们。这里展示另外一个例子,富类型如何将团队从巨大的操作开销中拯救出来,甚至防止安全漏洞。

相信很多系统中的代码库都有一个类似于字符串secretKey或字符串password的东西,它作为函数的参数。那么在什么情况下有可能出错呢?

如下(伪)代码:

try{var userLogin = new UserLogin{Username = usernamePassword=password}var success = _loginService.TryAuthenticate(userLogin);if(success)RedirectToHomeScreen(userLogin)。ReturnUnauthorized()。}catch (Exception ex){Logger.LogError(ex, "User login failed for {login}", userLogin);}1.2.3.4.5.6.7.8.9.10.11.12.13.14.15.16.

这里出现的问题是,如果在认证过程中抛出一个异常,那么这个应用程序将用户的明文密码写入日志。当然,这段代码一开始就不应该存在,这种情况会随着时间的推移而发生。大多数这样的错误都是随着时间的推移而逐步发生的。

最初,UserLogin类可以有一组不同的属性,在最初的代码审查中,这段代码可能没有问题。几年后,有人可能修改了UserLogin类以包括明文密码。这个功能甚至不会出现在代码提交的差异中,因此会逃过代码审查。于是就引入了安全漏洞。然而,如果引入一个富类型(专有类型),就可以避免类似错误的发生。

在C#中(以这个语言为例),当一个对象被写入日志时,ToString()方法会被自动调用。有了这些知识,我们就可以设计一个这样的密码类型。

public readonly record struct Password(){public override string ToString(){return "****"。}public string Cleartext(){return _cleartext。}}1.2.3.4.5.6.7.8.9.10.11.

虽然是一个微小的变化,但在系统的任何地方都不可能意外地输出一个明文密码。这不是很好吗?

当然,在实际的认证过程中,你可能仍然需要明文值,那么就需要通过非常明确的命名方法Cleartext()来实现的,所以对这个操作的敏感性没有任何含糊,它自动引导开发者有意和谨慎地使用这个方法。

处理用户的PII(如国家保险号、税号等)也是同样的原则。使用专门的类型对这些信息进行建模。覆盖默认函数,如.ToString()。ToString()的默认函数,并通过相应的命名函数暴露敏感数据。你永远不会把PII泄露到日志和其他地方,以后可能需要一个巨大的操作来再次刷掉它。

小伎俩发挥了大作用!

7、形成习惯

每当开发者处理那些有特殊规则、行为或敏感数据的时候,不妨考虑如何能通过创建一个显式类型来帮助自己。

让我们再举一个密码类型的例子,可以走得更远!

密码在被存储到数据库之前会进行散列计算,但这个哈希值不是一个简单的字符串。在某些时候,我们将不得不在登录过程中把以前存储的哈希值与新计算的哈希值进行比较。但并不是每个开发人员都是安全专家,比较两个哈希字符串可能会使代码受到攻击。

检查两个密码哈希值是否相等的推荐方法是以非优化的方式进行。

[MethodImpl(MethodImplOptions.NoInlining | MethodImplOptions.NoOptimization) ]

private static bool ByteArraysEqual(byte[] a, byte[] b){if(a == null &&b == null){return true;}if(a == null || b == null || a.Length != b.Length){return false;}var areSame = true;for (var i = 0; i < a.Length; i++){areSame &= (a[i] == b[i])。}return areSame。}1.2.3.4.5.6.7.8.9.10.11.12.13.14.15.16.17.

注:代码示例取自原始ASP.NET Core资源库

因此,将这一特殊功能编码为一个专门的类型才是合理的。

public readonly record struct PasswordHash{public override bool Equals(PasswordHash other){return ByteArraysEqual(this.Bytes(), other.Bytes())。}}1.2.3.4.5.6.7.

如果一个PasswordHasher只返回PasswordHash类型的值,即使是对业务不太了解的开发者也会使用一种安全的形式来检查相等。

在建立领域模型方面要考虑周全! 当然,编程中的一切都没有明确的对错之分,人们的个人使用情况总是有更多的细微差别,这些不是在一篇文章中所能表达的,但笔者建议是,考虑如何使类型系统对开发者的帮助很大。现在许多现代编程语言都有非常丰富的类型系统,我们可能忽视了它们没有利用好这些类型改进编码方式。

原文链接:https://dusted.codes/the-type-system-is-a-programmers-best-friend

译者介绍

(0)
上一篇 2023年1月17日 上午10:57
下一篇 2023年1月17日 上午11:04

相关推荐

  • 孪生素数问题解决了吗

    孪生素数问题与哥德巴赫猜想是一个方向的问题,但是进展似乎比哥德巴赫猜想好一点。什么是孪生素数呢,比方说3和5,这两个数都是素数,中间差了2,5和7也是两个素数,中间也差了2,29和…

    2023年4月14日
    0
  • 消毒喷雾枪怎么充电

    本文目录 1.步骤一:准备充电器 2.步骤二:将充电器插入消毒喷雾枪 3.步骤三:将充电器插入电源 4.步骤四:等待充电完成 消毒喷雾枪近年来越来越受到人们的欢迎,因为它可以快速、…

    生活百科 2023年6月16日
    0
  • 支付宝可以转账微信钱包

    10月16日,#支付宝已支持给微信QQ好友转账#话题登上热搜。有网友发现,支付宝已可以给微信好友转账,但不是直接转账,而是输入金额后会生成一个二维码,然后在微信等渠道可以扫码领取,…

    2023年3月13日
    0
  • 抖音最贵的礼物是不是嘉年华

    抖音作为一款国内最受欢迎的短视频分享应用,吸引了大量的用户,更是带动了一股全新的网络文化。在这个平台上,不仅有各种有趣、好玩的视频,还有丰富多样的礼物。其中,不少用户会选择送出一些…

    生活百科 2023年7月19日
    0
  • 吉利汽车怎么样,吉利汽车怎么样发电

    第一宗 离合重 原因分析:记者在和一些吉利车主沟通过程中了解到,2004年以前购买吉利的车主反映离合重的现象比较普遍,这些车主在维修过程中认为造成离合重的原因是因为离合器拉线的问题…

    生活百科 2023年6月2日
    0
  • 贷款买手机不还贷款的后果

    以现在的生活方式,手机已经成为了人们生活中不可或缺的一部分。然而,有些人为了购买高端手机,选择了贷款购买。但是,如果贷款买手机不还贷款,就会面临一系列的后果。 首先,不还贷款会导致…

    生活百科 2023年8月17日
    0
  • 鸭屎香柠檬茶是什么茶底

    鸭屎香柠檬茶是一种现在非常受欢迎的饮品,但是很多人都不知道它的茶底是什么。实际上,鸭屎香柠檬茶的茶底是绿茶和乌龙茶的混合物。 绿茶是一种非常健康的茶,因为它富含抗氧化剂和其他有益的…

    生活百科 2023年7月11日
    0
  • 肯德基爆汁三柠油柑茶多少钱一瓶

    肯德基(KFC)爆汁三柠油柑茶是近年来备受欢迎的饮品之一。这款饮品以其清新的柠檬香和浓郁的茶味,加上一些爆汁果粒,让人们在炎炎夏日感到清凉舒适。那么,肯德基爆汁三柠油柑茶的价格是多…

    生活百科 2023年9月23日
    0
  • 空调关机后外机还在运转是什么原因

    本文目录 1. 自带除湿功能 2. 维持室内温度 3. 空调故障 4.总结 在使用空调过程中,我们可能会遇到这样的情况:明明已经将空调关机,但是走到室外却发现外机还在运转,这是怎么…

    生活百科 2023年6月29日
    0
  • 韩综《我们结婚了》红薯夫妇是哪一季的演员

    韩综《我们结婚了》一共拍摄了四季,其中知名度最高的一季应该是第二季。在第二季中的孙佳仁和赵权的“亚当夫妇”,徐贤和郑容和的“红薯夫妇”,宋茜和尼坤的“维尼夫妇”被称为“我结”历史上…

    2023年5月26日
    0

发表评论

登录后才能评论