【问题标题】:RecyclerView Items' values reset when scrolling downRecyclerView 项目的值在向下滚动时重置
【发布时间】:2018-05-14 10:57:54
【问题描述】:

这是我正在尝试做的场景:我正在创建一个 RecyclerView,它可以添加或删除一个或多个具有 EditText 的子项/片段/视图但是......

这是我的问题:每当我在 RecyclerView 中向下滚动时,它的项目值都会重置。我认为这里的问题是 RecyclerView 适配器无法绑定或保存这些值,但我不知道我的代码中的问题到底出在哪里。

class StocksAdapter() : RecyclerView.Adapter<ViewHolder>() {


class StockFragmentHolder : ViewHolder {
    fun Double.format(digits: Int) = java.lang.String.format("%,.${digits}f", this)


    lateinit var numberOfShares: EditText;
    lateinit var buyPrice: EditText;
    lateinit var btnCompute: Button
    lateinit var btnRemove: Button
    lateinit var stockAveragePrice: TextView
    lateinit var stockTotalAmount: TextView
    private var savedStock : Stock? = null

    constructor(view: View) : super(view) {
        numberOfShares = view.findViewById(R.id.stockNumberOfShares)
        buyPrice = view.findViewById(R.id.stockBuyPrice)
        btnCompute = view.findViewById(R.id.btnCompute)
        btnRemove = view.findViewById(R.id.btnRemove)
        stockAveragePrice = view.findViewById(R.id.stockAveragePrice)
        stockTotalAmount = view.findViewById(R.id.stockTotalAmount)
    }


    fun populateView(stock: Stock, onStocksAdapterListener: OnStocksAdapterListener, position: Int) {

        if(savedStock == null || stock.buyPrice > 0 || stock.numberOfShares > 0) {
            savedStock = stock
        }

        this.buyPrice.setText(if (savedStock!!.buyPrice <= 0.0) "" else savedStock!!.buyPrice.toString())
        this.numberOfShares.setText(if (savedStock!!.numberOfShares <= 0.0) "" else savedStock!!.numberOfShares.toString())
        this.stockAveragePrice.text = "0"
        this.stockTotalAmount.text = "0"

        this.btnCompute.setOnClickListener(View.OnClickListener {

            var stock = Stock();

            stock.numberOfShares = this.numberOfShares.text.toString().toLongOrNull()
            stock.buyPrice = this.buyPrice?.text.toString().toDoubleOrNull()
            stock.sellPrice = 0.0


            if (stock.numberOfShares != null || stock.numberOfShares != 0L) {

                var buyTotalAmount: Double = 0.0;
                buyTotalAmount = StocksCalculator.calculateTotalSharesPrice(stock, StocksCalculator.TRANSACTION_FEE_BASE_VALUES, Constants.TRANSACTION_TYPE.TRANSACTION_TYPE_BUY)

                var averagePricePerShare = (buyTotalAmount / stock.numberOfShares)

                this.stockAveragePrice.text = averagePricePerShare.format(2);
                this.stockTotalAmount.text = "" + buyTotalAmount.format(2)

                onStocksAdapterListener.onStocksComputeButtonClicked(stock.numberOfShares, averagePricePerShare, buyTotalAmount, position)
            }
        })

        this.btnRemove.setOnClickListener(View.OnClickListener {
            onStocksAdapterListener.onStocksRemoveButtonClicked(position)
        })

    }
}


lateinit var onStocksAdapterListener: OnStocksAdapterListener;
lateinit var listStocks: ArrayList<Stock>

public constructor(listStocks: ArrayList<Stock>, onStocksAdapterListener: OnStocksAdapterListener) : this() {
    this.listStocks = listStocks
    this.onStocksAdapterListener = onStocksAdapterListener
    setHasStableIds(true)

}

override fun onCreateViewHolder(parent: ViewGroup?, viewType: Int): RecyclerView.ViewHolder {
    var itemView = LayoutInflater.from(parent?.getContext())
            .inflate(R.layout.fragment_stock, parent, false)
}

override fun getItemViewType(position: Int): Int {
    return position
}


override fun getItemCount(): Int {
    return listStocks.size
}

override fun onBindViewHolder(holder: RecyclerView.ViewHolder?, position: Int) {
    var stock: Stock = listStocks.get(position)
    holder.populateView(stock, this.onStocksAdapterListener, position)
}


public interface OnStocksAdapterListener {
    public fun onStocksAddButtonClicked()

