EntityExtensions.cs 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Reflection;
  5. using System.Text;
  6. using System.Threading.Tasks;
  7. namespace OASystem.Domain.Entities
  8. {
  9. /// <summary>
  10. /// 实体类完成状态验证扩展方法
  11. /// 提供便捷的方法来检查实体对象是否所有字段都已填写完成
  12. /// </summary>
  13. public static class EntityExtensions
  14. {
  15. /// <summary>
  16. /// 检查实体对象的所有字段是否都已填写完成
  17. /// 包括所有属性(字符串、值类型、引用类型)
  18. /// </summary>
  19. /// <param name="obj">要检查的实体对象</param>
  20. /// <returns>
  21. /// true: 所有字段都已正确填写
  22. /// false: 存在未填写或为默认值的字段
  23. /// </returns>
  24. /// <example>
  25. /// <code>
  26. /// var entity = new YourEntity { Name = "John", Age = 25 };
  27. /// if (entity.IsCompleted())
  28. /// {
  29. /// Console.WriteLine("所有字段已填写完成");
  30. /// }
  31. /// </code>
  32. /// </example>
  33. public static bool IsCompleted(this object obj)
  34. {
  35. // 获取实体类型的所有属性
  36. var properties = obj.GetType().GetProperties();
  37. // 使用LINQ的All方法检查所有属性是否都已填写
  38. // 如果所有属性都满足IsPropertyFilled条件,返回true;否则返回false
  39. return properties.All(property => IsPropertyFilled(property, obj));
  40. }
  41. /// <summary>
  42. /// 检查实体对象的必填字段是否都已填写完成
  43. /// 只检查标记了[RequiredField]特性的属性
  44. /// </summary>
  45. /// <param name="obj">要检查的实体对象</param>
  46. /// <returns>
  47. /// true: 所有必填字段都已正确填写
  48. /// false: 存在未填写的必填字段
  49. /// </returns>
  50. /// <example>
  51. /// <code>
  52. /// var entity = new YourEntity { Name = "John" }; // Age标记为[RequiredField]
  53. /// if (!entity.IsRequiredCompleted())
  54. /// {
  55. /// Console.WriteLine("请填写所有必填字段");
  56. /// }
  57. /// </code>
  58. /// </example>
  59. public static bool IsRequiredCompleted(this object obj)
  60. {
  61. // 获取实体类型的所有属性
  62. var properties = obj.GetType().GetProperties();
  63. // 筛选出标记了[RequiredField]特性的属性(必填字段)
  64. var requiredProperties = properties.Where(p => p.GetCustomAttribute<RequiredFieldAttribute>() != null);
  65. // 检查所有必填字段是否都已填写
  66. return requiredProperties.All(property => IsPropertyFilled(property, obj));
  67. }
  68. /// <summary>
  69. /// 检查单个属性是否已填写 - 使用模式匹配的安全版本
  70. /// 根据属性类型采用不同的验证规则
  71. /// </summary>
  72. /// <param name="property">要检查的属性信息</param>
  73. /// <param name="obj">包含该属性的对象实例</param>
  74. /// <returns>
  75. /// true: 属性已正确填写
  76. /// false: 属性未填写或为默认值
  77. /// </returns>
  78. /// <remarks>
  79. /// 验证规则:
  80. /// 1. 字符串类型:不能为null、空字符串或空白字符串
  81. /// 2. 值类型:不能等于该类型的默认值(如int不能为0,DateTime不能为MinValue等)
  82. /// 3. 布尔类型:总是返回true(因为bool总是有true或false值)
  83. /// 4. 引用类型:不能为null
  84. /// </remarks>
  85. private static bool IsPropertyFilled(PropertyInfo property, object obj)
  86. {
  87. // 获取属性的值
  88. var value = property.GetValue(obj);
  89. // 如果值为null,直接返回false
  90. if (value == null) return false;
  91. return property.PropertyType switch
  92. {
  93. // 字符串类型
  94. Type t when t == typeof(string) => !string.IsNullOrWhiteSpace((string)value),
  95. // 值类型使用模式匹配安全转换
  96. Type t when t == typeof(int) => value is int intValue && intValue != 0,
  97. Type t when t == typeof(long) => value is long longValue && longValue != 0,
  98. Type t when t == typeof(decimal) => value is decimal decimalValue && decimalValue != 0,
  99. Type t when t == typeof(double) => value is double doubleValue && doubleValue != 0,
  100. Type t when t == typeof(float) => value is float floatValue && floatValue != 0,
  101. Type t when t == typeof(DateTime) => value is DateTime dateValue && dateValue != DateTime.MinValue,
  102. Type t when t == typeof(bool) => value is bool, // bool只要不是null就认为已填写
  103. // 可空类型处理
  104. Type t when IsNullableType(t) => !IsDefaultNullableValue(value),
  105. // 其他值类型
  106. Type t when t.IsValueType => !value.Equals(Activator.CreateInstance(t)),
  107. // 引用类型(已经检查过null)
  108. _ => true
  109. };
  110. }
  111. /// <summary>
  112. /// 检查类型是否为可空类型(Nullable<T>)
  113. /// </summary>
  114. /// <param name="type">要检查的类型</param>
  115. /// <returns>是否为可空类型</returns>
  116. private static bool IsNullableType(Type type)
  117. {
  118. return type.IsGenericType && type.GetGenericTypeDefinition() == typeof(Nullable<>);
  119. }
  120. /// <summary>
  121. /// 检查可空类型的值是否为默认值(null或默认值)
  122. /// </summary>
  123. /// <param name="value">可空类型的值</param>
  124. /// <returns>是否为默认值</returns>
  125. private static bool IsDefaultNullableValue(object value)
  126. {
  127. var underlyingType = Nullable.GetUnderlyingType(value.GetType());
  128. if (underlyingType == null) return true;
  129. var defaultValue = Activator.CreateInstance(underlyingType);
  130. return value.Equals(defaultValue);
  131. }
  132. /// <summary>
  133. /// 快速检查实体是否包含数据(至少有一个字段已填写)
  134. /// 用于判断实体是否为空或未初始化
  135. /// </summary>
  136. /// <param name="obj">要检查的实体对象</param>
  137. /// <returns>
  138. /// true: 至少有一个字段已填写
  139. /// false: 所有字段都为默认值或空
  140. /// </returns>
  141. /// <example>
  142. /// <code>
  143. /// var entity = new YourEntity();
  144. /// if (!entity.HasData())
  145. /// {
  146. /// Console.WriteLine("实体没有任何数据");
  147. /// }
  148. /// </code>
  149. /// </example>
  150. public static bool HasData(this object obj)
  151. {
  152. // 获取所有属性
  153. var properties = obj.GetType().GetProperties();
  154. // 使用LINQ的Any方法检查是否存在至少一个已填写的属性
  155. return properties.Any(property => IsPropertyFilled(property, obj));
  156. }
  157. }
  158. /// <summary>
  159. /// 必填字段标记特性
  160. /// 用于标记实体类中必须填写的属性
  161. /// </summary>
  162. [AttributeUsage(AttributeTargets.Property)]
  163. public class RequiredFieldAttribute : Attribute
  164. {
  165. /// <summary>
  166. /// 验证失败时显示的错误消息
  167. /// </summary>
  168. public string ErrorMessage { get; }
  169. /// <summary>
  170. /// 初始化必填字段特性
  171. /// </summary>
  172. /// <param name="errorMessage">验证失败时显示的错误消息</param>
  173. public RequiredFieldAttribute(string errorMessage = "该字段为必填项")
  174. {
  175. ErrorMessage = errorMessage;
  176. }
  177. }
  178. }