Why I have to initialize my object twice while parsing RSS feeds

222 Views Asked by At

In my MainActivity.java I have written code for parsing like

public void parser() {
    try {
        /****** Creating a new instance of the SAX parser ****************/
        SAXParserFactory saxPF = SAXParserFactory.newInstance();
        SAXParser saxParser = saxPF.newSAXParser();
        XMLReader xmlReader = saxParser.getXMLReader();

        URL url = new URL("http://ibnlive.in.com/ibnrss/rss/world/world.xml");

        myXMLHandler = new FeedsXMLHandler();
        xmlReader.setContentHandler(myXMLHandler);
        xmlReader.parse(new InputSource(url.openStream()));     
    } catch (Exception e) {
        e.printStackTrace();
    }
    feedsData = myXMLHandler.getXMLData();
}

I am using IBN Live's rss feed

My handler class XMLHandler.java as follows :

@Override
public void startElement(String uri, String localName, String qName,
                     Attributes attributes) throws SAXException {
    elementValue = "";
    elementOn = true;

    if (localName.equalsIgnoreCase("rss")) {
        dataArray = new ArrayList<FeedsItems>();
        data = new FeedsItems();
        Log.v("Item", "I am in rss block");
    }

    if (localName.equalsIgnoreCase("channel")) {
       Log.v("Item", "I am in channel block");
    }

    if (localName.equalsIgnoreCase("item")) {
       data = new FeedsItems();
       Log.v("Item", "I am in item block");
    }

    if (localName.equalsIgnoreCase("description")) {
       bufferDesc = new StringBuffer();
       elementOn = true;
    }

    if (localName.equalsIgnoreCase("title")) {
       bufferTitle = new StringBuffer();
       elementOn = true;
    }

    if (localName.equalsIgnoreCase("link")) {
            bufferLink = new StringBuffer();
            elementOn = true;
    }
}

/*********** Method will be called when the tags of the XML end **************/
@Override
public void endElement(String uri, String localName, String qName) throws SAXException {
    elementOn = false;

    /*** Sets the values after retrieving the values from the XML tags ******/
    if (localName.equalsIgnoreCase("title")) {
        elementOn = false;

        data.setTitle(bufferTitle.toString());
        bufferTitle.delete(0, bufferTitle.length());
        Log.v("title", data.getTitle());
    }
    else if (localName.equalsIgnoreCase("link")) {
        elementOn = false;

        data.setFeedsUrl(bufferLink.toString());
        bufferLink.delete(0, bufferLink.length());
    }
    else if (localName.equalsIgnoreCase("description")) {
        elementOn = false;

        data.setDescription(bufferDesc.toString());
        bufferDesc.delete(0, bufferDesc.length());
    } else if (localName.equalsIgnoreCase("item")) {
        dataArray.add(data);
    }
}

public void characters(char[] ch, int start, int length) throws SAXException {
    if (elementOn) {
        if (bufferDesc != null) {
            bufferDesc.append(new String(ch, start, length).trim());
        }

        if (bufferTitle != null) {
            bufferTitle.append(new String(ch, start, length).trim());
        }

        if (bufferLink != null) {
            bufferLink.append(new String(ch, start, length).trim());
        }

    } else {
        elementValue = new String(ch, start, length);
    }
}

Problem is why I have to initialize my FeedsItems class twice , first while in 'rss' block and second in 'item' block ?

If I initialize it only in 'rss' block then same data repeatedly adding to my AarrayList and that is last feeds item from the response and if I initialize it only in 'item' block it giving me following error

java.lang.NullPointerException
at com.example.feedsdemo.FeedsXMLHandler.endElement(FeedsXMLHandler.java:83)
at org.apache.harmony.xml.ExpatParser.endElement(ExpatParser.java:160)
at org.apache.harmony.xml.ExpatParser.append(Native Method)
at org.apache.harmony.xml.ExpatParser.parseFragment(ExpatParser.java:521)
at org.apache.harmony.xml.ExpatParser.parseDocument(ExpatParser.java:482)
at org.apache.harmony.xml.ExpatReader.parse(ExpatReader.java:320)
at org.apache.harmony.xml.ExpatReader.parse(ExpatReader.java:277)
at com.example.feedsdemo.NewsFeedsActivity.parser(NewsFeedsActivity.java:65)
at com.example.feedsdemo.NewsFeedsActivity.onCreate(NewsFeedsActivity.java:36)
at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1047)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2627)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2679)
at android.app.ActivityThread.access$2300(ActivityThread.java:125)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2033)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:123)
at android.app.ActivityThread.main(ActivityThread.java:4627)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:521)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:868)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:626)
at dalvik.system.NativeStart.main(Native Method)
2

There are 2 best solutions below

0
On

You have to do it twice because both <channel> and <item> (and also <image>) use the <title> tag. If you don't instansiate it in

if (localName.equalsIgnoreCase("rss")) {
...
}

you'll get the NullPointerException that you posted when you find the first <title> within the first <channel>

What you want to do is either keep both, or remove the one for "rss" and add a check in endElement() that the title belongs to an <item> and not any other element.

0
On

Try this..

@Override
public void startElement(String uri, String localName, String qName,
                     Attributes attributes) throws SAXException {
    elementValue = "";
    elementOn = true;

    if (localName.equalsIgnoreCase("rss")) {
        dataArray = new ArrayList<FeedsItems>();
        data = new FeedsItems();
        bufferLink = new StringBuffer();
        bufferDesc = new StringBuffer();
        bufferTitle = new StringBuffer();
        Log.v("Item", "I am in rss block");
    }    
    else if (localName.equalsIgnoreCase("channel")) {
       Log.v("Item", "I am in channel block");
    }    
    else if (localName.equalsIgnoreCase("item")) {           
       Log.v("Item", "I am in item block");
    }    
    else if (localName.equalsIgnoreCase("description")) {          
       elementOn = true;
    }    
    else if (localName.equalsIgnoreCase("title")) {
       elementOn = true;
    }    
    else if (localName.equalsIgnoreCase("link")) {                
            elementOn = true;
    }
}