XmlSerializer带来的性能问题及解决办法

时间:2022-06-22 03:03:41
本文中的思路、二元hashtable等,大力感谢Leo Chen的帮助!
对于XmlSerializer带来的内存占用过高,最终导致Out Of Memory的问题,参见以前这个链接: http://www.cnblogs.com/juqiang/archive/2008/01/15/1039936.html
(但是那篇文章中对于XmlSerializer构造方法的说明,是错误的。那段代码没有问题,有问题的是下面的)

首先看System.Xml.Serialization.XmlSerializer的构造方法,一共分为三大类:

public XmlSerializer(Type type) : this(type, (string) null)
public XmlSerializer(Type type, string defaultNamespace)
这两个方法,采用了上文引用的那篇文章的处理方式,应用了cache,这是正确的,不会造成内存占用过高。(上文引用那篇文章里面这个地方解释错了)

另一大类的方法是:
public XmlSerializer(XmlTypeMapping xmlTypeMapping),这个方法里面会产生一个tempAssembly,但是没有用缓存方式来处理。

有问题的一类方法是:
public XmlSerializer(Type type, Type[] extraTypes) : this(type, null, extraTypes, null, null, null, null)
public XmlSerializer(Type type, XmlAttributeOverrides overrides) : this(type, overrides, new Type[0], null, null, null, null)
public XmlSerializer(Type type, XmlRootAttribute root) : this(type, null, new Type[0], root, null, null, null)
public XmlSerializer(Type type, XmlAttributeOverrides overrides, Type[] extraTypes, XmlRootAttribute root, string defaultNamespace) : this(type, overrides, extraTypes, root, defaultNamespace, null, null)
public XmlSerializer(Type type, XmlAttributeOverrides overrides, Type[] extraTypes, XmlRootAttribute root, string defaultNamespace, string location, Evidence evidence)

前4个都最终调用了最后一个构造方法,里面产生了一个tempAssembly,也没有用缓存方式来护理。

我上次提到的那个问题,问题代码如下:
public static TReturn Convert<TReturn, TInput>(TInput input) where TReturn: class, new() where TInput: IProvisioningObject
{
    using (MemoryStream stream = new MemoryStream())
    {
        new XmlSerializer(typeof(TInput)).Serialize((Stream) stream, input);
        stream.Position = 0L;
        XmlSerializer serializer = new XmlSerializer(typeof(TReturn), new XmlRootAttribute(input.GetType().Name));
        return (TReturn) serializer.Deserialize(stream);
    }
}
注意红色的代码,这里产生了一个tempAssembly,没有做缓存。这里的核心问题在于,tempAssembly不会被自动释放掉,除非appdomain被unload。
修正的方式类似于XmlSerializer的处理方式,采用一个二元Hashtable来做。代码修改如下:
 1 XmlSerializer带来的性能问题及解决办法  private   static  TempXmlSerializerCache cache  =   new  TempXmlSerializerCache();
 2 XmlSerializer带来的性能问题及解决办法
 3 XmlSerializer带来的性能问题及解决办法         public   static  TReturn Convert < TReturn, TInput > (TInput input)
 4 XmlSerializer带来的性能问题及解决办法             where  TReturn :  class new ()
 5 XmlSerializer带来的性能问题及解决办法             where  TInput : IProvisioningObject
 6 XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法         {
 7XmlSerializer带来的性能问题及解决办法            using (MemoryStream stream = new MemoryStream())
 8XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法            {
 9XmlSerializer带来的性能问题及解决办法                new XmlSerializer(typeof(TInput)).Serialize((Stream)stream, input);
10XmlSerializer带来的性能问题及解决办法                stream.Position = 0L;
11XmlSerializer带来的性能问题及解决办法
12XmlSerializer带来的性能问题及解决办法                XmlSerializer serializer = cache[typeof(TReturn).ToString(), input.GetType().Name];
13XmlSerializer带来的性能问题及解决办法                if (serializer == null)
14XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法                {
15XmlSerializer带来的性能问题及解决办法                    lock (cache)
16XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法                    {
17XmlSerializer带来的性能问题及解决办法                        serializer = cache[typeof(TReturn).ToString(), input.GetType().Name];
18XmlSerializer带来的性能问题及解决办法                        if (serializer == null)
19XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法                        {
20XmlSerializer带来的性能问题及解决办法                            serializer = new XmlSerializer(typeof(TReturn), new XmlRootAttribute(input.GetType().Name));
21XmlSerializer带来的性能问题及解决办法                            cache.Add(typeof(TReturn).ToString(), input.GetType().Name, serializer);
22XmlSerializer带来的性能问题及解决办法                        }

23XmlSerializer带来的性能问题及解决办法                    }

24XmlSerializer带来的性能问题及解决办法                }

25XmlSerializer带来的性能问题及解决办法                return (TReturn)serializer.Deserialize(stream);
26XmlSerializer带来的性能问题及解决办法            }

27XmlSerializer带来的性能问题及解决办法        }

