Android LayoutInflater 深度解析

在这里插入图片描述

在 Android 开发中,LayoutInflater 是一个非常重要的工具。它允许我们从 XML 布局文件中动态地创建 View 对象,从而使得 UI 的创建和管理更加灵活。本文将深入解析 android.view.LayoutInflater,包括它的基本用法、常见问题以及高级用法。

什么是 LayoutInflater?

LayoutInflater 是 Android 提供的一个类,用于将 XML 布局文件解析成对应的 View 对象。它通常用于 Activity 和 Fragment 中,通过代码动态地创建和操作 UI 元素。

基本用法

LayoutInflater 的基本用法非常简单,通常有以下几种方式:

从 Activity 获取 LayoutInflater

val inflater = LayoutInflater.from(this)
// 或者
val inflater = this.layoutInflater

从 Context 获取 LayoutInflater

val inflater = context.getSystemService(Context.LAYOUT_INFLATER_SERVICE) as LayoutInflater

从 Fragment 获取 LayoutInflater

val inflater = requireActivity().layoutInflater

常见用法示例

在 Activity 中使用 LayoutInflater

在 Activity 中,我们可以使用 LayoutInflater 来动态地加载布局,例如在一个 LinearLayout 中添加多个子视图:

class MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)val container = findViewById<LinearLayout>(R.id.container)val inflater = LayoutInflater.from(this)val itemView = inflater.inflate(R.layout.item_layout, container, false)container.addView(itemView)}
}

在 Fragment 中使用 LayoutInflater

在 Fragment 中,我们通常在 onCreateView 方法中使用 LayoutInflater 来加载布局:

class MyFragment : Fragment() {override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,savedInstanceState: Bundle?): View? {val view = inflater.inflate(R.layout.fragment_layout, container, false)// 初始化视图return view}
}

LayoutInflater 的高级用法

除了基本的用法,LayoutInflater 还有一些高级用法和技巧,可以帮助我们更高效地创建和管理视图。

视图缓存

在性能敏感的应用中,频繁地调用 LayoutInflater.inflate 可能会导致性能问题。为了提高性能,我们可以缓存已经加载的视图:

class MainActivity : AppCompatActivity() {private lateinit var cachedView: Viewoverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)val container = findViewById<LinearLayout>(R.id.container)val inflater = LayoutInflater.from(this)// 缓存视图if (!::cachedView.isInitialized) {cachedView = inflater.inflate(R.layout.item_layout, container, false)}container.addView(cachedView)}
}

自定义 LayoutInflater.Factory

我们可以通过实现 LayoutInflater.Factory 接口来自定义视图创建逻辑。例如,我们可以在视图创建时自动应用自定义字体:

class CustomFontFactory : LayoutInflater.Factory {override fun onCreateView(name: String, context: Context, attrs: AttributeSet): View? {val view = LayoutInflater.from(context).createView(name, null, attrs)if (view is TextView) {view.typeface = Typeface.createFromAsset(context.assets, "custom_font.ttf")}return view}
}class MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)val inflater = LayoutInflater.from(this)inflater.factory = CustomFontFactory()setContentView(R.layout.activity_main)}
}

常见问题和解决方案

inflate 方法的第三个参数 attachToRoot

LayoutInflater.inflate 方法的第三个参数 attachToRoot 经常会让人困惑。这个参数决定了新创建的视图是否应该被立即添加到传入的根视图中:

  • 如果 attachToRoottrue,新创建的视图会被立即添加到根视图中,且 inflate 方法会返回根视图。
  • 如果 attachToRootfalse,新创建的视图不会被添加到根视图中,且 inflate 方法会返回新创建的视图。

示例:

val container = findViewById<LinearLayout>(R.id.container)// attachToRoot = true
val view1 = LayoutInflater.from(this).inflate(R.layout.item_layout, container, true)
// view1 == container// attachToRoot = false
val view2 = LayoutInflater.from(this).inflate(R.layout.item_layout, container, false)
// view2 != container

自定义 View 和自定义属性

在使用 LayoutInflater 加载自定义 View 时,我们需要确保自定义属性可以正确应用。这通常通过在自定义 View 的构造函数中读取属性来实现:

