微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

序列化在C#中实现IEnumerable的自定义泛型类型

我有一个通用类型Foo1,它基本上是一个带有一些元数据和通用列表的容器.当我序列化该类型时,它将获得名称“ Foo1Of” TypeParameterName.

public class Foo1<T>
{
    public string Name { get; set; } = string.Empty;

    public List<T> List { get; set; } = new List<T>();
}

[TestMethod]
public void Serializefoo1()
{
    Foo1<string> foo = new Foo1<string>
    {
        Name = "Foo1",
        List =
        {
            "Bar",
            "Baz"
        }
    };

    XmlSerializer s = new XmlSerializer(foo.GetType());
    StringBuilder sb = new StringBuilder();
    XmlWriter w = XmlWriter.Create(sb);

    s.Serialize(w, foo);

    Console.WriteLine(sb.ToString());
}

这是旧版本,并返回:

<?xml version="1.0" encoding="utf-16"?>
<Foo1OfString xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
    <Name>Foo1</Name>
    <List><string>Bar</string><string>Baz</string></List>
</Foo1OfString>

为了更直接地访问内部列表,我实现了IEnumerable,IEnumerable和Add(…)方法.

public class Foo2<T> : IEnumerable<T>, IEnumerable
{
    public string Name { get; set; } = string.Empty;

    public List<T> List { get; set; } = new List<T>();

    public void Add(T item)
    {
        List.Add(item);
    }

    public IEnumerator<T> GetEnumerator()
    {
        return List.GetEnumerator();
    }

    IEnumerator IEnumerable.GetEnumerator()
    {
        return List.GetEnumerator();
    }
}

[TestMethod]
public void Serializefoo2()
{
    Foo2<string> foo = new Foo2<string>
    {
        Name = "Foo2",
        List =
        {
            "Bar",
            "Baz"
        }
    };

    XmlSerializer s = new XmlSerializer(foo.GetType());
    StringBuilder sb = new StringBuilder();
    XmlWriter w = XmlWriter.Create(sb);

    s.Serialize(w, foo);

    Console.WriteLine(sb.ToString());
}

之后,XML序列化的行为非常奇怪:

<?xml version="1.0" encoding="utf-16"?>
<arrayofstring xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
    <string>Bar</string><string>Baz</string>
</arrayofstring>

名称将变为“ ArrayOf” TypeParameterName,并且属性“ Name”将消失. XmlSerializer似乎认为,我的自定义类型是数组.有没有一种方法可以更改此行为,这意味着能够实现接口和Add(…)但可以获取旧的XML序列化?

解决方法:

当您实现IEnumerable< T>时,这是XmlSerializer的认行为.接口.由于IEnumerable< string>类型,XmlSerializer认为这是字符串的集合/数组.

如果必须实现IEnumerable< T&gt ;,则克服此认行为的一种方法是实现自定义序列化.为您的Foo1< T>实施IXmlSerializable.然后,您可以根据需要自定义Xml图,并返回所需的名称和列表.然后XmlSerializer将您的自定义实现.

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。