【问题标题】:Android ListView.notifyDataSetChanged causing incorrect rowsAndroid ListView.notifyDataSetChanged 导致不正确的行
【发布时间】:2020-02-05 14:47:37
【问题描述】:

我正在开发一个 android 项目,其中 ListFragment 正在从子活动更新。但是,虽然更新后元素的计数是正确的,但元素本身却不是。不正确的元素被放入列表并相互重复。

问题是,子activity修改文件后,调用populate()函数,然后我回到Fragments所在的主activity,问题就出现了。 例如,一开始它们的顺序是:[1, 2, 3]

现在子活动中的文件操作将项目“1”删除(不是直接调用remove,而是删除该文件,populate() 函数(如下所列)将生成一个没有该元素的列表),它应该变成[2, 3]。但它变成了[1, 2]

在一个按钮调用的完整刷新之后,包括清除和重新填充MetadataManager.packMapPackListManager.onlineListPackListManager.localList,然后重新启动MainActivity,它就会恢复正常。

现在我们有[2, 3]。子活动中的另一个文件操作应该返回“1”,它应该再次变为[1, 2, 3]。但是,它变成了[2, 3, 3]

元素来自单例中的 ArrayList。这是 ListFragment 的类:

class PackListFragment : ListFragment() {

    private lateinit var dataList: ArrayList<PackageMetadata>

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        val index = arguments?.getInt(ARG_SECTION_NUMBER) ?: 1
        if (index == 1){
            dataList = PackListManager.onlineList
        } else {
            dataList = PackListManager.localList
        }
    }

    override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        val adapter = PackMetadataAdapter(activity as Context, dataList)
        listAdapter = adapter
        return inflater.inflate(R.layout.fragment_main, container, false)
    }

    override fun onResume() {
        (listAdapter as ArrayAdapter<PackageMetadata>).notifyDataSetChanged()
        super.onResume()
    }

我的 ListView 使用自定义适配器。这是适配器:

class PackMetadataAdapter (context : Context, val values : List<PackageMetadata>)
    : ArrayAdapter<PackageMetadata>(context, -1, values) {

    override fun getView(position: Int, convertView: View?, parent: ViewGroup): View {
        if (convertView == null) {
            val inflater = context
                .getSystemService(Context.LAYOUT_INFLATER_SERVICE) as LayoutInflater
            /*val rowView : View = inflater.inflate(android.R.layout.simple_list_item_1, parent, false)
            val text1 = rowView.findViewById(android.R.id.text1) as TextView
            text1.text = values[position].Name_LO*/
            val metadata = values[position]
            val rowView: View = inflater.inflate(R.layout.listitem_route, parent, false)
            val textTitle = rowView.findViewById<View>(R.id.textTitle) as TextView
            val textAuthor = rowView.findViewById(R.id.textAuthor) as TextView
            val textVersion = rowView.findViewById(R.id.textVersion) as TextView
            val textTimestamp = rowView.findViewById(R.id.textTimestamp) as TextView
            val imageView: ImageView = rowView.findViewById<View>(R.id.imageThumbnail) as ImageView
            textTitle.text = metadata.Name_LO
            textAuthor.text = metadata.Author.Name_LO
            textVersion.text = metadata.Version.get()
            textTimestamp.text = SimpleDateFormat("yyyy-MM-dd", Locale.US)
                .format(values[position].Timestamp)
            DownloadImageTask(imageView).execute(metadata.Thumbnail)
            return rowView
        }
        return convertView
    }

}

单例中的列表是这样填充的:

object PackListManager {
    const val LOGCAT_TAG = "BCSPackListMan"
    val localList : ArrayList<PackageMetadata> = ArrayList()
    var onlineList : ArrayList<PackageMetadata> = ArrayList()

    fun populate(){
        localList.clear(); onlineList.clear()
        val localPacks = HashMap(PackLocalManager.getLocalPacks().map {
            val parts = stripExtension(it.nameWithoutExtension).split("_")
            if (parts.count() > 1) {
                parts[0] to Version(parts[1])
            } else {
                it.nameWithoutExtension to Version("0.0")
            }
        }.toMap())
        for (pack in MetadataManager.packMap){
            if (localPacks.containsKey(pack.key)){
                Log.i(LOGCAT_TAG, "Pack "+pack.key+" found on local disk")
                if (localPacks[pack.key]!! < pack.value.Version){
                    Log.i(LOGCAT_TAG, "Pack "+pack.key+" can be updated")
                }
                localPacks.remove(pack.key)
                localList.add(pack.value)
            } else {
                Log.i(LOGCAT_TAG, "Pack "+pack.key+" not installed")
                onlineList.add(pack.value)
            }
        }
    }

通过查看logcat,我可以说polulate()函数生成的列表是正确的,而如果没有调用上述手动刷新,则MetadataManager.packMap shuold保持不变。

我在谷歌上搜索过这样的问题,但我似乎找不到一个好的答案。我真的很抱歉发布这么长的代码,但我自己真的无法弄清楚。任何帮助将不胜感激!

顺便说一句,我不是母语人士,因此对于帖子中的错误深表歉意。

【问题讨论】:

  • 不要使用ListView,因为它已经过时了,而不是这个试试RecyclerView
  • @KostyaBakay 谢谢!将 ListView 替换为 RecyclerView 后,它立即生效!
  • 不客气 :) 我将添加我的评论作为答案

标签: java android kotlin android-listview


【解决方案1】:

不要使用ListView,因为它已经过时了,试试RecyclerView

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2011-11-16
    • 2011-04-21
    • 1970-01-01
    • 2014-02-05
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多