Jetpack Compose初体验
入门学习
由于工作需要,我们当前要在老代码的基础上使用 Compose 进行新页面的开发,这项工作主要落在我的身上。因此,我需要先了解 Compose。
这里我入门看的是写给初学者的Jetpack Compose教程,Lazy Layout,有兴趣可以看看。
了解 Compose
Compose 是 Google 推出的用于构建原生界面的现代工具包,专门用于 Android 平台的声明式 UI 开发。基于 Kotlin 语言,Compose 利用 Kotlin 的特性来简化 UI 开发过程,核心思想是使用函数来描述 UI 的外观和行为。
Compose 与传统 Android 开发的区别
1. 编程范式
- 传统 Android 开发:采用命令式编程范式,开发者需要手动管理 UI 组件的创建、更新和销毁。
- Compose 开发:采用声明式编程范式,开发者只需描述 UI 的最终状态,Compose 会自动更新 UI。
2. 布局系统
- 传统 Android 开发:使用 XML 布局文件和 Java/Kotlin 代码分离的方式。
- Compose 开发:使用 Kotlin 代码直接构建布局,布局代码和逻辑代码可以放在一起。
3. 状态管理
- 传统 Android 开发:手动处理状态的保存和恢复。
- Compose 开发:内置状态管理机制,使用
mutableStateOf,状态变化会自动更新 UI。
4. 开发效率
- 传统 Android 开发:代码量较大,开发效率相对较低。
- Compose 开发:代码简洁,声明式编程和内置状态管理机制提高了开发效率。
初试 Compose
从 AI 的搜索结果来看,Compose 开发具有以下特点:声明式编程、基于 Kotlin、高效的状态管理、简洁的布局系统、实时预览、与现有代码兼容。
环境搭建
想要进行 Compose 开发,首先要引入 Compose 环境。Android Studio 新版可以直接创建 Compose 项目,但我当前的版本暂未提供这一功能。可以通过在 build.gradle 文件中添加相关依赖来引入 Compose。
公共基类
我创建了一个 BaseComposeActivity 作为 Compose 页面的公共基类,它继承自 AppCompatActivity,并在 onCreate 方法中使用 setContent 来设置 Compose UI。
基本布局
根据产品的原型图,我基于平板和手机做出了不同的响应式布局。只需通过 @Composable 函数来组合不同的组件,即可实现 Activity 显示不同的信息。
主页面逻辑
我注册了一个主页面 DevicePolicyActivity,在其中定义了基本布局并对平板和手机进行了区分。通过 TopBar 和 Row 组件来安排导航栏和内容区域,使用 navigationNumber 和 hideSelectTac 来控制不同的页面显示。
响应式布局
- 平板端工作台:显示两列布局,左侧是目录,右侧是内容详情。
- 非平板端工作台:基于页面跳转来显示详情页面。
策略目录菜单
目录菜单通过 DirectoryItem 和 StrategyItem 组件来实现。通过点击菜单项更新选中状态,并相应地改变页面显示。对于非平板逻辑页面,点击后触发页面跳转,通过 forceRecompose 来手动触发页面重组。
DP根据逻辑生成的
这是ai根据我描述逻辑生成的,基本逻辑是一样的,但状态管理我都是基于remember和mutableIntStateOf进行管理的,其他意思到了就行:
代码实现细节
以下是基于 Compose 实现的核心代码逻辑和关键设计思路,结合具体场景进行说明:
1. BaseComposeActivity 基类
abstract class BaseComposeActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContent {MyAppTheme { // 自定义主题配置AppContent() // 主 UI 入口}}}@Composableabstract fun AppContent()
}
设计要点:
- 统一 Activity 入口模板
- 强制子类实现
AppContent方法作为 UI 入口 - 集成 Material Design 主题系统
2. 主页面 DevicePolicyActivity 实现
class DevicePolicyActivity : BaseComposeActivity() {private val viewModel: PolicyViewModel by viewModels()@Composableoverride fun AppContent() {val isTablet = LocalConfiguration.current.screenWidthDp >= 600if (isTablet) {TabletLayout(viewModel)} else {PhoneLayout(viewModel)}}
}
设备检测逻辑:
@Composable
fun isTabletMode(): Boolean {val configuration = LocalConfiguration.currentreturn configuration.screenWidthDp >= 600 || configuration.smallestScreenWidthDp >= 600
}
3. 平板双栏布局实现
@Composable
private fun TabletLayout(viewModel: PolicyViewModel) {Scaffold(topBar = { AppTopBar(viewModel.selectedTitle) }) { padding ->Row(Modifier.padding(padding)) {// 左侧目录(占30%宽度)NavigationColumn(Modifier.weight(0.3f),viewModel.policies,viewModel::selectPolicy)// 右侧详情(占70%宽度)PolicyDetail(Modifier.weight(0.7f),viewModel.selectedPolicy)}}
}
关键交互逻辑:
// 状态管理示例
class PolicyViewModel : ViewModel() {private val _selectedPolicy = mutableStateOf<Policy?>(null)val selectedPolicy: State<Policy?> = _selectedPolicyfun selectPolicy(policy: Policy) {_selectedPolicy.value = policy}
}
4. 手机端列表-详情跳转
@Composable
private fun PhoneLayout(viewModel: PolicyViewModel) {val navController = rememberNavController()NavHost(navController, startDestination = "policyList") {composable("policyList") {PolicyListScreen(viewModel) { policy ->navController.navigate("policyDetail/${policy.id}")}}composable("policyDetail/{policyId}") { backStackEntry ->PolicyDetailScreen(policyId = backStackEntry.arguments?.getString("policyId"),onBack = { navController.popBackStack() })}}
}
5. 策略目录组件实现
@Composable
fun NavigationColumn(modifier: Modifier = Modifier,policies: List<Policy>,onSelect: (Policy) -> Unit
) {LazyColumn(modifier) {items(policies) { policy ->DirectoryItem(policy = policy,isSelected = policy == viewModel.selectedPolicy,onClick = { onSelect(policy) })}}
}@Composable
private fun DirectoryItem(policy: Policy,isSelected: Boolean,onClick: () -> Unit
) {Surface(color = if (isSelected) MaterialTheme.colors.primary.copy(alpha = 0.1f) else Color.Transparent,modifier = Modifier.fillMaxWidth().clickable { onClick() }) {Text(text = policy.title,modifier = Modifier.padding(16.dp),style = MaterialTheme.typography.subtitle1)}
}
6. 强制重组逻辑优化
// 使用 remember 配合 key 控制重组
var recomposeKey by remember { mutableStateOf(0) }LaunchedEffect(Unit) {// 当需要强制重组时recomposeKey++
}Column(modifier = Modifier.remember(recomposeKey) { /*...*/ }) {// 动态内容
}
替代方案:
// 使用 derivedStateOf 精细化控制
val filteredPolicies = remember(policies, searchQuery) {derivedStateOf {policies.filter { it.contains(searchQuery) }}
}
遇到的典型问题与解决方案
问题 1:状态更新未触发重组
现象: 修改 mutableStateOf 的值后 UI 未更新
解决方案:
// 错误写法
var count = mutableStateOf(0)
Button(onClick = { count.value++ }) // 正确写法
val count = remember { mutableStateOf(0) }
Button(onClick = { count.value++ })
问题 2:列表性能问题
现象: LazyColumn 滚动卡顿
优化方案:
items(policies, key = { it.id }) { policy ->// 使用稳定唯一的 key
}
问题 3:主题配置冲突
解决方案: 创建独立主题配置
@Composable
fun MyAppTheme(content: @Composable () -> Unit) {MaterialTheme(colors = darkColors(primary = Color(0xFFBB86FC),secondary = Color(0xFF03DAC6)),typography = MyTypography,content = content)
}
相关文章:
Jetpack Compose初体验
入门学习 由于工作需要,我们当前要在老代码的基础上使用 Compose 进行新页面的开发,这项工作主要落在我的身上。因此,我需要先了解 Compose。 这里我入门看的是写给初学者的Jetpack Compose教程,Lazy Layout,有兴趣可…...
ceph部署-14版本(nautilus)-使用ceph-ansible部署实验记录
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、环境信息二、部署步骤2.1 基础环境准备2.2 各节点docker环境安装2.3 搭建互信集群2.4 下载ceph-ansible 三、配置部署文件3.1 使用本地docker3.2 配置hosts…...
【C++】C++ 旅馆管理系统(含 源码+报告)【独一无二】
👉博__主👈:米码收割机 👉技__能👈:C/Python语言 👉专__注👈:专注主流机器人、人工智能等相关领域的开发、测试技术。 系列文章目录 目录 系列文章目录一、设计要求二、设…...
快速排序
目录 什么是快速排序: 图解: 递归法: 方法一(Hoare法): 代码实现: 思路分析: 方法二(挖坑法): 代码实现: 思路分析: 非递…...
国内 ChatGPT Plus/Pro 订阅教程
1. 登录 chat.openai.com 依次点击 Login ,输入邮箱和密码 2. 点击升级 Upgrade 登录自己的 OpenAI 帐户后,点击左下角的 Upgrade to Plus,在弹窗中选择 Upgrade plan。 如果升级入口无法点击,那就访问这个网址,htt…...
易仓科技ai面试
请解释PHP中的面向对象编程的基本概念,并举例说明如何在PHP中定义一个类。 回答思路:需理解类、对象、继承和多态等基本概念,并能通过实例代码展示如何定义类及其属性和方法。 . 类(Class) 类是一个封装了数据和操作…...
LabVIEW用户界面(UI)和用户体验(UX)设计
作为一名 LabVIEW 开发者,满足功能需求、保障使用便捷与灵活只是基础要求。在如今这个用户体验至上的时代,为 LabVIEW 应用程序设计直观且具有美学感的界面,同样是不容忽视的关键任务。一个优秀的界面设计,不仅能提升用户对程序的…...
字玩FontPlayer开发笔记14 Vue3实现多边形工具
目录 字玩FontPlayer开发笔记14 Vue3实现多边形工具笔记整体流程临时变量多边形组件数据结构初始化多边形工具mousedown事件mousemove事件监听mouseup事件渲染控件将多边形转换为平滑的钢笔路径 字玩FontPlayer开发笔记14 Vue3实现多边形工具 字玩FontPlayer是笔者开源的一款字…...
低代码与 Vue.js:技术选型与架构设计
在当下数字化转型的浪潮中,企业对应用开发的效率和质量有着极高的追求。低代码开发平台的兴起,为企业提供了一条快速构建应用的捷径,而 Vue.js 作为热门的前端框架,与低代码开发平台的结合备受关注。如何做好两者的技术选型与架构…...
比较循环与迭代器的性能:Rust 零成本抽象的威力
一、引言 在早期的 I/O 项目中,我们通过对 String 切片的索引和 clone 操作来构造配置结构体,这种方法虽然能确保数据所有权的正确传递,但既显得冗长,又引入了不必要的内存分配。随着对 Rust 迭代器特性的深入了解,我…...
一文了解zookeeper
1.ZooKeeper是什么 简单来说,她是一个分布式的,开放源码的分布式应用程序协调服务 具体来说,他可以做如下事情: 分布式配置管理:ZooKeeper可以存储配置信息,应用程序可以动态读取配置信息。分布式同步&a…...
算法题(67):最长连续序列
审题: 需要我们在O(n)的时间复杂度下找到最长的连续序列长度 思路: 我们可以用两层for循环: 第一层是依次对每个数据遍历,让他们当序列的首元素。 第二层是访问除了该元素的其他元素 但是此时时间复杂度来到…...
大中型企业专用数据安全系统 | 天锐蓝盾终端安全 数据安全
天锐蓝盾系列产品是专门为大中型企业量身定制的数据安全防护产品体系,涵盖天锐蓝盾DLP、天锐蓝盾终端安全管理系统、天锐蓝盾NAC以及其他搭配产品,致力于实现卓越的数据安全防护、施行严格的网络准入控制以及构建稳固的终端安全管理体系。通过全方位的防…...
Deepseek解读 | UE像素流送与实时云渲染技术的差别
为了实现UE引擎开发的3D/XR程序推流,绝大多数开发者会研究像素流送(Pixel Streaming)的使用方法,并尝试将插件集成在程序中。对于短时、少并发、演示场景而言,像素流送可以满足基本需求。当3D/XR项目进入落地交付周期后…...
CTFSHOW-WEB入门-PHP特性109-115
题目:web 109 1. 题目: 2. 解题思路:题目要求获得两个参数,v1 v2,if语句中的意思是要求两个参数都包含字母,条件满足的话,执行 echo new 类名(方法()…...
模糊综合评价法:原理、步骤与MATLAB实现
引言 在复杂决策场景中,评价对象往往涉及多个相互关联的模糊因素。模糊综合评价法通过建立模糊关系矩阵,结合权重分配与合成算子,实现对多因素系统的科学评价。本文详细讲解模糊综合评价法的数学原理、操作步骤,并辅以MATLAB代码…...
【数据结构-红黑树】
文章目录 红黑树红黑树介绍红黑树的五个基本性质红黑树的平衡原理红黑树的操作红黑树的操作 代码实现节点实现插入和查询操作 红黑树 红黑树介绍 红黑树(Red-Black Tree)是一种自平衡的二叉查找树(Binary Search Tree, BST)&…...
【STM32】舵机SG90
1.舵机原理 舵机内部有一个电位器,当转轴随电机旋转,电位器的电压会发生改变,电压会带动转一定的角度,舵机中的控制板就会电位器输出的电压所代表的角度,与输入的PWM所代表的角度进行比较,从而得出一个旋转…...
【Linux】Socket编程—TCP
🔥 个人主页:大耳朵土土垚 🔥 所属专栏:Linux系统编程 这里将会不定期更新有关Linux的内容,欢迎大家点赞,收藏,评论🥳🥳🎉🎉🎉 文章目…...
c++11 for auto不定参数
数量不定的模板参数。参数分为一个和一包两部分。 冒号的左边声明一个变量。右手边必须是一个容器。从容器(某种数据结构)中找出每一个元素设置到左边这个变量。11之前可以用容器的迭代器去取数据。或者标准库里的foreach...
eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)
说明: 想象一下,你正在用eNSP搭建一个虚拟的网络世界,里面有虚拟的路由器、交换机、电脑(PC)等等。这些设备都在你的电脑里面“运行”,它们之间可以互相通信,就像一个封闭的小王国。 但是&#…...
PHP和Node.js哪个更爽?
先说结论,rust完胜。 php:laravel,swoole,webman,最开始在苏宁的时候写了几年php,当时觉得php真的是世界上最好的语言,因为当初活在舒适圈里,不愿意跳出来,就好比当初活在…...
关于iview组件中使用 table , 绑定序号分页后序号从1开始的解决方案
问题描述:iview使用table 中type: "index",分页之后 ,索引还是从1开始,试过绑定后台返回数据的id, 这种方法可行,就是后台返回数据的每个页面id都不完全是按照从1开始的升序,因此百度了下,找到了…...
鸿蒙DevEco Studio HarmonyOS 5跑酷小游戏实现指南
1. 项目概述 本跑酷小游戏基于鸿蒙HarmonyOS 5开发,使用DevEco Studio作为开发工具,采用Java语言实现,包含角色控制、障碍物生成和分数计算系统。 2. 项目结构 /src/main/java/com/example/runner/├── MainAbilitySlice.java // 主界…...
SAP学习笔记 - 开发26 - 前端Fiori开发 OData V2 和 V4 的差异 (Deepseek整理)
上一章用到了V2 的概念,其实 Fiori当中还有 V4,咱们这一章来总结一下 V2 和 V4。 SAP学习笔记 - 开发25 - 前端Fiori开发 Remote OData Service(使用远端Odata服务),代理中间件(ui5-middleware-simpleproxy)-CSDN博客…...
Hive 存储格式深度解析:从 TextFile 到 ORC,如何选对数据存储方案?
在大数据处理领域,Hive 作为 Hadoop 生态中重要的数据仓库工具,其存储格式的选择直接影响数据存储成本、查询效率和计算资源消耗。面对 TextFile、SequenceFile、Parquet、RCFile、ORC 等多种存储格式,很多开发者常常陷入选择困境。本文将从底…...
Docker 本地安装 mysql 数据库
Docker: Accelerated Container Application Development 下载对应操作系统版本的 docker ;并安装。 基础操作不再赘述。 打开 macOS 终端,开始 docker 安装mysql之旅 第一步 docker search mysql 》〉docker search mysql NAME DE…...
动态 Web 开发技术入门篇
一、HTTP 协议核心 1.1 HTTP 基础 协议全称 :HyperText Transfer Protocol(超文本传输协议) 默认端口 :HTTP 使用 80 端口,HTTPS 使用 443 端口。 请求方法 : GET :用于获取资源,…...
MySQL JOIN 表过多的优化思路
当 MySQL 查询涉及大量表 JOIN 时,性能会显著下降。以下是优化思路和简易实现方法: 一、核心优化思路 减少 JOIN 数量 数据冗余:添加必要的冗余字段(如订单表直接存储用户名)合并表:将频繁关联的小表合并成…...
怎么让Comfyui导出的图像不包含工作流信息,
为了数据安全,让Comfyui导出的图像不包含工作流信息,导出的图像就不会拖到comfyui中加载出来工作流。 ComfyUI的目录下node.py 直接移除 pnginfo(推荐) 在 save_images 方法中,删除或注释掉所有与 metadata …...
