- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个显示包含多个类别项目的垂直 RecyclerView 列表的 Activity 。此列表中的每个项目都是一个类别,显示“类别标题”及其对应的“图像 ID”。持有此 RecyclerView 的 Activity 是用户在移动设备上启动应用程序后出现的第一个 Activity 。此外,项目类别信息是通过 JSON 获取的,我在其中使用“Volley”库从 Web 上的 URL 获取数据。我的问题是我的 RecyclerView 显示未填充且为空,即使在网络上完美获取信息也是如此。我怀疑这是由于以下原因造成的:
在 JSON 请求完成后,如何更新 RecyclerView 以显示所有项目?我听说在 RecyclerView 适配器中使用 notifyDataSetChanged(),但我不知道如何使用此功能,最重要的是不知道将它放在我的代码中的什么位置。
下面是我的代码。非常感谢任何有关如何在这些情况下执行 RecyclerView 更新的详细帮助。
Activity
package example.co.uk.vapp;
import android.app.ProgressDialog;
import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.View;
import android.widget.Toast;
import com.android.volley.Response;
import com.android.volley.VolleyError;
import com.android.volley.toolbox.JsonArrayRequest;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import java.util.ArrayList;
import java.util.List;
import example.co.uk.vapp.adapters.GroupAdapter;
import example.co.uk.vapp.utils.RecyclerItemClickListener;
public class CategoryListActivity extends AppCompatActivity {
// Declare variables
private RecyclerView mRecyclerView;
private GridLayoutManager mGridLayoutManager;
// Tag for logging possible errors onto the Log
private static String TAG = CategoryListActivity.class.getSimpleName();
// Progress dialog
private ProgressDialog pDialog;
// JSON Array response url - ****fictitious url****
private static final String URL = "https://www.someurl.com";
// JSON Array containing the response
static JSONArray jsonArrayResponse;
// Json Object containing the category fields such as title and image-id
static JSONObject category;
// ArrayList containing the category titles
static ArrayList<String> categoryTitles;
// ArrayList containing the category image ids
private ArrayList<String> categoryImageId;
static GroupAdapter groupAdapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// Set layout
setContentView(R.layout.activity_category_list);
// Initialize ArrayList to contain category titles
categoryTitles = new ArrayList<String>();
categoryImageId = new ArrayList<String>();
// Reference variable with layout view
mRecyclerView = (RecyclerView) findViewById(R.id.cardList);
// Use this setting to improve performance if you know that changes
// in content do not change the layout size of the RecyclerView
mRecyclerView.setHasFixedSize(true);
// Display under one column
mGridLayoutManager = new GridLayoutManager(this, 1);
mRecyclerView.setLayoutManager(mGridLayoutManager);
// Set orientation
mGridLayoutManager.setOrientation(GridLayoutManager.VERTICAL);
mRecyclerView.setLayoutManager(mGridLayoutManager);
// Create list of "Group" type
final List<Group> groups = new ArrayList<Group>();
for (int i = 0; categoryTitles.size() < i && categoryImageId.size() < i; i++) {
String categoryName = categoryTitles.get(i);
int categoryImage = Integer.getInteger(categoryImageId.get(i));
groups.add(new Group(categoryName, categoryImage));
}
// Set Adapter
groupAdapter = new GroupAdapter(groups);
mRecyclerView.setAdapter(groupAdapter);
// Create click listener for each item on the list
mRecyclerView.addOnItemTouchListener(
new RecyclerItemClickListener(this, new RecyclerItemClickListener.OnItemClickListener() {
@Override
public void onItemClick(View view, int position) {
Intent intent = new Intent(CategoryListActivity.this, groups.get(position).getClass());
startActivity(intent);
}
})
);
// Shows message to user while makeJsonObjectRequest() is still running
pDialog = new ProgressDialog(this);
pDialog.setMessage("Getting exchange rates...");
pDialog.setCancelable(false);
// Retrieves JSON format exchange rates from Yahoo Finance
makeJsonArrayRequest();
}
private void makeJsonArrayRequest() {
// Show dialog while the request is made
showpDialog();
final JsonArrayRequest jsonArrayReq = new JsonArrayRequest(URL,
new Response.Listener<JSONArray>() {
@Override
public void onResponse(JSONArray response) {
// Log response
Log.d(TAG, response.toString());
// Set "response" to jsonArrayResponse global variable so that we can use it on other methods in this class
jsonArrayResponse = response;
try {
// Get array with all categories (raw)
for (int i = 0; i < jsonArrayResponse.length(); i++) {
category = jsonArrayResponse.getJSONObject(i);
String title = category.getString("category-localized-title");
categoryTitles.add(title);
String imageId = category.getString("product-category-image-id");
categoryImageId.add(imageId);
}
} catch (JSONException e) {
e.printStackTrace();
Toast.makeText(gevapplicationContext(),
"Error: " + e.getMessage(),
Toast.LENGTH_LONG).show();
}
for (int i = 0; i < categoryTitles.size(); i++) {
Log.i("categoryTitles", categoryTitles.get(i).toString());
Log.i("categoryImageIds", categoryImageId.get(i).toString());
}
// Hide dialog after information has been requested
hidepDialog();
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) {
// Warn user
Toast.makeText(gevapplicationContext(),
"No internet connection", Toast.LENGTH_LONG).show();
// Log error
Log.e("test", error.getMessage());
// hide the progress dialog
hidepDialog();
}
});
// Adding request to request queue
AppController.getInstance().addToRequestQueue(jsonArrayReq);
}
/**
* Method for showing dialog
*/
private void showpDialog() {
if (!pDialog.isShowing())
pDialog.show();
}
/**
* Method for hiding dialog
*/
private void hidepDialog() {
if (pDialog.isShowing())
pDialog.dismiss();
}
}
RecyclerView 适配器
package example.co.uk.vapp.adapters;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;
import java.util.List;
import example.co.uk.vapp.Group;
import example.co.uk.vapp.R;
public class GroupAdapter extends RecyclerView.Adapter<GroupAdapter.GroupViewHolder> {
private List<Group> groupList;
public GroupAdapter(List<Group> groupList) {
this.groupList = groupList;
}
@Override
public int getItemCount() {
return groupList.size();
}
@Override
public void onBindViewHolder(GroupViewHolder holder, int position) {
Group group = groupList.get(position);
holder.vGroupTitle.setText(group.getGroupTitle());
holder.vGroupImage.setImageResource(group.getGroupImage());
}
@Override
public GroupViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View itemView = LayoutInflater.from(parent.getContext()).inflate(R.layout.group_list_card_layout, parent, false);
return new GroupViewHolder(itemView);
}
// Create the ViewHolder class "pattern"
public static class GroupViewHolder extends RecyclerView.ViewHolder {
protected TextView vGroupTitle;
protected ImageView vGroupImage;
public GroupViewHolder(View v) {
super(v);
vGroupTitle = (TextView) v.findViewById(R.id.title);
vGroupImage = (ImageView) v.findViewById(R.id.image);
}
}
}
最佳答案
在您的适配器中,声明一个方法,一旦您获取了新的或初始数据集,您将使用该方法更新适配器。它应该看起来像这样:
public void swapDataSet(list<Group> newData){
this.groupList = newData;
//now, tell the adapter about the update
notifyDataSetChanged();
}
上述方法可行,但它不一定是后续更改的最有效方法,因为假设整个数据集已更改,适配器将更新所有项目。首次设置适配器后,最好使用更具体的更改通知,如 notifyItemInsered(int pos)
、notifyItemDeleted(int pos)
等,如果可能的话。通过这种方式,您无需付出额外的努力也可以获得漂亮的动画。
更具体地说,在您的情况下,您在有任何标题或类别之前更新组。为什么不这样做:
for (int i = 0; i < jsonArrayResponse.length(); i++) {
category = jsonArrayResponse.getJSONObject(i);
String title = category.getString("category-localized-title");
String imageId = category.getString("product-category-image-id");
categories.add(new Group(title, imageId));
}
//now that you have fresh groups
GroupAdapter.swapDataSet(groups);
关于android - 如何更新 RecyclerView - notifyDataSetChanged(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33726125/
我正在尝试在我的 recyclerview 方法上实现此方法 Changing background color of selected item in recyclerview 我正确地遵循了脚本,
我有一个列表,其中填充了自定义 ArrayAdapter。 我想每秒更新一个进度条,所以我尝试使用 notifyDataSetChanged() 来重新绘制列表,但它似乎不起作用。 最佳答案 您如何更
我正在改进我的应用程序稳定性和性能,但现在我遇到了来自 Android Studio 的警告。请考虑以下适配器类: private class CoinsAdapter(private val fra
我正在制作一个Android来做名为list track的列表应用程序,现在我仍在研究该应用程序的基础知识。无论如何,我为列表 View 创建了一个上下文菜单,该上下文菜单只有1个选项“删除”,它的意
我想从另一个线程调用adapter.notifyDataSetChanged()。我读到我应该使用 AsyncTask 并在执行后执行 adapter.notifyDataSetChanged() 。
我知道这个问题已被问过数千次,但由于某种原因我找不到适合我的情况的答案。 我拥有的是一个从 Web 服务获取数据并用信息填充列表的线程。然后我想按下一个按钮并调用同一线程来获取更多数据并将其添加到列表
friend , 我正在为 android ListView 使用以下自定义适配器我面临的问题是调用 notifydatasetchanged() 后位置错误; 我第一次将数据与列表位置绑定(bind
我在 windows 7 64bit 下使用 android 1.6当我从扩展 BaseAdapter 的适配器对象调用 notifyDataSetChanged() 时,我得到了运行时异常 我尝试使
我有一个类 Myadapter extends BaseAdapter.in Myadapter 我更改了 MyListView 数据然后调用 notifyDataSetChanged() 方法,比如
我一直在寻找年龄并尝试了几个月。我永远无法让我的观点(任何观点)无效。这里有什么问题,任何人都可以告诉我吗?谁能告诉我如何使用这些无效方法。提前致谢! if(resul
在我的 Activity 中,我有两个 ListView ,因此我使用了两个不同的适配器。我的要求是在两个列表项中我都有一个按钮。单击任何 ListView 中的按钮,两个 ListView 中的数据
我有一个用于我的 ListView 的适配器,然后我需要使用 SensorChange 方法中的 notifydatasetchanged 方法刷新它。你知道 SensorChange 方法被频繁快速
我正在使用 FragmentStatePagerAdapter 来实现 VerticalPageAdapter 的目的,我必须为与 Button< 相关的相同 Fragment 创建动态编号 被按下,
这个问题在这里已经有了答案: Custom Adapter getView() method is not called (6 个答案) 关闭 7 年前。 这是我的代码我想在 ListView 中显
我正在尝试做的事情: 按下添加按钮时,将值从 EditText 插入到数据库。 用数据库中的数据填充 ListView 。 问题: 当我单击添加按钮时,它会将值插入到数据库中,但不会将其填充到 Lis
我正在尝试制作一个像 tinder 一样刷卡的应用程序。我有一个 Texview 上面有一些文字,当我刷下一张卡片时有另一个文字等等。我还想做的是当我单击一个按钮以删除屏幕上的卡片并转到下一张时。我正
我正在制作一个程序,该程序将寻找与手机蓝牙设备配对的程序,显示它们及其地址以供用户查看,并且当用户已经在他的手机中打开 Bt 时它工作正常。当他不这样做时会出现问题,导致添加到 arrayList 中
有人可以向我解释以下问题的来源: 我有一个列表,根据某些偏好显示多个项目。用户可以通过 SlidingDrawer 更改这些首选项。每当发生这种情况时,我都会获得与新首选项相对应的项目,然后使用以下代
我有一个 ListView ,它使用不同类中的基本适配器显示来 self 的数据库的多个数据。每次我删除或添加一个项目到我的数据库时,我都想刷新我的 ListView 。 这是我的代码,但它没有刷新我
在我的代码中,我有两个 Activity 。 一个是 ListView Activity ,另一个 Activity 用于修改数据。修改数据后,用户将返回到 ListView Activity 。 我
我是一名优秀的程序员,十分优秀!