【问题标题】:Parsing JSON data with recyclerview and volley error使用 recyclerview 和 volley 错误解析 JSON 数据
【发布时间】:2021-04-13 16:35:32
【问题描述】:

我正在尝试按照this 将 JSON 数据从 mySQL 提取到 Android 应用程序。我收到此错误E/RecyclerView: No adapter attached; skipping layout。我尝试查找其他帖子的建议,但仍然无法解决问题。以下是我正在处理的代码:

import androidx.recyclerview.widget.RecyclerView;

import android.os.Bundle;

import com.android.volley.Request;
import com.android.volley.RequestQueue;
import com.android.volley.Response;
import com.android.volley.VolleyError;
import com.android.volley.toolbox.StringRequest;
import com.android.volley.toolbox.Volley;

import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity {
    private static final String HI ="http://xxx/phpMyAdmin-5.0.4/Appi.php" ;   //xxx is ip and the user name
    private List<List_Data> list_data;
    private RecyclerView rv;
    private MyAdapter adapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        rv=(RecyclerView)findViewById(R.id.recyclerView);
        rv.setHasFixedSize(true);
        rv.setLayoutManager(new LinearLayoutManager(this));
        list_data=new ArrayList<>();
        adapter=new MyAdapter(list_data);

        getMovieData();

    }

    private void getMovieData() {
        StringRequest stringRequest=new StringRequest(Request.Method.GET, HI, new Response.Listener<String>() {
            @Override
            public void onResponse(String response) {
                try {
                    JSONObject jsonObject=new JSONObject(response);
                    JSONArray array=jsonObject.getJSONArray("data");
                    for (int i=0; i<array.length(); i++ ){
                        JSONObject ob=array.getJSONObject(i);
                        List_Data listData=new List_Data(ob.getString("name")
                                ,ob.getString("moive"));
                        list_data.add(listData);
                    }
                    rv.setAdapter(adapter);
                } catch (JSONException e) {
                    e.printStackTrace();
                }

            }
        }, new Response.ErrorListener() {
            @Override
            public void onErrorResponse(VolleyError error) {

            }
        });
        RequestQueue requestQueue= Volley.newRequestQueue(this);
        requestQueue.add(stringRequest);
    }
}

JSON 格式为:

 [{"ID":"1","name":"ab","movie":"cd"},{"ID":"2","name":"ef","movie":"gh"}]

适配器是:

import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.LinearLayout;
import android.widget.ListView;
import android.widget.TextView;
import androidx.recyclerview.widget.RecyclerView;

import java.util.List;

public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder>{
    private List<List_Data>list_data;

    public MyAdapter(List<List_Data> list_data) {
        this.list_data = list_data;
    }

    @Override
    public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view= LayoutInflater.from(parent.getContext()).inflate(R.layout.list_data,parent,false);
        return new ViewHolder(view);
    }

    @Override
    public void onBindViewHolder(ViewHolder holder, int position) {
        List_Data listData=list_data.get(position);
        holder.txtname.setText(listData.getName());
        holder.txtmovie.setText(listData.getMoviename());
    }

    @Override
    public int getItemCount() {

        return list_data.size();
    }

    public class ViewHolder extends RecyclerView.ViewHolder{
        private TextView txtname,txtmovie;
        public ViewHolder(View itemView) {
            super(itemView);
            txtname=(TextView)itemView.findViewById(R.id.txt_name);
            txtmovie=(TextView)itemView.findViewById(R.id.txt_moviename);
        }
    }
}

另外,我这样设置类:

    public class List_Data {
    private String name;
    private String moviename;

    public List_Data(String name, String moviename) {
        this.name = name;
        this.moviename = moviename;
    }

    public String getName() {
        return name;
    }

    public String getMoviename() {
        return moviename;
    }
}

而activity_main.xml是:

<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/recyclerView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>

</RelativeLayout>

还有lis_data.xml:

<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal">
<androidx.cardview.widget.CardView
    android:padding="10dp"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    >
    <LinearLayout
        android:padding="10dp"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical">
        <LinearLayout
            android:layout_marginLeft="10dp"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:orientation="horizontal">
            <TextView
                android:textColor="#111"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="name:"/>

            <TextView
                android:id="@+id/txt_name"
                android:textColor="#111"
                android:layout_marginLeft="10dp"
                style="@style/Base.TextAppearance.AppCompat.Medium"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                />
        </LinearLayout>
        <LinearLayout
            android:layout_marginTop="10dp"
            android:layout_marginLeft="10dp"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:orientation="horizontal">
            <LinearLayout
                android:layout_marginLeft="10dp"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:orientation="horizontal">
            <TextView
                android:textColor="#111"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="moivename:"/>

            <TextView
                android:id="@+id/txt_moviename"
                android:textColor="#111"
                android:layout_marginLeft="10dp"
                style="@style/Base.TextAppearance.AppCompat.Medium"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                />
        </LinearLayout>
    </LinearLayout>
    </LinearLayout>
</androidx.cardview.widget.CardView>

提前感谢您的建议!

【问题讨论】:

  • 在调试模式下运行并在每个函数上设置检查点始终是一个不错的选择,这样您就可以了解应用程序到达的位置或将日志语句放入函数中

标签: java android android-recyclerview android-volley


【解决方案1】:

在您的代码中,适配器总是会在获得新数据时重新设置。最好设置一次适配器,并在获得新数据时通知数据更改。 这是我的建议。 on onCreate 设置回收器视图适配器

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    rv=(RecyclerView)findViewById(R.id.recyclerView);
    rv.setHasFixedSize(true);
    rv.setLayoutManager(new LinearLayoutManager(this));
    list_data=new ArrayList<>();
    adapter=new MyAdapter(list_data);
    // set adapter
    rv.setAdapter(adapter);
    getMovieData();

}

