溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

JAVA基于SnakeYAML實(shí)現(xiàn)解析與序列化YAML

發(fā)布時(shí)間:2020-09-18 13:25:05 來(lái)源:腳本之家 閱讀:202 作者:JadePeng 欄目:編程語(yǔ)言

這篇文章主要介紹了JAVA基于SnakeYAML實(shí)現(xiàn)解析與序列化YAML,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

1.概述

本文,我們將學(xué)習(xí)如何使用SnakeYAML庫(kù)將

YAML文檔轉(zhuǎn)換為Java對(duì)象,以及JAVA對(duì)象如何序列化為YAML文檔。

2.項(xiàng)目設(shè)置

要在項(xiàng)目中使用SnakeYAML,需要添加Maven依賴(lài)項(xiàng)(可在此處找到最新版本):

<dependency>
  <groupId>org.yaml</groupId>
  <artifactId>snakeyaml</artifactId>
  <version>1.25</version>
</dependency>

3.入口點(diǎn)

該YAML類(lèi)是API的入口點(diǎn):

Yaml yaml = new Yaml()

由于實(shí)現(xiàn)不是線(xiàn)程安全的,因此不同的線(xiàn)程必須具有自己的Yaml實(shí)例。

4.加載YAML文檔

SnakeYAML支持從String或InputStream加載文檔,我們從定義一個(gè)簡(jiǎn)單的YAML文檔開(kāi)始,然后將文件命名為customer.yaml:

firstName: "John"
lastName: "Doe"
age: 20

4.1基本用法

現(xiàn)在,我們將使用Yaml類(lèi)來(lái)解析上述YAML文檔:

Yaml yaml = new Yaml();
InputStream inputStream = this.getClass()
 .getClassLoader()
 .getResourceAsStream("customer.yaml");
Map<String, Object> obj = yaml.load(inputStream);
System.out.println(obj);

上面的代碼生成以下輸出:

{firstName=John, lastName=Doe, age=20}

默認(rèn)情況下,load()方法返回一個(gè)Map對(duì)象。查詢(xún)Map對(duì)象時(shí),我們需要事先知道屬性鍵的名稱(chēng),否則容易出錯(cuò)。更好的辦法是自定義類(lèi)型。

4.2自定義類(lèi)型解析

SnakeYAML提供了一種將文檔解析為自定義類(lèi)型的方法

讓我們定義一個(gè)Customer類(lèi),然后嘗試再次加載該文檔:

public class Customer {
 
  private String firstName;
  private String lastName;
  private int age;
 
  // getters and setters
}

現(xiàn)在我么來(lái)加載:

Yaml yaml = new Yaml();
InputStream inputStream = this.getClass()
 .getClassLoader()
 .getResourceAsStream("customer.yaml");
Customer customer = yaml.load(inputStream);

還有一種方法是使用Constructor:

Yaml yaml = new Yaml(new Constructor(Customer.class));

4.3隱式類(lèi)型

如果沒(méi)有為給定屬性定義類(lèi)型,則庫(kù)會(huì)自動(dòng)將值轉(zhuǎn)換為隱式type。

例如:

1.0 -> Float
42 -> Integer
2009-03-30 -> Date

讓我們使用一個(gè)TestCase來(lái)測(cè)試這種隱式類(lèi)型轉(zhuǎn)換:

@Test
public void whenLoadYAML_thenLoadCorrectImplicitTypes() {
  Yaml yaml = new Yaml();
  Map<Object, Object> document = yaml.load("3.0: 2018-07-22");
 
  assertNotNull(document);
  assertEquals(1, document.size());
  assertTrue(document.containsKey(3.0d));  
}

4.4 嵌套對(duì)象

SnakeYAML 支持嵌套的復(fù)雜類(lèi)型。

讓我們向“ customer.yaml”添加“ 聯(lián)系方式” 和“ 地址” 詳細(xì)信息,并將新文件另存為customer_with_contact_details_and_address.yaml.。

現(xiàn)在,我們將分析新的YAML文檔:

firstName: "John"
lastName: "Doe"
age: 31
contactDetails:
  - type: "mobile"
   number: 123456789
  - type: "landline"
   number: 456786868
homeAddress:
  line: "Xyz, DEF Street"
  city: "City Y"
  state: "State Y"
  zip: 345657

我們來(lái)更新java類(lèi):

public class Customer {
  private String firstName;
  private String lastName;
  private int age;
  private List<Contact> contactDetails;
  private Address homeAddress;  
  // getters and setters
}

public class Contact {
  private String type;
  private int number;
  // getters and setters
}

public class Address {
  private String line;
  private String city;
  private String state;
  private Integer zip;
  // getters and setters
}

現(xiàn)在,我們來(lái)測(cè)試下Yaml#load():

