Zde je ukázkový serializátor pro System.Type, který serializuje název typu jako řetězec BSON. To má určitá omezení v tom, že metoda Deserialize selže, pokud název typu není systémový typ nebo ve stejném sestavení, ale můžete tento ukázkový serializátor vyladit tak, aby místo toho zapisoval AssemblyQualifiedName.
public class TypeSerializer : IBsonSerializer
{
public object Deserialize(BsonReader reader, Type nominalType, IBsonSerializationOptions options)
{
var actualType = nominalType;
return Deserialize(reader, nominalType, actualType, options);
}
public object Deserialize(BsonReader reader, Type nominalType, Type actualType, IBsonSerializationOptions options)
{
if (reader.CurrentBsonType == BsonType.Null)
{
return null;
}
else
{
var fullName = reader.ReadString();
return Type.GetType(fullName);
}
}
public bool GetDocumentId(object document, out object id, out Type idNominalType, out IIdGenerator idGenerator)
{
throw new InvalidOperationException();
}
public void Serialize(BsonWriter writer, Type nominalType, object value, IBsonSerializationOptions options)
{
if (value == null)
{
writer.WriteNull();
}
else
{
writer.WriteString(((Type)value).FullName);
}
}
public void SetDocumentId(object document, object id)
{
throw new InvalidOperationException();
}
}
Trik je v tom, aby to bylo správně zaregistrováno. Musíte jej zaregistrovat pro System.Type i System.RuntimeType, ale System.RuntimeType není veřejný, takže na něj nemůžete odkazovat ve svém kódu. Ale můžete se k tomu dostat pomocí Type.GetType. Zde je kód pro registraci serializátoru:
var typeSerializer = new TypeSerializer();
BsonSerializer.RegisterSerializer(typeof(Type), typeSerializer);
BsonSerializer.RegisterSerializer(Type.GetType("System.RuntimeType"), typeSerializer);
Použil jsem tuto testovací smyčku k ověření, že to funguje:
var types = new Type[] { typeof(int), typeof(string), typeof(Guid), typeof(C) };
foreach (var type in types)
{
var json = type.ToJson();
Console.WriteLine(json);
var rehydratedType = BsonSerializer.Deserialize<Type>(json);
Console.WriteLine("{0} -> {1}", type.FullName, rehydratedType.FullName);
}
kde C je jen prázdná třída:
public static class C
{
}