getMovieData()数据更新后,调用adapter.notifyDataSetChanged()而不是设置新的适配器

private void getMovieData() {
    StringRequest stringRequest=new StringRequest(Request.Method.GET, HI, new Response.Listener<String>() {
        @Override
        public void onResponse(String response) {
            try {
                JSONObject jsonObject=new JSONObject(response);
                JSONArray array=jsonObject.getJSONArray("data");
                for (int i=0; i<array.length(); i++ ){
                    JSONObject ob=array.getJSONObject(i);
                    List_Data listData=new List_Data(ob.getString("name")
                            ,ob.getString("moive"));
                    list_data.add(listData);
                }
                adapter.notifyDataSetChanged()
            } catch (JSONException e) {
                e.printStackTrace();
            }

        }
    }, new Response.ErrorListener() {
        @Override
        public void onErrorResponse(VolleyError error) {

        }
    });
    RequestQueue requestQueue= Volley.newRequestQueue(this);
    requestQueue.add(stringRequest);
}

【讨论】:

  • 感谢您的回复!这似乎有助于解决E/RecyclerView: 问题,但数据仍然没有显示在模拟器上。你知道为什么会发生这种情况吗?我遵循与教程相同的步骤..
  • 请确保函数onResponse被调用(而不是错误),当adapter.notifyDataSetChanged()被调用时list_data也不为空
  • 嗨@swiftlearneer 对不起。只是为了确保,不要忘记在AndroidManifes.xml文件&lt;uses-permission android:name="android.permission.INTERNET" /&gt;中添加互联网权限@
  • 确保 json 数据是 { "data": [{"ID":"1","name":"ab","movie":"cd"},{"ID":"2","name":"ef","movie":"gh"}] } 而不是像你上面的问题中的 [{"ID":"1","name":"ab","movie":"cd"},{"ID":"2","name":"ef","movie":"gh"}]
  • 因为我不知道你是如何在你的服务器端(php)实现它的,所以我无能为力。但是,您也可以在解析 json 时修改您的 android 代码,而不是从字符串创建 json 对象,您可以直接创建 jsonarray。删除此代码JSONObject jsonObject=new JSONObject(response); 并将JSONArray array=jsonObject.getJSONArray("data"); 修改为JSONArray array=new JSONArray(response);
【解决方案2】:

更新:我尝试了@hakim 的建议。应用仍在运行,但数据未显示。

我还发现 new Response.Listener&lt;String&gt;()new Response.ErrorListener() 在下面的 getMoiveData() 中显示为灰色:

private void getMovieData() {
    StringRequest stringRequest=new StringRequest(Request.Method.GET, HI, new Response.Listener<String>() {
        @Override
        public void onResponse(String response) {
            try {
                JSONArray array=new JSONArray(response);
                for (int i=0; i<array.length(); i++ ){
                    JSONObject ob=array.getJSONObject(i);
                    List_Data listData=new List_Data(ob.getString("name")
                            ,ob.getString("moive"));
                    list_data.add(listData);
                }
                adapter.notifyDataSetChanged();
            } catch (JSONException e) {
                e.printStackTrace();
            }

        }
    }, new Response.ErrorListener() {
        @Override
        public void onErrorResponse(VolleyError error) {

        }
    });
    RequestQueue requestQueue= Volley.newRequestQueue(this);
    requestQueue.add(stringRequest);
}

【讨论】:

    【解决方案3】:

    你可以使用下面的代码

      public class MainActivity extends AppCompatActivity {
        private static final String HI ="http://xxx/phpMyAdmin-5.0.4/Appi.php" ;   //xxx 
           is ip and the user name
        private List<List_Data> list_data;
        private RecyclerView rv;
        private MyAdapter adapter;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            rv=(RecyclerView)findViewById(R.id.recyclerView);
            rv.setHasFixedSize(true);
            rv.setLayoutManager(new LinearLayoutManager(this));
            list_data=new ArrayList<>();
         
            getMovieData();
    
        }
    
        private void getMovieData() {
            StringRequest stringRequest=new StringRequest(Request.Method.GET, HI, new Response.Listener<String>() {
                @Override
                public void onResponse(String response) {
                    try {
                        JSONObject jsonObject=new JSONObject(response);
                        JSONArray array=jsonObject.getJSONArray("data");
                        for (int i=0; i<array.length(); i++ ){
                            JSONObject ob=array.getJSONObject(i);
                            List_Data listData=new List_Data(ob.getString("name")
                                    ,ob.getString("moive"));
                            list_data.add(listData);
                        }
                           setAdapter(listData)
                    } catch (JSONException e) {
                        e.printStackTrace();
                    }
    
                }
            }, new Response.ErrorListener() {
                @Override
                public void onErrorResponse(VolleyError error) {
    
                }
            });
            RequestQueue requestQueue= Volley.newRequestQueue(this);
            requestQueue.add(stringRequest);
        }
    
         public void setAdapter(List<List_Data> list_data){
    
      adapter=new MyAdapter(list_data);
        rv.setAdapter(adapter);
        
    
       }
       }
    

    【讨论】:

    • 感谢您的回复!我试过了,但应用程序仍然运行良好(没有错误),但它是一个空白页面,没有显示 MySQL 数据.....
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2018-06-30
    • 1970-01-01
    • 2019-12-19
    相关资源
    最近更新 更多