2016-09-16 72 views
0

我有一個CSV文件看起來像這樣:如何將錶轉換爲地圖的地圖

request.car.model (STRING),request.car.make (STRING),request.person.lastname (STRING),other.person.birthdate (DATE),other.person.length (INTEGER) 
BMW,7 series,Doe,31/12/1980,170 
Tesla,Model S,Smith,1/1/1975,172 
Volvo,C40,Johnson,13/11/1982,189 

第一行頭,與被括號之間解析類型。標題使用點符號,非常像Javascript會解析一個複雜的對象。

我需要做的是爲CSV文件中的每一行創建一個Java Map<String, Serializable>。例如,第一行需要導致map在Java中看起來像這樣:

Map<String, Serializable> requestCarMap = new HashMap<>(); 
requestCarMap.put("make", "BMW"); 
requestCarMap.put("model", "7 series"); 
Map<String, Serializable> requestPersonMap = new HashMap<>(); 
requestPersonMap.put("lastName", "Doe"); 
Map<String, Serializable> requestMap = new HashMap<>(); 
requestMap.put("car", (HashMap) requestCarMap); 
requestMap.put("pesron", (HashMap) requestPersonMap); 
Map<String, Serializable> otherPersonMap = new HashMap<>(); 
otherPersonMap.put("birthdate", new Date(1980,12,31)); 
Map<String, Serializable> otherMap = new HashMap<>(); 
otherMap.put("person", (HashMap) otherPersonMap); 
Map<String, Serializable> map = new HashMap<>(); 
map.put("request", (HashMap) requestMap); 
map.put("other", (HashMap) otherMap); 

我目前有創建一個基於價值觀的地圖,但是,如果沒有分裂頭,並把它在子圖的方法:

protected static Map<String, Serializable> parseRecord(CSVRecord record, CSVRecord header) { 
    final Map<String, Serializable> map = new HashMap<>(); 
    for (int i = 0; i < record.size(); i++) { 
     String headerField = getHeader(header.get(i)); 
     String[] headerFieldPart = headerField.split("\\."); 
     String headerFieldType = getHeaderFieldType(header.get(i)); 
     Object recordField = getRecordField(headerFieldType, record.get(i)); 
     if (recordField != null) { 
      map.put(headerField, (Serializable) recordField); 
     } else { 
      log.warn("Skipped value for record item: " 
        + record.get(i)); 
     } 
    } 
    return map; 
} 

我怎樣才能把在子圖中的值作爲例如在通用的方式?我不知道列的名稱或地圖的前面嵌套深度。我無法弄清楚在subMap不存在,存在等情況下工作的邏輯。任何智能解決方案?任何能幫助我解決這個問題的庫?

回答

2

基本上,在點上分割標題,迭代零件,並根據相關零件切換目標映射指針。

像這樣(假設你得到了CSV-物降):

public static void main(String[] args) { 
    String[] headers = { "request.car.model", "request.car.make", "request.buyer" }; 
    String[] values = { "a", "b", "c" }; 

    Map<String, Serializable> outer = new HashMap<>(); 

    for(int i = 0; i < headers.length; i++) { 
     String header = headers[i]; 
     String value = values[i]; 

     String[] parts = header.split("\\."); 
     Map<String, Serializable> targetMap = outer; 
     for(int j = 0; j < parts.length - 1; j++) 
      targetMap = (Map<String, Serializable>) targetMap.computeIfAbsent(parts[j], x -> new HashMap<>()); 
     targetMap.put(parts[parts.length - 1], value); 
    } 

    System.out.println(outer.get("request")); 
} 

注意,如果頭是一致的,這隻會工作,也就是說,作爲「AA值沒有這樣的事情「之後是」aab「的值。

+0

與所提供的CSV測試,它的工作原理就像一個魅力。非常感謝你! –

0

僅供參考下面是完整的方法:

protected static Map<String, Serializable> parseRecord(CSVRecord record, CSVRecord fullHeader) { 
    if (record == null) { 
     log.warn("Record is null."); 
     return null; 
    } 
    log.info("Parsing record number: " + (record.getRecordNumber() - 1)); 
    log.debug(" with content: " + record.toString()); 
    final Map<String, Serializable> map = new HashMap<>(); 
    for (int i = 0; i < record.size(); i++) { 
     String headerFieldType = getHeaderFieldType(fullHeader.get(i)); 
     Object recordField = getRecordField(headerFieldType, record.get(i)); 
     if (recordField != null) { 
      String headerField = getHeaderField(fullHeader.get(i)); 
      String[] headerFieldPart = headerField.split("\\."); 
      Map<String, Serializable> targetMap = map; 
      for (int j = 0; j < headerFieldPart.length - 1; j++) { 
       targetMap = (Map<String, Serializable>) targetMap.computeIfAbsent(headerFieldPart[j], x -> new HashMap<>()); 
      } 
      targetMap.put(headerFieldPart[headerFieldPart.length - 1], (Serializable) recordField); 
     } else { 
      log.warn("Skipped value for record item: " 
        + record.get(i)); 
     } 
    } 
    log.debug("Parsed map: " + map); 
    return map; 
} 

而生成的日誌:

INFO Parsing record number: 1 
DEBUG with content: CSVRecord [comment=null, mapping=null, recordNumber=2, values=[BMW, 7 series, Doe, 31/12/1980, 170]] 
INFO Parsed map: {request={car={model=BMW, make=7 series}, person={lastname=Doe}}, other={person={birthdate=Thu Jan 31 00:12:00 CET 1980, length=170}}} 
INFO Parsing record number: 2 
DEBUG with content: CSVRecord [comment=null, mapping=null, recordNumber=3, values=[Tesla, Model S, Smith, 1/1/1975, 172]] 
INFO Parsed map: {request={car={model=Tesla, make=Model S}, person={lastname=Smith}}, other={person={birthdate=Wed Jan 01 00:01:00 CET 1975, length=172}}} 
INFO Parsing record number: 3 
DEBUG with content: CSVRecord [comment=null, mapping=null, recordNumber=4, values=[Volvo, C40, Johnson, 13/11/1982, 189]] 
INFO Parsed map: {request={car={model=Volvo, make=C40}, person={lastname=Johnson}}, other={person={birthdate=Wed Jan 13 00:11:00 CET 1982, length=189}}}