Android第六次面试总结(Java设计模式二)
在 Android 开发里,ListView 和 RecyclerView 是常用的视图组件,用于展示大量数据列表。不过,这些视图组件本身无法直接展示原始数据源,需要借助 Adapter(适配器)把数据源适配成视图能够展示的数据,这便是适配器模式的实际应用。下面详细介绍 Adapter 在 ListView 和 RecyclerView 中的使用。
适配器模式原理
适配器模式的核心在于把一个类的接口转换为客户期望的另一个接口,让原本由于接口不兼容而不能一起工作的那些类可以协同工作。在 ListView 和 RecyclerView 的场景中,数据源(如数组、集合)和视图组件(ListView、RecyclerView)的接口不匹配,Adapter 就充当了中间的转换器,将数据源适配成视图组件能够识别和展示的形式。
ListView 中的适配器模式
示例代码
import android.app.Activity;
import android.os.Bundle;
import android.widget.ArrayAdapter;
import android.widget.ListView;import java.util.ArrayList;
import java.util.List;public class ListViewAdapterExample extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(android.R.layout.activity_list_item);// 数据源List<String> dataSource = new ArrayList<>();dataSource.add("Item 1");dataSource.add("Item 2");dataSource.add("Item 3");// 创建适配器ArrayAdapter<String> adapter = new ArrayAdapter<>(this, android.R.layout.simple_list_item_1, dataSource);// 获取 ListView 实例ListView listView = findViewById(android.R.id.list);// 设置适配器listView.setAdapter(adapter);}
}
代码解释
- 数据源:
List<String> dataSource是一个字符串列表,代表原始的数据集合。 - 适配器:
ArrayAdapter是 Android 提供的一个适配器类,它将List<String>类型的数据源适配成ListView可以展示的形式。ArrayAdapter接收三个参数:上下文、列表项的布局资源和数据源。 - ListView:通过
findViewById方法获取ListView实例,然后调用setAdapter方法将适配器设置给ListView,这样ListView就可以根据适配器提供的数据来展示列表项。
RecyclerView 中的适配器模式
示例代码
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;
import java.util.List;// 自定义适配器类
class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyViewHolder> {private List<String> dataSource;private Context context;public MyAdapter(Context context, List<String> dataSource) {this.context = context;this.dataSource = dataSource;}@NonNull@Overridepublic MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {View view = LayoutInflater.from(context).inflate(android.R.layout.simple_list_item_1, parent, false);return new MyViewHolder(view);}@Overridepublic void onBindViewHolder(@NonNull MyViewHolder holder, int position) {String item = dataSource.get(position);holder.textView.setText(item);}@Overridepublic int getItemCount() {return dataSource.size();}static class MyViewHolder extends RecyclerView.ViewHolder {TextView textView;public MyViewHolder(@NonNull View itemView) {super(itemView);textView = itemView.findViewById(android.R.id.text1);}}
}
import android.os.Bundle;
import android.widget.LinearLayout;
import androidx.appcompat.app.AppCompatActivity;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import java.util.ArrayList;
import java.util.List;public class RecyclerViewAdapterExample extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(android.R.layout.activity_list_item);// 数据源List<String> dataSource = new ArrayList<>();dataSource.add("Item 1");dataSource.add("Item 2");dataSource.add("Item 3");// 获取 RecyclerView 实例RecyclerView recyclerView = findViewById(android.R.id.list);// 设置布局管理器recyclerView.setLayoutManager(new LinearLayoutManager(this));// 创建适配器MyAdapter adapter = new MyAdapter(this, dataSource);// 设置适配器recyclerView.setAdapter(adapter);}
}
代码解释
- 数据源:同样是
List<String>类型的dataSource,代表原始的数据集合。 - 适配器:自定义的
MyAdapter类继承自RecyclerView.Adapter,它需要实现三个重要方法:onCreateViewHolder:用于创建ViewHolder实例,ViewHolder是一个用于缓存视图组件的容器。onBindViewHolder:用于将数据源中的数据绑定到ViewHolder中的视图组件上。getItemCount:返回数据源的大小。
- RecyclerView:通过
findViewById方法获取RecyclerView实例,先设置布局管理器(如LinearLayoutManager),然后调用setAdapter方法将适配器设置给RecyclerView,从而展示数据列表。
适配器模式的优势
- 解耦数据源和视图:通过适配器模式,数据源和视图组件之间的耦合度降低,使得数据源和视图可以独立变化。
- 可扩展性:可以根据需要自定义适配器,实现不同的视图展示效果,提高代码的可扩展性。
- 代码复用:适配器可以被多个视图组件复用,提高代码的复用性。
在 Android 的 Jetpack 组件中,观察者模式有着广泛的应用,其中 LiveData 就是一个典型的例子,它很好地体现了观察者模式,并且借助注解来提升代码的安全性与可读性。下面详细介绍 LiveData 中的观察者模式以及相关注解的使用。
观察者模式在 LiveData 中的应用
原理
LiveData 是一个可观察的数据持有者类,它遵循观察者模式。LiveData 持有一个数据对象,当这个数据对象的状态发生变化时,所有注册的观察者(即 Observer)都会收到通知并更新自身状态。LiveData 还具备生命周期感知能力,它只会在观察者的生命周期处于活跃状态(如 STARTED 或 RESUMED)时才会通知观察者,避免了内存泄漏和不必要的更新。
示例代码
import androidx.appcompat.app.AppCompatActivity;
import androidx.lifecycle.LiveData;
import androidx.lifecycle.MutableLiveData;
import androidx.lifecycle.Observer;
import android.os.Bundle;
import android.widget.TextView;public class MainActivity extends AppCompatActivity {private MutableLiveData<String> dataLiveData;private TextView textView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// 初始化 LiveDatadataLiveData = new MutableLiveData<>();// 获取 TextView 实例textView = findViewById(R.id.textView);// 创建观察者Observer<String> observer = new Observer<String>() {@Overridepublic void onChanged(String newData) {// 当 LiveData 中的数据发生变化时,更新 TextView 的文本textView.setText(newData);}};// 注册观察者dataLiveData.observe(this, observer);// 更新 LiveData 中的数据updateData();}private void updateData() {// 模拟数据更新dataLiveData.setValue("New data");}
}
代码解释
LiveData的创建:使用MutableLiveData类创建一个可变的LiveData对象dataLiveData,用于持有数据。- 观察者的创建:创建一个
Observer对象,实现onChanged方法,当LiveData中的数据发生变化时,该方法会被调用,从而更新TextView的文本。 - 观察者的注册:调用
dataLiveData.observe(this, observer)方法将观察者注册到LiveData上,其中this表示当前的Activity,用于提供生命周期信息。 - 数据的更新:调用
dataLiveData.setValue("New data")方法更新LiveData中的数据,此时所有注册的观察者都会收到通知。
Jetpack 注解的应用
@NonNull 和 @Nullable 注解
在使用 LiveData 时,为了确保数据的非空性,可以使用 @NonNull 和 @Nullable 注解。这些注解可以帮助开发者在编译时发现潜在的空指针异常。
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.lifecycle.LiveData;
import androidx.lifecycle.MutableLiveData;public class MyViewModel {private MutableLiveData<@NonNull String> nonNullLiveData = new MutableLiveData<>();private MutableLiveData<@Nullable String> nullableLiveData = new MutableLiveData<>();public LiveData<@NonNull String> getNonNullLiveData() {return nonNullLiveData;}public LiveData<@Nullable String> getNullableLiveData() {return nullableLiveData;}public void setNonNullData(@NonNull String data) {nonNullLiveData.setValue(data);}public void setNullableData(@Nullable String data) {nullableLiveData.setValue(data);}
}
代码解释
@NonNull注解:表示该变量或参数不能为null,如果尝试将null值赋给被@NonNull注解修饰的变量或参数,编译器会发出警告。@Nullable注解:表示该变量或参数可以为null。
@MainThread 和 @WorkerThread 注解
在更新 LiveData 时,需要注意更新操作必须在主线程中进行。可以使用 @MainThread 和 @WorkerThread 注解来明确方法的调用线程。
import androidx.annotation.MainThread;
import androidx.annotation.WorkerThread;
import androidx.lifecycle.MutableLiveData;public class MyViewModel {private MutableLiveData<String> liveData = new MutableLiveData<>();@MainThreadpublic void updateDataOnMainThread(String data) {liveData.setValue(data);}@WorkerThreadpublic void updateDataOnWorkerThread(String data) {// 在工作线程中更新 LiveData 需要使用 postValue 方法liveData.postValue(data);}
}
代码解释
@MainThread注解:表示该方法必须在主线程中调用,setValue方法必须在主线程中调用,因此使用@MainThread注解进行标记。@WorkerThread注解:表示该方法必须在工作线程中调用,postValue方法可以在工作线程中调用,因此使用@WorkerThread注解进行标记。
总结
LiveData 是 Jetpack 中实现观察者模式的重要组件,它通过观察者模式实现了数据的实时更新和通知。同时,Jetpack 提供的注解(如 @NonNull、@Nullable、@MainThread、@WorkerThread 等)可以帮助开发者提高代码的安全性和可读性,避免一些常见的错误。
相关文章:
Android第六次面试总结(Java设计模式二)
在 Android 开发里,ListView 和 RecyclerView 是常用的视图组件,用于展示大量数据列表。不过,这些视图组件本身无法直接展示原始数据源,需要借助 Adapter(适配器)把数据源适配成视图能够展示的数据…...
一站式电脑工具箱,功能全面且实用
小明工具箱是一款集成了系统设置、维护工具、实用工具、图像处理等四大类工具的电脑工具箱,涵盖了上百种实用工具,能够满足用户在文件管理、文本处理、系统优化、图像处理等多方面的需求。 初次使用,需双击软件,便会自动将工具解压…...
那些正常的动态规划
文章目录 前言动态规划到底是啥? 线性dp最长上升子序列子集和子序列和子串的区别内容分析 最大上升子序列例题1——[NOIP2004 提高组] 合唱队形分析 最长公共子序列最长公共子串 平面dp例题2——[NOIP2000 提高组] 方格取数分析 例题3——[NOIP2008 提高组] 传纸条分…...
Opencv计算机视觉编程攻略-第二节 图像像素操作
第二节 图像像素操作 1.访问像素值2.用指针扫描图像3.扫描图像并访问相邻像素4.实现简单的图像运算5.图像重映射 1.访问像素值 以椒盐噪声为例展示像素值访问的几种方法 void salt(cv::Mat image, int n) {// C11 random number generatorstd::default_random_engine generat…...
华为交换相关
端口模式 (1)access:只能属于单个VLAN,一般用于连接计算机端口 (2)trunk:端口允许多个VLAN通过,可以接收和发送多个VLAN报文,默认情况下只有管理VLAN不携带标签信息 &…...
Chrome Performance 面板完全指南:从卡顿到丝滑的终极调试术
1.写在前面 前端性能调试是优化网页加载速度和运行效率的关键步骤,Chrome DevTools 的 Performance 面板 是核心工具; 2.Performance 面板使用步骤 ★ 基础 打开面板 在 Chrome 中按 F12 → 切换到 Performance 标签页。 开始录制 方式一:点击 ⚫️ 圆…...
idea中快速注释函数
在IntelliJ IDEA中,有多种方法可以快速注释函数。 使用快捷键 你可以使用以下快捷键来快速注释函数[3]: 行注释:使用Ctrl/(Windows系统)或Command/(Mac系统)可以在当前行前添加或删除单行注释…...
深入解析Linux网络、安全与容器技术
1. Netfilter:Linux内核的包处理框架 Netfilter 是Linux内核中用于控制网络数据包的核心机制,负责处理数据包的过滤、修改和转发。其核心功能包括: 包过滤(Packet Filtering):根据规则允许或拒绝数据包通过…...
JDK 24:Java 24 中的新功能
🧑 博主简介:CSDN博客专家,历代文学网(PC端可以访问:历代文学,移动端可微信小程序搜索“历代文学”)总架构师,15年工作经验,精通Java编程,高并发设计…...
java中的枚举类型和c,c++的有区别吗?c,c++的枚举,结构体,联合体,三种数据有什么区别和联系
Java 枚举类型与 C、C 枚举类型的区别 1. 类型安全 Java:Java 的枚举类型是类型安全的。枚举常量是枚举类型的实例,编译器会严格检查传递的参数是否为该枚举类型的有效常量。例如: java Apply enum Color { RED, GREEN, BLUE } // 编译器会检…...
ubuntu服务器server版安装,ssh远程连接xmanager管理,改ip网络连接。图文教程
ventoy启动服务器版iso镜像,注意看server名称,跟之前desktop版ubuntu不一样。没有gui界面。好,进入命令行界面。语言彻底没汉化了,选英文吧,别的更看不懂。 跟桌面版ubuntu类似,选择是否精简系统࿰…...
什么叫税务黑名单?详解税务黑名单的来源。
一、什么叫税务黑名单? 1、税务黑名单是指由税务部门根据相关法律法规和税收管理策,对违反税收法规、逃避纳税义务或其他严重违法违规行为的个人或企业进行记录和公示的名单。 2、被列入税务黑名单意味着该个人或企业在税务方面存在严重的不诚信行为&a…...
计算机二级:基础操作题
一 sinfoinput() info_listsinfo.split(,) print("姓名,年龄") for strname in info_list:snamestrname[:-2]sagestrname[-2:]print("{},{}".format(sname,sage))二 import random as r r.seed(1) sinput("请输入三个整数n,m,…...
python机器学习——新手入门学习笔记
一,概论 1.什么是机器学习 定义: 机器学习是从数据中自动分析获得模型,并利用模型对未知数据进行预测。 其实就是通过问题和数据,发现规律,并进行预测,与人脑相似。目的就是从历史数据当中获得规律&#x…...
LabVIEW 与 PLC 通讯的常见方式
在工业自动化和数据采集系统中,PLC(可编程逻辑控制器) 广泛用于控制和监测各种设备,而 LabVIEW 作为强大的图形化编程工具,常用于上位机数据处理和可视化。为了实现 LabVIEW 与 PLC 的高效通讯,常见的方法包…...
深度学习 Deep Learning 第9章 卷积网络 CNN
深度学习 Deep Learning 第9章 卷积网络 章节概述 本章深入探讨了卷积网络的原理、变体及其在深度学习中的应用。卷积网络通过卷积操作实现了参数共享和稀疏连接,显著提高了模型的效率和性能。本章首先介绍了卷积操作的基本形式及其在不同数据维度上的应用&#x…...
Tekton系列之实践篇-从触发到完成的完整执行过程
以下介绍的是基于 Gitee 仓库 的 Tekton 工作流程 操作流程 定义task 克隆代码的task # task-clone.yaml apiVersion: tekton.dev/v1beta1 kind: Task metadata:name: git-clone spec:workspaces:- name: source # 工作目录params:- name: repo-url # 你的 Gitee 仓库地址…...
【简单学习】Prompt Engineering 提示词工程
一、Prompt 1、Prompt 是什么? Prompt 是一种人为构造的输入序列,用于引导 GPT 模型根据先前输入的内容生成相关的输出。简单来说,就是你向模型提供的 “提示词”。 在 ChatGpt 中,我们可以通过设计不同的 prompt,让…...
neo4j删除所有数据
neo4j删除所有数据 一次性删除 MATCH (n) DETACH DELETE n ;分批次删除 先删除关系 MATCH ()-[r]->()WITH r LIMIT 100000DELETE rRETURN count(r)在删除节点 MATCH (n)WITH n LIMIT 100000DELETE nRETURN count(n)验证 查询节点总数 MATCH (n) RETURN count(n) AS node…...
零基础入门网络爬虫第5天:Scrapy框架
4周 Srapy爬虫框架 不是一个简单的函数功能库,而是一个爬虫框架 安装:pip install scrapy 检测:scrapy -h Scrapy爬虫框架结构 爬虫框架 爬虫框架是实现爬虫功能的一个软件结构和功能组件集合爬虫框架是一个半成品,能够帮助…...
ARCGIS PRO DSK 栅格数据(Raster)
ArcGIS Pro 中与栅格相关的功能可以在两个单独程序集中的两个命名空间中找到。 1、ArcGIS.Core.dll 中的 ArcGIS.Core.Data.Raster 命名空间提供了栅格类和成员,用于处理栅格数据集、内存栅格、像素块和光标。 2、ArcGIS.Desktop.Mapping.dll 中的 ArcGIS.Desktop.M…...
C#设计模式快速回顾
知识点来源:人间自有韬哥在,豆包 目录 一、七大原则1. 单一职责原则 (Single Responsibility Principle)2. 开放封闭原则 (Open-Closed Principle)3. 里氏替换原则 (Liskov Substitution Principle)4. 接口隔离原则 (Interface Segregation Principle)5…...
分页查询互动问题(用户端)
文章目录 概要整体架构流程技术细节小结 概要 需求分析以及接口设计 技术细节 1.Controller层 GetMapping("/page")ApiOperation("分页查询问题")public PageDTO<QuestionVO> queryQuestionPage(QuestionPageQuery query){return questionService…...
【全队项目】智能学术海报生成系统PosterGenius(项目介绍)
🌈 个人主页:十二月的猫-CSDN博客 🔥 系列专栏:🏀大模型实战训练营_十二月的猫的博客-CSDN博客 💪🏻 十二月的寒冬阻挡不了春天的脚步,十二点的黑夜遮蔽不住黎明的曙光 目录 1. 前…...
P5356 [Ynoi Easy Round 2017] 由乃打扑克 Solution
Description 给定序列 a ( a 1 , a 2 , ⋯ , a n ) a(a_1,a_2,\cdots,a_n) a(a1,a2,⋯,an),有 m m m 个操作分两种: add ( l , r , x ) \operatorname{add}(l,r,x) add(l,r,x):对每个 i ∈ [ l , r ] i\in[l,r] i∈[l,r] 执行 …...
【线程安全问题的原因和方法】【java形式】【图片详解】
在本章节中采用实例图片的方式,以一个学习者的姿态进行描述问题解决问题,更加清晰明了,以及过程中会发问的问题都会一一进行呈现 目录 线程安全演示线程不安全情况图片解释: 将上述代码进行修改【从并行转化成穿行的方式】不会出…...
MySQL-----视图与索引
目录 视图 1.视图 2.操作 11.索引 1.定义 2.优缺点: 3.分类 4.索引的设计原则 5.索引的使用 作业 视图 1.视图 ❓如果需要在原表中隐藏部分字段时,怎么办? 视图 📖视图: 是一个没有存储任何数据的表,可以对其CRUD视图…...
【差分隐私相关概念】约束下的列联表边缘分布计算方法
列联表及其边缘分布的详细解释 一、列联表的定义 列联表(Contingency Table) 是一种用于表示 多个分类变量联合分布 的表格。其核心是通过多维数组记录不同属性组合的频次。以下是关键点: 分类属性: 设有 k k k 个分类属性 A …...
解决IDEA中maven找不到依赖项的问题
直接去官网找到对应的依赖项jar包,并且下载到本地,然后安装到本地厂库中。 Maven官网:https://mvnrepository.com/ 一、使用mvn install:install-file命令 Maven提供了install:install-file插件,用于手动将jar包安装到本地仓库…...
pyside6的QGraphicsView体系,当鼠标位于不同的物体,显示不同的右键菜单
代码: # 设置样本图片的QGraphicsView模型 from PySide6.QtCore import Qt, QRectF, QObject from PySide6.QtGui import QPainter, QPen, QColor, QAction, QMouseEvent from PySide6.QtWidgets import QGraphicsView, QGraphicsScene, QGraphicsPixmapItem, QGra…...
