2

SAX で xml ファイルを解析しようとしています。なぜ例外になるのかわかりません。このコードは、数週間前に書いた別のアプリで動作します。今回は XML ファイルを作成します。

wifi.xml

<?xml version="1.0" encoding="UTF-8"?>
<main>

<wifi bssid="..." >

    <latitude>
        ...
    </latitude>

    <longitude>
        ...
    </longitude>

    <ssid>
        ...
    </ssid>
</wifi>

<wifi bssid="..." >

    <latitude>
        ...
    </latitude>

    <longitude>
        ...
    </longitude>

    <ssid>
        ...
    </ssid>
</wifi>

</main>

このファイルを解析するには、次のコードを使用します。

WifiParser.java

    ...
    public static Vector<WifiDescription> parse(InputStream file) {
    Vector<WifiDescription> wifiDescription = new Vector<WifiDescription>();
    Log.d("XML", "test Parsage");
    try {
        // create a XMLReader from SAXParser
        XMLReader xmlReader = SAXParserFactory.newInstance().newSAXParser().getXMLReader();
        Log.e("XML", "xmlReader");
        WifiHandler wifiHandler = new WifiHandler();
        Log.e("XML", "WifiHandler");
        xmlReader.setContentHandler(wifiHandler);
        Log.e("XML", "setContentHandler");
        xmlReader.parse(new InputSource(file));
        Log.e("XML", "parse");
        wifiDescription = wifiHandler.getWifiDescription();
        Log.e("XML", "wifi description");           
    } 

    catch(Exception ex) {
        Log.d("XML", "WifiParser: parse() failed");
    }

    return wifiDescription;
}
    ...

WifiHandler .java

public class WifiHandler extends DefaultHandler {

public boolean isMain,isWifi,isLatitude,isLongitude,isSSID;

public int i;

public String tmpVal;

private Vector<WifiDescription> mWifiDescription;
private WifiDescription currentWifi;

/**
 * @return
 */
public Vector<WifiDescription> getWifiDescription() {
    Log.v("test parsage getWifiDescription","");
    return mWifiDescription;
}

@Override
public void startDocument() throws SAXException {
    // create new object
    i=0;
    Log.v("test parsage Start","");
    this.mWifiDescription = new Vector<WifiDescription>();
}


@Override
public void endDocument() throws SAXException {
    // nothing we need to do here
    Log.v("test parsage End","");
}

@Override
public void startElement(String namespaceURI, String localName, String qName, Attributes atts)
        throws SAXException {
    Log.v("test parsage StartElement","");
    if(localName.equals("main")) {
        isMain=true;
        Log.v("test parsage main","");
    }

    if(localName.equals("wifi") && isMain ) {
        this.currentWifi = new WifiDescription();

        tmpVal = atts.getValue(WifiDescription.MSSID);
        this.currentWifi.mBSSID=tmpVal;
        isWifi=true;
    }

    else if(localName.equals("latitude") && isWifi ) {
        tmpVal = atts.getValue(WifiDescription.LATITUDE);
        this.currentWifi.mLatitude=Double.parseDouble(tmpVal);
    }
    else if(localName.equals("longitude") && isWifi ) {
        tmpVal = atts.getValue(WifiDescription.LONGITUDE);
        this.currentWifi.mLongitude=Double.parseDouble(tmpVal);
    }
    else if(localName.equals("ssid") && isWifi ) {
        tmpVal = atts.getValue(WifiDescription.SSID);
        this.currentWifi.mSSID=tmpVal;
    }
}

@Override
public void endElement(String namespaceURI, String localName, String qName) throws SAXException {

    Log.v("test parsage EndElement","");
    if(localName.equals("wifi")) {
        //System.out.println("passe dans End landmark");
        isWifi=false;
        i++;
        this.mWifiDescription.add(currentWifi);
    }
    Log.v("XML", "DONE \n");
}

}

そして、mainActivity で次のように呼び出します。

MainActivity.java

    ...
    public void returnTableParseXml(){
    try {
        wifis = WifiParser.parse(getAssets().open("wifi.xml"));
    }
    catch (IOException e) 
    {
        Log.d("XML Main","onCreate(): parse() failed");
        return;
    }
}
    ...

ログキャット:


     07-10 11:00:07.460: E/XML(16482): xmlReader
     07-10 11:00:07.460: E/XML(16482): WifiHandler
     07-10 11:00:07.460: E/XML(16482): setContentHandler
     07-10 11:00:07.465: D/XML ex.getMessage(16482): null
     07-10 11:00:07.465: D/XML(16482): WifiParser: parse() failed

ex.printStackTrace();

07-10 13:33:35.480: W/System.err(29590): java.lang.NullPointerException
07-10 13:33:35.480: W/System.err(29590):    at java.lang.StringToReal.parseDouble(StringToReal.java:244)
07-10 13:33:35.480: W/System.err(29590):    at java.lang.Double.parseDouble(Double.java:295)
07-10 13:33:35.480: W/System.err(29590):    at com.stage.ecolenavale.parser.WifiHandler.startElement(WifiHandler.java:66)
07-10 13:33:35.480: W/System.err(29590):    at org.apache.harmony.xml.ExpatParser.startElement(ExpatParser.java:143)
07-10 13:33:35.480: W/System.err(29590):    at org.apache.harmony.xml.ExpatParser.appendBytes(Native Method)
07-10 13:33:35.480: W/System.err(29590):    at org.apache.harmony.xml.ExpatParser.parseFragment(ExpatParser.java:513)
07-10 13:33:35.480: W/System.err(29590):    at org.apache.harmony.xml.ExpatParser.parseDocument(ExpatParser.java:474)
07-10 13:33:35.480: W/System.err(29590):    at org.apache.harmony.xml.ExpatReader.parse(ExpatReader.java:321)
07-10 13:33:35.480: W/System.err(29590):    at org.apache.harmony.xml.ExpatReader.parse(ExpatReader.java:279)
07-10 13:33:35.480: W/System.err(29590):    at com.stage.ecolenavale.parser.WifiParser.parse(WifiParser.java:28)
07-10 13:33:35.480: W/System.err(29590):    at com.stage.ecolenavale.MainActivity.returnTableParseXml(MainActivity.java:420)
07-10 13:33:35.480: W/System.err(29590):    at com.stage.ecolenavale.MainActivity.onCreate(MainActivity.java:161)

