安卓Volley空指针异常

10

我试图将之前基于Activity的教程转换为Fragment,但在适配器上仍然遇到NullPointerException错误。

这个教程是基于这里的,而我已经缩减了适配器的构造函数,因为它以前是调用一个Activity的。

这里有一个可能的解决方案,但没有人知道如何处理同样的问题。

我想把一切都转换成可工作的Fragment。如果需要更多信息,请告诉我。

主要转换的类:

public class mainViewController2 extends Fragment {
    private static final String TAG = mainViewController2.class.getSimpleName();
    private ListView listView;
    private FeedListAdapter listAdapter;
    private List<FeedItem> feedItems;
    private String URL_FEED = "http://api.androidhive.info/feed/feed.json";

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                    Bundle savedInstanceState) {

        View v = inflater.inflate(R.layout.mainviewcontroller2_fragment, container, false);

        listView = (ListView) v.findViewById(R.id.list);

        feedItems = new ArrayList<FeedItem>();


        //where the error shows up?
        listAdapter = new FeedListAdapter(feedItems);
        listView.setAdapter(listAdapter);



        // We first check for cached request
        Cache cache = AppController.getInstance().getRequestQueue().getCache();
        Entry entry = cache.get(URL_FEED);
        if (entry != null) {
            // fetch the data from cache
            try {
                String data = new String(entry.data, "UTF-8");
                try {
                    parseJsonFeed(new JSONObject(data));
                } catch (JSONException e) {
                    e.printStackTrace();
                }
            } catch (UnsupportedEncodingException e) {
                e.printStackTrace();
            }

        } else {
            // making fresh volley request and getting json
            JsonObjectRequest jsonReq = new JsonObjectRequest(Method.GET,
                    URL_FEED, null, new Response.Listener<JSONObject>() {

                        @Override
                        public void onResponse(JSONObject response) {
                            VolleyLog.d(TAG, "Response: " + response.toString());
                            if (response != null) {
                                parseJsonFeed(response);
                            }
                        }
                    }, new Response.ErrorListener() {

                        @Override
                        public void onErrorResponse(VolleyError error) {
                            VolleyLog.d(TAG, "Error: " + error.getMessage());
                        }
                    });

            // Adding request to volley request queue
            AppController.getInstance().addToRequestQueue(jsonReq);
        }

        return v;
    }

    /**
     * Parsing json reponse and passing the data to feed view list adapter
     * */
    private void parseJsonFeed(JSONObject response) {
        try {
            JSONArray feedArray = response.getJSONArray("feed");

            for (int i = 0; i < feedArray.length(); i++) {
                JSONObject feedObj = (JSONObject) feedArray.get(i);

                FeedItem item = new FeedItem();
                item.setId(feedObj.getInt("id"));
                item.setName(feedObj.getString("name"));

                // Image might be null sometimes
                String image = feedObj.isNull("image") ? null : feedObj
                        .getString("image");
                item.setImge(image);
                item.setStatus(feedObj.getString("status"));
                item.setProfilePic(feedObj.getString("profilePic"));
                item.setTimeStamp(feedObj.getString("timeStamp"));

                // url might be null sometimes
                String feedUrl = feedObj.isNull("url") ? null : feedObj
                        .getString("url");
                item.setUrl(feedUrl);

                feedItems.add(item);
            }

            // notify data changes to list adapater
            listAdapter.notifyDataSetChanged();
        } catch (JSONException e) {
            e.printStackTrace();
        }
    }
}