这里的二元Hashtable,是从XmlSerializer里面扒出来的,稍微修改了一下而已:

XmlSerializer带来的性能问题及解决办法    public class TempXmlSerializerCache
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法    
{
XmlSerializer带来的性能问题及解决办法        
private Hashtable cache = new Hashtable();
XmlSerializer带来的性能问题及解决办法
XmlSerializer带来的性能问题及解决办法        
public void Add(string ns, object o, XmlSerializer serializer)
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法        
{
XmlSerializer带来的性能问题及解决办法            XmlSerializerCacheKey key 
= new XmlSerializerCacheKey(ns, o);
XmlSerializer带来的性能问题及解决办法            
lock (this)
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法            
{
XmlSerializer带来的性能问题及解决办法                
if (this.cache[key] != serializer)
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法                
{
XmlSerializer带来的性能问题及解决办法                    Hashtable hashtable 
= new Hashtable();
XmlSerializer带来的性能问题及解决办法                    
foreach (object obj2 in this.cache.Keys)
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法                    
{
XmlSerializer带来的性能问题及解决办法                        hashtable.Add(obj2, 
this.cache[obj2]);
XmlSerializer带来的性能问题及解决办法                    }

XmlSerializer带来的性能问题及解决办法                    
this.cache = hashtable;
XmlSerializer带来的性能问题及解决办法                    
this.cache[key] = serializer;
XmlSerializer带来的性能问题及解决办法                }

XmlSerializer带来的性能问题及解决办法            }

XmlSerializer带来的性能问题及解决办法        }

XmlSerializer带来的性能问题及解决办法
XmlSerializer带来的性能问题及解决办法        
public XmlSerializer this[string ns, object o]
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法        
{
XmlSerializer带来的性能问题及解决办法            
get
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法            
{
XmlSerializer带来的性能问题及解决办法                
return (XmlSerializer)this.cache[new XmlSerializerCacheKey(ns, o)];
XmlSerializer带来的性能问题及解决办法            }

XmlSerializer带来的性能问题及解决办法        }

XmlSerializer带来的性能问题及解决办法    }

XmlSerializer带来的性能问题及解决办法

以及:
XmlSerializer带来的性能问题及解决办法    public   class  XmlSerializerCacheKey
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法    
{
XmlSerializer带来的性能问题及解决办法        
private string ns;
XmlSerializer带来的性能问题及解决办法        
private object type;
XmlSerializer带来的性能问题及解决办法
XmlSerializer带来的性能问题及解决办法        
public XmlSerializerCacheKey(string ns, object type)
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法        
{
XmlSerializer带来的性能问题及解决办法            
this.type = type;
XmlSerializer带来的性能问题及解决办法            
this.ns = ns;
XmlSerializer带来的性能问题及解决办法        }

XmlSerializer带来的性能问题及解决办法
XmlSerializer带来的性能问题及解决办法        
public override bool Equals(object o)
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法        
{
XmlSerializer带来的性能问题及解决办法            XmlSerializerCacheKey key 
= o as XmlSerializerCacheKey;
XmlSerializer带来的性能问题及解决办法            
if (key == null)
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法            
{
XmlSerializer带来的性能问题及解决办法                
return false;
XmlSerializer带来的性能问题及解决办法            }

XmlSerializer带来的性能问题及解决办法            
return ((key.type == this.type) && (key.ns == this.ns));
XmlSerializer带来的性能问题及解决办法        }

XmlSerializer带来的性能问题及解决办法
XmlSerializer带来的性能问题及解决办法        
public override int GetHashCode()
XmlSerializer带来的性能问题及解决办法XmlSerializer带来的性能问题及解决办法        
{
XmlSerializer带来的性能问题及解决办法            
return (((this.ns != null? this.ns.GetHashCode() : 0^ ((this.type != null? this.type.GetHashCode() : 0));
XmlSerializer带来的性能问题及解决办法        }

XmlSerializer带来的性能问题及解决办法    }


经过修改、编译、发布,再次测试,问题得到了解决,哈哈!!!

本文中的思路、二元hashtable等,大力感谢Leo Chen的帮助!