У меня есть следующий код:
[DataContract]
class TestContract {
private String _Name;
private Int32 _Age;
[DataMember( Name = "Name" )]
public String Name {
get { return _Name; }
set { _Name = value; }
}
[DataMember( Name = "Age" )]
public Int32 Age {
get { return _Age; }
set { _Age = value; }
}
}
[Serializable]
public class DNCJsonDictionary<K, V> : ISerializable {
Dictionary<K, V> dict = new Dictionary<K, V>();
public DNCJsonDictionary() { }
protected DNCJsonDictionary( SerializationInfo info, StreamingContext context ) {
}
public void GetObjectData( SerializationInfo info, StreamingContext context ) {
foreach( K key in dict.Keys ) {
info.AddValue( key.ToString(), dict[ key ] );
}
}
public void Add( K key, V value ) {
dict.Add( key, value );
}
public V this[ K index ] {
set { dict[ index ] = value; }
get { return dict[ index ]; }
}
}
public class MainClass {
public static String Serialize( Object data ) {
var serializer = new DataContractJsonSerializer( data.GetType() );
var ms = new MemoryStream();
serializer.WriteObject( ms, data );
return Encoding.UTF8.GetString( ms.ToArray() );
}
public static void Main() {
DNCJsonDictionary<String, Object> address = new DNCJsonDictionary<String, Object>();
address[ "Street" ] = "30 Rockefeller Plaza";
address[ "City" ] = "New York City";
address[ "State" ] = "NY";
TestContract test = new TestContract();
test.Name = "CsDJ";
test.Age = 28;
DNCJsonDictionary<String, Object> result = new DNCJsonDictionary<String, Object>();
result[ "foo" ] = "bar";
result[ "Name" ] = "John Doe";
result[ "Age" ] = 32;
result[ "Address" ] = address;
// ** --- THIS THROWS AN EXCEPTION!!! --- **
result[ "test" ] = test;
Console.WriteLine( Serialize( result ) );
Console.ReadLine();
}
}
При запуске я получаю следующее исключение:
Введите 'Json_Dictionary_Test.TestContract' с именем контракта данных 'TestContract: http://schemas.datacontract.org/2004/07/Json_Dictionary_Test' не ожидается.Добавьте любые типы, которые не известны статически, в список известных типов - например, с помощью атрибута KnownTypeAttribute или добавив их в список известных типов, передаваемых в DataContractSerializer.
Но я не понимаютот!Как я знаю, KnownTypeAttribute используется только для десериализации, и если есть наследование, не так ли?Но здесь только сериализация.И без члена datacontract работает нормально.
Есть идеи?
Я понял что-то, что работает!Существует родительский класс со списком KnownTypes, который я заполняю всеми дочерними классами и который будет использоваться при сериализации:
[DataContract]
[KnownType( "GetKnownTypes" )] // for serialization
class ResultContract {
private static List<Type> KnownTypes { get; set; }
public static List<Type> GetKnownTypes() {
return KnownTypes;
}
static ResultContract() {
KnownTypes = new List<Type>();
try {
foreach( Type type in Assembly.GetExecutingAssembly().GetTypes() ) {
if( !type.IsAbstract && type.IsSubclassOf( typeof( ResultContract ) ) ) {
KnownTypes.Add( type );
}
}
} catch( Exception ex ) {
Console.WriteLine( "Fatal error!" );
}
}
}
[DataContract]
class TestContract : *ResultContract* {
...
}
...