(十)安卓开发中的Activity之间的通信使用详解

news2025/4/18 3:52:41

在 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() {
        // 假设我们获得了用户的用户名和ID
        String 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);

        // 启动 MainActivity
        startActivity(intent);
    }
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_login);
        // 模拟登录按钮点击后进行登录操作
    }
}

MainActivity.java:

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // 获取启动该 Activity 的 Intent
        Intent 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;

    @Override
    protected 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);
        });
    }

    // 接收返回结果
    @Override
    protected 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 {

    @Override
    protected 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 -> {
            // 在退出当前页面之前,回传数据给 ActivityA
            Intent 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();
                }
            }
        }
    );

    @Override
    protected 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)

在实际应用中,有时需要传递不只是基本数据类型,而是自定义对象。这时可以让对象实现 SerializableParcelable 接口。

使用 Serializable 示例

User.java:

public class User implements Serializable {
    private String name;
    private int age;

    // 构造器、getter 与 setter
    public 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>() {
        @Override
        public User createFromParcel(Parcel in) {
            return new User(in);
        }
        @Override
        public User[] newArray(int size) {
            return new User[size];
        }
    };

    @Override
    public int describeContents() {
        return 0;
    }
    
    @Override
    public 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)可以降低耦合度,便于维护和扩展。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于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 修改登录密码&#xff0c;可以使用生成按钮生成密码&#xff0c;连接数据库&#xff0c;默认设置为neo4j…

【学习笔记】文件上传漏洞--二次渲染、.htaccess、变异免杀

目录 第十二关 远程包含地址转换 第十三关 突破上传删除 条件竞争 第十四关 二次渲染 第十五关 第十六关 第十七关 .htaccess 第十八关 后门免杀 第十九关 日志包含 第十二关 远程包含地址转换 延续第十一关&#xff0c;加一个文件头&#xff0c;上传成功&#xff0c…

【OS】Process Management(3)

《计算机操作系统&#xff08;第三版&#xff09;》&#xff08;汤小丹&#xff09;学习笔记 文章目录 5、进程通信&#xff08;Inter-Process Communication&#xff09;5.1、进程通信的类型5.1.1、共享存储器系统&#xff08;Shared Memory System&#xff09;5.1.2、消息传递…

单reactor实战

前言&#xff1a;reactor作为一种高性能的范式&#xff0c;值得我们学习 本次目标 实现一个基于的reactor 具备echo功能的服务器 核心组件 Reactor本身是靠一个事件驱动的框架,无疑引出一个类似于moduo的"EventLoop "以及boost.asio中的context而言&#xff0c;不断…

初阶C++笔记第一篇:C++基础语法

虽然以下大多数知识点都在C语言中学过&#xff0c;但还是有一些知识点和C语言不同&#xff0c;比如&#xff1a;代码格式、头文件、关键字、输入输出、字符串类型等... 1. 初识C 1.1 第一个C程序 编写C分为4个步骤&#xff1a; 创建项目创建文件编写代码运行程序 C的第一条…

无需libpacp库,BPF指令高效捕获指定数据包

【环境】无libpacp库的Linux服务器 【要求】高效率读取数据包&#xff0c;并过滤指定端口和ip 目前遇到两个问题 一是手写BPF&#xff0c;难以兼容&#xff0c;有些无法正常过滤二是性能消耗问题&#xff0c;尽可能控制到1% 大方向&#xff1a;过滤数据包要在内核层处理&…

react实现上传图片到阿里云OSS以及问题解决(保姆级)

一、优势 提高上传速度&#xff1a;前端直传利用了浏览器与 OSS 之间的直接连接&#xff0c;能够充分利用用户的网络带宽。相比之下&#xff0c;后端传递文件时&#xff0c;文件需要经过后端服务器的中转&#xff0c;可能会受到后端服务器网络环境和处理能力的限制&#xff0c;…

Python 字典和集合(常见的映射方法)

本章内容的大纲如下&#xff1a; 常见的字典方法 如何处理查找不到的键 标准库中 dict 类型的变种set 和 frozenset 类型 散列表的工作原理 散列表带来的潜在影响&#xff08;什么样的数据类型可作为键、不可预知的 顺序&#xff0c;等等&#xff09; 常见的映射方法 映射类型…

Matlab轴承故障信号仿真与故障分析

1.摘要 本文介绍了一个基于Matlab的轴承故障信号仿真与分析程序&#xff0c;旨在模拟和分析轴承内圈故障信号的特征。程序首先通过生成故障信号、共振信号和调制信号&#xff0c;添加噪声和离散化处理&#xff0c;构建模拟的振动信号&#xff0c;并保存相关数据。通过快速傅里…

