View Full Version : [JAVA] Leggere file XML help
ho creato questo file xml:
<?xml version="1.0"?>
<libri>
<libro>
<autore>pippo</autore>
<titolo>primo titolo</titolo>
<costo>11</costo>
</libro>
<libro>
<autore>sandro</autore>
<titolo>secondo titolo</titolo>
<costo>34</costo>
</libro>
</libri>
adesso dovrei creare una classe che mi legga il file e mi printa i nodi come posso fare?
adesso dovrei creare una classe che mi legga il file e mi printa i nodi come posso fare?Devi usare un apposito parser XML. Ci sono principalmente due tipi di parser: DOM e SAX. Puoi usare parser di terze parti (librerie esterne) o usare i parser disponibili in Java SE presenti fin dalla versione 1.4. Esiste anche un terzo tipo di parser StAX ma è disponibile solo da Java 6 in poi.
Se non hai mai usato questi parser, ti conviene studiare e valutare bene i vari tipi per stabilire cosa è meglio usare.
Inizia da:
http://en.wikipedia.org/wiki/Java_API_for_XML_Processing
SAX è un parser procedurale, si può usare per file molto grandi (genere un gigabyte) ma penso che DOM sia più comprensibile in quanto dichiara l'esistenza di nodi, elementi e attributi che hanno un'immediata corrispondenza con la struttura di un documento XML.
Quanto alla differenza C#, Java, le due lingue sono praticamente identiche, probabilmente è una questione di librerie.
Potresti provare a guardare se la fondazione Apache ha le librerie XML DOM per C#: sono le stesse usate da Java.
Comunque, il grosso della lettura di un documento XML in Java con DOM sta nell'ottenere l'oggetto Document. Bisogna passare per un po' di intermediari. Supponiamo che il file si chiami "documento.xml". La lettura di questo documento si farebbe con:
DocumentBuilderFactory factory;
DocumentBuilder builder;
Document document;
try {
factory = DocumentBuilderFactory.newInstance();
builder = factory.newDocumentBuilder();
document = builder.parse(new java.io.File("documento.xml"));
gestioneDocumento(document); //un metodo definito a parte
} catch(Exception ex) {
ex.printStackTrace();
//varie
}
Una volta che hai il Document (org.w3c.Document) "lavori" con Element, Node, NodeList, AttributeSet e String.
Ad esempio, per quel documento, il nodo "libri", radice del documento, è:
Element radice = document.getDocumentElement();
Puoi recuperare direttamente i nodi "libro" con il metodo getElementsByTagName(string) di Document:
NodeList elencoLibri = document.getElementsByTagName("libro");
Il NodeList è una lista che ha nodeList.getLength() elementi ognuno dei quali accessibile con lista.item(indice).
Per prelevare il testo contenuto in un nodo puoi usare il metodo getTextContent().
Ad esempio per il nodo:
<autore>Pippo</autore>
il metodo getTextContent restituisce Pippo.
I nodi come autore, costo, libro, libri, titolo, sono degli elementi. E' utile saperlo perche gli elementi (Element) hanno dei metodi in più rispetto ai nodi (Node), in particolare ti permettono di ottenere una lista di tutti i figli di nome N.
Scrivo un esempio di lettura del documento xml che hai incollato.
import org.w3c.dom.*;
import javax.xml.parsers.*;
public class Main {
public static void main(String[] args) {
DocumentBuilderFactory factory;
DocumentBuilder parser;
Document document;
try {
factory = DocumentBuilderFactory.newInstance();
parser = factory.newDocumentBuilder();
document = parser.parse(new java.io.File("documento.xml"));
handleDocument(document);
} catch(Exception ex) {
System.out.println("Errore.");
ex.printStackTrace();
}
}
private static void handleDocument(Document document) {
/* Tutti i nodi contenuti in document che si chiamano "libro" */
NodeList libri = document.getElementsByTagName("libro");
for(int i = 0; i < libri.getLength(); i++) {
Element libro = (Element)libri.item(i);
String autore = libro.getElementsByTagName("autore").item(0).getTextContent();
String titolo = libro.getElementsByTagName("titolo").item(0).getTextContent();
String prezzo = libro.getElementsByTagName("costo").item(0).getTextContent();
System.out.println("Autore: " + autore);
System.out.println("Titolo: " + titolo);
System.out.println("Prezzo: " + prezzo);
}
}
}
Le linee:
libro.getElementsByTagName(X).item(0).getTextContent() sono la concatenazione di:
NodeList figli = libro.getElementsByTagName(X);
Node x = figli.item(0);
String testo = x.getTextContent();
Cioè dato il nodo libro, che sarebbe:
<libro>
<autore>pippo</autore>
<titolo>primo titolo</titolo>
<costo>11</costo>
</libro>
getElementsByTagName("autore") restituisce, in forma di NodeList, tutti i figli di libro che si chiamano "autore". Uno solo, quindi item(0) su quel NodeList restituisce il primo (e unico) nodo che si chiama "autore" in libro. Lo stesso vale per gli altri due. Uso questo getElementsByTagName perchè libro potrebbe avere più figli di quanti appaiano. In particolare se lo schema del documento non "scarta" gli spazi vuoti allora libro ha dei figli di tipo Text che contengono i caratteri vuoti. In questo caso dire:
libro.getChildNodes().item(0);
non corrisponderebbe ad ottenere il nodo "autore" ma uno strambo nodo che c'è ma non si vede. Cercando i figli per nome sei certo di andare a pescare dove dovresti.
vi ringrazio ;) provo a fare una classe e chiedo se mai :D
Errori:
init:
deps-jar:
Compiling 1 source file to C:\Documents and Settings\edpsam\Desktop\Progetti JAVA\Gestione XML\leggereXML\build\classes
compile:
Errore.
java.io.FileNotFoundException: C:\Documents and Settings\edpsam\Desktop\Progetti JAVA\Gestione XML\leggereXML\dati.xml (The system cannot find the file specified)
at java.io.FileInputStream.open(Native Method)
at java.io.FileInputStream.<init>(FileInputStream.java:106)
at java.io.FileInputStream.<init>(FileInputStream.java:66)
at sun.net.www.protocol.file.FileURLConnection.connect(FileURLConnection.java:70)
at sun.net.www.protocol.file.FileURLConnection.getInputStream(FileURLConnection.java:161)
at com.sun.org.apache.xerces.internal.impl.XMLEntityManager.setupCurrentEntity(XMLEntityManager.java:973)
at com.sun.org.apache.xerces.internal.impl.XMLVersionDetector.determineDocVersion(XMLVersionDetector.java:184)
at com.sun.org.apache.xerces.internal.parsers.XML11Configuration.parse(XML11Configuration.java:798)
at com.sun.org.apache.xerces.internal.parsers.XML11Configuration.parse(XML11Configuration.java:764)
at com.sun.org.apache.xerces.internal.parsers.XMLParser.parse(XMLParser.java:148)
at com.sun.org.apache.xerces.internal.parsers.DOMParser.parse(DOMParser.java:250)
at com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderImpl.parse(DocumentBuilderImpl.java:292)
at javax.xml.parsers.DocumentBuilder.parse(DocumentBuilder.java:172)
at Main.Main.main(Main.java:28)
debug:
BUILD SUCCESSFUL (total time: 1 second)
Codice:
package Main;
/**
*
* @author edpsam
*/
import org.w3c.dom.*;
import javax.xml.parsers.*;
public class Main {
public static void main(String[] args) {
DocumentBuilderFactory factory;
DocumentBuilder parser;
Document document;
try {
factory = DocumentBuilderFactory.newInstance();
parser = factory.newDocumentBuilder();
document = parser.parse(new java.io.File("dati.xml"));
handleDocument(document);
} catch(Exception ex) {
System.out.println("Errore.");
ex.printStackTrace();
}
}
private static void handleDocument(Document document) {
/* Tutti i nodi contenuti in document che si chiamano "libro" */
NodeList libri = document.getElementsByTagName("libro");
for(int i = 0; i < libri.getLength(); i++) {
Element libro = (Element)libri.item(i);
String autore = libro.getElementsByTagName("autore").item(0).getTextContent();
String titolo = libro.getElementsByTagName("titolo").item(0).getTextContent();
String prezzo = libro.getElementsByTagName("costo").item(0).getTextContent();
System.out.println("Autore: " + autore);
System.out.println("Titolo: " + titolo);
System.out.println("Prezzo: " + prezzo);
}
}
}
xche?
FileNotFoundException
Non trova il file. Dove si trova?
new java.io.File("dati.xml")
"dati.xml" è un percorso relativo. Per trovare un file ci vuole un percorso assoluto. Se specifichi un percorso relativo esso viene trasformato in un percorso assoluto usando come base il percorso della cartella di lavoro del programma.
La cartella di lavoro del programma è una variabile associata al processo. Può essere maneggiata e rimaneggiata ma, di solito, è la stessa del processo "padre".
Senza star qui a rotolarci nella mota, specifica un percorso assoluto e sei a posto. Tipo:
"C:\\Documents and Settings\\edpsam\\Desktop\\Progetti JAVA\\dati.xml"
al posto di
"dati.xml"
Questo non si deve fare in un contesto reale ma a te interessa, credo, acquisire dimestichezza con la manipolazione di documenti xml in Java.
FileNotFoundException
Non trova il file. Dove si trova?
new java.io.File("dati.xml")
"dati.xml" è un percorso relativo. Per trovare un file ci vuole un percorso assoluto. Se specifichi un percorso relativo esso viene trasformato in un percorso assoluto usando come base il percorso della cartella di lavoro del programma.
La cartella di lavoro del programma è una variabile associata al processo. Può essere maneggiata e rimaneggiata ma, di solito, è la stessa del processo "padre".
Senza star qui a rotolarci nella mota, specifica un percorso assoluto e sei a posto. Tipo:
"C:\\Documents and Settings\\edpsam\\Desktop\\Progetti JAVA\\dati.xml"
al posto di
"dati.xml"
Questo non si deve fare in un contesto reale ma a te interessa, credo, acquisire dimestichezza con la manipolazione di documenti xml in Java.
per funzionare funziona ;) ma ho provato a mettere il file dati.xml nella rot del progetto e mettere dati.xml" ma nn me lo prende.::::?
Come prima istruzione del metodo main metti:
System.out.println(System.getProperty("user.dir"));
Il percorso che viene stampato sulla console è quello della cartella di lavoro. Quel percorso è la base rispetto a cui è risolto il nome del file.
Comunque, ribadisco, questa risoluzione dei percorsi relativi rispetto alla cartella di lavoro è una questione che affronti solo in via sperimentale. Le applicazioni Java, ma non solo, hanno altri mezzi per accedere a risorse esterne che si trovino sul filesystem.
vBulletin® v3.6.4, Copyright ©2000-2025, Jelsoft Enterprises Ltd.