c# XML反序列化,及轉換錯誤的解決方法(空值,數組等)

目錄

另一篇文章: 可以修改前綴的反序列化方法

轉換方法

問題及解決

問題:數值類型爲空時, 轉換出錯

問題:在轉換帶有列表的節點時識別錯誤(list,數組)

問題:在XML序列化時怎樣判斷 必填屬性是否被填寫

問題:當類賦值後,轉換爲xml的字符串中 節點多了xmlns屬性

xml序列化時類的自定義屬性

空值標識:IsNullable

指定序列化名稱:ElementName

取消字段的正反序列化:[XmlIgnore]


轉換方法

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Xml.Serialization;

namespace WindowsFormsApplication1
{
    class XmlSerializeHelper
    {
        #region 使用演示
        //aaa  類名
        //Example 實例名
        //strxml xml字符串
        
        string strxml;
        //反序列化演示
        //aaa Example = XmlSerializeHelper.DESerializer<aaa>(strxml); 
        //序列化演示
        //string strxml = XmlSerializeHelper.XmlSerialize<aaa>(Example);
        #endregion

        /// <summary>
        /// 實體類轉換成XML
        /// </summary>
        /// <typeparam name="T">類名</typeparam>
        /// <param name="obj">T類名的實例</param>
        /// <returns></returns>
        public static string XmlSerialize<T>(T obj)
        {
            using (StringWriter sw = new StringWriter())
            {
                Type t = obj.GetType();
                XmlSerializer serializer = new XmlSerializer(obj.GetType());
                serializer.Serialize(sw, obj);
                sw.Close();
                return sw.ToString();
            }
        }
        



        /// <summary>
        /// XML轉換成實體類-方法1
        /// </summary>
        /// <typeparam name="T">對應的類</typeparam>
        /// <param name="strXML">XML字符串</param>
        /// <returns></returns>
        public static T DESerializer<T>(string strXML) where T : class
        {
            try
            {
                using (StringReader sr = new StringReader(strXML))
                {
                    XmlSerializer serializer = new XmlSerializer(typeof(T));
                    return serializer.Deserialize(sr) as T;
                }
            }
            catch (Exception ex)
            {
                return null;
            }
        }

        /// <summary>
        /// XML轉換成實體類-方法2
        /// </summary>
        /// <param name="xmlStr"></param>
        /// <param name="type"></param>
        /// <returns></returns>
        public static object DeserializeFromXml(string xmlStr, Type type)
        {
            try
            {
                using (StringReader sr = new StringReader(xmlStr))
                {
                    XmlSerializer xs = new XmlSerializer(type);
                    return xs.Deserialize(sr);
                }
            }
            catch (Exception ex)
            {
                throw (ex);
            }
        }
        //Request
        //


    }
}

小技巧:在建立xml對應類時可自動建立,方法如下:

  1. 將完整的XML文本複製下來
  2. 新建一個類,將光標放在準備粘貼的位置
  3. 點擊VS左上角的 [編輯] - [選擇性粘貼] - [將XML粘貼爲類]

注:如果沒有 [將xml粘貼爲類] 的選項,右鍵  將 項目-屬性-應用程序 中的目標框架改爲.NET Framework 4.5 即可。

問題及解決

  • 問題:數值類型爲空時, 轉換出錯

        在轉換後的實體類中,有些屬性因爲業務需求,需要修改類型爲Int32,Uint32等數字類型,但是給出的xml文本中,此項可能會沒有值,導致在反序列化時會出錯(即使在get方法中容錯也不行)

  • 解決:因爲在反序列化的時候接受的屬性是數字類型且xml文本中爲空,就會出錯,無論是否在get方法中容錯;所以用屬性類型爲string的屬性進行反序列化的對接,具體容錯和賦值可在該string類型內的get方法中進行(可賦值給其他數值類型屬性,string類型屬性僅當做反序列化時的對接口),具體操作如下

舉例說明:反序列化的類中有屬性int num 和 string str 兩個屬性

//對應的實體類
public partial class aaa
    {
        public string str { get; set; }

        [XmlIgnore]//取消反序列化
        public int num;

        //用string類型的屬性來反序列化,將值轉賦給原數值屬性num
        [XmlElement(ElementName = "num")]//反序列化對應的屬性名
        public string numStr
        {
            get
            {
                return this.num.ToString();
            }
            set
            {
                Int32 num1;
                Int32.TryParse(value.ToString(), out num1);
                this.num = num1;
            }
        }
        
    }

注意:在XML反序列化中對應的類的屬性必須爲public才能被反序列化,private不能被反序列化

  • 問題:在轉換帶有列表的節點時識別錯誤(list,數組)

  • 解決:在列表標籤下加2個及以上的子循環標籤,進行轉換時即可自動轉換爲數組

舉例說明:轉換對照

<eee:qqq xmlns:rrr="http://rrr" xmlns:eee="http://eee">
  <eee:vvv>
    <!--重複1-->
    <eee:mmm />
    <!--重複2-->
    <eee:mmm />
  </eee:vvv>
</eee:qqq>

