什么是.NET的强类型字符串(Strongly typed string)?

一给  金牌会员 | 2024-11-29 18:47:38 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 798|帖子 798|积分 2394

在.NET中,强类型字符串(Strongly typed string)并不是一个官方的概念,是指利用特定的结构来表示某种类型字符串数据的编码实践。类似于枚举,可以提供编译时检查类型,淘汰运行时错误,以及更好的可读性和维护性。相比于枚举,具有更好的扩展性以及更强的束缚性。
枚举

枚举提供了一种便捷的方法来利用相关常数集并将常数值与名称相关联,具有类型安全、可读性高以及编译时检查等优点。但是枚举类型不能定义任何方法、属性或变乱,只能通过扩展方法功能模仿向枚举类型添加方法。
只管枚举提供了编译时检查,但对输入值的束缚是有限的。例如,下面这个枚举有四个值,默认情况下是int类型。取值范围为0 ~ 3。
  1. public enum Roles {
  2.     Author,
  3.     Editor,
  4.     Administrator,
  5.     SalesRepresentative
  6. }
复制代码
然后,有一个方法接受这个枚举类型的参数:
  1. public string DoSomething(Roles role) {
  2.     return role.ToString();
  3. }
复制代码
许多开辟人员大概不会检查传入值是否为实际有效的枚举值。任何int类型都可以转换,大概出现下边这种代码:
  1. var result = myObject.DoSomething((Roles)10);
复制代码
输出的效果是 “10”,如果后续代码中有基于这个枚举的分支语句或者条件判断,将产生错误的效果。对于这种情况,强类型字符串是一个不错的选择。
强类型字符串(Strongly typed string)

强类型字符串要声明成带有字符串构造函数的不可变值类型(struct),即要在该类型上用  readonly 修饰符,并为其实现 IEquatable 接口。要覆写强类型字符串的 ToString() 方法,以返回隐式的字符串值。并将已知的强类型字符串通过静态只读属性声明到该类型上。
为了让强类型字符串在通用代码的语言结构上看起来更像字符串或者枚举,需要为强类型字符串覆写相等运算符。
以下就是 .NET 源码中加密哈希算法的名称强类型字符串HashAlgorithmName的代码
  1. using System.Diagnostics.CodeAnalysis;
  2. namespace System.Security.Cryptography
  3. {
  4.    
  5.     public readonly struct HashAlgorithmName : IEquatable<HashAlgorithmName>
  6.     {
  7.         public static HashAlgorithmName MD5 { get { return new HashAlgorithmName("MD5"); } }
  8.         public static HashAlgorithmName SHA1 { get { return new HashAlgorithmName("SHA1"); } }
  9.         public static HashAlgorithmName SHA256 { get { return new HashAlgorithmName("SHA256"); } }
  10.         public static HashAlgorithmName SHA384 { get { return new HashAlgorithmName("SHA384"); } }
  11.         public static HashAlgorithmName SHA512 { get { return new HashAlgorithmName("SHA512"); } }
  12.         public static HashAlgorithmName SHA3_256 => new HashAlgorithmName("SHA3-256");
  13.         public static HashAlgorithmName SHA3_384 => new HashAlgorithmName("SHA3-384");
  14.         public static HashAlgorithmName SHA3_512 => new HashAlgorithmName("SHA3-512");
  15.         private readonly string? _name;
  16.         public HashAlgorithmName(string? name)
  17.         {
  18.             // Note: No validation because we have to deal with default(HashAlgorithmName) regardless.
  19.             _name = name;
  20.         }
  21.         public string? Name
  22.         {
  23.             get { return _name; }
  24.         }
  25.         public override string ToString()
  26.         {
  27.             return _name ?? string.Empty;
  28.         }
  29.         public override bool Equals([NotNullWhen(true)] object? obj)
  30.         {
  31.             return obj is HashAlgorithmName && Equals((HashAlgorithmName)obj);
  32.         }
  33.         public bool Equals(HashAlgorithmName other)
  34.         {
  35.             // NOTE: intentionally ordinal and case sensitive, matches CNG.
  36.             return _name == other._name;
  37.         }
  38.         public override int GetHashCode()
  39.         {
  40.             return _name == null ? 0 : _name.GetHashCode();
  41.         }
  42.         public static bool operator ==(HashAlgorithmName left, HashAlgorithmName right)
  43.         {
  44.             return left.Equals(right);
  45.         }
  46.         public static bool operator !=(HashAlgorithmName left, HashAlgorithmName right)
  47.         {
  48.             return !(left == right);
  49.         }
  50.         //其他扩展功能
  51.         public static bool TryFromOid(string oidValue, out HashAlgorithmName value)
  52.         {
  53.             ArgumentNullException.ThrowIfNull(oidValue);
  54.             switch (oidValue)
  55.             {
  56.                 case Oids.Md5:
  57.                     value = MD5;
  58.                     return true;
  59.                 case Oids.Sha1:
  60.                     value = SHA1;
  61.                     return true;
  62.                 case Oids.Sha256:
  63.                     value = SHA256;
  64.                     return true;
  65.                 case Oids.Sha384:
  66.                     value = SHA384;
  67.                     return true;
  68.                 case Oids.Sha512:
  69.                     value = SHA512;
  70.                     return true;
  71.                 case Oids.Sha3_256:
  72.                     value = SHA3_256;
  73.                     return true;
  74.                 case Oids.Sha3_384:
  75.                     value = SHA3_384;
  76.                     return true;
  77.                 case Oids.Sha3_512:
  78.                     value = SHA3_512;
  79.                     return true;
  80.                 default:
  81.                     value = default;
  82.                     return false;
  83.             }
  84.         }
  85.         public static HashAlgorithmName FromOid(string oidValue)
  86.         {
  87.             if (TryFromOid(oidValue, out HashAlgorithmName value))
  88.             {
  89.                 return value;
  90.             }
  91.             throw new CryptographicException(SR.Format(SR.Cryptography_InvalidHashAlgorithmOid, oidValue));
  92.         }
  93.     }
  94. }
复制代码
这段代码更好地束缚了加密哈希算法名称的输入,同时还扩展了其他功能。但比枚举繁琐不少。
根据《框架计划指南》发起:当基类支持一组固定的输入参数,但是派生类需要支持更多的参数时,发起利用强类型字符串;当仅由密封类型利用时,只需要利用预定义的值,枚举将是更好的选择。
此外,枚举通常定义的是封闭的选项集,对于操作体系版本这种开放聚集,也发起利用强类型字符串。控件库 HandyControl 中的 SystemVersionInfo正是这样的例子。
参考

Enum Alternatives in C# | Blog
利用枚举类(而不是枚举类型) - .NET | Microsoft Learn

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

一给

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表