class CustomView @JvmOverloads constructor(context: Context,attrs: AttributeSet? = null,defStyleAttr: Int = 0
) : View(context, attrs, defStyleAttr) {init {context.theme.obtainStyledAttributes(attrs,R.styleable.CustomView,0, 0).apply {try {val customAttribute = getString(R.styleable.CustomView_customAttribute)// 使用自定义属性} finally {recycle()}}}
}

在 XML 中使用自定义 View 和属性:

<com.example.CustomViewandroid:layout_width="match_parent"android:layout_height="wrap_content"app:customAttribute="Hello, Custom!" />

结论

LayoutInflater 是 Android 开发中不可或缺的工具,通过理解和掌握它的用法,可以大大提高 UI 开发的效率和灵活性。无论是基本的布局加载,还是高级的自定义视图创建,LayoutInflater 都提供了强大的功能和灵活性。希望本文能帮助你更好地理解和使用 LayoutInflater,提升你的安卓开发技能。

Best regards!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/367836.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

idea xml ctrl+/ 注释格式不对齐

处理前 处理后 解决办法 取消这两个勾选

【UE5.3】笔记6-创建可自由控制Pawn类

搭建场景 搭建一个场景&#xff1a;包含地板、围墙。可以根据喜好加一些自发光的效果。 增加食物 创建食物蓝图类&#xff0c;在场景里放置一些食物以供我们player去吃掉获取分值。 创建可控制的layer 我们先右键创建一个蓝图继承自pawn类&#xff0c;起名BP_Player&#xf…

深度学习之半监督学习:一文梳理目标检测中的半监督学习策略

什么是半监督目标检测&#xff1f; 传统机器学习根据训练数据集中的标注情况&#xff0c;有着不同的场景&#xff0c;主要包括&#xff1a;监督学习、弱监督学习、弱半监督学习、半监督学习。由于目标检测任务的特殊性&#xff0c;在介绍半监督目标检测方法之前&#xff0c;我…

视频融合共享平台LntonCVS统一视频接入平台智慧安防应用方案

安防视频监控平台LntonCVS是一款拥有强大拓展性和灵活部署能力的综合管理平台。它支持多种主流标准协议&#xff0c;包括国标GB28181、RTSP/Onvif、RTMP等&#xff0c;同时兼容各厂家的私有协议和SDK&#xff0c;如海康Ehome、海大宇等。LntonCVS不仅具备传统安防视频监控功能&…

PHP电商系统开发指南最佳实践

电子商务系统开发的最佳实践包括&#xff1a;数据库设计&#xff1a;选择适合关系型数据库&#xff0c;优化数据结构&#xff0c;考虑表分区&#xff1b;安全&#xff1a;加密数据&#xff0c;防止 sql 注入&#xff0c;处理会话管理&#xff1b;用户界面&#xff1a;遵循 ux 原…

mysql-sql-第十四周

学习目标&#xff1a; sql 学习内容&#xff1a; 40.查询学过「哈哈」老师授课的同学的信息 Select * from students left join score on students.stunmscore.stunm where counm (select counm from teacher left join course on teacher.teanmcourse.teanm where teache…

【深度学习】Transformer

李宏毅深度学习笔记 https://blog.csdn.net/Tink1995/article/details/105080033 https://blog.csdn.net/leonardotu/article/details/135726696 https://blog.csdn.net/u012856866/article/details/129790077 Transformer 是一个基于自注意力的序列到序列模型&#xff0c;与基…

伺服调试三环讲解

在伺服调试过程中,有些项目要求不高,采用伺服自整定就可以调试好伺服,但有些项目对伺服有着比较高的要求,于是需要采取手动调试伺服参数,下面就介绍一下伺服三环参数的调试的方法。 三环指:电流环、速度环、位置环 带宽关系:电流环带宽>速度环带宽>位置环带宽 三环控…

C语言单链表的算法之插入节点

一&#xff1a;访问各个节点中的数据 &#xff08;1&#xff09;访问链表中的各个节点的有效数据&#xff0c;这个访问必须注意不能使用p、p1、p2&#xff0c;而只能使用phead &#xff08;2&#xff09;只能用头指针不能用各个节点自己的指针。因为在实际当中我们保存链表的时…

偏微分方程笔记

极小位能原理&#xff1a; C 2 C^2 C2 是一个集合符号&#xff0c;表示所有二阶连续可微函数的集合 弱导数 C 2 C^2 C2 是一个集合符号&#xff0c;表示所有二阶连续可微函数的集合。 C 0 ∞ ( I ) C^{\infty}_0(I) C0∞​(I)表示于 I I I上无穷可微&#xff0c;且在端点a&…

使用pyinstaller 如何打包python项目

参考&#xff1a;【python项目正确打包方法-哔哩哔哩】 https://b23.tv/EDB6zbG Pyinstaller 详解多种打包过程(去坑,填坑)。_pyinstaller -f -w-CSDN博客 1.打开命令提示符&#xff1a; 找到python项目所在位置&#xff0c;输入cmd即可 2. 安装pipenv: 在命令提示符&#…

【强化学习的数学原理】课程笔记--2(贝尔曼最优公式,值迭代与策略迭代)

目录 贝尔曼最优公式最优 Policy求解贝尔曼最优公式求解最大 State Value v ∗ v^* v∗根据 v ∗ v^* v∗ 求解最佳 Policy π ∗ \pi^* π∗一些证明过程 一些影响 π ∗ \pi^* π∗ 的因素如何让 π ∗ \pi^* π∗ 不 “绕弯路” γ \gamma γ 的影响reward 的影响 值迭…

UiPath+Appium实现app自动化测试

一、环境准备工作 1.1 完成appium环境的搭建 参考&#xff1a;pythonappiumpytestallure模拟器(MuMu)自动化测试环境搭建_appium mumu模拟器-CSDN博客 1.2 完成uipath的安装 登录官网&#xff0c;完成注册与软件下载安装。 UiPath业务自动化平台&#xff1a;先进的RPA及自动…

Visual Studio 中的键盘快捷方式

1. Visual Studio 中的键盘快捷方式 1.1. 可打印快捷方式备忘单 1.2. Visual Studio 的常用键盘快捷方式 本部分中的所有快捷方式都将全局应用&#xff08;除非另有指定&#xff09;。 “全局”上下文表示该快捷方式适用于 Visual Studio 中的任何工具窗口。 生成&#xff1…

第十四章 集合(List)

一、集合框架体系 集合&#xff1a; &#xff08;1&#xff09;可以动态保存任意多个对象 &#xff08;2&#xff09;提供了一系列方便的操作对象的方法 二、Collection 1. Collection 接口常用方法 &#xff08;1&#xff09;add&#xff1a;添加单个元素 &#xff08;2…

Cypress测试:7个快速解决问题的调试技巧!

以下为作者观点&#xff1a; 快速编写代码是一项宝贵的技能&#xff0c;但能够有效调试和解决错误和bug&#xff0c;更是一个软件开发人员具有熟练技能的标志。调试是开发过程中的一个关键环节&#xff0c;可以确保软件按预期运行并满足用户需求。 Cypress 调试简介 Cypress …

【Linux】—Xshell、Xftp安装

文章目录 前言一、下载Xshell、Xftp二、安装Xshell三、使用XShell连接Linux服务器四、修改windows的主机映射文件&#xff08;hosts文件&#xff09;五、远程连接hadoop102/hadoop103/hadoop104服务器六、安装Xftp 前言 XShell远程管理工具&#xff0c;可以在Windows界面下来访…

uniapp + vue3 + Script Setup 写法变动 (持续更新)

一、uniapp 应用生命周期&#xff1a; https://uniapp.dcloud.net.cn/tutorial/vue3-composition-api.html 注意&#xff1a; 应用生命周期仅可在App.vue中监听&#xff0c;在其它页面监听无效。 二 、uniapp页面生命周期&#xff1a; https://uniapp.dcloud.net.cn/tutori…

3.js - 色调映射(renderer.toneMapping)

// ts-nocheck// 引入three.js import * as THREE from three// 导入轨道控制器 import { OrbitControls } from three/examples/jsm/controls/OrbitControls// 导入lil.gui import { GUI } from three/examples/jsm/libs/lil-gui.module.min.js// 导入tween import * as TWEEN…

昇思25天学习打卡营第11天|基于MindSpore通过GPT实现情感分类

学AI还能赢奖品&#xff1f;每天30分钟&#xff0c;25天打通AI任督二脉 (qq.com) 基于MindSpore通过GPT实现情感分类 %%capture captured_output # 实验环境已经预装了mindspore2.2.14&#xff0c;如需更换mindspore版本&#xff0c;可更改下面mindspore的版本号 !pip uninsta…