java - 将新元素合并到 xml 文件并按年份分组

标签 java xpath jdom

我的问题是,我想通过 XML 文件中的 Data 存储我的 Expenditure 对象,但对于每个 Expenditure 实例,都会创建新的 Year 元素,如下所示:

<Expenditures xmlns="testing">
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2017">
    <Expenditure>
      <date>03-01-2017</date>
      <money>USD 2</money>
      <description>parking ticket</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2019">
    <Expenditure>
      <date>09-01-2019</date>
      <money>PLN 1.5</money>
      <description>Batonik czekoladowy</description>
      <type>FOOD</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2019">
    <Expenditure>
      <date>09-01-2019</date>
      <money>USD 1.5</money>
      <description>guma balonowa</description>
      <type>FOOD</type>
    </Expenditure>
  </year>
</Expenditures>

我的 XML 应该如下所示:

<Expenditures xmlns="testing">
  <year xmlns="" y="2014">
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
        <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
        <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
    <Expenditure>
      <date>01-01-2014</date>
      <money>EUR 1</money>
      <description>bilet autobusowy</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2017">
    <Expenditure>
      <date>03-01-2017</date>
      <money>USD 2</money>
      <description>parking ticket</description>
      <type>TRANSPORT</type>
    </Expenditure>
  </year>
  <year xmlns="" y="2019">
    <Expenditure>
      <date>09-01-2019</date>
      <money>PLN 1.5</money>
      <description>Batonik czekoladowy</description>
      <type>FOOD</type>
    </Expenditure>
    <Expenditure>
      <date>09-01-2019</date>
      <money>USD 1.5</money>
      <description>guma balonowa</description>
      <type>FOOD</type>
    </Expenditure>
  </year>
</Expenditures>

这是我的代码:

    public static void mergeToXMLTesting() throws Exception {
        User user = User.mock();
        Document doc = new Document();
        doc.setRootElement(new Element("Expenditures", Namespace.getNamespace("testing")));
        InputStream in = Main.class.getResourceAsStream(System.getProperty("user.dir") + "/testxml2.xml");

        SAXBuilder builder = new SAXBuilder();
        File xmlFile = new File(System.getProperty("user.dir") + "/testxml2.xml");
        Document document = (Document) builder.build(xmlFile);
        Element rootNode = document.getRootElement();

        for (Expenditure e : user.getExpenditures()) {
            int year = e.getDate().getYear();
            String query = "//year[@y='" + year + "']";
            XPathExpression<Element> xpe = XPathFactory.instance().compile(query, Filters.element());
            Element thisYear = null;
            if (!xpe.evaluate(document).isEmpty()) {
                thisYear = xpe.evaluate(document).get(0);

                System.out.println("found year " + year);

                Element expenditure = new Element("Expenditure");
                expenditure.addContent(new Element("date")
                        .setText(Utils.getFormattedDateTime("dd-MM-yyyy", new DateTime(e.getDate()))));
                expenditure.addContent(new Element("money").setText(e.getMonetaryAmount().toString()));
                expenditure.addContent(new Element("description").setText(e.getDescription()));
                expenditure.addContent(new Element("type").setText(e.getType().name()));
                thisYear.addContent(expenditure);
            } else {
                thisYear = new Element("year");
                thisYear.setAttribute("y", year + "");
                Element expenditure = new Element("Expenditure");
                expenditure.addContent(new Element("date")
                        .setText(Utils.getFormattedDateTime("dd-MM-yyyy", new DateTime(e.getDate()))));
                expenditure.addContent(new Element("money").setText(e.getMonetaryAmount().toString()));
                expenditure.addContent(new Element("description").setText(e.getDescription()));
                expenditure.addContent(new Element("type").setText(e.getType().name()));
                thisYear.addContent(expenditure);
                //doc.getRootElement().addContent(thisYear);
            }
            doc.getRootElement().addContent(thisYear);
        }
        XMLOutputter xmlOutputter = new XMLOutputter(Format.getPrettyFormat());
        xmlOutputter.output(doc, new FileOutputStream(System.getProperty("user.dir") + "/testxml2.xml"));
    }

最佳答案

问题是您将 thisYear 元素放在 for (Expenditure e : user.getExpenditures()) 循环的每次迭代中。

我建议用另一种方式重写整个逻辑: 在迭代之前按年份对支出进行分组。

Map<Integer, List<Expenditure>> groups = expenditures.stream().collect(Collectors.groupingBy(e -> e.getDate().getYear()));
for (Entry<Integer, List<Expenditure>> entry : groups.entrySet()) {
    thisYear = new Element("year");
    thisYear.setAttribute("y", year + "");
    entry.getValues().forEach( e -> {
        Element expenditure = new Element("Expenditure");
        expenditure.addContent(new Element("date")
                .setText(Utils.getFormattedDateTime("dd-MM-yyyy", new DateTime(e.getDate()))));
        expenditure.addContent(new Element("money").setText(e.getMonetaryAmount().toString()));
        expenditure.addContent(new Element("description").setText(e.getDescription()));
        expenditure.addContent(new Element("type").setText(e.getType().name()));
        thisYear.addContent(expenditure);
    });
    doc.getRootElement().addContent(thisYear);

}

抱歉,如果有任何编译错误,我没有完整的代码上下文。

关于java - 将新元素合并到 xml 文件并按年份分组,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54402911/

相关文章:

html - Xpath用于2个不同标签中的两个不同属性

java - 在java中读取xml时junit nullpointerException

java - 使用 JDOM 解析 XML 时出错 - 序言中不允许内容

java - 这个演示中的 JVM 内存分配和释放解释?

java - 如何通过 Acegi 的 beans 手动查询 LDAP?

java - 解码具有单个属性的空元素

java - Jpanel 元素尺寸过大

python - 如何验证模式对话框的存在

javascript - 在 javascript 中使用 xPath 解析具有默认命名空间的 XML

java - 如果 XML 文件有 "xmlns",我如何使用 XPath 从 XML 文件获取文本