(十)安卓开发中的Activity之间的通信使用详解
在 Android 开发中,Activity 之间的通信是非常常见且核心的功能之一,常见的方式包括:
- 使用显式 Intent 传递数据
- 使用隐式 Intent 实现跨组件调用
- 使用 startActivityForResult(或新版 Activity Result API)回传数据
- 传递复杂数据(如 Serializable/Parcelable 对象)
下面将结合代码和具体使用场景,详细讲解这几种方式。
1. 使用显式 Intent 传递数据
场景说明
假设有两个界面:登录界面(LoginActivity)和主界面(MainActivity)。在登录成功后,LoginActivity 需要将用户信息传递给 MainActivity。
示例代码
LoginActivity.java:
public class LoginActivity extends AppCompatActivity {// 模拟登录成功后传递数据private void loginSuccess() {// 假设我们获得了用户的用户名和IDString username = "JohnDoe";int userId = 123;Intent intent = new Intent(LoginActivity.this, MainActivity.class);// 将数据存入 Intent 的 Extras 里,Key-Value 方式intent.putExtra("EXTRA_USERNAME", username);intent.putExtra("EXTRA_USER_ID", userId);// 启动 MainActivitystartActivity(intent);}@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_login);// 模拟登录按钮点击后进行登录操作}
}
MainActivity.java:
public class MainActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// 获取启动该 Activity 的 IntentIntent intent = getIntent();if (intent != null) {// 使用 getStringExtra、getIntExtra 等方法获取传递的数据String username = intent.getStringExtra("EXTRA_USERNAME");int userId = intent.getIntExtra("EXTRA_USER_ID", -1); // -1 为默认值// 根据获得的数据执行后续逻辑,比如显示欢迎信息TextView welcomeTextView = findViewById(R.id.welcomeTextView);welcomeTextView.setText("欢迎 " + username + ", 你的用户ID是:" + userId);}}
}
原理说明
- Intent:用于启动另一个 Activity。通过
putExtra()方法,可以将数据以键值对(Key-Value)的形式传递。 - Extras:在目标 Activity 中,通过
getIntent()获取传递过来的 Intent,再通过相关的getXXXExtra()方法获取数据。
2. 使用 startActivityForResult 回传数据
场景说明
有两个界面:A 页面和 B 页面。A 页面需要启动 B 页面获取用户的选择或输入,并在 B 页面关闭后拿到返回的数据。
示例代码
ActivityA.java:
public class ActivityA extends AppCompatActivity {private static final int REQUEST_CODE = 1001;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_a);Button openBButton = findViewById(R.id.openBButton);openBButton.setOnClickListener(v -> {// 启动 ActivityB 并等待返回结果Intent intent = new Intent(ActivityA.this, ActivityB.class);intent.putExtra("EXTRA_MESSAGE", "来自ActivityA的消息");startActivityForResult(intent, REQUEST_CODE);});}// 接收返回结果@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);if (requestCode == REQUEST_CODE && resultCode == RESULT_OK) {// 从 data 中取出 B 页面回传的数据String resultMessage = data.getStringExtra("RESULT_MESSAGE");Toast.makeText(this, "返回信息:" + resultMessage, Toast.LENGTH_LONG).show();}}
}
ActivityB.java:
public class ActivityB extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_b);// 获取 A 页面传递过来的数据Intent intent = getIntent();String message = intent.getStringExtra("EXTRA_MESSAGE");// 显示 A 页面传过来的数据TextView tvMessage = findViewById(R.id.tvMessage);tvMessage.setText(message);// 假设用户在此页面进行选择或输入Button sendResultButton = findViewById(R.id.sendResultButton);sendResultButton.setOnClickListener(v -> {// 在退出当前页面之前,回传数据给 ActivityAIntent data = new Intent();data.putExtra("RESULT_MESSAGE", "这是来自ActivityB的回传数据");setResult(RESULT_OK, data);finish(); // 结束当前 Activity});}
}
原理说明
- startActivityForResult:在 ActivityA 中启动 ActivityB,同时传递一个请求码。
- setResult():在 ActivityB 中设置返回数据以及结果状态,然后调用
finish()关闭当前 Activity。 - onActivityResult():ActivityA 重写该方法接收 B 页面返回的数据,其中请求码和结果码用来判断返回数据的合法性。
注意: 从 AndroidX 开始,官方推荐使用 Activity Result APIs 替代传统的
startActivityForResult,该 API 更容易管理并能有效防止内存泄漏和其他生命周期相关问题。
3. 使用 Activity Result API(新版方式)
场景说明
与上面的例子类似,假设在 A 页面启动 B 页面获取用户数据,不过采用新版 API。
示例代码
ActivityA.java:
public class ActivityA extends AppCompatActivity {// 注册 ActivityResultLauncher 用于启动另一个 Activity 并接收结果private ActivityResultLauncher<Intent> launcher = registerForActivityResult(new ActivityResultContracts.StartActivityForResult(),result -> {if (result.getResultCode() == Activity.RESULT_OK) {Intent data = result.getData();if (data != null) {String resultMessage = data.getStringExtra("RESULT_MESSAGE");Toast.makeText(ActivityA.this, "返回信息:" + resultMessage, Toast.LENGTH_LONG).show();}}});@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_a);Button openBButton = findViewById(R.id.openBButton);openBButton.setOnClickListener(v -> {Intent intent = new Intent(ActivityA.this, ActivityB.class);launcher.launch(intent);});}
}
ActivityB.java:
代码与使用 startActivityForResult 的版本相同,设置结果并调用 finish() 完成返回操作。
4. 传递复杂数据(Serializable/Parcelable)
在实际应用中,有时需要传递不只是基本数据类型,而是自定义对象。这时可以让对象实现 Serializable 或 Parcelable 接口。
使用 Serializable 示例
User.java:
public class User implements Serializable {private String name;private int age;// 构造器、getter 与 setterpublic User(String name, int age) {this.name = name;this.age = age;}public String getName() { return name; }public int getAge() { return age; }
}
传递数据示例:
// ActivityA 中
User user = new User("Alice", 28);
Intent intent = new Intent(ActivityA.this, ActivityB.class);
intent.putExtra("EXTRA_USER", user);
startActivity(intent);// ActivityB 中
User user = (User) getIntent().getSerializableExtra("EXTRA_USER");
if (user != null) {Log.d("ActivityB", "User Name: " + user.getName() + ", Age: " + user.getAge());
}
使用 Parcelable 示例
相比于 Serializable,Parcelable 的效率更高,因为它是专门为 Android 设计的序列化方案。编写 Parcelable 需要实现更多接口方法,但可以使用 Android Studio 插件自动生成代码。
public class User implements Parcelable {private String name;private int age;// 构造器public User(String name, int age) {this.name = name;this.age = age;}protected User(Parcel in) {name = in.readString();age = in.readInt();}public static final Creator<User> CREATOR = new Creator<User>() {@Overridepublic User createFromParcel(Parcel in) {return new User(in);}@Overridepublic User[] newArray(int size) {return new User[size];}};@Overridepublic int describeContents() {return 0;}@Overridepublic void writeToParcel(Parcel dest, int flags) {dest.writeString(name);dest.writeInt(age);}// getter 方法public String getName() { return name; }public int getAge() { return age; }
}
传递 Parcelable 对象:
// ActivityA 中传递对象
User user = new User("Bob", 35);
Intent intent = new Intent(ActivityA.this, ActivityB.class);
intent.putExtra("EXTRA_USER", user);
startActivity(intent);// ActivityB 中接收对象
User user = getIntent().getParcelableExtra("EXTRA_USER");
if (user != null) {Log.d("ActivityB", "User Name: " + user.getName() + ", Age: " + user.getAge());
}
5. 小结与使用建议
-
明确场景需求:
如果只是简单数据传递,使用基本类型的 extra 即可;如果需要回传结果,则采用 startActivityForResult 或新版 Activity Result API。 -
注意数据安全与优化:
大量数据或复杂对象传递时,尽量使用 Parcelable 来提升性能,同时避免在 Intent 中传递过大的数据(如大图片、文件等),推荐使用全局单例、数据库或文件方式传递数据。 -
代码维护和扩展:
将通信逻辑和数据传输封装成统一的工具类或使用第三方库(如 EventBus)可以降低耦合度,便于维护和扩展。
相关文章:
(十)安卓开发中的Activity之间的通信使用详解
在 Android 开发中,Activity 之间的通信是非常常见且核心的功能之一,常见的方式包括: 使用显式 Intent 传递数据使用隐式 Intent 实现跨组件调用使用 startActivityForResult(或新版 Activity Result API)回传数据传递…...
python 浅拷贝copy与深拷贝deepcopy 理解
一 浅拷贝与深拷贝 1. 浅拷贝 浅拷贝只复制了对象本身(即c中的引用)。 2. 深拷贝 深拷贝创建一个新的对象,同时也会创建所有子对象的副本,因此新对象与原对象之间完全独立。 二 代码理解 1. 案例一 a 10 b a b 20 print…...
基于neo4j存储知识树-mac
1、安装jdk21 for mac(jdk-21_macos-aarch64_bin.dmg) 2、安装neo4j for mac(neo4j-community-5.26.0-unix.tar.gz) 3、使用默认neo4j/neo4j登录http://localhost:7474 修改登录密码,可以使用生成按钮生成密码,连接数据库,默认设置为neo4j…...
Tiktok 关键字 视频及评论信息爬虫(1) [2025.04.07]
🙋♀️Tiktok APP的基于关键字检索的视频及评论信息爬虫共分为两期,希望对大家有所帮助。 第一期见下文。 第二期:基于视频URL的评论信息爬取 1. Node.js环境配置 首先配置 JavaScript 运行环境(如 Node.js)&#x…...
基于人工智能的高中教育评价体系重构研究
基于人工智能的高中教育评价体系重构研究 一、引言 1.1 研究背景 在科技飞速发展的当下,人工智能技术已广泛渗透至各个领域,教育领域亦不例外。人工智能凭借其强大的数据处理能力、智能分析能力和个性化服务能力,为教育评价体系的创新与发…...
【学习笔记】文件上传漏洞--二次渲染、.htaccess、变异免杀
目录 第十二关 远程包含地址转换 第十三关 突破上传删除 条件竞争 第十四关 二次渲染 第十五关 第十六关 第十七关 .htaccess 第十八关 后门免杀 第十九关 日志包含 第十二关 远程包含地址转换 延续第十一关,加一个文件头,上传成功,…...
C++ 基础进阶
C 基础进阶 内容概述: 函数重载:int add(int x, inty);,long long add(long long x, long long y);,double add(double x, double y);模板函数:template<typename T> 或 template<class T>结构体&#x…...
【OS】Process Management(3)
《计算机操作系统(第三版)》(汤小丹)学习笔记 文章目录 5、进程通信(Inter-Process Communication)5.1、进程通信的类型5.1.1、共享存储器系统(Shared Memory System)5.1.2、消息传递…...
单reactor实战
前言:reactor作为一种高性能的范式,值得我们学习 本次目标 实现一个基于的reactor 具备echo功能的服务器 核心组件 Reactor本身是靠一个事件驱动的框架,无疑引出一个类似于moduo的"EventLoop "以及boost.asio中的context而言,不断…...
初阶C++笔记第一篇:C++基础语法
虽然以下大多数知识点都在C语言中学过,但还是有一些知识点和C语言不同,比如:代码格式、头文件、关键字、输入输出、字符串类型等... 1. 初识C 1.1 第一个C程序 编写C分为4个步骤: 创建项目创建文件编写代码运行程序 C的第一条…...
java基础 流(Stream)
Stream Stream 的核心概念核心特点 Stream 的操作分类中间操作(Intermediate Operations)终止操作(Terminal Operations) Stream 的流分类顺序流(Sequential Stream)并行流(Parallel Stream&…...
【AI】prompt engineering
prompt engineering ## prompt engineering ## prompt engineering ## prompt engineering 一、定义 Prompt 工程(Prompt Engineering)是指在使用语言模型(如 ChatGPT、文心一言等)等人工智能工具时,设计和优化输入提…...
无需libpacp库,BPF指令高效捕获指定数据包
【环境】无libpacp库的Linux服务器 【要求】高效率读取数据包,并过滤指定端口和ip 目前遇到两个问题 一是手写BPF,难以兼容,有些无法正常过滤二是性能消耗问题,尽可能控制到1% 大方向:过滤数据包要在内核层处理&…...
LeetCode算法题(Go语言实现)_36
题目 给定一个二叉树的根节点 root ,和一个整数 targetSum ,求该二叉树里节点值之和等于 targetSum 的 路径 的数目。 路径 不需要从根节点开始,也不需要在叶子节点结束,但是路径方向必须是向下的(只能从父节点到子节点…...
react实现上传图片到阿里云OSS以及问题解决(保姆级)
一、优势 提高上传速度:前端直传利用了浏览器与 OSS 之间的直接连接,能够充分利用用户的网络带宽。相比之下,后端传递文件时,文件需要经过后端服务器的中转,可能会受到后端服务器网络环境和处理能力的限制,…...
无法看到新安装的 JDK 17
在 Linux 系统中使用 update-alternatives --config java 无法看到新安装的 JDK 17,可能是由于 JDK 未正确注册到系统备选列表中。 一、原因分析 JDK 未注册到 update-alternatives update-alternatives 工具需要手动注册 JDK 路径后才能识别新版本。如果仅安装 JDK…...
LeetCode 3396.使数组元素互不相同所需的最少操作次数:O(n)一次倒序遍历
【LetMeFly】3396.使数组元素互不相同所需的最少操作次数:O(n)一次倒序遍历 力扣题目链接:https://leetcode.cn/problems/minimum-number-of-operations-to-make-elements-in-array-distinct/ 给你一个整数数组 nums,你需要确保数组中的元素…...
Vue2 快速过度 Vue3 教程 (后端学习)
隔好长一段时间没有写文章了,因为最近公司一个项目进度很赶,导致一直加班,没有时间空出来学习新的东西,这次趁着周末,赶紧补一下之前落下的一直想重新学一下整个大前端生态的想法,这次写一篇自己学习Vue3的…...
供应链管理-职业规划:数字化供应链管理专家 / 供应链管理商业模式专家 / 供应链管理方案专家
一、背景阐述 依据联合国产业分类标准,工业体系被细致划分为41个工业大类、207个工业中类以及666个工业小类。中国凭借其独特的产业布局,成为全球唯一一个全面涵盖所有这些门类的国家,成功构建起独立且完备的现代工业体系。这一辉煌成就&…...
无状态版的DHCPv6是不是SLAAC? 笔记250405
无状态版的DHCPv6是不是SLAAC? 笔记250405 无状态版 DHCPv6 不是 SLAAC,但二者在 IPv6 网络中可协同工作。以下是核心区别与协作关系: 本质区别 特性SLAAC无状态 DHCPv6主要功能生成 IPv6 地址(基于路由器通告的前缀)分发 DNS、…...
遍历算法及其应用详解
李升伟 整理 什么是遍历? 遍历是指按照某种规则或顺序,系统地访问数据结构(如树、图等)中的每个节点一次且仅一次的过程。遍历是算法设计中的基本操作,用于访问、检查或修改数据结构中的所有元素。 主要遍历算法 1…...
Python 字典和集合(常见的映射方法)
本章内容的大纲如下: 常见的字典方法 如何处理查找不到的键 标准库中 dict 类型的变种set 和 frozenset 类型 散列表的工作原理 散列表带来的潜在影响(什么样的数据类型可作为键、不可预知的 顺序,等等) 常见的映射方法 映射类型…...
基于大模型的ALS预测与手术优化系统技术方案
目录 技术方案文档:基于大模型的ALS预测与手术优化系统1. 数据预处理与特征工程模块流程图伪代码2. 多模态融合预测模型模型架构图伪代码3. 术中实时监测与动态干预系统系统流程图伪代码4. 统计验证与可解释性模块验证流程图伪代码示例(SHAP分析)5. 健康教育与交互系统系统架…...
创建一个简单的HTML游戏站
创建一个简单的HTML游戏站涉及多个步骤,包括规划网站结构、设计用户界面、编写游戏逻辑以及测试和部署。下面是一个详细的步骤指南: 1. 规划网站结构 确定目标受众:了解你的目标用户群体。选择游戏类型:决定你要开发的游戏类型&…...
Matlab轴承故障信号仿真与故障分析
1.摘要 本文介绍了一个基于Matlab的轴承故障信号仿真与分析程序,旨在模拟和分析轴承内圈故障信号的特征。程序首先通过生成故障信号、共振信号和调制信号,添加噪声和离散化处理,构建模拟的振动信号,并保存相关数据。通过快速傅里…...
Linux 进程 | 概念 / 特征 / 状态 / 优先级 / 空间
注: 本文为 “Linux 进程” 相关文章合辑。 未整理去重。 Linux 进程概念(精讲) A little strawberry 于 2021-10-15 10:23:55 发布 基本概念 课本概念:程序的一个执行实例,正在执行的程序等。 内核观点ÿ…...
项目中如何防止超卖
什么是超卖?假如只剩下一个库存,却被多个订单买到了,简单理解就是库存不够了还能正常下单。 方案1:数据库行级锁 1. 实体类 Data TableName("product") public class Product {TableId(type IdType.AUTO)private Lon…...
重回全面发展亲自操刀
项目场景: 今年工作变动,优化后在一家做国有项目的私人公司安顿下来了。公司环境不如以前,但是好在瑞欣依然可以每天方便的买到。人文氛围挺好,就是工时感觉有点紧,可能长期从事产品迭代开发,一下子转变做项…...
3D珠宝渲染用什么软件比较好?渲染100邀请码1a12
印度珠宝商 Mohar Fine Jewels 和英国宝石商 Gemfields 在今年推出了合作珠宝系列——「Emeralds in Full Bloom」,它的灵感源自花草绽放的春季田野,共有 39 件作品,下面这个以植物为主题的开口手镯就是其中一件。 在数字时代,像这…...
【数据结构】邻接矩阵完全指南:原理、实现与稠密图优化技巧
邻接矩阵 导读一、图的存储结构1.1 分类 二、邻接矩阵法2.1 邻接矩阵2.2 邻接矩阵存储网 三、邻接矩阵的存储结构四、算法评价4.1 时间复杂度4.2 空间复杂度 五、邻接矩阵的特点5.1 特点1解析5.2 特点2解析5.3 特点3解析5.4 特点4解析5.5 特点5解析5.6 特点6解析 结语 导读 大…...
