我正在尝试在自定义适配器中将文本设置为EditText
,并且正在获取此堆栈:
java.lang.NullPointerException: Attempt to read from field 'android.support.v7.widget.ViewInfoStore android.support.v7.widget.RecyclerView.mViewInfoStore' on a null object reference
at android.support.v7.widget.RecyclerView$LayoutManager.addViewInt(RecyclerView.java:8194)
at android.support.v7.widget.RecyclerView$LayoutManager.addView(RecyclerView.java:8180)
at android.support.v7.widget.RecyclerView$LayoutManager.addView(RecyclerView.java:8168)
at android.support.v7.widget.LinearLayoutManager.layoutChunk(LinearLayoutManager.java:1573)
at android.support.v7.widget.LinearLayoutManager.fill(LinearLayoutManager.java:1519)
at android.support.v7.widget.LinearLayoutManager.onLayoutChildren(LinearLayoutManager.java:614)
at android.support.v7.widget.RecyclerView.dispatchLayoutStep2(RecyclerView.java:3812)
at android.support.v7.widget.RecyclerView.onMeasure(RecyclerView.java:3225)
at android.view.View.measure(View.java:17547)
at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5535)
at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1436)
at android.widget.LinearLayout.measureVertical(LinearLayout.java:722)
at android.widget.LinearLayout.onMeasure(LinearLayout.java:613)
at android.view.View.measure(View.java:17547)
at android.widget.ScrollView.measureChildWithMargins(ScrollView.java:1260)
at android.widget.FrameLayout.onMeasure(FrameLayout.java:436)
at android.widget.ScrollView.onMeasure(ScrollView.java:337)
at android.view.View.measure(View.java:17547)
at android.support.constraint.ConstraintLayout.onMeasure(ConstraintLayout.java:1676)
at android.view.View.measure(View.java:17547)
at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5535)
at android.widget.FrameLayout.onMeasure(FrameLayout.java:436)
at android.support.v7.widget.ContentFrameLayout.onMeasure(ContentFrameLayout.java:141)
at android.view.View.measure(View.java:17547)
at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5535)
at android.support.v7.widget.ActionBarOverlayLayout.onMeasure(ActionBarOverlayLayout.java:400)
at android.view.View.measure(View.java:17547)
at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5535)
at android.widget.FrameLayout.onMeasure(FrameLayout.java:436)
at android.view.View.measure(View.java:17547)
at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5535)
at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1436)
at android.widget.LinearLayout.measureVertical(LinearLayout.java:722)
at android.widget.LinearLayout.onMeasure(LinearLayout.java:613)
at android.view.View.measure(View.java:17547)
at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5535)
at android.widget.FrameLayout.onMeasure(FrameLayout.java:436)
at com.android.internal.policy.impl.PhoneWindow$DecorView.onMeasure(PhoneWindow.java:2615)
at android.view.View.measure(View.java:17547)
at android.view.ViewRootImpl.performMeasure(ViewRootImpl.java:2015)
at android.view.ViewRootImpl.measureHierarchy(ViewRootImpl.java:1173)
at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1379)
at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1061)
at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:5885)
at android.view.Choreographer$CallbackRecord.run(Choreographer.java:767)
at android.view.Choreographer.doCallbacks(Choreographer.java:580)
at android.view.Choreographer.doFrame(Choreographer.java:550)
at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:753)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:135)
at android.app.ActivityThread.main(ActivityThread.java:5254)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
at com.andro
02-21 14:14:29.702 1707-1707/com.android.inputmethod.latin E/RichInputConnection: Unable to connect to the editor to retrieve text.
02-21 14:14:31.705 1707-1707/com.android.inputmethod.latin E/RichInputConnection: Unable to connect to the editor to retrieve text.
02-21 14:14:33.706 1707-1707/com.android.inputmethod.latin E/RichInputConnection: Unable to connect to the editor to retrieve text.
02-21 14:14:35.709 1707-1707/com.android.inputmethod.latin E/RichInputConnection: Unable to connect to the editor to retrieve text.
我不确定是什么问题,我将文本设置为TextView
的方式完全相同,而且效果很好。我看过论坛和文章,也看到有类似问题的人,但似乎他们没有正确地将视图与我的ViewHolder
属性绑定在一起,或者试图访问
这是我的自定义适配器类:
public class ProductListAdapter extends RecyclerView.Adapter<ProductListViewHolder>{
private List<Product> productList;
private ProductItemManager productItemManager;
public ProductListAdapter(List<Product> productList, ProductItemManager productItemManager) {
this.productList = productList;
this.productItemManager = productItemManager;
}
@NonNull
@Override
public ProductListViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View productListView = LayoutInflater.from(parent.getContext())
.inflate(R.layout.adapter_product_list, parent, false);
return new ProductListViewHolder(productListView, productItemManager);
}
@Override
public void onBindViewHolder(@NonNull ProductListViewHolder productListViewHolder, final int position) {
Product product = productList.get(position);
Double finalPrice = product.getSellingPrice() * product.getQuantity();
productListViewHolder.productDescription.setText(product.getDescription());
productListViewHolder.productSellingPrice.setText("un. " + Utils.doubleToReal(product.getSellingPrice()));
productListViewHolder.productFinalPrice.setText(Utils.doubleToReal(finalPrice));
productListViewHolder.productQuantity.setText(Double.toString(product.getQuantity()));
}
@Override
public int getItemCount() {
return productList.size();
}
}
这是我的ViewHolder
public class ProductListViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
public TextView productDescription, productSellingPrice, productFinalPrice;
public ImageView removeProductIcon;
public EditText productQuantity;
private ProductItemManager productItemManager;
public ProductListViewHolder(View itemView, ProductItemManager productItemManager) {
super(itemView);
productDescription = itemView.findViewById(R.id.productDescription);
productSellingPrice = itemView.findViewById(R.id.productSellingPrice);
productFinalPrice = itemView.findViewById(R.id.productFinalPrice);
removeProductIcon = itemView.findViewById(R.id.removeProductIcon);
productQuantity = itemView.findViewById(R.id.productQuantity);
this.productItemManager = productItemManager;
removeProductIcon.setOnClickListener(this);
this.productChangeListener();
}
private void productChangeListener() {
productQuantity.addTextChangedListener(new TextWatcher() {
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
Double quantity = Double.parseDouble(productQuantity.getText().toString());
productItemManager.setProductQuantity(getAdapterPosition(), quantity);
}
@Override
public void afterTextChanged(Editable s) {
}
});
}
@Override
public void onClick(View v) {
productItemManager.removeProduct(getAdapterPosition());
}
}
答案 0 :(得分:3)
查看绑定后,应调用TextWatch上设置的侦听器。在onBindViewHolder
中致电您的注册。
......
@Override
public void onBindViewHolder(@NonNull ProductListViewHolder productListViewHolder, final int position) {
......
productListViewHolder.productChangeListener();
....
我还建议更改注册侦听器的功能。在RecyclerView
创建期间,如果不创建数百个项目。
答案 1 :(得分:2)
有几件事需要引起您的注意:
在RecyclerViews中(通常是列表)具有EditText通常不是一个好主意,因为它们具有UX。您要么必须使用户停止滚动列表,要么基于滚动进行一些操作,以使事情变得更加顺畅(因为他们可能正在输入内容)。
代码中有一个非常重要的问题,它是对正在回收的编辑文本的“ TextWatcher”的引用。回收视图时,您需要删除文本观察器。如果不删除TextWatcher,会发生什么?由于它不像OnClickListener
(它不是set
函数,而是add
函数),因此每次重新创建单元格时,都会使用以下命令将一个新的文本观察器添加到EditText中:它是对旧ViewHolder的引用。
要解决此问题,您需要向ViewHolder中添加一个方法以解除绑定它所保存的任何内容:
public class ProductListViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
public TextView productDescription, productSellingPrice, productFinalPrice;
public ImageView removeProductIcon;
public EditText productQuantity;
private ProductItemManager productItemManager;
private TextWatcher textWatcher;
public ProductListViewHolder(View itemView, ProductItemManager productItemManager) {
super(itemView);
productDescription = itemView.findViewById(R.id.productDescription);
productSellingPrice = itemView.findViewById(R.id.productSellingPrice);
productFinalPrice = itemView.findViewById(R.id.productFinalPrice);
removeProductIcon = itemView.findViewById(R.id.removeProductIcon);
productQuantity = itemView.findViewById(R.id.productQuantity);
this.productItemManager = productItemManager;
removeProductIcon.setOnClickListener(this);
this.productChangeListener();
}
private void productChangeListener() {
textWatcher = new TextWatcher() {
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
Double quantity = Double.parseDouble(productQuantity.getText().toString());
productItemManager.setProductQuantity(getAdapterPosition(), quantity);
}
@Override
public void afterTextChanged(Editable s) {
}
};
productQuantity.addTextChangedListener(textWatcher);
}
public void unbind(){
productQuantity.removeTextChangedListener(textWatcher);
}
@Override
public void onClick(View v) {
productItemManager.removeProduct(getAdapterPosition());
}
}
然后,您可以使用RecyclerView的onViewRecycled
方法来知道何时回收视图并删除TextWatcher:
@Override
public void onViewRecycled(ProductListViewHolder productListViewHolder) {
super.onViewRecycled(holder);
productListViewHolder.unbind();
}
答案 2 :(得分:0)
每次观看的文本更改时,TextWatcher
都会被调用。这不仅意味着用户编辑将调用监视程序(良好),而且每次视图持有者反弹时,都会调用监视程序(效果不佳)。这也意味着productItemManager.setProductQuantity()
将被不必要地调用。 (我说“不必要”,但是,也许这是您打算发生的事情。)
绑定视图持有者时避免文本监视程序的一种方法是在View.OnFocusChangeListener()
上设置productQuantity
,当该字段获得焦点或丢失焦点时,将应用和删除文本监视程序。换句话说,只有在编辑EditText
时,文本观察器才会处于活动状态。
productQuantity.setOnFocusChangeListener(new View.OnFocusChangeListener() {
@Override
public void onFocusChange(View v, boolean hasFocus) {
if (hasFocus) {
productQuantity.addTextChangedListener(textWatcher);
} else {
productQuantity.removeTextChangedListener(textWatcher);
}
}
});
这可能无法解决您遇到的问题,但是它将消除可能是造成问题的不必要工作。
我认为发布setProductQuantity()
可以帮助某人找到问题的根源。
答案 3 :(得分:0)
我想同意@Cheticamp提出的实现。根据我的经验,文本更改侦听器可能在不应该调用的不同情况下触发。 EditText
通常涉及自动更正单词并提供建议,并且可以在初始化期间调用。因此,我建议您将以下内容放入布局文件的EditText
声明中。
android:inputType="textNoSuggestions"
除了使用setOnFocusChangeListener
之外,您还可以考虑像下面一样在文本观察器中进行null
的签入。
private void productChangeListener() {
productQuantity.addTextChangedListener(new TextWatcher() {
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
String editTextContent = productQuantity.getText().toString();
if(editTextContent != null && editTextContent.trim().length() > 0) {
Double quantity = Double.parseDouble(productQuantity.getText().toString());
productItemManager.setProductQuantity(getAdapterPosition(), quantity);
}
}
@Override
public void afterTextChanged(Editable s) {
}
});
}
希望有帮助!
答案 4 :(得分:0)
//update this line in your code, you will get your answer :)
productListViewHolder.productQuantity.setText(Double.toString(productList.get(position).getQuantity()));