生成結果: 

/// <remarks/>
    [System.Xml.Serialization.XmlTypeAttribute(AnonymousType = true, Namespace = "http://eee")]
    [System.Xml.Serialization.XmlRootAttribute(Namespace = "http://eee", IsNullable = false)]
    public partial class qqq
    {
        private object[] vvvField;

        /// <remarks/>
        //public List<object> vvv { get; set; }
        [System.Xml.Serialization.XmlArrayItemAttribute("mmm", IsNullable = false)]
        public object[] vvv
        {
            get
            {
                return this.vvvField;
            }
            set
            {
                this.vvvField = value;
            }
        }
        
    }

  • 問題:在XML序列化時怎樣判斷 必填屬性是否被填寫

  • 解決:利用自定義屬性和get方法,
    • 1.將需要必填的類中屬性添加 IsNullable = true 的自定義屬性,當序列化爲xml後,爲null的屬性的節點標籤有 xsi:nil="true" 屬性顯示
    • 2.將需要必填的類中屬性的get方法中添加約束,如果value爲空時添加標識
    • 3.在序列化後的XML中查找指定的標識,即可判斷必填屬性是否都已填寫

所需方法:

/// <summary>
/// 檢查是否存在必填項未填錯誤
/// </summary>
/// <param name="XmlStr">帶有標識,從類中轉換出的xml數據</param>
/// <param name="ErrorKeyword">指定錯誤標識</param>
/// <returns></returns>
private static bool IsRequiredItemError(string XmlStr, string ErrorKeyword)
{
	XDocument xmlDocument = XDocument.Parse(XmlStr);
	//查詢指定空識別
	if (XmlStr.Contains(ErrorKeyword))
	{
		return true;
	}
	//遍歷節點屬性是否有空
	List<XElement> elementList = new List<XElement>();
	GetElements(xmlDocument.Root.Elements(), "Objects", elementList, "Left");
	List<XAttribute> XAList = GetXattributes(elementList);

	foreach (XAttribute item in XAList)
	{
		string XAname = item.Name.LocalName;
		string XAvalue = item.Value;
		if (XAname == "nil" && XAvalue=="true")
		{
			return true;
		}
	}
	return false;
}

/// <summary>
/// 獲取所有節點的所有屬性
/// </summary>
/// <param name="elementList"></param>
/// <returns></returns>
public static List<XAttribute> GetXattributes(List<XElement> elementList)
{
	List<XAttribute> LXA = new List<XAttribute>();
	foreach (XElement item in elementList)
	{
		IEnumerable<XAttribute> dd = item.Attributes();
		var ie = dd.GetEnumerator();                      // 通過GetEnumerator()方法返回IEnumerator迭代器
		while (ie.MoveNext())
		{
			XAttribute XA = ie.Current;
			LXA.Add(XA);
		}
	}
	return LXA;
}



/// <summary>
/// 獲取嵌套遍歷XML中所有節點
/// </summary>
/// <param name="xElements"></param>
/// <param name="parentNodeName"></param>
/// <param name="elementList">接收</param>
/// <param name="dictionaryValue"></param>
public static void GetElements(IEnumerable<XElement> xElements, string parentNodeName, List<XElement> elementList, string dictionaryValue)
{
	foreach (XElement item in xElements)
	{
		if (item.HasElements)
		{
			GetElements(item.Elements(), parentNodeName + item.Name.ToString(), elementList, dictionaryValue);
		}
		else
		{
			elementList.Add(item);
		}
	}
}

使用:

//讀取XML
string inputtext = System.IO.File.ReadAllText("./XMLFile1.xml");
//將xml轉換爲類再轉換爲xml(可將必填項設置標識)
string sttt = XmlSerializeHelper.XmlSerialize(XmlSerializeHelper.DeserializeFromXml(inputtext, typeof(qqq)), Encoding.UTF8, new Dictionary<string, string>());
//必填項是否通還有空值
bool BL = IsRequiredItemError(sttt, "標識");
//需要注意的是,數組爲空時不會被標註出來,如果大家有什麼好的方法的話 歡迎留言
  • 問題:當類賦值後,轉換爲xml的字符串中 節點多了xmlns屬性

         原因:當給類中Object類型的屬性賦值時, 節點就會自動添加xmlns屬性

         解決:將Object類型改爲string類型即可

xml序列化時類的自定義屬性

空值標識:IsNullable

[XmlElement(IsNullable = true)]   當示例轉爲XML文本時,帶有此自定義屬性的屬性如果爲null則會生成  xsi:nil = true 的屬性

使用方式:[XmlElement(IsNullable = true)]
        public string Str

生成結果:<str xsi:nil="true" />

指定序列化名稱:ElementName

[XmlElement(ElementName = "str")]
public string aaastr

序列化結果: <str/>

 : 不同參數間可以用"," 間隔 如: [XmlElement(ElementName = "str", IsNullable = true)]

取消字段的正反序列化:[XmlIgnore]

使用方法: 在需要忽略序列化的字段上加上[XmlIgnore]即可
        [XmlIgnore]
        public string Str

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章