JSON.NET VS BinaryFormatter 性能

时间:2021-12-13 02:31:53

近期有个性能调优工作。通过dottrace 分析,发现几处问题,其中json.net 在序列化和反序列化的时候也比较耗性能,所以考虑能不能通过其它序列化方式来提高性能。

1 object 序列化代码

    public class ObjectConvert
{
/// <summary>
/// 将一个object对象序列化,返回一个byte[]
/// </summary>
/// <param name="obj">能序列化的对象</param>
/// <returns></returns>
public static byte[] ObjectToBytes(object obj)
{
using (MemoryStream ms = new MemoryStream())
{
IFormatter formatter = new BinaryFormatter();
formatter.Serialize(ms, obj); return ms.GetBuffer();
}
} /// <summary>
/// 将一个序列化后的byte[]数组还原
/// </summary>
/// <param name="Bytes"></param>
/// <returns></returns>
public static object BytesToObject(byte[] Bytes)
{
using (MemoryStream ms = new MemoryStream(Bytes))
{
IFormatter formatter = new BinaryFormatter();
return formatter.Deserialize(ms);
}
}
}

2 Test 方法,用了 老赵写的codetimer

    [TestClass]
public class SerializeTestor
{
private List<UserInfo> InitData(int count)
{
UserInfo user = null;
List<UserInfo> users = new List<UserInfo>(); for (int i = 0; i < count; i++)
{
user = new UserInfo();
user.Age = new Random().Next(10, 100);
user.Name = "yuan";
user.CreateTime = DateTime.Now;
users.Add(user);
} return users;
}
private List<UserInfo> InitBigStringData(int count)
{
UserInfo user = null;
List<UserInfo> users = new List<UserInfo>(); for (int i = 0; i < count; i++)
{
user = new UserInfo();
user.Age = new Random().Next(10, 100);
//user.Name = "yuan";
user.Name = "yudddddddddddddddddddddddddddyudddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddanyudddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddanyudddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddanyudddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddanddddddddddddddddddddddddddddddddddddddddddddddddddan";
user.CreateTime = DateTime.Now;
users.Add(user);
} return users;
} [TestMethod]
public void Serialize_十万次_Test()
{
var users = InitData(100000);
// object序列化
CodeTimer.Time("object序列化", 1, () =>
{
var temps = ObjectConvert.ObjectToBytes(users);
});
// newtonsoft.json
CodeTimer.Time("json.net序列化", 1, () =>
{
var temps = JsonConvert.SerializeObject(users);
});
} [TestMethod]
public void Serialize_百万次_Test()
{
var users = InitData(1000000);
// object序列化
CodeTimer.Time("object序列化", 1, () =>
{
var temps = ObjectConvert.ObjectToBytes(users);
});
// newtonsoft.json
CodeTimer.Time("json.net序列化", 1, () =>
{
var temps = JsonConvert.SerializeObject(users);
});
}
[TestMethod]
public void SerializeBigString_十万次_Test()
{
var users = InitBigStringData(100000);
// object序列化
CodeTimer.Time("object序列化", 1, () =>
{
var temps = ObjectConvert.ObjectToBytes(users);
});
// newtonsoft.json
CodeTimer.Time("json.net序列化", 1, () =>
{
var temps = JsonConvert.SerializeObject(users);
});
}
[TestMethod]
public void SerializeBigString_百万次_Test()
{
var users = InitBigStringData(1000000);
// object序列化
CodeTimer.Time("object序列化", 1, () =>
{
var temps = ObjectConvert.ObjectToBytes(users);
});
// newtonsoft.json
CodeTimer.Time("json.net序列化", 1, () =>
{
var temps = JsonConvert.SerializeObject(users);
});
}
}

  

3 测试结果:

Serialize_十万次_Test:
object序列化
Time Elapsed: 188ms
Time Elapsed (one time):188ms
CPU time: 187,500,000ns
CPU time (one time): 187,500,000ns
Gen 0: 1
Gen 1: 0
Gen 2: 0 json.net序列化
Time Elapsed: 138ms
Time Elapsed (one time):138ms
CPU time: 140,625,000ns
CPU time (one time): 140,625,000ns
Gen 0: 3
Gen 1: 1
Gen 2: 0 ================================================
Serialize_百万次_Test
object序列化
Time Elapsed: 2,073ms
Time Elapsed (one time):2,073ms
CPU time: 2,078,125,000ns
CPU time (one time): 2,078,125,000ns
Gen 0: 23
Gen 1: 1
Gen 2: 0 json.net序列化
Time Elapsed: 1,449ms
Time Elapsed (one time):1,449ms
CPU time: 1,437,500,000ns
CPU time (one time): 1,437,500,000ns
Gen 0: 28
Gen 1: 27
Gen 2: 0 ===================================================
SerializeBigString_十万次_Test
object序列化
Time Elapsed: 187ms
Time Elapsed (one time):187ms
CPU time: 187,500,000ns
CPU time (one time): 187,500,000ns
Gen 0: 3
Gen 1: 2
Gen 2: 2 json.net序列化
Time Elapsed: 339ms
Time Elapsed (one time):339ms
CPU time: 328,125,000ns
CPU time (one time): 328,125,000ns
Gen 0: 19
Gen 1: 14
Gen 2: 3 ======================================================
SerializeBigString_百万次_Test
object序列化
Time Elapsed: 2,099ms
Time Elapsed (one time):2,099ms
CPU time: 2,046,875,000ns
CPU time (one time): 2,046,875,000ns
Gen 0: 26
Gen 1: 25
Gen 2: 3 json.net序列化
System.OutOfMemoryException: 引发类型为“System.OutOfMemoryException”的异常。

  

4 结论:

JSON.NET 在小对象处理时,性能比BinaryFormatter好

JSON.NET 比 BinaryFormatter  耗内存

JSON.NET  GC中带回收的对象会比较影响整体性能