FeedListAdapter:

 public class FeedListAdapter extends BaseAdapter { 
    private Activity activity;
    private LayoutInflater inflater;
    private List<FeedItem> feedItems;

    Context context;

    ImageLoader imageLoader;

     public FeedListAdapter(Context ctx,List<FeedItem> feedItems) {
        this.context= ctx;
        this.feedItems = feedItems;
        imageLoader = AppController.getInstance().getImageLoader();
        inflater = LayoutInflater.from(context);
     }

    @Override
    public int getCount() {
        return feedItems.size();
    }

    @Override
    public Object getItem(int location) {
        return feedItems.get(location);
    }
        @Override
        public long getItemId(int position) {
            return position;
        }

        @SuppressLint("InflateParams")
        @Override
        public View getView(int position, View convertView, ViewGroup parent) {

            if (inflater == null)
                inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
            if (convertView == null)
                convertView = inflater.inflate(R.layout.feed_item, null);

            if (imageLoader == null)
                imageLoader = AppController.getInstance().getImageLoader();

            TextView name = (TextView) convertView.findViewById(R.id.name);
            TextView timestamp = (TextView) convertView
                    .findViewById(R.id.timestamp);
            TextView statusMsg = (TextView) convertView
                    .findViewById(R.id.txtStatusMsg);
            TextView url = (TextView) convertView.findViewById(R.id.txtUrl);
            NetworkImageView profilePic = (NetworkImageView) convertView
                    .findViewById(R.id.profilePic);
            FeedImageView feedImageView = (FeedImageView) convertView
                    .findViewById(R.id.feedImage1);

            FeedItem item = feedItems.get(position);

            name.setText(item.getName());

            // Converting timestamp into x ago format
            CharSequence timeAgo = DateUtils.getRelativeTimeSpanString(
                    Long.parseLong(item.getTimeStamp()),
                    System.currentTimeMillis(), DateUtils.SECOND_IN_MILLIS);
            timestamp.setText(timeAgo);

            // Chcek for empty status message
            if (!TextUtils.isEmpty(item.getStatus())) {
                statusMsg.setText(item.getStatus());
                statusMsg.setVisibility(View.VISIBLE);
            } else {
                // status is empty, remove from view
                statusMsg.setVisibility(View.GONE);
            }

            // Checking for null feed url
            if (item.getUrl() != null) {
                url.setText(Html.fromHtml("<a href=\"" + item.getUrl() + "\">"
                        + item.getUrl() + "</a> "));

                // Making url clickable
                url.setMovementMethod(LinkMovementMethod.getInstance());
                url.setVisibility(View.VISIBLE);
            } else {
                // url is null, remove from the view
                url.setVisibility(View.GONE);
            }

            // user profile pic
            profilePic.setImageUrl(item.getProfilePic(), imageLoader);

            // Feed image
            if (item.getImge() != null) {
                feedImageView.setImageUrl(item.getImge(), imageLoader);
                feedImageView.setVisibility(View.VISIBLE);
                feedImageView
                        .setResponseObserver(new FeedImageView.ResponseObserver() {
                            @Override
                            public void onError() {
                            }

                            @Override
                            public void onSuccess() {
                            }
                        });
            } else {
                feedImageView.setVisibility(View.GONE);
            }

            return convertView;
        }

    }

日志记录错误:

09-06 02:51:55.823: E/AndroidRuntime(8279): FATAL EXCEPTION: main
09-06 02:51:55.823: E/AndroidRuntime(8279): Process: com.rynovation.kline, PID: 8279
09-06 02:51:55.823: E/AndroidRuntime(8279): java.lang.NullPointerException
09-06 02:51:55.823: E/AndroidRuntime(8279):     at androidFeedClasses.FeedListAdapter.<init>(FeedListAdapter.java:35)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at com.rynovation.kline.mainViewController2.onCreateView(mainViewController2.java:51)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.app.Fragment.performCreateView(Fragment.java:1700)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:890)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:1062)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.app.BackStackRecord.run(BackStackRecord.java:684)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1453)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.app.FragmentManagerImpl$1.run(FragmentManager.java:443)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.os.Handler.handleCallback(Handler.java:733)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.os.Handler.dispatchMessage(Handler.java:95)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.os.Looper.loop(Looper.java:146)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at android.app.ActivityThread.main(ActivityThread.java:5487)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at java.lang.reflect.Method.invokeNative(Native Method)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at java.lang.reflect.Method.invoke(Method.java:515)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1283)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1099)
09-06 02:51:55.823: E/AndroidRuntime(8279):     at dalvik.system.NativeStart.main(Native Method)