    public fun onStocksComputeButtonClicked(numberOfShares: Long?, averagePricePerShare: Double?, averageTotalAmount: Double?, position: Int)

    public fun onStocksRemoveButtonClicked(position: Int)
}

这是我在 Fragment 中创建 recyclerView 的方法:

    val mLayoutManager = LinearLayoutManager(this.context)
    var recyclerView: RecyclerView = view.findViewById(R.id.recyclerView)

    val itemDecorator = DividerItemDecoration(context, DividerItemDecoration.VERTICAL)
    itemDecorator.setDrawable(ContextCompat.getDrawable(context, R.drawable.divider))
    recyclerView.addItemDecoration(itemDecorator)
    recyclerView.setLayoutManager(mLayoutManager)
    recyclerView.setItemAnimator(DefaultItemAnimator())
    recyclerView.adapter = stocksAdapter

我已尝试禁用 ViewHolder 的 setIsRecyclable,但问题仍然存在。

防止 RecyclerView 停止重置值的最佳方法是什么?还是应该用 ListView 替换 RecyclerView?

【问题讨论】:

  • 尝试删除 if(savedStock == null || stock.buyPrice > 0 || stock.numberOfShares > 0) { savedStock = stock } 并且只有 savedStock = stock
  • @SahilManchanda 嗨,我已经做到了。问题依然存在:(
  • 当你说它重置值是什么意思?
  • @Eminem EditText 值重置为默认值 0 时它们不应该是因为我已经将那里的值设置为无 0。我认为这是因为 RecyclerView 已经销毁了最顶层的项目,这就是为什么有值被重置

标签: android kotlin android-recyclerview


【解决方案1】:

我找到了一种解决方法,方法是增加 recyclerview 的缓存大小。只需确保缓存大小对您的应用程序来说足够好即可。

代码:

recyclerView.setItemViewCacheSize(int);

【讨论】:

    【解决方案2】:

    正如我在您的代码中看到的那样,您正试图在活动加载后添加一个项目。因此,当您再次滚动活动时,它会再次调用 onCreate 活动,并且再次从存储或网络加载数据,因此请以特定方法或 OnActiviyt Result 方法加载数据。

    【讨论】:

    • 您好,我已经在fragment的onCreateView中初始化了数据和recycler。我已经弄清楚为什么子项目会重置,因为回收站视图在添加新项目后已经回收了以前的项目。我找到了解决方法,增加 recyclerView.setItemViewCacheSize(int); 的大小确保recyclerview不会销毁之前的物品
    • 只要在recyclerView中添加一个item,创建后或者使用任何方法,app就不能再次自动加载数据。
    【解决方案3】:

    我不知道在你的代码中什么是savedStock 或者你用它做什么。但是您的 populateView 方法无法按预期工作。

    如果你只使用这个savedStock进行渲染,请将其完全移除,否则,不要在populateView中使用它。

    populateView 中依赖于根据你获得的位置传递的stock 来绘制值。

    你的代码应该是这样的

    this.buyPrice.setText(if (stock!!.buyPrice <= 0.0) "" else stock!!.buyPrice.toString())
    this.numberOfShares.setText(if (stock!!.numberOfShares <= 0.0) "" else stock!!.numberOfShares.toString())
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2015-06-10
      • 2022-01-01
      • 2015-01-20
      • 1970-01-01
      • 1970-01-01
      • 2016-06-19
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多