web-dev-qa-db-ja.com

Android Volley Null Pointer Exception

以前のアクティビティベースのチュートリアルをフラグメントに変換しようとしていますが、アダプターでNullPointerExceptionエラーが発生し続けます。

チュートリアルは this に基づいており、以前はアクティビティを呼び出していたため、アダプターのコンストラクターをスリム化しました。

考えられる解決策が1つあります ここ しかし、同じ考えられる質問を進める方法を誰も知りません。

すべてを機能するフラグメントに変換したいと思います。私からのより多くの情報が必要な場合は私に知らせてください。

主な変換されたクラス:

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;
        }

    }

LogCatエラー:

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)
10
user2793987

マニフェストファイルの簡単なコードを見逃したと思います。 AppControllerパッケージはApplicationを拡張しているため、マニフェストのアプリケーションタグはAppControllerを検索していますが、見つけることができません。

この簡単なコードをAndroidManifest.xmlに書くだけです

Android:name = your.package.AppController

27
user3902144

少し遅れるかもしれませんが、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がnullである理由です。

お役に立てれば :)

11
David Passmore

エラーを見ると、getActivity()内でonCreateView()を呼び出しているようです。代わりに、nullであってはならないonActivityCreatedでアクティビティを取得する必要があります。

1
bph

これをから変更します

_listView = (ListView) getActivity().findViewById(R.id.list);
_

_listView = (ListView)v.findViewById(R.id.list);
_

これは、getActivity()が一般的にフラグメントのContextとして使用されるためです。 FragmentでUI要素のIDを見つけたら、onCreateView()メソッドで返される_View's_オブジェクトの参照を指定する必要があります。

更新:

アダプターを次のように変更します。

_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として使用します。

1
Piyush

すべてのパラメータを注意深く確認してください。パラメータにnull値を送信しないように注意してください。保存されている値を使用する場合は、次のような条件を使用してください。

@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;
 }
0
Mehroz Munir