[C#] XML 문서를 역 직렬화하는 방법

이 XML 문서를 역 직렬화하는 방법 :

<?xml version="1.0" encoding="utf-8"?>
<Cars>
  <Car>
    <StockNumber>1020</StockNumber>
    <Make>Nissan</Make>
    <Model>Sentra</Model>
  </Car>
  <Car>
    <StockNumber>1010</StockNumber>
    <Make>Toyota</Make>
    <Model>Corolla</Model>
  </Car>
  <Car>
    <StockNumber>1111</StockNumber>
    <Make>Honda</Make>
    <Model>Accord</Model>
  </Car>
</Cars>

내가 이거 가지고있어:

[Serializable()]
public class Car
{
    [System.Xml.Serialization.XmlElementAttribute("StockNumber")]
    public string StockNumber{ get; set; }

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

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

.

[System.Xml.Serialization.XmlRootAttribute("Cars", Namespace = "", IsNullable = false)]
public class Cars
{
    [XmlArrayItem(typeof(Car))]
    public Car[] Car { get; set; }

}

.

public class CarSerializer
{
    public Cars Deserialize()
    {
        Cars[] cars = null;
        string path = HttpContext.Current.ApplicationInstance.Server.MapPath("~/App_Data/") + "cars.xml";

        XmlSerializer serializer = new XmlSerializer(typeof(Cars[]));

        StreamReader reader = new StreamReader(path);
        reader.ReadToEnd();
        cars = (Cars[])serializer.Deserialize(reader);
        reader.Close();

        return cars;
    }
}

작동하지 않는 것 같습니다 🙁



답변

작동하는 버전이 있습니다. XML에서 StockNumber, Make 및 Model 값이 속성이 아닌 요소이기 때문에 XmlElementAttribute레이블을 변경했습니다 XmlElement. 또한 reader.ReadToEnd();(이 함수 는 전체 스트림을 읽고 문자열을 반환하므로 Deserialize()함수는 더 이상 리더를 사용할 수 없습니다 … 위치는 스트림의 끝에있었습니다). 나는 또한 :)라는 이름으로 자유를 얻었습니다.

수업은 다음과 같습니다.

[Serializable()]
public class Car
{
    [System.Xml.Serialization.XmlElement("StockNumber")]
    public string StockNumber { get; set; }

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

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


[Serializable()]
[System.Xml.Serialization.XmlRoot("CarCollection")]
public class CarCollection
{
    [XmlArray("Cars")]
    [XmlArrayItem("Car", typeof(Car))]
    public Car[] Car { get; set; }
}

역 직렬화 기능 :

CarCollection cars = null;
string path = "cars.xml";

XmlSerializer serializer = new XmlSerializer(typeof(CarCollection));

StreamReader reader = new StreamReader(path);
cars = (CarCollection)serializer.Deserialize(reader);
reader.Close();

그리고 약간 수정 된 xml (<Cars>를 감싸기 위해 새로운 요소를 추가해야했습니다 … Net은 배열의 직렬화 해제에 까다 롭습니다).

<?xml version="1.0" encoding="utf-8"?>
<CarCollection>
<Cars>
  <Car>
    <StockNumber>1020</StockNumber>
    <Make>Nissan</Make>
    <Model>Sentra</Model>
  </Car>
  <Car>
    <StockNumber>1010</StockNumber>
    <Make>Toyota</Make>
    <Model>Corolla</Model>
  </Car>
  <Car>
    <StockNumber>1111</StockNumber>
    <Make>Honda</Make>
    <Model>Accord</Model>
  </Car>
</Cars>
</CarCollection>


답변

XML을 파일로 저장하고 xsd 를 사용 하여 C # 클래스를 생성하는 것은 어떻습니까?

  1. 파일을 디스크에 씁니다 (이름은 foo.xml 임)
  2. xsd를 생성하십시오. xsd foo.xml
  3. C #을 생성하십시오. xsd foo.xsd /classes

E. Voila-및 C # 코드 파일을 통해 데이터를 읽을 수 있어야합니다 XmlSerializer.

    XmlSerializer ser = new XmlSerializer(typeof(Cars));
    Cars cars;
    using (XmlReader reader = XmlReader.Create(path))
    {
        cars = (Cars) ser.Deserialize(reader);
    }

(프로젝트에 생성 된 foo.cs 포함)


답변

두 가지 가능성이 있습니다.

방법 1. XSD 도구


이 위치에 XML 파일이 있다고 가정하십시오. C:\path\to\xml\file.xml

  1. 열기 개발자는 명령 프롬프트
    당신은에서 찾을 수 있습니다 Start Menu > Programs > Microsoft Visual Studio 2012 > Visual Studio Tools
    당신은 그냥 입력을 시작할 수있는 Windows 8이있는 경우 또는 개발자 명령 프롬프트 에서 시작 화면
  2. 다음을 입력하여 XML 파일 디렉토리로 위치를 변경하십시오. cd /D "C:\path\to\xml"
  3. 다음 을 입력하여 XML 파일 에서 XSD 파일 을 작성하십시오.xsd file.xml
  4. 입력하여 C # 클래스 만들기xsd /c file.xsd

그리고 그게 다야! xml 파일에서 C # 클래스를 생성했습니다.C:\path\to\xml\file.cs

방법 2-특수 붙여 넣기


필요한 Visual Studio 2012 이상

  1. XML 파일의 내용을 클립 보드로 복사
  2. 새로운 빈 클래스 파일 ( Shift+ Alt+ C)을 솔루션에 추가
  3. 해당 파일을 열고 메뉴에서 클릭 Edit > Paste special > Paste XML As Classes
    여기에 이미지 설명을 입력하십시오

그리고 그게 다야!

용법


이 도우미 클래스는 사용법이 매우 간단합니다.

using System;
using System.IO;
using System.Web.Script.Serialization; // Add reference: System.Web.Extensions
using System.Xml;
using System.Xml.Serialization;

namespace Helpers
{
    internal static class ParseHelpers
    {
        private static JavaScriptSerializer json;
        private static JavaScriptSerializer JSON { get { return json ?? (json = new JavaScriptSerializer()); } }

        public static Stream ToStream(this string @this)
        {
            var stream = new MemoryStream();
            var writer = new StreamWriter(stream);
            writer.Write(@this);
            writer.Flush();
            stream.Position = 0;
            return stream;
        }


        public static T ParseXML<T>(this string @this) where T : class
        {
            var reader = XmlReader.Create(@this.Trim().ToStream(), new XmlReaderSettings() { ConformanceLevel = ConformanceLevel.Document });
            return new XmlSerializer(typeof(T)).Deserialize(reader) as T;
        }

        public static T ParseJSON<T>(this string @this) where T : class
        {
            return JSON.Deserialize<T>(@this.Trim());
        }
    }
}

지금해야 할 일은 :

    public class JSONRoot
    {
        public catalog catalog { get; set; }
    }
    // ...

    string xml = File.ReadAllText(@"D:\file.xml");
    var catalog1 = xml.ParseXML<catalog>();

    string json = File.ReadAllText(@"D:\file.json");
    var catalog2 = json.ParseJSON<JSONRoot>();


답변

다음 스 니펫은 트릭을 수행해야합니다 (대부분의 직렬화 속성을 무시할 수 있음).

public class Car
{
  public string StockNumber { get; set; }
  public string Make { get; set; }
  public string Model { get; set; }
}

[XmlRootAttribute("Cars")]
public class CarCollection
{
  [XmlElement("Car")]
  public Car[] Cars { get; set; }
}

using (TextReader reader = new StreamReader(path))
{
  XmlSerializer serializer = new XmlSerializer(typeof(CarCollection));
  return (CarCollection) serializer.Deserialize(reader);
}


답변

이것이 도움이되는지 확인하십시오.

[Serializable()]
[System.Xml.Serialization.XmlRootAttribute("Cars", Namespace = "", IsNullable = false)]
public class Cars
{
    [XmlArrayItem(typeof(Car))]
    public Car[] Car { get; set; }
}

.

[Serializable()]
public class Car
{
    [System.Xml.Serialization.XmlElement()]
    public string StockNumber{ get; set; }

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

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

그리고 실패하면 Visual Studio와 함께 제공되는 xsd.exe 프로그램을 사용하여 해당 xml 파일을 기반으로 스키마 문서를 만든 다음 다시 사용하여 스키마 문서를 기반으로 클래스를 만듭니다.


답변

.net이 ‘배열을 직렬화 해제하는 데 까다 롭습니다.’라고 생각하지 않습니다. 첫 번째 xml 문서가 제대로 구성되지 않았습니다. 존재하는 것처럼 보이지만 루트 요소는 없습니다. 표준 xml 문서에는 루트와 하나 이상의 요소 (있는 경우)가 있습니다. 귀하의 예에서 :

<Root> <-- well, the root
  <Cars> <-- an element (not a root), it being an array
    <Car> <-- an element, it being an array item
    ...
    </Car>
  </Cars>
</Root>


답변

.xml 파일이 디스크 어딘가에 생성되었고 다음을 사용한 경우이 코드 블록을 사용해보십시오 List<T>.

//deserialization

XmlSerializer xmlser = new XmlSerializer(typeof(List<Item>));
StreamReader srdr = new StreamReader(@"C:\serialize.xml");
List<Item> p = (List<Item>)xmlser.Deserialize(srdr);
srdr.Close();`

참고 : C:\serialize.xml내 .xml 파일의 경로입니다. 필요에 따라 변경할 수 있습니다.