使用XmlAttributeOverrides预编译XmlSerializers

时间:2009-03-24 18:43:50

标签: .net xml-serialization sgen

在.NET中构造XmlSerializer实例时,会动态生成用于序列化和反序列化指定类型的程序集。这是一个耗时的过程。 Microsoft的sgen.exe工具可用于预编译XmlSerializer实例,以便以后使用它们而无需动态生成它们。不幸的是,对于使用XmlAttributeOverrides的XmlSerializer实例,这是不可能的。

有没有办法预编译这些XmlSerializer实例以避免在运行时生成?

1 个答案:

答案 0 :(得分:6)

Andreas,这不是sgen工具本身的问题,这是由于XmlSerializer的实现。

当您创建XmlSerializer的实例时,使用仅具有一个Type参数的构造函数,它会检查缓存并查找预生成的程序集。 但是当你使用带有XmlAttributeOverrides的构造函数时,XmlSerializer不会检查任何缓存并立即生成临时程序集。

最有可能的原因是,由于序列化逻辑中的相当激进的变化,您可以使用XmlAttributeOverrides参数来实现,这在sgen等工具的编译时无法“预见”。

如果你需要预编译的东西,你[叹气]必须避免使用XmlAttributeOverrides。如果无法做到这一点,请尝试提前创建所需的XmlSerializer实例,可能在后台线程中。

为了您的兴趣,这里是默认构造函数的代码(检查缓存并尝试查找预先生成的程序集):

public XmlSerializer(Type type, string defaultNamespace)
{
    this.events = new XmlDeserializationEvents();
    if (type == null)
    {
        throw new ArgumentNullException("type");
    }
    this.mapping = GetKnownMapping(type, defaultNamespace);
    if (this.mapping != null)
    {
        this.primitiveType = type;
    }
    else
    {
        this.tempAssembly = cache[defaultNamespace, type];
        if (this.tempAssembly == null)
        {
            lock (cache)
            {
                this.tempAssembly = cache[defaultNamespace, type];
                if (this.tempAssembly == null)
                {
                    XmlSerializerImplementation implementation;
                    Assembly assembly = TempAssembly.LoadGeneratedAssembly(type, defaultNamespace, out implementation);
                    if (assembly == null)
                    {
                        this.mapping = new XmlReflectionImporter(defaultNamespace).ImportTypeMapping(type, null, defaultNamespace);
                        this.tempAssembly = GenerateTempAssembly(this.mapping, type, defaultNamespace);
                    }
                    else
                    {
                        this.mapping = XmlReflectionImporter.GetTopLevelMapping(type, defaultNamespace);
                        this.tempAssembly = new TempAssembly(new XmlMapping[] { this.mapping }, assembly, implementation);
                    }
                }
                cache.Add(defaultNamespace, type, this.tempAssembly);
            }
        }
        if (this.mapping == null)
        {
            this.mapping = XmlReflectionImporter.GetTopLevelMapping(type, defaultNamespace);
        }
    }
}

这是与XmlAttributeOverrides一起使用的构造函数(总是生成序列化程序集):

public XmlSerializer(Type type, XmlAttributeOverrides overrides, Type[] extraTypes, XmlRootAttribute root, string defaultNamespace, string location, Evidence evidence)
{
    this.events = new XmlDeserializationEvents();
    if (type == null)
    {
        throw new ArgumentNullException("type");
    }
    XmlReflectionImporter importer = new XmlReflectionImporter(overrides, defaultNamespace);
    for (int i = 0; i < extraTypes.Length; i++)
    {
        importer.IncludeType(extraTypes[i]);
    }
    this.mapping = importer.ImportTypeMapping(type, root, defaultNamespace);
    if (location != null)
    {
        this.DemandForUserLocation();
    }
    this.tempAssembly = GenerateTempAssembly(this.mapping, type, defaultNamespace, location, evidence);
}