将两个不同的适配器合并到一个ListView中 (Java Android)

将两个不同的适配器合并到一个listview中 (java android)

本文介绍了如何在Android应用中将两个不同数据源的适配器合并到一个ListView中显示。通过创建自定义适配器,利用不同的ViewHolder来处理不同类型的数据,可以实现将多个数据源整合到一个ListView中展示的需求。本文将提供详细的步骤和代码示例,帮助开发者轻松实现这一功能。

在Android开发中,有时我们需要在一个ListView中展示来自不同数据源的数据。直接使用两个不同的Adapter是不行的,我们需要创建一个自定义的Adapter来处理这种情况。以下是一种实现方案,它使用了ViewHolder模式,并且可以轻松地扩展到更多的数据源。虽然RecyclerView在性能和灵活性上更胜一筹,但ListView在某些场景下仍然适用,因此理解这种合并适配器的方法仍然很有价值。

1. 创建数据模型类

首先,定义代表不同数据类型的数据模型类。例如,基于问题描述,我们有两个数据源:customerLoanApplicationList 和 customerLoanList。我们需要定义相应的数据模型类:

public class LoanApplication {    // LoanApplication 的属性,例如:    private String applicationId;    private String applicantName;    // 构造函数、getter 和 setter 方法    public LoanApplication(String applicationId, String applicantName) {        this.applicationId = applicationId;        this.applicantName = applicantName;    }    public String getApplicationId() {        return applicationId;    }    public String getApplicantName() {        return applicantName;    }}public class Loan {    // Loan 的属性,例如:    private String loanId;    private double loanAmount;    // 构造函数、getter 和 setter 方法    public Loan(String loanId, double loanAmount) {        this.loanId = loanId;        this.loanAmount = loanAmount;    }    public String getLoanId() {        return loanId;    }    public double getLoanAmount() {        return loanAmount;    }}

2. 创建自定义适配器

立即学习“Java免费学习笔记(深入)”;

接下来,创建一个自定义的Adapter,它将处理两种类型的数据。我们需要定义不同的ViewHolder来对应不同的数据类型。

import android.content.Context;import android.view.LayoutInflater;import android.view.View;import android.view.ViewGroup;import android.widget.BaseAdapter;import android.widget.TextView;import java.util.List;public class CombinedAdapter extends BaseAdapter {    private Context context;    private List data; // 使用 Object 类型来存储不同类型的数据    private LayoutInflater inflater;    private static final int TYPE_LOAN_APPLICATION = 0;    private static final int TYPE_LOAN = 1;    private static final int TYPE_MAX_COUNT = TYPE_LOAN + 1;  // 适配器支持的类型数量    public CombinedAdapter(Context context, List data) {        this.context = context;        this.data = data;        this.inflater = LayoutInflater.from(context);    }    @Override    public int getCount() {        return data.size();    }    @Override    public Object getItem(int position) {        return data.get(position);    }    @Override    public long getItemId(int position) {        return position;    }    @Override    public int getItemViewType(int position) {        Object item = getItem(position);        if (item instanceof LoanApplication) {            return TYPE_LOAN_APPLICATION;        } else if (item instanceof Loan) {            return TYPE_LOAN;        }        return -1; // 或者抛出异常,表明不支持的数据类型    }    @Override    public int getViewTypeCount() {        return TYPE_MAX_COUNT;    }    @Override    public View getView(int position, View convertView, ViewGroup parent) {        ViewHolder holder = null;        int type = getItemViewType(position);        if (convertView == null) {            switch (type) {                case TYPE_LOAN_APPLICATION:                    convertView = inflater.inflate(R.layout.item_loan_application, parent, false);                    holder = new LoanApplicationViewHolder();                    ((LoanApplicationViewHolder) holder).applicationId = convertView.findViewById(R.id.text_application_id);                    ((LoanApplicationViewHolder) holder).applicantName = convertView.findViewById(R.id.text_applicant_name);                    break;                case TYPE_LOAN:                    convertView = inflater.inflate(R.layout.item_loan, parent, false);                    holder = new LoanViewHolder();                    ((LoanViewHolder) holder).loanId = convertView.findViewById(R.id.text_loan_id);                    ((LoanViewHolder) holder).loanAmount = convertView.findViewById(R.id.text_loan_amount);                    break;            }            if(holder != null) {                convertView.setTag(holder);            }        } else {            holder = (ViewHolder) convertView.getTag();        }        if (holder instanceof LoanApplicationViewHolder) {            LoanApplicationViewHolder loanApplicationViewHolder = (LoanApplicationViewHolder) holder;            LoanApplication loanApplication = (LoanApplication) getItem(position);            loanApplicationViewHolder.applicationId.setText(loanApplication.getApplicationId());            loanApplicationViewHolder.applicantName.setText(loanApplication.getApplicantName());        } else if (holder instanceof LoanViewHolder) {            LoanViewHolder loanViewHolder = (LoanViewHolder) holder;            Loan loan = (Loan) getItem(position);            loanViewHolder.loanId.setText(loan.getLoanId());            loanViewHolder.loanAmount.setText(String.valueOf(loan.getLoanAmount()));        }        return convertView;    }    // ViewHolder 接口    private interface ViewHolder {}    // LoanApplication 的 ViewHolder    private static class LoanApplicationViewHolder implements ViewHolder{        TextView applicationId;        TextView applicantName;    }    // Loan 的 ViewHolder    private static class LoanViewHolder implements ViewHolder{        TextView loanId;        TextView loanAmount;    }}

在这个Adapter中,我们定义了两个内部类 LoanApplicationViewHolder 和 LoanViewHolder,分别对应 LoanApplication 和 Loan 数据类型。getItemViewType() 方法根据数据类型返回不同的类型值,getViewTypeCount() 返回支持的类型总数。在 getView() 方法中,我们根据类型值来创建或复用相应的ViewHolder,并设置数据。

3. 定义布局文件

琅琅配音 琅琅配音

全能AI配音神器

琅琅配音 208 查看详情 琅琅配音

确保你已经创建了 item_loan_application.xml 和 item_loan.xml 布局文件,它们分别对应 LoanApplication 和 Loan 数据类型的显示。

item_loan_application.xml:

        

item_loan.xml:

        

4. 使用适配器

在Activity或Fragment中,创建数据列表,并将数据添加到ListView中。

import android.os.Bundle;import android.widget.ListView;import androidx.appcompat.app.AppCompatActivity;import java.util.ArrayList;import java.util.List;public class MainActivity extends AppCompatActivity {    private ListView listView;    private CombinedAdapter adapter;    private List dataList;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        listView = findViewById(R.id.list_view);        dataList = new ArrayList();        // 添加 LoanApplication 数据        dataList.add(new LoanApplication("APP001", "John Doe"));        dataList.add(new LoanApplication("APP002", "Jane Smith"));        // 添加 Loan 数据        dataList.add(new Loan("LOAN001", 10000.0));        dataList.add(new Loan("LOAN002", 20000.0));        adapter = new CombinedAdapter(this, dataList);        listView.setAdapter(adapter);    }}

确保你已经创建了 activity_main.xml 布局文件,其中包含一个ListView。


注意事项和总结

ViewHolder模式: 使用ViewHolder模式可以避免重复查找View,提高ListView的滚动性能。数据类型判断: getItemViewType() 方法是关键,它决定了每个Item使用哪个ViewHolder和布局。Object 类型: List 允许存储不同类型的数据,但需要进行类型转换。布局文件: 确保为每种数据类型创建对应的布局文件。RecyclerView: 虽然本文展示了ListView的解决方案,但在新项目中,推荐使用RecyclerView,它提供了更好的性能和灵活性。RecyclerView同样可以使用多ViewHolder来实现类似的功能,而且更加高效。错误处理: 在getItemViewType中,如果遇到未知的数据类型,应该返回一个默认值或者抛出异常,避免程序崩溃。

通过以上步骤,你就可以成功地将两个不同数据源的适配器合并到一个ListView中显示了。这种方法可以灵活地处理多种数据类型,并且可以方便地扩展到更多的数据源。记住,虽然ListView在某些场景下仍然适用,但在新项目中,推荐使用RecyclerView。

以上就是将两个不同的适配器合并到一个ListView中 (Java Android)的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/344564.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月5日 19:49:27
下一篇 2025年11月5日 19:50:51

相关推荐

  • 如何利用 Service Worker 实现可靠的离线应用和资源缓存?

    Service Worker 是实现 Web 应用离线可用的核心,通过注册并激活代理、缓存关键资源、拦截请求返回缓存内容,并在更新时清理旧缓存,确保离线体验稳定可靠。 要让 Web 应用在离线状态下依然可用,Service Worker 是核心工具。它充当浏览器与网络之间的代理,能拦截请求并返回缓存…

    2025年12月20日
    000
  • 如何利用 JavaScript 实现一个简单的区块链数据结构?

    区块包含索引、时间戳、数据、前一区块哈希和自身哈希,通过SHA-256计算;2. 区块链由区块链接构成,首块为创世块;3. 验证链完整性需检查每块哈希及前后连接一致性。 要实现一个简单的区块链数据结构,核心是理解区块链的基本组成:区块、哈希计算和链式连接。下面用 JavaScript 实现一个基础版…

    2025年12月20日
    000
  • 如何实现一个虚拟列表(Virtual List)以优化长列表渲染性能?

    虚拟列表通过只渲染可视区域元素来减少DOM数量,提升长列表性能。1. 核心原理:根据滚动位置、容器尺寸和项高度计算可见范围,用占位模拟整体高度并复用节点。2. 实现结构:定义itemHeight、containerHeight、scrollTop等变量,通过Math.floor(scrollTop …

    2025年12月20日
    000
  • 如何利用 JavaScript 的 Service Worker 实现离线可用的 Web 应用?

    Service Worker通过拦截请求和缓存资源实现离线访问,需在HTTPS环境下注册sw.js文件;安装时预缓存核心资源,激活后采用缓存优先策略响应请求,并在版本更新时清理旧缓存,从而提升Web应用的离线可用性。 要让 Web 应用在离线状态下依然可用,Service Worker 是关键。它是…

    2025年12月20日
    000
  • JavaScript中的CSS Houdini如何突破样式限制?

    CSS Houdini通过Typed OM、自定义属性、Paint Worklet和Animation Worklet让JS深度参与样式布局,1. Typed OM提升样式操作安全性与计算能力;2. 自定义属性支持继承与动画;3. Paint Worklet实现高性能动态绘图;4. Animatio…

    2025年12月20日
    000
  • 如何实现一个基于 JavaScript 的简单解释器或领域特定语言?

    答案:实现一个基于JavaScript的DSL需经历词法分析、语法分析、构建AST和解释执行四步。首先通过tokenizer将源码拆分为token流,如标识符、数字、操作符;接着parser利用递归下降法根据语法规则生成AST,处理表达式优先级与赋值结构;最后evaluate函数遍历AST,在环境对…

    2025年12月20日
    000
  • 如何利用机器学习库(如TensorFlow.js)在浏览器中运行AI模型?

    答案:在浏览器中运行AI模型需将模型转为TensorFlow.js格式,通过异步加载、输入预处理和predict推理实现,结合WebGL加速与内存优化提升性能。 在浏览器中运行AI模型已成为前端智能化的重要方向,借助TensorFlow.js这类机器学习库,开发者可以直接在网页中加载和执行训练好的模…

    2025年12月20日
    000
  • 使用 CSS Grid 实现父容器高度自适应子内容高度

    本文介绍如何使用 CSS Grid 布局来实现父容器的高度自适应其子内容的高度,即使子内容可能超出视口范围,且无需使用 JavaScript。通过将父容器设置为 Grid 容器,并将背景和文本内容都放置在同一行和列中,可以轻松实现背景高度与文本内容高度一致的效果。 使用 CSS Grid 实现高度自…

    2025年12月20日
    000
  • 解决点击按钮时元素跳动问题的CSS对齐技巧

    本文探讨了在网页开发中,点击按钮时元素发生跳动的问题,特别是当按钮状态切换导致CSS属性(如border-style和padding)变化时。通过深入分析其根本原因——内联元素基线对齐和盒模型变化,文章提供了使用vertical-align: middle;这一CSS属性的解决方案,确保按钮及其周围…

    2025年12月20日
    000
  • 如何利用JavaScript操作二进制数据,如ArrayBuffer和Blob?

    ArrayBuffer是二进制数据容器,需通过TypedArray或DataView视图操作;Blob用于封装文件类二进制对象,可与ArrayBuffer互转,适用于文件传输与网络通信。 JavaScript 提供了多种方式来操作二进制数据,特别是在处理文件、网络传输、图像音频处理等场景中,Arra…

    2025年12月20日
    000
  • ElectronJS中渲染进程调用主进程多线程函数的IPC实践

    ElectronJS教程:本文详细介绍了如何在ElectronJS应用中,通过渲染进程安全有效地调用主进程中包含多线程逻辑的函数。核心在于利用Electron的ipcRenderer和ipcMain模块建立进程间通信,允许渲染进程发送请求,主进程接收并执行基于threads.js的异步任务,从而实现…

    2025年12月20日
    000
  • 使用 CSS Grid 实现父容器高度与内容自适应

    本文介绍如何使用 CSS Grid 布局,在不使用 JavaScript 的情况下,使父容器的高度自动适应其内容的高度,即使内容超出视口也能正确撑开父容器。通过将父容器设置为 Grid 布局,并将其子元素置于同一行和列,可以轻松实现背景与内容高度的完美匹配。 在网页开发中,经常会遇到需要让背景元素的…

    2025年12月20日
    000
  • 如何用Web Locks API管理资源并发访问?

    Web Locks API 是一种浏览器提供的机制,通过互斥锁协调同源下页面与 Worker 对共享资源的访问。它不锁定硬件资源,而是提供逻辑同步,确保关键代码串行执行,避免竞态条件。核心方法为 navigator.locks.request(lockName, options?, callback…

    2025年12月20日
    000
  • Highcharts浮动条形图(范围条形图)实现指南

    本教程详细介绍了如何在Highcharts中创建浮动条形图,即水平方向的范围条形图。通过利用Highcharts的bar系列类型,并结合数据点的y(高值)和low(低值)属性,可以轻松实现指定左右边界的条形图效果,克服了columnrange仅支持垂直方向的限制。 理解浮动条形图的需求 在数据可视化…

    2025年12月20日
    000
  • Nightwatch.js中高效管理元素选择器:告别重复定义

    本教程探讨Nightwatch.js中避免重复使用元素选择器的方法。针对在同一元素上执行多项操作时选择器冗余的问题,文章提供了两种核心解决方案:通过常量变量复用选择器,以及利用页面对象(Page Objects)进行集中管理。同时,教程也解释了Nightwatch.js与Cypress在命令链式调用…

    2025年12月20日
    000
  • iFrame交互后页面滚动位置自动恢复教程:利用URL变化监听与自定义事件

    本教程旨在解决iFrame内操作导致主页面URL更新及滚动位置重置的问题。我们将探讨从最初的基于加载事件的尝试,到通过实时监控URL变化来触发滚动,最终引入更优雅的自定义事件和哈希变化监听机制,确保用户在iFrame内容更新后能自动回到正确视图,提升用户体验。 理解问题:iFrame交互与页面滚动重…

    2025年12月20日
    000
  • 优化React表单输入与API请求:useEffect的正确姿势与提交处理

    本文深入探讨了React函数组件中处理表单输入与API请求的常见陷阱,特别是useEffect钩子的不当使用。我们将通过一个实际案例,详细阐述useEffect应如何正确放置、表单如何有效提交以及如何将用户输入与API调用逻辑无缝集成,以避免不必要的页面刷新和渲染问题,确保数据正确加载。 引言 在r…

    2025年12月20日
    000
  • 如何编写跨浏览器兼容的现代ES6+ JavaScript代码?

    使用Babel转译ES6+代码并配置目标浏览器范围,结合core-js按需引入polyfill,通过构建工具打包模块化代码,运行时检测不兼容API并降级处理,确保跨浏览器兼容性。 编写跨浏览器兼容的现代ES6+ JavaScript代码,关键在于平衡语言新特性与浏览器支持之间的差距。虽然现代浏览器已…

    2025年12月20日
    000
  • Nightwatch.js中优化元素选择器复用:变量与页面对象实践

    本教程旨在解决Nightwatch.js测试中元素选择器重复使用的问题。我们将探讨两种主要策略:通过常量变量存储选择器以实现代码简洁,以及利用页面对象模式提升大型项目中的可维护性和可重用性。同时,文章也将解释Nightwatch.js独特的命令链式调用哲学及其对测试编写的影响,帮助开发者编写更高效、…

    2025年12月20日
    000
  • CSS Grid布局:优雅解决背景层高度自适应内容层的问题

    本文将介绍如何仅使用CSS Grid布局,无需JavaScript,实现背景层Div的高度与前景内容Div的高度保持一致。通过将背景和前景元素置于同一网格单元中,即使内容溢出视口,也能确保背景完美覆盖,提供一种高效且响应式的布局解决方案。 挑战:背景层与内容层高度同步 在网页设计中,我们经常遇到需要…

    2025年12月20日
    000

发表回复

登录后才能评论
关注微信