前言:

我们在很多项目场景中使用对象映射工具,那么使用最多的OOM对象工具也就那几个。今天所说的EmitMapper 和TinyMapper 两者的性能都是很高的,相比autoMapper 速度不知道快了多少倍,因为我平时使用的最多EmitMapper,所在业余时间做了一下测试两者对比。

测试数据:10万条和1000万条,分手动,TinyMapper,EmitMapper分别进行测试。22次用例结果:平均值:(注autoMapper不参与,因为笔者认为性能暂时无法和以上相比)

SELECT   TOP (200) typename, AVG(esTime) AS estimed
FROM emitTinyMapperData
GROUP BY typename
ORDER BY 2

结果显示:

手动生成 458
TinyMapper 535
EmitMapper 618

数据加大到:1000万条:

手动生成    458
TinyMapper 525
EmitMapper 593

TestModel数据对象(为了方便看就写在一起了):

 namespace YGOP.PayCenter.MapperTest
{
/// <summary>
/// 测试对象
/// </summary>
public class TestModel
{
public string modelName { get; set; }
} /// <summary>
/// 测试对象A
/// </summary>
public class TestModelA: TestModel
{
public string modelNameA { get; set; }
public int age { get; set; }
public int sex { get; set; }
/// <summary>
/// 注意此处
/// </summary>
public OrderItem orderItem { get; set; }
} public class TestModelB : TestModel
{
public string modelNameB { get; set; } public long sex { get; set; }
public int ageB { get; set; }
/// <summary>
/// 注意此处
/// </summary>
public string gender { get; set; }
public string bak { get; set; }
/// <summary>
/// 注意此处
/// </summary>
public OrderItem2 orderItem2 { get; set; } }
/// <summary>
/// 订单子对象测试
/// </summary>
public class OrderItem
{
public string orderId { get;set;}
public string subId { get; set; } public int qty { get; set; }
public double price { get; set; }
public string bak { get; set; } public string qq { get; set; }
public bool isMem { get; set; } } /// <summary>
/// 订单子对象测试
/// </summary>
public class OrderItem2
{
public string orderId { get; set; }
public string itemId { get; set; } public string qq { get; set; }
public int amount { get; set; }
public decimal unitprice { get; set; } public bool isMem { get; set; } }

测试用例代码:

0.手动生成对象

    /// <summary>
/// 手工生成对象
/// </summary>
/// <param name="max"></param>
/// <returns></returns>
private List<TestModelA> HanderCustomObject(int max= )
{
List<TestModelA> modelAlist = new List<TestModelA>(); for (int i = ; i < max; i++)
{
modelAlist.Add(new TestModelA()
{
age = i + ,
modelNameA = "测试a" + i,
modelName = "测试a" + i,
sex = i,
orderItem = new OrderItem()
{
bak = "" + i,
orderId = "orderId" + DateTime.Now.ToString() + i.ToString(),
price = i + new Random().Next(, ),
qty = i,
qq = "" + i,
subId = "subId" + i
,
isMem = i % ==
}
});
} return modelAlist; }

1.给 TinyMapper 做对象的关系映射

         /// <summary>
/// 给 TinyMapper 做对象的关系映射
/// </summary>
private void InitTinyMapperCustomMapper()
{ TinyMapper.Bind<OrderItem, OrderItem2>(mapconfig2Order =>
{
mapconfig2Order.Bind(a => a.qty, b => b.amount);
mapconfig2Order.Bind(a => a.price, b => b.unitprice);
mapconfig2Order.Bind(a => a.subId, b => b.itemId);
mapconfig2Order.Bind(a => a.bak, b => b.amount);
}); TinyMapper.Bind<TestModelA, TestModelB>(mapConfig =>
{
mapConfig.Ignore(src => src.modelName);
mapConfig.Bind(a => a.age, b => b.ageB);
mapConfig.Bind(a => a.sex, b => b.gender);
mapConfig.Bind(a => a.modelNameA, b => b.modelNameB);
mapConfig.Bind(a => a.orderItem, b => b.orderItem2);
}); }

2.给EmitMapper 做对象关系的映射

        /// <summary>
/// 给EmitMapper 做对象关系的映射
/// </summary>
/// <returns></returns>
private IMappingConfigurator InitEmitCustomMapper2()
{
IMappingConfigurator conf = new DefaultMapConfig().ConvertUsing<TestModelA, TestModelB>(
v => new TestModelB()
{
ageB = v.age,
gender = (v.sex % == ? "男" : "女"),
modelNameB = v.modelNameA,
orderItem2 = new OrderItem2()
{
amount = v.orderItem.qty,
itemId = v.orderItem.subId,
unitprice = Convert.ToDecimal(v.orderItem.price),
qq = v.orderItem.qq,
orderId = v.orderItem.orderId,
isMem = v.orderItem.isMem
} }); return conf;
}

3.最终的调用

       private void button1_Click(object sender, EventArgs e)
{
Stopwatch stopwatch = new Stopwatch();
stopwatch.Start(); int max = ;
var modelAlist = HanderCustomObject();
stopwatch.Stop();
string ok1 = string.Format(max + "次,手动生成 所用时间 :{0} 毫秒", stopwatch.ElapsedMilliseconds);
textBox1.AppendText(ok1 + "\r\n");
stopwatch.Start();
InitTinyMapperCustomMapper();
var ss = TinyMapperHelper.Instance.Convert<List<TestModelA>, List<TestModelB>>(modelAlist);
stopwatch.Stop();
string ok = string.Format(max + "次,TinyMapper 所用时间 :{0} 毫秒", stopwatch.ElapsedMilliseconds);
textBox1.AppendText(ok + "\r\n"); //Thread.Sleep(1000);
stopwatch.Start();
var conf= InitEmitCustomMapper2();
var ccc = EmitMapperHelper.Instance.Convert<List<TestModelA>, List<TestModelB>>(modelAlist, conf); stopwatch.Stop();
string ok2 = string.Format(max + "次,EmitMapper 所用时间 :{0} 毫秒", stopwatch.ElapsedMilliseconds);
textBox1.AppendText(ok2 + "\r\n"); return;
}

使用22次后,结果

 方式/10万次采样    耗时/毫秒
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper
手动生成
TinyMapper
EmitMapper

EmitMapper 和TinyMapper 两者简单对比-LMLPHP

EmitMapper 和TinyMapper 两者简单对比-LMLPHP

个人使用总结:

tinyMapper 真的挺快的,EmitMapper 仅比其差了一点点而已(10万-1000万条数据中的映射场景并不多见,70多毫秒也不会太在意)。

因为EmitMapper的作者在2011年就不在维护了,我认为EmitMapper还是够吊的了。

但tinyMapper 刚出来不久,但性能真的是无与伦对的美丽!!赞,希望能作者能改善。

最后给出我心中,以上工具的打分和优点缺点简单评。

EmitMapper 和TinyMapper 两者简单对比-LMLPHP

05-11 17:26