Clean Code that Works.

어쩌다 보니 간단한 XML 에디터 까지 개발해야 해서...
완전 삽질중. -_-;;
작년 9월부터 시작한걸 아직까지 잡고 있다.
뭐 중간에 캐나다고 갔다오고.. 하기 싫어서 다른 책도 보고 했지만..
이제 슬슬 끝이 보인듯 싶다. 역시 GUI는 번거롭다.

여튼 각설하고 XPATH로 원하는 노드를 찾은 후 XML 문서에 저장 하는 방법을 알아보자.

일단 처음엔 XPATH를 사용해야하기 때문에 XPathFactory 로 새 인스턴스를 만든 후, XPath도 새로 만들어 준다.

XPathFactory xpathFactory = XPathFactory.newInstance();
xpath = xpathFactory.newXPath();
xmlSource = new InputSource(Main.TESTSOURCE);
// 인풋 소스는 해당 xml 파일.

이렇게 해서 각각 인스턴스들을 만든후.
xpath 검색을 해서 엘리먼트에 할당한 다음에
이 엘리먼트에서 수정을 한 후.

Element element = (Element) xpath.evaluate(path, xmlSource, XPathConstants.NODE);
//xpath를 사용해서 검색.
Element root = (Element) element.getParentNode().getParentNode()
    .getParentNode().getParentNode();
//루트는 새로 문서를 저장할 때 필요하기 때문에 필요하다. 나 같은 경우에는 xpath를 사용해서 검색한 깊이가 4단계 이기 때문에 4번째 위의 부모 노드가 루트 노드이다.
Element feature = (Element) element.getChildNodes().item(3);
feature.setTextContent("원하는 데이터");
//해당 노드를 얻어 와서 setTextContent를 사용해서 값을 할당


이렇게 하면 일단 메모리에 올라와 있는 element 들은 다 수정이 되어 있다.
그 다음에는 TransformerFactory 를 사용해서 문서를 다시 저장해 주어야 한다.

TransformerFactory transFactory = TransformerFactory.newInstance();
Transformer transformer = transFactory.newTransformer();
DOMSource source = new DOMSource(root);
StreamResult result = new StreamResult(new File(Main.TESTSOURCE));
transformer.transform(source, result);


DOM 객체를 수정, 추가, 제거하는 것은 모두 메모리상에서 이루어지고 이것을 영구 보본하기 위해서는 파일의 형태로 XML 문서로 저장해야 한다.

다양한 방법으로 문서를 저장 할 수 있지만 위의 방법은 XSLT변환기를 사용하여 파일로 저장한다.

XPATH 로 검색하면.

속성값은 정확히 일치 하는것을 찾을 수 있고.
내용값은 검색어를 포함한 것을 찾을 수 있다.

예제 추가는 나중에 ㅋ

간단한 XML 에디터를 만들어 가고 있는중에...
XPATH를 사용하면 훨씬더 쉽게 검색할 수 있다는 사실을 발견...

OTL 그럼 전에 캐 삽질 DOM 트리 검색은....0_0?
XPATH로 고치는거 고고싱? ㅠ_ㅠ
귀찮다 OTL...일단 에디터 다 만들고 생각해 봐야지 ..

사용자 삽입 이미지

TreeViewer로 구현


마음 같아서야 이클립스에 포함된 XML 에디터 처럼 만들고 싶지만..
아직 허접해서 -_-;;

저기 펼쳐져 있는 트리중에 사진자료에 새로운 사진 노드를 추가하는 것부터 시작..

XPATH를 사용해서 할려고 하기때문에.. 대 난관에 봉착했지만..

다행이 해결 -ㅁ-v

일단 추가는 됬으니 자질구레한거 정리해서 하면 될것 같다..

이제그만 각설하고 소스를 살펴보면

일단 주요 내용을 살펴보면

문서를 파싱한다.
XPath로 객채를 만든다.
XPath표현식을 작성한다.
검색한 결과를 object로 저장하고.
여기에 새 엘리멘트를 추가한다.
그리고 최종적으로 메모리에 올라가 있는 돔 을 파일에 쓴다.


DocumentBuilderFactory domFactory = DocumentBuilderFactory
    .newInstance();

  //domFactory.setNamespaceAware(true); // never forget this!

  DocumentBuilder builder = domFactory.newDocumentBuilder();

  Document doc = builder.parse("VirusInformation.xml");

  XPathFactory factory = XPathFactory.newInstance();

  XPath xpath = factory.newXPath();

  XPathExpression expr = xpath.compile("//작물[@종류='채소']/병해/채소[@종류='오이']/병[@병명='노균병']/사진자료");

  Object result = expr.evaluate(doc, XPathConstants.NODESET);

  NodeList nodes = (NodeList) result;
 
  for (int i = 0; i < nodes.getLength(); i++) {
   Element e = (Element)nodes.item(i);
   Element newElement = doc.createElement("사진");
   Text objText = doc.createTextNode("사진 추가 연습");
   newElement.appendChild(objText);
   newElement.setAttribute("파일명", "1300.jpg");
   e.appendChild(newElement);
   System.out.println(nodes.item(i));
  }
 


  // 메모리에 추가한후 파일에 다시 쓰기.
 
  TransformerFactory tFactory = TransformerFactory.newInstance();
        Transformer transformer;
  try {
   transformer = tFactory.newTransformer();
   transformer.transform(new DOMSource(doc), new StreamResult(new FileOutputStream("VirusInformation.xml")));
  } catch (TransformerConfigurationException e) {
   e.printStackTrace();
  } catch (TransformerException e) {
   e.printStackTrace();
  }



이런 형식으로?
아 오전내내 삽질해서 성공했구나. OTL