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로 변경했습니다. 또한 독자를 제거했습니다 .ReadToEnd (); (이 함수 는 전체 스트림을 읽고 문자열을 반환하므로 Deserialze () 함수는 더 이상 리더를 사용할 수 없습니다 ... 위치는 스트림의 끝에있었습니다). 나는 또한 :)라는 이름으로 자유를 얻었습니다.
수업은 다음과 같습니다.
[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 # 클래스를 생성하는 것은 어떻습니까?
- 파일을 디스크에 씁니다 (이름은 foo.xml)
- xsd를 생성하십시오.
xsd foo.xml
- 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
- 열기 개발자는 명령 프롬프트
당신은에서 찾을 수 있습니다Start Menu > Programs > Microsoft Visual Studio 2012 > Visual Studio Tools
당신은 그냥 입력을 시작할 수있는 Windows 8이있는 경우 또는 개발자 명령 프롬프트 에서 시작 화면 - 다음을 입력하여 XML 파일 디렉토리로 위치를 변경하십시오.
cd /D "C:\path\to\xml"
- 다음 을 입력하여 XML 파일 에서 XSD 파일 을 작성하십시오.
xsd file.xml
- 입력하여 C # 클래스 만들기
xsd /c file.xsd
그리고 그게 다야! xml 파일에서 C # 클래스를 생성했습니다.C:\path\to\xml\file.cs
방법 2-특수 붙여 넣기
필요한 Visual Studio 2012 이상
- XML 파일의 내용을 클립 보드로 복사
- 새로운 빈 클래스 파일 ( Shift+ Alt+ C)을 솔루션에 추가
- 해당 파일을 열고 메뉴에서 클릭
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 파일의 경로입니다. 필요에 따라 변경할 수 있습니다.
Kevin의 anser는 사실을 제외하고는 실제 세계에서는 사용자의 요구에 맞게 원본 XML을 변경할 수없는 경우가 많습니다.
원본 XML에 대한 간단한 솔루션도 있습니다.
[XmlRoot("Cars")]
public class XmlData
{
[XmlElement("Car")]
public List<Car> Cars{ get; set; }
}
public class Car
{
public string StockNumber { get; set; }
public string Make { get; set; }
public string Model { get; set; }
}
그런 다음 간단히 전화 할 수 있습니다.
var ser = new XmlSerializer(typeof(XmlData));
XmlData data = (XmlData)ser.Deserialize(XmlReader.Create(PathToCarsXml));
Xml 직렬화 및 역 직렬화에 대한이 일반 클래스를 사용해보십시오.
public class SerializeConfig<T> where T : class
{
public static void Serialize(string path, T type)
{
var serializer = new XmlSerializer(type.GetType());
using (var writer = new FileStream(path, FileMode.Create))
{
serializer.Serialize(writer, type);
}
}
public static T DeSerialize(string path)
{
T type;
var serializer = new XmlSerializer(typeof(T));
using (var reader = XmlReader.Create(path))
{
type = serializer.Deserialize(reader) as T;
}
return type;
}
}
초보자를위한
나는 여기에 대한 답변이 매우 도움이된다는 것을 알았습니다. 따라서 누군가에게 도움이되는 경우 작업 솔루션을 설명합니다.
원래 질문의 XML. xml은 Class1.xml 파일에 있으며이 파일에 대한 a path
는이 xml 파일을 찾는 코드에서 사용됩니다.
@erymski의 대답을 사용 하여이 작업을 수행하므로 Car.cs라는 파일을 만들고 다음을 추가했습니다.
using System.Xml.Serialization; // Added 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; } }
@erymski가 제공하는 다른 코드 비트 ...
using (TextReader reader = new StreamReader(path)) { XmlSerializer serializer = new XmlSerializer(typeof(CarCollection)); return (CarCollection) serializer.Deserialize(reader); }
... static CarCollection XCar()
다음과 같이 기본 프로그램 (Program.cs)으로 이동 합니다.
using System;
using System.IO;
using System.Xml.Serialization;
namespace ConsoleApp2
{
class Program
{
public static void Main()
{
var c = new CarCollection();
c = XCar();
foreach (var k in c.Cars)
{
Console.WriteLine(k.Make + " " + k.Model + " " + k.StockNumber);
}
c = null;
Console.ReadLine();
}
static CarCollection XCar()
{
using (TextReader reader = new StreamReader(@"C:\Users\SlowLearner\source\repos\ConsoleApp2\ConsoleApp2\Class1.xml"))
{
XmlSerializer serializer = new XmlSerializer(typeof(CarCollection));
return (CarCollection)serializer.Deserialize(reader);
}
}
}
}
그것이 도움이되기를 바랍니다 :-)
아이디어는 직렬화 해제를 위해 모든 수준을 처리하는 것입니다. 비슷한 문제를 해결 한 샘플 솔루션을 참조하십시오.
<?xml version="1.0" ?>
<TRANSACTION_RESPONSE>
<TRANSACTION>
<TRANSACTION_ID>25429</TRANSACTION_ID>
<MERCHANT_ACC_NO>02700701354375000964</MERCHANT_ACC_NO>
<TXN_STATUS>F</TXN_STATUS>
<TXN_SIGNATURE>a16af68d4c3e2280e44bd7c2c23f2af6cb1f0e5a28c266ea741608e72b1a5e4224da5b975909cc43c53b6c0f7f1bbf0820269caa3e350dd1812484edc499b279</TXN_SIGNATURE>
<TXN_SIGNATURE2>B1684258EA112C8B5BA51F73CDA9864D1BB98E04F5A78B67A3E539BEF96CCF4D16CFF6B9E04818B50E855E0783BB075309D112CA596BDC49F9738C4BF3AA1FB4</TXN_SIGNATURE2>
<TRAN_DATE>29-09-2015 07:36:59</TRAN_DATE>
<MERCHANT_TRANID>150929093703RUDZMX4</MERCHANT_TRANID>
<RESPONSE_CODE>9967</RESPONSE_CODE>
<RESPONSE_DESC>Bank rejected transaction!</RESPONSE_DESC>
<CUSTOMER_ID>RUDZMX</CUSTOMER_ID>
<AUTH_ID />
<AUTH_DATE />
<CAPTURE_DATE />
<SALES_DATE />
<VOID_REV_DATE />
<REFUND_DATE />
<REFUND_AMOUNT>0.00</REFUND_AMOUNT>
</TRANSACTION>
</TRANSACTION_RESPONSE>
위의 XML은 두 가지 수준으로 처리됩니다.
[XmlType("TRANSACTION_RESPONSE")]
public class TransactionResponse
{
[XmlElement("TRANSACTION")]
public BankQueryResponse Response { get; set; }
}
내부 레벨
public class BankQueryResponse
{
[XmlElement("TRANSACTION_ID")]
public string TransactionId { get; set; }
[XmlElement("MERCHANT_ACC_NO")]
public string MerchantAccNo { get; set; }
[XmlElement("TXN_SIGNATURE")]
public string TxnSignature { get; set; }
[XmlElement("TRAN_DATE")]
public DateTime TranDate { get; set; }
[XmlElement("TXN_STATUS")]
public string TxnStatus { get; set; }
[XmlElement("REFUND_DATE")]
public DateTime RefundDate { get; set; }
[XmlElement("RESPONSE_CODE")]
public string ResponseCode { get; set; }
[XmlElement("RESPONSE_DESC")]
public string ResponseDesc { get; set; }
[XmlAttribute("MERCHANT_TRANID")]
public string MerchantTranId { get; set; }
}
같은 방법으로 당신은 여러 수준의 필요한 car as array
확인을 다단계 직렬화 복원이 예제를
xsd.exe를 사용하여 xsd 파일을 작성하는 중에 오류가 발생하면 msdn에 언급 된 대로 XmlSchemaInference 클래스를 사용하십시오 . 다음은 시연하기위한 단위 테스트입니다.
using System.Xml;
using System.Xml.Schema;
[TestMethod]
public void GenerateXsdFromXmlTest()
{
string folder = @"C:\mydir\mydata\xmlToCSharp";
XmlReader reader = XmlReader.Create(folder + "\some_xml.xml");
XmlSchemaSet schemaSet = new XmlSchemaSet();
XmlSchemaInference schema = new XmlSchemaInference();
schemaSet = schema.InferSchema(reader);
foreach (XmlSchema s in schemaSet.Schemas())
{
XmlWriter xsdFile = new XmlTextWriter(folder + "\some_xsd.xsd", System.Text.Encoding.UTF8);
s.Write(xsdFile);
xsdFile.Close();
}
}
// now from the visual studio command line type: xsd some_xsd.xsd /classes
Cars car 속성의 속성 하나를 XmlArrayItem에서 XmlElment로 변경할 수 있습니다. 즉,
[System.Xml.Serialization.XmlRootAttribute("Cars", Namespace = "", IsNullable = false)]
public class Cars
{
[XmlArrayItem(typeof(Car))]
public Car[] Car { get; set; }
}
에
[System.Xml.Serialization.XmlRootAttribute("Cars", Namespace = "", IsNullable = false)]
public class Cars
{
[XmlElement("Car")]
public Car[] Car { get; set; }
}
내 해결책 :
Edit > Past Special > Paste XML As Classes
코드에서 클래스를 얻는 데 사용- 다음과 같이 해보십시오. 해당 클래스의 목록 (
List<class1
>)을 만든XmlSerializer
다음를 사용하여 해당 목록을xml
파일 로 직렬화 하십시오. - 이제 파일의 본문을 데이터로 바꾸고 시도
deserialize
하십시오.
암호:
StreamReader sr = new StreamReader(@"C:\Users\duongngh\Desktop\Newfolder\abc.txt");
XmlSerializer xml = new XmlSerializer(typeof(Class1[]));
var a = xml.Deserialize(sr);
sr.Close();
참고 : 루트 이름에주의를 기울여야하며 변경하지 마십시오. 내 "ArrayOfClass1"입니다
XML 문서를 직렬화 해제하는 일반 클래스는 어떻습니까?
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// Generic class to load any xml into a class
// used like this ...
// YourClassTypeHere InfoList = LoadXMLFileIntoClass<YourClassTypeHere>(xmlFile);
using System.IO;
using System.Xml.Serialization;
public static T LoadXMLFileIntoClass<T>(string xmlFile)
{
T returnThis;
XmlSerializer serializer = new XmlSerializer(typeof(T));
if (!FileAndIO.FileExists(xmlFile))
{
Console.WriteLine("FileDoesNotExistError {0}", xmlFile);
}
returnThis = (T)serializer.Deserialize(new StreamReader(xmlFile));
return (T)returnThis;
}
이 부분은 필요하거나 필요하지 않을 수 있습니다. Visual Studio에서 XML 문서를 열고 XML을 마우스 오른쪽 단추로 클릭 한 다음 속성을 선택하십시오. 그런 다음 스키마 파일을 선택하십시오.
async public static Task<JObject> XMLtoNETAsync(XmlDocument ToConvert)
{
//Van XML naar JSON
string jsonText = await Task.Run(() => JsonConvert.SerializeXmlNode(ToConvert));
//Van JSON naar .net object
var o = await Task.Run(() => JObject.Parse(jsonText));
return o;
}
짧막 한 농담:
var object = (Cars)new XmlSerializer(typeof(Cars)).Deserialize(new StringReader(xmlString));
참고 URL : https://stackoverflow.com/questions/364253/how-to-deserialize-xml-document
도와주세요.
'development' 카테고리의 다른 글
PuTTYgen (Windows)을 사용하여 생성 된 SSH 키 쌍을 ssh-agent 및 Keychain (Linux)에서 사용하는 키 쌍으로 변환하는 방법 (0) | 2020.02.17 |
---|---|
부모와 커밋을 어떻게 다른가요? (0) | 2020.02.17 |
한 배열 위치에서 다른 배열 위치로 배열 요소 이동 (0) | 2020.02.17 |
Docker 컨테이너의 런타임 성능 비용은 얼마입니까? (0) | 2020.02.17 |
데이터베이스에 비밀번호를 저장하는 가장 좋은 방법 (0) | 2020.02.17 |