@Test
public void
 whenLoadYAMLDocumentWithTopLevelClass_thenLoadCorrectJavaObjectWithNestedObjects() {
 
  Yaml yaml = new Yaml(new Constructor(Customer.class));
  InputStream inputStream = this.getClass()
   .getClassLoader()
   .getResourceAsStream("yaml/customer_with_contact_details_and_address.yaml");
  Customer customer = yaml.load(inputStream);
 
  assertNotNull(customer);
  assertEquals("John", customer.getFirstName());
  assertEquals("Doe", customer.getLastName());
  assertEquals(31, customer.getAge());
  assertNotNull(customer.getContactDetails());
  assertEquals(2, customer.getContactDetails().size());
   
  assertEquals("mobile", customer.getContactDetails()
   .get(0)
   .getType());
  assertEquals(123456789, customer.getContactDetails()
   .get(0)
   .getNumber());
  assertEquals("landline", customer.getContactDetails()
   .get(1)
   .getType());
  assertEquals(456786868, customer.getContactDetails()
   .get(1)
   .getNumber());
  assertNotNull(customer.getHomeAddress());
  assertEquals("Xyz, DEF Street", customer.getHomeAddress()
   .getLine());
}

4.5類(lèi)型安全的集合

當(dāng)給定Java類(lèi)的一個(gè)或多個(gè)屬性是泛型集合類(lèi)時(shí),需要通過(guò)TypeDescription來(lái)指定泛型類(lèi)型,以以便可以正確解析。

讓我們假設(shè)一個(gè) 一個(gè)Customer擁有多個(gè)Contact:

firstName: "John"
lastName: "Doe"
age: 31
contactDetails:
  - { type: "mobile", number: 123456789}
  - { type: "landline", number: 123456789}

為了能正確解析,我們可以在頂級(jí)類(lèi)上為給定屬性指定TypeDescription :

Constructor constructor = new Constructor(Customer.class);
TypeDescription customTypeDescription = new TypeDescription(Customer.class);
customTypeDescription.addPropertyParameters("contactDetails", Contact.class);
constructor.addTypeDescription(customTypeDescription);
Yaml yaml = new Yaml(constructor);

4.6載入多個(gè)文件

在某些情況下,單個(gè)文件中可能有多個(gè)YAML文檔,而我們想解析所有文檔。所述YAML類(lèi)提供了一個(gè)LOADALL()方法來(lái)完成這種類(lèi)型的解析。

假設(shè)下面的內(nèi)容在一個(gè)文件中:

---
firstName: "John"
lastName: "Doe"
age: 20
---
firstName: "Jack"
lastName: "Jones"
age: 25

我們可以使用loadAll()方法解析以上內(nèi)容,如以下代碼示例所示:

@Test
public void whenLoadMultipleYAMLDocuments_thenLoadCorrectJavaObjects() {
  Yaml yaml = new Yaml(new Constructor(Customer.class));
  InputStream inputStream = this.getClass()
   .getClassLoader()
   .getResourceAsStream("yaml/customers.yaml");
 
  int count = 0;
  for (Object object : yaml.loadAll(inputStream)) {
    count++;
    assertTrue(object instanceof Customer);
  }
  assertEquals(2,count);
}

5.生成YAML文件

SnakeYAML 支持 將java對(duì)象序列化為yml。

5.1基本用法

我們將從一個(gè)將Map <String,Object>的實(shí)例轉(zhuǎn)儲(chǔ)到Y(jié)AML文檔(String)的簡(jiǎn)單示例開(kāi)始:

@Test
public void whenDumpMap_thenGenerateCorrectYAML() {
  Map<String, Object> data = new LinkedHashMap<String, Object>();
  data.put("name", "Silenthand Olleander");
  data.put("race", "Human");
  data.put("traits", new String[] { "ONE_HAND", "ONE_EYE" });
  Yaml yaml = new Yaml();
  StringWriter writer = new StringWriter();
  yaml.dump(data, writer);
  String expectedYaml = "name: Silenthand Olleander\nrace: Human\ntraits: [ONE_HAND, ONE_EYE]\n";
 
  assertEquals(expectedYaml, writer.toString());
}

上面的代碼產(chǎn)生以下輸出(請(qǐng)注意,使用LinkedHashMap的實(shí)例將保留輸出數(shù)據(jù)的順序):

name: Silenthand Olleander
race: Human
traits: [ONE_HAND, ONE_EYE]

5.2自定義Java對(duì)象

我們還可以選擇將自定義Java類(lèi)型轉(zhuǎn)儲(chǔ)到輸出流中。

@Test
public void whenDumpACustomType_thenGenerateCorrectYAML() {
 Customer customer = new Customer();
 customer.setAge(45);
 customer.setFirstName("Greg");
 customer.setLastName("McDowell");
 Yaml yaml = new Yaml();
 StringWriter writer = new StringWriter();
 yaml.dump(customer, writer);  
 String expectedYaml = "!!com.baeldung.snakeyaml.Customer {age: 45, contactDetails: null, firstName: Greg,\n homeAddress: null, lastName: McDowell}\n";
 
 assertEquals(expectedYaml, writer.toString());
}

生成內(nèi)容會(huì)包含!!com.baeldung.snakeyaml.Customer,為了避免在輸出文件中使用標(biāo)簽名,我們可以使用庫(kù)提供的 dumpAs()方法。

因此,在上面的代碼中,我們可以進(jìn)行以下調(diào)整以刪除標(biāo)記:

yaml.dumpAs(customer, Tag.MAP, null);

六 結(jié)語(yǔ)

本文說(shuō)明了SnakeYAML庫(kù)解析和序列化YAML文檔。

所有示例都可以在GitHub項(xiàng)目中找到。

英文原文: Parsing YAML with SnakeYAML

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI