9

私は次のコードを持っています:

BaseContent.cs

public class BaseContent
{
   // Some auto properties
}

News.cs

public class News : BaseContent
{
   // Some more auto properties
}

Events.cs

public class Event : BaseContent
{
   // Some more auto properites
}

GenericResponse.cs

public class GenericResponse<T> 
{
  [XmlArray("Content")]
  [XmlArrayItem("NewsObject", typeof(News)]
  [XmlArrayItem("EventObject", typeof(Event)]
  public List<T> ContentItems { get; set; }
}

NewsResponse.cs

public class NewsResponse : GenericResponse<News> {}

EventResponse.cs

public class EventResponse : GenericResponse<Event> {}

ご覧のとおり、基本クラスBaseContentと、そこから派生する 2 つのクラスがあります。次に、xml ファイルの構造は常に同じですが、一部のプロパティが異なるため、一般的な応答クラスがあります。

[XmlArrayItem]特定のクラスに使用する名前を指定できると思いました。しかし今、私はエラーが発生します:

System.InvalidOperationException: 一時クラスを生成できません (結果 = 1)。エラー CS0012: 型 'System.Object' は、参照されていないアセンブリで定義されています。アセンブリ 'System.Runtime, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a' への参照を追加する必要があります。

Windows 8 アプリで作業しているため、この参照を追加できません。

そのうちの1つをコメントアウトすると、[XmlArrayItem]うまく機能しています。

誰でもこの問題を解決するためのアイデアを持っていますか?

更新XmlAttributesを使用する必要があるため、DataContractSerializer を使用 できません

4

4 に答える 4

11

編集:デモプロジェクトをダウンロードしてください

オブジェクトのすべてのプロパティを提供しなかったので、例としていくつか追加させてください。

public class BaseContent
{
    [XmlAttribute("Name")]
    public string Name { get; set; }
}

[XmlType(TypeName = "EventObject")]
public class Event : BaseContent
{
    [XmlAttribute("EventId")]
    public int EventId { get; set; }
}

[XmlType(TypeName = "NewsObject")]
public class News : BaseContent
{
    [XmlAttribute("NewsId")]
    public int NewsId { get; set; }
}

GenericResponse.csは、次のように定義できます。配列項目のtypeofを指定する必要はありません。

public class GenericResponse<T>
{
    [XmlArray("Content")]
    public List<T> ContentItems { get; set; }

    public GenericResponse()
    {
        this.ContentItems = new List<T>();
    }
}

そして、応答クラスがあります。

public class EventResponse : GenericResponse<Event>
{
}

public class NewsResponse : GenericResponse<News>
{
}

例1:EventResponseオブジェクトをシリアル化する

var response = new EventResponse
{
    ContentItems = new List<Event> 
    {
        new Event {
            EventId = 1,
            Name = "Event 1"
        },

        new Event {
            EventId = 2,
            Name = "Event 2"
        }
    }
};

string xml = XmlSerializer<EventResponse>.Serialize(response);

出力XML:

<?xml version="1.0" encoding="utf-8"?>
<EventResponse xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
    <Content>
        <EventObject Name="Event 1" EventId="1" />
        <EventObject Name="Event 2" EventId="2" />
    </Content>
</EventResponse>

NewsResponseで同じことを試してみると、問題なく動作します。ところで、私は一般的なXmlSerializerを使用しています。リンクをクリックして、詳細を確認してください。

XmlSerializer.cs:

/// <summary>
/// XML serializer helper class. Serializes and deserializes objects from/to XML
/// </summary>
/// <typeparam name="T">The type of the object to serialize/deserialize.
/// Must have a parameterless constructor and implement <see cref="Serializable"/></typeparam>
public class XmlSerializer<T> where T: class, new()
{
    /// <summary>
    /// Deserializes a XML string into an object
    /// Default encoding: <c>UTF8</c>
    /// </summary>
    /// <param name="xml">The XML string to deserialize</param>
    /// <returns>An object of type <c>T</c></returns>
    public static T Deserialize(string xml)
    {
        return Deserialize(xml, Encoding.UTF8, null);
    }

    /// <summary>
    /// Deserializes a XML string into an object
    /// Default encoding: <c>UTF8</c>
    /// </summary>
    /// <param name="xml">The XML string to deserialize</param>
    /// <param name="encoding">The encoding</param>
    /// <returns>An object of type <c>T</c></returns>
    public static T Deserialize(string xml, Encoding encoding)
    {
        return Deserialize(xml, encoding, null);
    }

    /// <summary>
    /// Deserializes a XML string into an object
    /// </summary>
    /// <param name="xml">The XML string to deserialize</param>
    /// <param name="settings">XML serialization settings. <see cref="System.Xml.XmlReaderSettings"/></param>
    /// <returns>An object of type <c>T</c></returns>
    public static T Deserialize(string xml, XmlReaderSettings settings)
    {
        return Deserialize(xml, Encoding.UTF8, settings);
    }

    /// <summary>
    /// Deserializes a XML string into an object
    /// </summary>
    /// <param name="xml">The XML string to deserialize</param>
    /// <param name="encoding">The encoding</param>
    /// <param name="settings">XML serialization settings. <see cref="System.Xml.XmlReaderSettings"/></param>
    /// <returns>An object of type <c>T</c></returns>
    public static T Deserialize(string xml, Encoding encoding, XmlReaderSettings settings)
    {
        if (string.IsNullOrEmpty(xml))
            throw new ArgumentException("XML cannot be null or empty", "xml");

        XmlSerializer xmlSerializer = new XmlSerializer(typeof(T));

        using (MemoryStream memoryStream = new MemoryStream(encoding.GetBytes(xml)))
        {
            using (XmlReader xmlReader = XmlReader.Create(memoryStream, settings))
            {
                return (T) xmlSerializer.Deserialize(xmlReader);
            }
        }
    }

    /// <summary>
    /// Deserializes a XML file.
    /// </summary>
    /// <param name="filename">The filename of the XML file to deserialize</param>
    /// <returns>An object of type <c>T</c></returns>
    public static T DeserializeFromFile(string filename)
    {
        return DeserializeFromFile(filename, new XmlReaderSettings());
    }

    /// <summary>
    /// Deserializes a XML file.
    /// </summary>
    /// <param name="filename">The filename of the XML file to deserialize</param>
    /// <param name="settings">XML serialization settings. <see cref="System.Xml.XmlReaderSettings"/></param>
    /// <returns>An object of type <c>T</c></returns>
    public static T DeserializeFromFile(string filename, XmlReaderSettings settings)
    {
        if (string.IsNullOrEmpty(filename))
            throw new ArgumentException("filename", "XML filename cannot be null or empty");

        if (! File.Exists(filename))
            throw new FileNotFoundException("Cannot find XML file to deserialize", filename);

        // Create the stream writer with the specified encoding
        using (XmlReader reader = XmlReader.Create(filename, settings))
        {
            System.Xml.Serialization.XmlSerializer xmlSerializer = new System.Xml.Serialization.XmlSerializer(typeof(T));
            return (T) xmlSerializer.Deserialize(reader);
        }
    }

    /// <summary>
    /// Serialize an object
    /// </summary>
    /// <param name="source">The object to serialize</param>
    /// <returns>A XML string that represents the object to be serialized</returns>
    public static string Serialize(T source)
    {
        // indented XML by default
        return Serialize(source, null, GetIndentedSettings());
    }

    /// <summary>
    /// Serialize an object
    /// </summary>
    /// <param name="source">The object to serialize</param>
    /// <param name="namespaces">Namespaces to include in serialization</param>
    /// <returns>A XML string that represents the object to be serialized</returns>
    public static string Serialize(T source, XmlSerializerNamespaces namespaces)
    {
        // indented XML by default
        return Serialize(source, namespaces, GetIndentedSettings());
    }

    /// <summary>
    /// Serialize an object
    /// </summary>
    /// <param name="source">The object to serialize</param>
    /// <param name="settings">XML serialization settings. <see cref="System.Xml.XmlWriterSettings"/></param>
    /// <returns>A XML string that represents the object to be serialized</returns>
    public static string Serialize(T source, XmlWriterSettings settings)
    {
        return Serialize(source, null, settings);
    }

    /// <summary>
    /// Serialize an object
    /// </summary>
    /// <param name="source">The object to serialize</param>
    /// <param name="namespaces">Namespaces to include in serialization</param>
    /// <param name="settings">XML serialization settings. <see cref="System.Xml.XmlWriterSettings"/></param>
    /// <returns>A XML string that represents the object to be serialized</returns>
    public static string Serialize(T source, XmlSerializerNamespaces namespaces, XmlWriterSettings settings)
    {
        if (source == null)
            throw new ArgumentNullException("source", "Object to serialize cannot be null");

        string xml = null;
        XmlSerializer serializer = new XmlSerializer(source.GetType());

        using (MemoryStream memoryStream = new MemoryStream())
        {
            using (XmlWriter xmlWriter = XmlWriter.Create(memoryStream, settings))
            {
                System.Xml.Serialization.XmlSerializer x = new System.Xml.Serialization.XmlSerializer(typeof(T));
                x.Serialize(xmlWriter, source, namespaces);

                memoryStream.Position = 0; // rewind the stream before reading back.
                using (StreamReader sr = new StreamReader(memoryStream))
                {
                    xml = sr.ReadToEnd();
                } 
            }
        }

        return xml;
    }

    /// <summary>
    /// Serialize an object to a XML file
    /// </summary>
    /// <param name="source">The object to serialize</param>
    /// <param name="filename">The file to generate</param>
    public static void SerializeToFile(T source, string filename)
    {
        // indented XML by default
        SerializeToFile(source, filename, null, GetIndentedSettings());
    }

    /// <summary>
    /// Serialize an object to a XML file
    /// </summary>
    /// <param name="source">The object to serialize</param>
    /// <param name="filename">The file to generate</param>
    /// <param name="namespaces">Namespaces to include in serialization</param>
    public static void SerializeToFile(T source, string filename, XmlSerializerNamespaces namespaces)
    {
        // indented XML by default
        SerializeToFile(source, filename, namespaces, GetIndentedSettings());
    }

    /// <summary>
    /// Serialize an object to a XML file
    /// </summary>
    /// <param name="source">The object to serialize</param>
    /// <param name="filename">The file to generate</param>
    /// <param name="settings">XML serialization settings. <see cref="System.Xml.XmlWriterSettings"/></param>
    public static void SerializeToFile(T source, string filename, XmlWriterSettings settings)
    {
         SerializeToFile(source, filename, null, settings);
    }

    /// <summary>
    /// Serialize an object to a XML file
    /// </summary>
    /// <param name="source">The object to serialize</param>
    /// <param name="filename">The file to generate</param>
    /// <param name="namespaces">Namespaces to include in serialization</param>
    /// <param name="settings">XML serialization settings. <see cref="System.Xml.XmlWriterSettings"/></param>
    public static void SerializeToFile(T source, string filename, XmlSerializerNamespaces namespaces, XmlWriterSettings settings)
    {
        if (source == null)
            throw new ArgumentNullException("source", "Object to serialize cannot be null");

        XmlSerializer serializer = new XmlSerializer(source.GetType());

        using (XmlWriter xmlWriter = XmlWriter.Create(filename, settings))
        {
            System.Xml.Serialization.XmlSerializer x = new System.Xml.Serialization.XmlSerializer(typeof(T));
            x.Serialize(xmlWriter, source, namespaces);
        }
    }

    #region Private methods


    private static XmlWriterSettings GetIndentedSettings()
    {
        XmlWriterSettings xmlWriterSettings = new XmlWriterSettings();
        xmlWriterSettings.Indent = true;
        xmlWriterSettings.IndentChars = "\t";

        return xmlWriterSettings;
    }

    #endregion
}
于 2012-12-30T23:33:37.053 に答える
3

私は自分の解決策を見つけました。最高ではありませんが、機能しています。

私はIXmlSerializableを実装し、自分でそれを処理しました。

public void ReadXml(System.Xml.XmlReader reader)
{
    reader.Read();
    reader.MoveToContent();

    if (reader.LocalName == "AnotherNode")
    {
        var innerXml = Serializer<AnotherClass>.CreateSerializer();
        Remove = (AnotherClass) innerXml.Deserialize(reader);
        reader.MoveToContent();
    }

    reader.Read();

    // Here is the trick
    if (reader.IsStartElement())
    {
        do
        {
            var innerXml = Serializer<T>.CreateSerializer();

            var obj = (T) innerXml.Deserialize(reader);
            Updates.Add(obj);
        } while (reader.MoveToContent() == XmlNodeType.Element);
    }
}

public void WriteXml(System.Xml.XmlWriter writer)
{
    var removeSerializer = Serializer<RemoveElement>.CreateSerializer();
    removeSerializer.Serialize(writer, Remove);

    if (Updates.Any())
    {
        var innerXml = Serializer<T>.CreateSerializer();
        writer.WriteStartElement("ContentUpdates");
        foreach (var update in Updates)
        {
            innerXml.Serialize(writer, update);
        }
        writer.WriteEndElement();
    }
}
于 2012-12-27T08:44:27.823 に答える
0

XmlSerializer の代わりに DataContractSerializer の使用を検討してください。

XmlSerializer は、後でシリアル化 - 逆シリアル化プロセスを高速化するために、実行時に一時アセンブリを生成しています。したがって、コードのコンパイルが必要です。

一方、DataContractSerializer はそうではありません。DataContractSerializer アプローチを使用する場合は、「継承」の問題を制御できるように、適切な属性をいじる必要があります。

于 2012-12-23T11:25:39.677 に答える
0

これは、特に Windows ストア プラットフォームのコード エミッターのバグのように聞こえXmlSerializerますが、通常の .NET でも失敗するため、少し意味がありませんが、別のメッセージが表示されます。

一時クラスを生成できません (結果 = 1)。

エラー CS0030: タイプ 'ニュース' を 'イベント' に変換できません

エラー CS1502: 'System.Collections.Generic.List.Add(News)' に最適なオーバーロードされたメソッドの一致には、無効な引数がいくつかあります

エラー CS1503: 引数 1: 'イベント' から 'ニュース' に変換できません

したがって、基本的には、2 つのシナリオで異なる形で現れる、サポートされていないシナリオのように見えます。基本的に、問題は、注釈が (for T=のNews場合) 「これNewsが の場合News、要素 'NewsObject' を呼び出します。これが の場合、要素 'EventObject' を呼び出します」Newsと言っているということです。などEventNewsEvent

ただし、良いニュースは、あなたがやりたいことは を使用するだけで実行できるということです[XmlType(...)]:

[XmlType("NewsObject")]
public class News : BaseContent
{
    // Some more auto properties
    public int B { get; set; }
}
[XmlType("EventObject")]
public class Event : BaseContent
{
    // Some more auto properites
    public int C { get; set; }
}
...
public class GenericResponse<T>
{
    [XmlArray("Content")]
    public List<T> ContentItems { get; set; }
}

注:[XmlArrayItem(...)]上記ではありません。

次に出力します(フォーマット、クリーニングなどの後):

<NewsResponse>
    <Content>
        <NewsObject><A>1</A><B>2</B></NewsObject>
    </Content>
</NewsResponse>

テストコード経由:

var obj = new NewsResponse { ContentItems = new List<News> {
    new News { A = 1, B = 2 } } };

var sw = new StringWriter();
using (var xw = System.Xml.XmlWriter.Create(sw))
{
    var ser = new XmlSerializer(obj.GetType());
    ser.Serialize(xw, obj);
}
string xml = sw.ToString();

それ以上の制御が必要な場合は、確認する必要がありますXmlAttributeOverrides。ただし、アセンブリの出血やメモリ リークを避けるために、 で作成されたシリアライザーをキャッシュする必要があります。XmlAttributeOverrides

于 2013-01-02T08:11:33.360 に答える