【转载】对象克隆(C# 快速高效率复制对象另一种方式 表达式树转)

Linqpad 代码

Benmarker测试结果耗时改善很明显,而且还可以支持深度复制,看来果然很优秀

BenchmarkDotNet=v0.12.0, OS=Windows 10.0.18362
Intel Core i7-9700 CPU 3.00GHz, 1 CPU, 8 logical and 8 physical cores
.NET Core SDK=3.1.201
[Host] : .NET Core 3.1.3 (CoreCLR 4.700.20.11803, CoreFX 4.700.20.12001), X64 RyuJIT



|              Method |          Mean |      Error |     StdDev |
|

-------------------- |--------------:|-----------:|-----------:|

| TestTransReflection |   1,176.83 ns |  15.771 ns |  14.752 ns |
|      TestExpression | 105,372.34 ns | 560.277 ns | 437.427 ns |
|        TestTransExp |     272.02 ns |   5.334 ns |   6.746 ns |
|      TestTransExpV2 |      14.51 ns |   0.349 ns |   0.466 ns |

<Query Kind="Program">
  <NuGetReference>BenchmarkDotNet</NuGetReference>
  <Namespace>System.Windows.Forms.DataVisualization.Charting</Namespace>
  <Namespace>BenchmarkDotNet.Running</Namespace>
  <Namespace>System.Security.Cryptography</Namespace>
  <Namespace>BenchmarkDotNet.Attributes</Namespace>
</Query>

#LINQPad optimize+
/*

对象克隆(C# 快速高效率复制对象另一种方式 表达式树转) 
https://www.cnblogs.com/lsgsanxiao/p/8205096.html
*/

void Main()
{
    Student s = new Student() { Age = 20, Id = 1, Name = "Emrys" , StudentAddress = new Address(){address="testaddress"}};
	//StudentSecond ss= TransReflection<Student, StudentSecond>(s);

	StudentSecond ss= TransExpV2<Student, StudentSecond>.Trans(s);
	ss.Dump();
	
	 var summary = BenchmarkRunner.Run<CloneBenchmarks>();
	 
}


public class Student
{
    public int Id { get; set; }
    public string Name { get; set; } 
    public int Age { get; set; } 
	public Address StudentAddress { get; set; } 
}

public class StudentSecond
{
    public int Id { get; set; }
    public string Name { get; set; }
    public int Age { get; set; } 
	public Address StudentAddress { get; set; } 
}

public class Address 
{ 
    public string address { get; set; } 
} 


public class CloneBenchmarks
{

	[Benchmark]
	public object TestTransReflection(){
	    Student s = new Student() { Age = 20, Id = 1, Name = "Emrys" };
		StudentSecond ss= TransReflection<Student, StudentSecond>(s);
		return ss;
	}
	
	[Benchmark]
	public object TestExpression(){
	    Student s = new Student() { Age = 20, Id = 1, Name = "Emrys" };
		Expression<Func<Student, StudentSecond>> ss = (x) => new StudentSecond { Age = x.Age, Id = x.Id, Name = x.Name };
        var f = ss.Compile();
        StudentSecond studentSecond = f(s);
		return studentSecond;
	}
	
	private static Dictionary<string, object> _Dic = new Dictionary<string, object>();


	[Benchmark]
	public object TestTransExp(){
	    Student s = new Student() { Age = 20, Id = 1, Name = "Emrys" };
		StudentSecond ss= TransExp<Student, StudentSecond>(s);
		return ss;
	}
	
	[Benchmark]
	public object TestTransExpV2(){
	    Student s = new Student() { Age = 20, Id = 1, Name = "Emrys" };
		StudentSecond ss= TransExpV2<Student, StudentSecond>.Trans(s);
		return ss;
	}
	
	public  TOut TransReflection<TIn, TOut>(TIn tIn)
	{
	    TOut tOut = Activator.CreateInstance<TOut>();
	    var tInType = tIn.GetType();
	    foreach (var itemOut in tOut.GetType().GetProperties())
	    {
	        var itemIn = tInType.GetProperty(itemOut.Name); ;
	        if (itemIn != null)
	        {
	            itemOut.SetValue(tOut, itemIn.GetValue(tIn));
	        }
	    }
	    return tOut;
	}
	
   private static TOut TransExp<TIn, TOut>(TIn tIn)
    {
        string key = string.Format("trans_exp_{0}_{1}", typeof(TIn).FullName, typeof(TOut).FullName);
        if (!_Dic.ContainsKey(key))
        {
            ParameterExpression parameterExpression = Expression.Parameter(typeof(TIn), "p");
            List<MemberBinding> memberBindingList = new List<MemberBinding>();

            foreach (var item in typeof(TOut).GetProperties())
            {  

				if (!item.CanWrite)
				continue; 
                MemberExpression property = Expression.Property(parameterExpression, typeof(TIn).GetProperty(item.Name));
                MemberBinding memberBinding = Expression.Bind(item, property);
                memberBindingList.Add(memberBinding);
            }

            MemberInitExpression memberInitExpression = Expression.MemberInit(Expression.New(typeof(TOut)), memberBindingList.ToArray());
            Expression<Func<TIn, TOut>> lambda = Expression.Lambda<Func<TIn, TOut>>(memberInitExpression, new ParameterExpression[] { parameterExpression });
            Func<TIn, TOut> func = lambda.Compile();

            _Dic[key] = func;
        }
        return ((Func<TIn, TOut>)_Dic[key])(tIn);
    }
	


}

public static class TransExpV2<TIn, TOut>
{

    private static readonly Func<TIn, TOut> cache = GetFunc();
    private static Func<TIn, TOut> GetFunc()
    {
        ParameterExpression parameterExpression = Expression.Parameter(typeof(TIn), "p");
        List<MemberBinding> memberBindingList = new List<MemberBinding>();

        foreach (var item in typeof(TOut).GetProperties())
        {
if (!item.CanWrite)
continue;

            MemberExpression property = Expression.Property(parameterExpression, typeof(TIn).GetProperty(item.Name));
            MemberBinding memberBinding = Expression.Bind(item, property);
            memberBindingList.Add(memberBinding);
        }

        MemberInitExpression memberInitExpression = Expression.MemberInit(Expression.New(typeof(TOut)), memberBindingList.ToArray());
        Expression<Func<TIn, TOut>> lambda = Expression.Lambda<Func<TIn, TOut>>(memberInitExpression, new ParameterExpression[] { parameterExpression });

        return lambda.Compile();
    }
	
	public static TOut Trans(TIn tIn)
    {
        return cache(tIn);
    }
}

  

猜你喜欢

转载自www.cnblogs.com/sui84/p/12815283.html