Linux 进程 | 概念 / 特征 / 状态 / 优先级 / 空间

注&#xff1a; 本文为 “Linux 进程” 相关文章合辑。 未整理去重。 Linux 进程概念&#xff08;精讲&#xff09; A little strawberry 于 2021-10-15 10:23:55 发布 基本概念 课本概念&#xff1a;程序的一个执行实例&#xff0c;正在执行的程序等。 内核观点&#xff…

重回全面发展亲自操刀

项目场景&#xff1a; 今年工作变动&#xff0c;优化后在一家做国有项目的私人公司安顿下来了。公司环境不如以前&#xff0c;但是好在瑞欣依然可以每天方便的买到。人文氛围挺好&#xff0c;就是工时感觉有点紧&#xff0c;可能长期从事产品迭代开发&#xff0c;一下子转变做项…

3D珠宝渲染用什么软件比较好?渲染100邀请码1a12

印度珠宝商 Mohar Fine Jewels 和英国宝石商 Gemfields 在今年推出了合作珠宝系列——「Emeralds in Full Bloom」&#xff0c;它的灵感源自花草绽放的春季田野&#xff0c;共有 39 件作品&#xff0c;下面这个以植物为主题的开口手镯就是其中一件。 在数字时代&#xff0c;像这…

【数据结构】邻接矩阵完全指南:原理、实现与稠密图优化技巧​

邻接矩阵 导读一、图的存储结构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解析 结语 导读 大…

【嵌入式-stm32电位器控制以及旋转编码器控制LED亮暗】

嵌入式-stm32电位器控制LED亮暗 任务1代码1Key.cKey.hTimer.cTimer.hPWM.cPWM.hmain.c 实验现象1任务2代码2Key.cKey.hmain.c 实验现象2问题与解决总结 源码框架取自江协科技&#xff0c;在此基础上做扩展开发。 任务1 本文主要介绍利用stm32f103C8T6实现电位器控制PWM的占空比…

Uniapp 集成极光推送(JPush)完整指南

文章目录 前言一、准备工作1. 注册极光开发者账号2. 创建应用3. Uniapp项目准备 二、集成极光推送插件方法一&#xff1a;使用UniPush&#xff08;推荐&#xff09;方法二&#xff1a;手动集成极光推送SDK 三、配置原生平台参数四、核心功能实现1. 获取RegistrationID2. 设置别…

2025年常见渗透测试面试题-sql(题目+回答)

网络安全领域各种资源&#xff0c;学习文档&#xff0c;以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具&#xff0c;欢迎关注。 目录 SQLi 一、发现test.jsp?cid150 注入点的5种WebShell获取思路 1. 文件写入攻击 2. 日志文件劫持 3.…

【RabbitMQ】队列模型

1.概述 RabbitMQ作为消息队列&#xff0c;有6种队列模型&#xff0c;分别在不同的场景进行使用&#xff0c;分别是Hello World&#xff0c;Work queues&#xff0c;Publish/Subscribe&#xff0c;Routing&#xff0c;Topics&#xff0c;RPC。 下面就分别对几个模型进行讲述。…

StarRocks 助力首汽约车精细化运营

作者&#xff1a;任智红&#xff0c;首汽约车大数据负责人 更多交流&#xff0c;联系我们&#xff1a;https://wx.focussend.com/weComLink/mobileQrCodeLink/334%201%202/ffbe5 导读&#xff1a; 本文整理自首汽约车大数据负责人任智红在 StarRocks 年度峰会上的演讲&#xf…

痉挛性斜颈康复助力:饮食调养指南

痉挛性斜颈患者除了积极治疗&#xff0c;合理饮食也能辅助缓解症状&#xff0c;提升生活质量。其健康饮食可从以下方面着手&#xff1a; 高蛋白质食物助力肌肉修复 痉挛性斜颈会导致颈部肌肉异常收缩&#xff0c;消耗较多能量&#xff0c;蛋白质有助于肌肉的修复与维持。日常可…

mysql镜像创建docker容器,及其可能遇到的问题

前提&#xff0c;已经弄好基本的docker服务了。 一、基本流程 1、目录准备 我自己的资料喜欢放在 /data 目录下&#xff0c;所以老规矩&#xff1a; 先进入 /data 目录&#xff1a; cd /data 创建 mysql 目录并进入&#xff1a; mkdir mysql cd mysql 2、镜像查找 docke…