How to change the specific line in an XML file?

2.3k Views Asked by At

I have a really long config file(not my own) and I want to change a specific line in it.

I need to change the ServerPort value to 12345 for example, but without the code knowing the actual number..(the 26900).

this is a very small part of the config file:

<?xml version="1.0"?>
<ServerSettings>
  <property name="ServerPort"               value="26900"/>             <!-- Port you want the server to listen on. -->
  <property name="ServerIsPublic"           value="true"/>              <!-- Should this server register to master server -->
  <property name="ServerName"               value="My Game Host"/>      <!-- Whatever you want the name to be. -->
  <property name="ServerPassword"           value=""/>                  <!-- Password to gain entry to the server -->
  <property name="ServerMaxPlayerCount"     value="8"/>                 <!-- Maximum Concurrent Players -->
  <property name="ServerDescription"        value="A 7 Days to Die server"/> <!-- Whatever you want the description to be. -->
  <property name="ServerWebsiteURL"         value=""/>                  <!-- Website URL for the server -->
4

There are 4 best solutions below

7
On BEST ANSWER

The simple answer is use XML parser and change the desired attribute value. To search for node - use XPath.

Let's take your XML file:

xmlfile.xml

<ServerSettings>
  <property name="ServerPort"               value="26900"/>             <!-- Port you want the server to listen on. -->
  <property name="ServerIsPublic"           value="true"/>              <!-- Should this server register to master server -->
  <property name="ServerName"               value="My Game Host"/>      <!-- Whatever you want the name to be. -->
  <property name="ServerPassword"           value=""/>                  <!-- Password to gain entry to the server -->
  <property name="ServerMaxPlayerCount"     value="8"/>                 <!-- Maximum Concurrent Players -->
  <property name="ServerDescription"        value="A 7 Days to Die server"/> <!-- Whatever you want the description to be. -->
  <property name="ServerWebsiteURL"         value=""/>                  <!-- Website URL for the server -->
</ServerSettings>

For example using DOM parser and using XPath to get the node

    File file = new File("xmlfile.xml");
    DocumentBuilder dBuilder = DocumentBuilderFactory.newInstance().newDocumentBuilder();
    Document xml = dBuilder.parse(file);
    XPath xpath = XPathFactory.newInstance().newXPath();
    Node portNode = (Node) xpath.compile("/ServerSettings/property[@name='ServerPort']").evaluate(xml, XPathConstants.NODE);
    portNode.getAttributes().getNamedItem("value").setNodeValue("8080");

    // Saving changed XML back
    Transformer transformer = TransformerFactory.newInstance().newTransformer();
    Result output = new StreamResult(new File("xmlfile.xml"));
    Source input = new DOMSource(xml);
    transformer.transform(input, output);

will change from 26900 to 8080.
If your file is really big (like tens of megabytes) better use SAX parser or try to search for it using regular expression. If it's regular settings file worth couple kilobytes - code above is the most simple.

2
On

In this specific case, you need XML parser. Can't compare XML parsing to regular file modification.

I am using jdom for xml parsing. You can download this jdom.jar from web

import java.io.File;
import java.io.FileWriter;
import java.io.IOException;

import org.jdom.Document;
import org.jdom.Element;
import org.jdom.JDOMException;
import org.jdom.input.SAXBuilder;
import org.jdom.output.Format;
import org.jdom.output.XMLOutputter;

public class Sample1 {
    public static void main(String[] args) throws IOException, JDOMException {

        File file = new File("c:\\file.xml");
        Document doc = new SAXBuilder().build(file);

        Element rootElement = doc.getRootElement();

        for (Object child : rootElement.getChildren("property")) {
            Element el = (Element) child;
            if (el.getAttributeValue("name").equalsIgnoreCase("ServerPort")) {
                el.setAttribute("value", "12345");
            }
        }

        XMLOutputter xmlOutput = new XMLOutputter();

        xmlOutput.setFormat(Format.getPrettyFormat());
        xmlOutput.output(doc, new FileWriter("c:\\file.xml"));
    }

}
0
On

File you have mentioned is an xml file. Just you use an xml parser+builder and it will be much easier for you to read and update fields.

3
On

You can use a regular expression to find and replace the port number:

pattern to search for:

(<property name="ServerPort"\s+?value=)"\d+?"

and replace with:

$1"12345"

12345 being an example port.

String fileContent = // open, read file.  Either line by line or the entirety
fileContent = fileContent.replaceAll("(<property name=\"ServerPort\"\\s+?value=)\"\\d+?\"", "$1\"12345\"");
// write fileContent back to disk.