Deserialize XML data into object

由于工作的需要,自己写了一个exmaple on 反序列 xml data 成 自定义的一个object:


主体:

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


namespace NextLabs.RemindwithEmail
{    
    class Program
    {
        static void Main(string[] args)
        {
            emailcollection emails = null;
            string path = Directory.GetCurrentDirectory().ToString() + @"\" + "EmailInformation.xml";


            XmlSerializer serializer = new XmlSerializer(typeof(emailcollection));
            StreamReader reader = new StreamReader(path);
            emails = (emailcollection)serializer.Deserialize(reader);
            reader.Close();


            foreach (email myemail in emails)
            {
                Console.WriteLine(myemail.smtp);
                Console.WriteLine(myemail.to);
                Console.WriteLine(myemail.port);
                Console.WriteLine(myemail.subject);
            }
        }
    }
}


辅助:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Xml.Serialization;
using System.Collections;


namespace NextLabs.RemindwithEmail
{
    [Serializable]
    public class email
    {
        [System.Xml.Serialization.XmlElement("smtp")]
        public string smtp { get; set; }


        [System.Xml.Serialization.XmlElement("port")]
        public string port { get; set; }


        [System.Xml.Serialization.XmlElement("to")]
        public string to { get; set; }


        [System.Xml.Serialization.XmlElement("subject")]
        public string subject { get; set; }
    }


    [Serializable]
    [System.Xml.Serialization.XmlRoot("emailcollection")]
    public class emailcollection : IEnumerable
    {
        [System.Xml.Serialization.XmlArray("emailcollection")]
        [System.Xml.Serialization.XmlArrayItem("email",typeof(email))]
        public List<email> emails
        { 
            get 
            {
                return _emails;
            }
            set
            {
                _emails = value;
            }
          }


        private List<email> _emails;


        public void Add(object o) 
        {
            if (_emails == null)
                _emails = new List<email>();


            _emails.Add(o as email);
            
        }


        public emailEnum GetEnumerator()
        {
            return new emailEnum(emails);
        }


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


    public class emailEnum : IEnumerator
    {
        public List<email> _emailcollection;


        int postion = -1;


        public emailEnum(List<email> list)
        {
            _emailcollection = list;
        }


        public void Reset()
        {
            postion = -1;
        }


        public bool MoveNext()
        {
            postion++;
            return (postion < _emailcollection.Count);
        }


        object IEnumerator.Current
        {
            get
            {
                return Current;
            }
        }


        public email Current
        {
            get
            {
                try
                {
                    return _emailcollection[postion];
                }
                catch (IndexOutOfRangeException)
                {
                    throw new InvalidOperationException();
                }
            }
        }
    }
}


在这里我遇到几个问题:

1.  反序列时候 XmlArray("emailcollection") 对应的变量不能被定义为 List, 只能定义为 array。 其实是错的。 可以被定义为 list。 

[System.Xml.Serialization.XmlArray("emailcollection")]
        [System.Xml.Serialization.XmlArrayItem("email",typeof(email))]
        public List<email> emails


2. 自定义的collection 要能使用foreach statoment,必须实现Getenumerator() 方法


微软解释

Type:  System.Collections.IEnumerator
An  IEnumerator object that can be used to iterate through the collection.

The foreach statement of the C# language (For Each in Visual Basic) hides the complexity of the enumerators. Therefore, using foreach is recommended, instead of directly manipulating the enumerator.

Enumerators can be used to read the data in the collection, but they cannot be used to modify the underlying collection.

Initially, the enumerator is positioned before the first element in the collection. The Reset method also brings the enumerator back to this position. At this position, the Current property is undefined. Therefore, you must call the MoveNext method to advance the enumerator to the first element of the collection before reading the value of Current.

Current returns the same object until either MoveNext or Reset is called. MoveNext sets Current to the next element.

If MoveNext passes the end of the collection, the enumerator is positioned after the last element in the collection and MoveNext returns false. When the enumerator is at this position, subsequent calls to MoveNext also return false. If the last call to MoveNext returns falseCurrent is undefined. To set Current to the first element of the collection again, you can callReset followed by MoveNext.

An enumerator remains valid as long as the collection remains unchanged. If changes are made to the collection, such as adding, modifying, or deleting elements, the enumerator is irrecoverably invalidated and its behavior is undefined.

The enumerator does not have exclusive access to the collection; therefore, enumerating through a collection is intrinsically not a thread-safe procedure. To guarantee thread safety during enumeration, you can lock the collection during the entire enumeration. To allow the collection to be accessed by multiple threads for reading and writing, you must implement your own synchronization.



3. 反序列后,object 需要被添加。 添加时需要判定当前collection的状态,然后实习add,否则会进入deadlock然后得到stackoverexception

public void Add(object o) 
        {
            if (_emails == null)
                _emails = new List<email>();


            _emails.Add(o as email);            
        }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值