using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.Linq; using System.Linq.Expressions; using System.Reflection; using static System.Linq.Expressions.Expression; namespace SqlSugar { internal static class FastCopy { static ConcurrentDictionary copiers = new ConcurrentDictionary(); /// /// 复制两个对象同名属性值 /// /// /// /// 源对象 /// 目标对象 /// 源对象属性值为null时,是否将值复制给目标对象 public static void Copy(S source, T target, bool copyNull = true) { string name = string.Format("{0}_{1}_{2}", typeof(S), typeof(T), copyNull); object targetCopier; if (!copiers.TryGetValue(name, out targetCopier)) { Action copier = CreateCopier(copyNull); copiers.TryAdd(name, copier); targetCopier = copier; } Action action = (Action)targetCopier; action(source, target); } /// /// 为指定的两种类型编译生成属性复制委托 /// /// /// /// 源对象属性值为null时,是否将值复制给目标对象 /// private static Action CreateCopier(bool copyNull) { ParameterExpression source = Parameter(typeof(S)); ParameterExpression target = Parameter(typeof(T)); var sourceProps = typeof(S).GetProperties(BindingFlags.Instance | BindingFlags.Public).Where(p => p.CanRead).ToList(); var targetProps = typeof(T).GetProperties(BindingFlags.Instance | BindingFlags.Public).Where(p => p.CanWrite).ToList(); // 查找可进行赋值的属性 var copyProps = targetProps.Where(tProp => sourceProps.Where(sProp => sProp.Name == tProp.Name// 名称一致 且 && ( sProp.PropertyType == tProp.PropertyType// 属性类型一致 或 || sProp.PropertyType.IsAssignableFrom(tProp.PropertyType) // 源属性类型 为 目标属性类型 的 子类;eg:object target = string source; 或 || (tProp.PropertyType.IsValueType && sProp.PropertyType.IsValueType && // 属性为值类型且基础类型一致,但目标属性为可空类型 eg:int? num = int num; ((tProp.PropertyType.GenericTypeArguments.Length > 0 ? tProp.PropertyType.GenericTypeArguments[0] : tProp.PropertyType) == sProp.PropertyType)) )).Count() > 0); List expressionList = new List(); foreach (var prop in copyProps) { if (prop.PropertyType.IsValueType)// 属性为值类型 { PropertyInfo sProp = typeof(S).GetProperty(prop.Name); PropertyInfo tProp = typeof(T).GetProperty(prop.Name); if (sProp.PropertyType == tProp.PropertyType)// 属性类型一致 eg:int num = int num; 或 int? num = int? num; { var assign = Assign(Property(target, prop.Name), Property(source, prop.Name)); expressionList.Add(assign); } else if (sProp.PropertyType.GenericTypeArguments.Length <= 0 && tProp.PropertyType.GenericTypeArguments.Length > 0)// 属性类型不一致且目标属性类型为可空类型 eg:int? num = int num; { var convert = Convert(Expression.Property(source, prop.Name), tProp.PropertyType); var cvAssign = Assign(Expression.Property(target, prop.Name), convert); expressionList.Add(cvAssign); } } else// 属性为引用类型 { var assign = Assign(Property(target, prop.Name), Property(source, prop.Name));// 编译生成属性赋值语句 target.{PropertyName} = source.{PropertyName}; var sourcePropIsNull = Equal(Constant(null, prop.PropertyType), Property(source, prop.Name));// 判断源属性值是否为Null;编译生成 source.{PropertyName} == null var setNull = IsTrue(Constant(copyNull));// 判断是否复制Null值 编译生成 copyNull == True var setNullTest = IfThen(setNull, assign); var condition = IfThenElse(sourcePropIsNull, setNullTest, assign); /** * 编译生成 * if(source.{PropertyName} == null) * { * if(setNull) * { * target.{PropertyName} = source.{PropertyName}; * } * } * else * { * target.{PropertyName} = source.{PropertyName}; * } */ expressionList.Add(condition); } } var block = Block(expressionList.ToArray()); Expression> lambda = Lambda>(block, source, target); return lambda.Compile(); } internal static List GetDiff(T item, T trackingData) where T : class, new() { List result = new List(); foreach (var t in typeof(T).GetProperties()) { var leftvalue = t.GetValue(item); var rightvalue = t.GetValue(trackingData); if ((leftvalue == null && rightvalue != null)) { result.Add(t.Name); } else if (leftvalue!=null&&!leftvalue.Equals(rightvalue)) { result.Add(t.Name); } } return result; } } }