开发过程中经常需要返回某实体类的列表,公司通常用的都是XML格式的接口,小猪借鉴了公司前辈留下的代码一直是类似这么写的:
public static string GetXMLList(IList<Article> articlelist) { using (MemoryStream memoryStream = new MemoryStream()) { XmlWriterSettings xmlWriterSettings = new XmlWriterSettings(); xmlWriterSettings.Indent = true; xmlWriterSettings.Encoding = new UTF8Encoding(false); xmlWriterSettings.NewLineChars = Environment.NewLine; using (XmlWriter xmlWriter = XmlWriter.Create(memoryStream, xmlWriterSettings)) { xmlWriter.WriteStartDocument(true); xmlWriter.WriteStartElement("Articles"); if (articlelist != null) { foreach (var article in articlelist) { xmlWriter.WriteStartElement("Article"); xmlWriter.WriteStartAttribute("id"); xmlWriter.WriteString(article.Id.ToString()); xmlWriter.WriteEndAttribute(); xmlWriter.WriteStartElement("Title"); xmlWriter.WriteCData(article.Title); xmlWriter.WriteEndElement(); xmlWriter.WriteStartElement("Summary"); xmlWriter.WriteCData(article.Summary); xmlWriter.WriteEndElement(); xmlWriter.WriteStartElement("Author"); xmlWriter.WriteCData(article.Author); xmlWriter.WriteEndElement(); xmlWriter.WriteStartElement("CreateDate"); xmlWriter.WriteCData(article.CreateDate.ToString("yyyy-MM-dd")); xmlWriter.WriteEndElement(); xmlWriter.WriteStartElement("BannerURL"); xmlWriter.WriteCData(article.BannerURL); xmlWriter.WriteEndElement(); xmlWriter.WriteStartElement("ImageURL"); xmlWriter.WriteCData(article.ImageURL); xmlWriter.WriteEndElement(); xmlWriter.WriteStartElement("ImageAuthor"); xmlWriter.WriteCData(article.ImageAuthor); xmlWriter.WriteEndElement(); xmlWriter.WriteStartElement("Category"); xmlWriter.WriteCData(article.Category.ToString()); xmlWriter.WriteEndElement(); xmlWriter.WriteEndElement(); } } xmlWriter.WriteEndElement(); xmlWriter.WriteEndDocument(); } string xml = Encoding.UTF8.GetString(memoryStream.ToArray()); return xml; } }
生成的代码:
<?xml version="1.0" encoding="utf-8" standalone="yes"?> <Articles> <Article id="83"> <Title><![CDATA[存储]]></Title> <Summary><![CDATA[asdsa]]></Summary> <Author><![CDATA[asdsa]]></Author> <CreateDate><![CDATA[2013-12-30]]></CreateDate> <BannerURL><![CDATA[/Images/2013年12月/20131230153738573.png]]></BannerURL> <ImageURL><![CDATA[/Images/2013年12月/20131230153731941.png]]></ImageURL> <ImageAuthor><![CDATA[sda]]></ImageAuthor> <Category><![CDATA[1]]></Category> </Article> <Article id="81"> <Title><![CDATA[存储]]></Title> <Summary><![CDATA[asdsa]]></Summary> <Author><![CDATA[asdsa]]></Author> <CreateDate><![CDATA[2013-12-30]]></CreateDate> <BannerURL><![CDATA[/Images/2013年12月/20131230153738573.png]]></BannerURL> <ImageURL><![CDATA[/Images/2013年12月/20131230153731941.png]]></ImageURL> <ImageAuthor><![CDATA[sda]]></ImageAuthor> <Category><![CDATA[1]]></Category> </Article> </Articles>
代码一直延续到昨天!小猪决定重构他!
首先上述代码最大的问题就是大量的复制粘贴,违背了DRY(Don't Repeated Yourself)原则。多个字段就要多粘贴一次,为了解决这个问题在遍历实体列表时使用下属代码:
using (MemoryStream memoryStream = new MemoryStream()) { XmlWriterSettings xmlWriterSettings = new XmlWriterSettings(); xmlWriterSettings.Indent = true; xmlWriterSettings.Encoding = new UTF8Encoding(false); xmlWriterSettings.NewLineChars = Environment.NewLine; using (XmlWriter xmlWriter = XmlWriter.Create(memoryStream, xmlWriterSettings)) { xmlWriter.WriteStartDocument(true); xmlWriter.WriteStartElement("Articles"); if (articlelist != null) { foreach (var article in articlelist) { Type type = typeof(Article); foreach (PropertyInfo propertyInfo in type.GetProperties()) { object ob = propertyInfo.GetValue(article, null); if (null != ob) { xmlWriter.WriteStartElement(propertyInfo.Name); xmlWriter.WriteCData(ob.ToString()); xmlWriter.WriteEndElement(); } } xmlWriter.WriteEndElement(); } } xmlWriter.WriteEndElement(); xmlWriter.WriteEndDocument(); } string xml = Encoding.UTF8.GetString(memoryStream.ToArray()); return xml; }
可是这样只能遍历Article类型,其他类型还是使用不了这个方法
在最原始的代码中每为一个实体增加类似功能的时候都要把那一整块代码复制过来然后做修改,我们在重构一中还是没有解决这个问题,为了使上面的方法能够在以后被重复利用我们加入泛型
public static string GetXMLList<T>(IList<T> articlelist) { using (MemoryStream memoryStream = new MemoryStream()) { XmlWriterSettings xmlWriterSettings = new XmlWriterSettings(); xmlWriterSettings.Indent = true; xmlWriterSettings.Encoding = new UTF8Encoding(false); xmlWriterSettings.NewLineChars = Environment.NewLine; using (XmlWriter xmlWriter = XmlWriter.Create(memoryStream, xmlWriterSettings)) { xmlWriter.WriteStartDocument(true); xmlWriter.WriteStartElement("Roots"); Type type = typeof(T); if (articlelist != null) { foreach (var article in articlelist) { foreach (PropertyInfo propertyInfo in type.GetProperties()) { object ob = propertyInfo.GetValue(article, null); if (null != ob) { xmlWriter.WriteStartElement(propertyInfo.Name); xmlWriter.WriteCData(ob.ToString()); xmlWriter.WriteEndElement(); } } xmlWriter.WriteEndElement(); } } xmlWriter.WriteEndElement(); xmlWriter.WriteEndDocument(); } string xml = Encoding.UTF8.GetString(memoryStream.ToArray()); return xml; } }
这样代码完成了多类型的使用,但是却把所有的共有属性都写进了XML,在实际使用中我们可能不希望把所有属性都列进来,例如是否推荐字段,阅读权限字段等等~
重构三:
...
为了使每个节点上面增加Id属性,定义一个抽象类Listable
。抽象类中包涵自动Id,让需要提供列表的实体类继承至这个抽象类:
/*========================================================== *作者:SmallerPig *时间:2013/12/30 17:26:01 *版权所有:无锡睿阅数字科技有限公司 ============================================================*/ namespace RY.Entity { public abstract class Listable { public int Id { get; set; } } }
实体类来继承它。例如:
public class Article : Listable { public string Title { get; set; } public string Summary { get; set; } }
然后给泛型方法加上约束,在该指定Id的地方加上id属性!
static string ToXML<T>(IList<T> TList, string ingor) where T : Listable { using (MemoryStream memoryStream = new MemoryStream()) { XmlWriterSettings xmlWriterSettings = new XmlWriterSettings(); xmlWriterSettings.Indent = true; xmlWriterSettings.Encoding = new UTF8Encoding(false); xmlWriterSettings.NewLineChars = Environment.NewLine; using (XmlWriter xmlWriter = XmlWriter.Create(memoryStream, xmlWriterSettings)) { xmlWriter.WriteStartDocument(true); xmlWriter.WriteStartElement("Roots"); if (TList != null) { Type type = typeof(T); foreach (T t in TList) { xmlWriter.WriteStartElement(type.Name); xmlWriter.WriteStartAttribute("id"); xmlWriter.WriteString(t.Id.ToString()); xmlWriter.WriteEndAttribute(); foreach (PropertyInfo propertyInfo in type.GetProperties()) { if (propertyInfo.CanRead && propertyInfo.Name.ToLower() != ingor.ToLower()) { if (propertyInfo.PropertyType == typeof(DateTime)) { xmlWriter.WriteStartElement(propertyInfo.Name); DateTime dt = Convert.ToDateTime(propertyInfo.GetValue(t, null)); xmlWriter.WriteCData(dt.ToString("yyyy-MM-dd hh:mm:ss")); xmlWriter.WriteEndElement(); } if (propertyInfo.PropertyType == typeof(String) || propertyInfo.PropertyType == typeof(int)) { object ob = propertyInfo.GetValue(t, null); if (null != ob) { xmlWriter.WriteStartElement(propertyInfo.Name); xmlWriter.WriteCData(ob.ToString()); xmlWriter.WriteEndElement(); } } } } xmlWriter.WriteEndElement(); } } xmlWriter.WriteEndElement(); xmlWriter.WriteEndDocument(); } string xml = Encoding.UTF8.GetString(memoryStream.ToArray()); return xml; } }
最后效果:
<?xml version="1.0" encoding="utf-8" standalone="yes"?> <Roots> <Article id="83"> <Title><![CDATA[存储]]></Title> <Summary><![CDATA[asdsa]]></Summary> <Author><![CDATA[asdsa]]></Author> <Category><![CDATA[1]]></Category> <CreateDate><![CDATA[2013-12-30 03:37:52]]></CreateDate> <BannerURL><![CDATA[/Images/2013年12月/20131230153738573.png]]></BannerURL> <ImageURL><![CDATA[/Images/2013年12月/20131230153731941.png]]></ImageURL> <ImageAuthor><![CDATA[sda]]></ImageAuthor> <Id><![CDATA[83]]></Id> </Article> <Article id="82"> <Title><![CDATA[存储]]></Title> <Summary><![CDATA[asdsa]]></Summary> <Author><![CDATA[asdsa]]></Author> <Category><![CDATA[1]]></Category> <CreateDate><![CDATA[2013-12-30 03:37:51]]></CreateDate> <BannerURL><![CDATA[/Images/2013年12月/20131230153738573.png]]></BannerURL> <ImageURL><![CDATA[/Images/2013年12月/20131230153731941.png]]></ImageURL> <ImageAuthor><![CDATA[sda]]></ImageAuthor> <Id><![CDATA[82]]></Id> </Article> </Roots>