理由がわかりません。私のXMLファイルの問題ですか?

4

2 に答える 2

0

SAXでファイルを解析するために使用したコードは次のとおりです。

WifiHandler.java


public class WifiHandler extends DefaultHandler {

public boolean isMain;

public int i;

public String tmpVal;

private Vector<WifiDescription> mWifiDescription;
private WifiDescription currentWifi;

/**
 * @return
 */
public Vector<WifiDescription> getWifiDescription() {
    Log.v("test parsage getWifiDescription","");
    return mWifiDescription;
}

@Override
public void startDocument() throws SAXException {
    // create new object
    i=0;
    Log.v("test parsage Start","");
    this.mWifiDescription = new Vector<WifiDescription>();
}


@Override
public void endDocument() throws SAXException {
    // nothing we need to do here
    Log.v("test parsage End","");
}

@Override
public void startElement(String namespaceURI, String localName, String qName, Attributes atts)
        throws SAXException {
    Log.v("test parsage StartElement","");
    if(localName.equals("main")) {
        isMain=true;
        Log.v("test parsage main","");
    }

    if(localName.equals("wifi") && isMain ) {
        this.currentWifi = new WifiDescription();

        tmpVal = atts.getValue(WifiDescription.MSSID);
        this.currentWifi.mBSSID=tmpVal;

        tmpVal = atts.getValue(WifiDescription.LATITUDE);
        this.currentWifi.mLatitude=tmpVal;

        tmpVal = atts.getValue(WifiDescription.LONGITUDE);
        this.currentWifi.mLongitude=tmpVal;

        tmpVal = atts.getValue(WifiDescription.SSID);
        this.currentWifi.mSSID=tmpVal;
        isWifi=true;
    }
}

@Override
public void endElement(String namespaceURI, String localName, String qName) throws SAXException {

    Log.v("test parsage EndElement","");
    if(localName.equals("wifi")) {
        //System.out.println("passe dans End landmark");
        isWifi=false;
        i++;
        this.mWifiDescription.add(currentWifi);
    }
    Log.v("XML", "DONE \n");
}

}

WifiParser.java


public class WifiParser  {

public static Vector<WifiDescription> parse(InputStream file) {
    Vector<WifiDescription> wifiDescription = new Vector<WifiDescription>();
    Log.d("XML", "test Parsage");
    try {
        // create a XMLReader from SAXParser
        XMLReader xmlReader = SAXParserFactory.newInstance().newSAXParser().getXMLReader();
        Log.e("XML", "xmlReader");
        WifiHandler wifiHandler = new WifiHandler();
        Log.e("XML", "WifiHandler");
        xmlReader.setContentHandler(wifiHandler);
        Log.e("XML", "setContentHandler");
        xmlReader.parse(new InputSource(file));
        Log.e("XML", "parse");
        wifiDescription = wifiHandler.getWifiDescription();
        Log.e("XML", "wifi description");           
    } 

    catch(Exception ex) {
        Log.d("XML ex.getMessage", "" + ex.getMessage());
        ex.printStackTrace();

        Log.d("XML", "WifiParser: parse() failed");
    }

    return wifiDescription;
}

}

MainActivity.java


    public void returnTableParseXml(){
    try {
        wifis = WifiParser.parse(getAssets().open("wifi.xml"));
    }
    catch (IOException e) 
    {
        Log.d("XML Main","onCreate(): parse() failed");
        return;
    }
}

WifiDescription.java


パブリック クラスの WifiDescription {

    // Description Wifi
    public String mLatitude;
    public String mLongitude;
    public String mBSSID;
    public String mSSID;
    public String mName;


    public static final String WIFI = "wifi";

    // vector usage
    public static final String LATITUDE = "latitude";
    public static final String LONGITUDE ="longitude";
    public static final String SSID = "ssid";
    public static final String MSSID ="bssid";
    public static final String NAME ="name";

}

Wifi.xml


<main>
    <wifi
        bssid="..:..:..:..:..:.."
        latitude="..."
        longitude="..."
        ssid="..." >
    </wifi>
       ...
</main>

于 2013-07-10T12:09:19.590 に答える
0

XML ファイルに存在しない属性値を変換しようとしています:

else if(localName.equals("latitude") && isWifi ) {
    tmpVal = atts.getValue(WifiDescription.LATITUDE);
    this.currentWifi.mLatitude=Double.parseDouble(tmpVal);
}

ここでtmpValは、null になるため、例外になります。

以前、あなたが持っていた

<wifi
    bssid="..:..:..:..:..:.."
    latitude="..."
    longitude="..."
    ssid="..." >
</wifi>

今、あなたは持っています

<wifi bssid="..." >
   <latitude>
       ...
   </latitude>
</wifi>

したがって、必要な値は属性ではなく、 <lattitude> 要素の文字コンテンツです。characters()メソッドを使用してそのコンテンツを読み取る必要があります。文字コンテンツを収集し、それを の正しい変数に読み込む必要があるため、コードがより複雑になりますendElement

于 2013-07-10T12:17:46.277 に答える