你的适配器代码很混乱。请了解ViewHolder模式并尝试创建类似的东西。这是Google版本,但也可以尝试搜索其他示例:http://developer.android.com/training/improving-layouts/smooth-scrolling.html - Itai Hanski
6个回答

27

我猜你在manifest文件中遗漏了一个简单的代码。你的AppController包扩展了Application,所以manifest文件中的应用程序标签正在搜索AppController,但无法找到它。

只需将这个简单的代码写入你的AndroidManifest.xml文件中:

android:name = your.package.AppController


1
先生,您是省时的人。 - Murat

11

可能有点晚了,但是这是对user3902144的回答的代表性答案:

类似的问题答案:链接

你可能在清单文件中漏掉了这个:

 <application
        android:name="<package-name>.app.AppController" // <-- this one
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme">

这就是为什么onCreate从未被调用,而mInstance为空。

希望这可以帮到你 :)


谢谢!永远不晚哈哈 - CodeGuru
回答有点晚了,但还是帮了我 :) - Ranjithkumar
如果因为有多 dex 无法进行呢? - user3402040
@delive 你能详细说明一下吗? - David Passmore

1

将此更改为

listView = (ListView) getActivity().findViewById(R.id.list);

listView = (ListView)v.findViewById(R.id.list);

这是因为在Fragment中,通常使用getActivity()作为Context。当您在Fragment中查找UI元素的ID时,必须提供View对象的引用,该对象将在onCreateView()方法中返回。请注意保留HTML标记。
更新:
请更改您的适配器如下,
listAdapter = new FeedListAdapter(getActivity(),feedItems);

现在处于构造函数中。
 ImageLoader imageLoader;

 public FeedListAdapter(Context ctx,List<FeedItem> feedItems) {
    this.context= ctx;
    this.feedItems = feedItems;
    imageLoader = AppController.getInstance().getImageLoader()
 }

现在,在您的适配器类中使用context变量作为Context

谢谢,但我仍然遇到相同的错误。我认为这可能是在FeedListAdapter中引起的? - user2793987
好的,我也更新了我的帖子以反映您的更改。然而,现在问题似乎出现在构造函数中。请参见上面更新的日志记录。 - user2793987
在您的适配器构造函数中添加这一行 inflater = LayoutInflater.from(context)。 - Piyush
好的,我已经这样做了,现在我通过新的logcat/更新的代码看到一个指向那里的错误。请参见上面。 - user2793987
还有什么想法吗?我按照你说的做了,但是还是出现了相同的错误。 - user2793987

1
看起来你在onCreateView()中调用了getActivity(),导致出现错误。你应该在onActivityCreated中获取activity,这样就不会为空了。

你有什么建议?它应该是一个Fragment,而不是一个Activity。如果可能的话,你能写一个例子吗? - user2793987
你能解释一下为什么需要进入Activity来查找ListView吗?ListView不在R.layout.mainviewcontroller2_fragment中吗? - bph
它显然不存在。你有什么想象它应该是什么样子的吗? - user2793987
你能点击它以转到方法声明吗?它应该存在于某个地方。否则,你的代码将无法编译。 - bph
它进入相同的FeedListAdapter,没有一个特定的允许在构造函数中使用一个参数的适配器。 - user2793987
显示剩余2条评论

0

//我曾遇到这个错误,只需将其写入您的AppController类中即可解决。

@Override
    public void onCreate() {
        super.onCreate();
        instance = this;
    }
}

0
请仔细检查所有参数,确保在参数中不发送任何空值,如果使用一些存储的值,则使用以下条件:
@Override
protected Map<String, String> getParams() {
 Map<String, String> params = new HashMap<String, String>();
 if(paramValue!= null)
     params.put("paramKey", paramValue);
  else
     params.put("paramKey", "anyOtherDefaultValue");
 return params;
 }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接