小程序如何实现跨页面通信

前言

最近有很多同学问,小程序里面如何进行跨页面通信。看了下之前的老代码,基本都是基于onShow或者localStorage。虽然可以实现,但是并不怎么优雅。

今天就来聊一聊,小程序的跨页面通信的几种实现方案。或许会有你想要的方案(优雅…)

方式一:onShow + localStorage

业务场景:页面一未登录跳转至登录页面,登录成功后返回页面一,页面一需要更新当前登录态

<!-- 页面一 -->
<template><view><text>{{ name }}</text><view class="login_text">当前是否登录:<text>{{ isLogin ? '是' : '否' }}</text></view><button @tap="gotoLogin">跳转登录</button></view>
</template><script setup lang="ts">
import { ref } from 'vue'
import taro, { useDidShow } from '@tarojs/taro'
const name = ref('前端南玖---小程序页面通信')
const loginStatus = taro.getStorageSync('isLogin') || false
const isLogin = ref<boolean>(loginStatus)const gotoLogin = () => {taro.navigateTo({url: '/pages/login/index'})
}
// 小程序onshow生命周期,从localStorage获取是否登录,更新页面
useDidShow(() => {const loginStatus = taro.getStorageSync('isLogin') || falseisLogin.value = loginStatus
})
</script>
<!--登录页-->
<template><view>登录页面<button @tap="login">登录</button></view>
</template><script setup lang="ts">
import taro from '@tarojs/taro'
const login = () => {taro.login({success: function (res) {console.log('登录成功', res)taro.setStorageSync('isLogin', true)taro.navigateBack()},fail: function (res) {console.log('登录失败', res)}})
}
</script>

在这里插入图片描述

优点: 这种方案可能是最简单的通信方案,比较容易理解

缺点: 如果完成通信后,没有即时清除通信数据,可能会出现问题。另外因为依赖localStorage,而localStorage可能出现读写失败,从面造成通信失败

方式二:onShow + globalData

业务场景同上

这个方案与第一个方案差不多,只不过是将localStorage换成了globalData

Taro框架想要使用小程序的globalData需要使用Taro提供的插件 setGlobalDataPlugin

// app.ts
import { setGlobalDataPlugin } from '@tarojs/taro'const App = createApp({
})// 注册全局数据
App.use(setGlobalDataPlugin, {isLogin: false, // 是否登录
})
// 页面一
// ...import { ref } from 'vue'
import taro, { useDidShow } from '@tarojs/taro'
const app = taro.getApp()
const name = ref('前端南玖---小程序页面通信')
const loginStatus = taro.getStorageSync('isLogin') || false
const isLogin = ref<boolean>(loginStatus)const gotoLogin = () => {taro.navigateTo({url: '/pages/login/index'})
}// 使用globalData
useDidShow(() => {// const loginStatus = taro.getStorageSync('isLogin') || falseconsole.log('app.globalData', app.isLogin)const loginStatus = app.isLogin || falseisLogin.value = loginStatus
})
// 登录页
import taro from '@tarojs/taro'
const app = taro.getApp()
const login = () => {taro.login({success: function (res) {console.log('登录成功', res)app.isLogin = truetaro.navigateBack()},fail: function (res) {console.log('登录失败', res)}})
}

在这里插入图片描述

优点: 实现简单,容易理解。因为不用读写localStorage,直接操作内存,所以相比方式1,速度更快,更可靠

缺点: 同方式1一样,要注意globalData污染

方式三:eventBus发布订阅

我们还可以通过实现一个中央事件总线,通过发布订阅实现跨页面通信。

// eventBus
export default class EventBus {private static instance: EventBusprivate listeners: Record<string, Function[]>private constructor() {this.listeners = {}}public static getInstance() {if (!EventBus.instance) {EventBus.instance = new EventBus()}return EventBus.instance}public on(event: string, callback: Function) {if (!this.listeners[event]) {this.listeners[event] = []}this.listeners[event].push(callback)}public off(event: string, callback: Function) {if (!this.listeners[event]) {return}const index = this.listeners[event].findIndex((listener) => listener === callback)if (index !== -1) {this.listeners[event].splice(index, 1)}}public emit(event: string, ...args: any[]) {if (!this.listeners[event]) {return}this.listeners[event].forEach((listener) => listener(...args))}
}
// app.ts
import EventBus from './utils/eventBus'
// 注册全局事件总线
App.config.globalProperties.$bus = EventBus.getInstance()
// 页面一
import { onMounted, ref, getCurrentInstance } from 'vue'
import taro, { useDidShow } from '@tarojs/taro'
const $bus = getCurrentInstance()?.appContext.config.globalProperties.$busonMounted(() => {// 订阅登录状态isLogin.value = $bus.on('loginStatus', (status: boolean) => {console.log('$bus', status)isLogin.value = status})
})
// 登录页
import taro from '@tarojs/taro'
import { getCurrentInstance } from 'vue'
const $bus = getCurrentInstance()?.appContext.config.globalProperties.$bus
const login = () => {taro.login({success: function (res) {console.log('登录成功', res)// 发布登录状态$bus.emit('loginStatus', true)taro.navigateBack()},fail: function (res) {console.log('登录失败', res)}})
}

在这里插入图片描述

这种方式看着是比前两种优雅了不少,但缺点是需要维护发布的事件,避免重复绑定。

方式四:Taro.eventCenter(taro提供的发布订阅)

Taro 提供了 Taro.Events 来实现消息机制,同时 Taro 还提供了一个全局消息中心 Taro.eventCenter 以供使用,它是 Taro.Events 的实例

import Taro, { Events } from '@tarojs/taro'const events = new Events()// 监听一个事件,接受参数
events.on('eventName', (arg) => {// doSth
})// 监听同个事件,同时绑定多个 handler
events.on('eventName', handler1)
events.on('eventName', handler2)
events.on('eventName', handler3)// 触发一个事件,传参
events.trigger('eventName', arg)// 触发事件,传入多个参数
events.trigger('eventName', arg1, arg2, ...)// 取消监听一个事件
events.off('eventName')// 取消监听一个事件某个 handler
events.off('eventName', handler1)// 取消监听所有事件
events.off()
// 页面一
onMounted(() => {// 订阅登录状态taro.eventCenter.on('loginStatusTaro', (status: boolean) => {console.log('eventCenter', status)isLogin.value = status})
})
// 登录页
const login = () => {taro.login({success: function (res) {console.log('登录成功', res)// 向首页发送数据// eventChannel.emit('acceptDataFromLoginPage', { data: res.code, loginStatus: true })// 触发事件,传递参数taro.eventCenter.trigger('loginStatusTaro', true)// 发布登录状态// $bus.emit('loginStatus', true)taro.navigateBack()},fail: function (res) {console.log('登录失败', res)}})
}

方式五:小程序的EventChannel

页面间事件通信通道

  • EventChannel.emit(string eventName, any args):触发一个事件

  • EventChannel.on(string eventName, function fn):持续监听一个事件

  • EventChannel.once(string eventName, function fn):监听一个事件一次,触发后失效

  • EventChannel.off(string eventName, function fn):取消监听一个事件。给出第二个参数时,只取消给出的监听函数,否则取消所有监听函数

EventChannel借助wx.navigateTo方法,在两个页面之间构建起了数据通道,互相可以通过“派发事件”及“注册这些事件的监听器”来实现基于事件的页面通信。

// 页面一const gotoLogin = () => {taro.navigateTo({url: '/pages/login/index',events: {// 为指定事件添加一个监听器,获取被打开页面传送到当前页面的数据acceptDataFromLoginPage: function(data) {console.log('来自登录页的数据', data)isLogin.value = data.loginStatus},},success: function(res) {// 通过eventChannel向被打开页面传送数据res.eventChannel.emit('acceptDataFromIndexPage', { data: 'nanjiu from index' })}})
}
// 登录页
import taro, { getCurrentPages } from '@tarojs/taro'
const current = getCurrentPages().pop()
const eventChannel = current?.getOpenerEventChannel()
eventChannel.on('acceptDataFromIndexPage', function(data) {console.log('来自首页的数据', data)
})
const login = () => {taro.login({success: function (res) {console.log('登录成功', res)eventChannel.emit('acceptDataFromLoginPage', { data: res.code, loginStatus: true })taro.navigateBack()},fail: function (res) {console.log('登录失败', res)}})
}

在这里插入图片描述

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

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

相关文章

【工具】win-画图 保留图片信息并仅改变图片比例的工具

Windows 系统自带的“画图”工具 Windows 系统自带的“画图”&#xff08;Paint&#xff09;工具可以进行简单的图片编辑&#xff0c;包括调整图片大小和比例。 使用方法&#xff1a; 打开“画图”工具&#xff08;可以通过在开始菜单中搜索“画图”或“Paint”&#xff09;。…

如何编辑autodl中以.bashrc结尾的隐藏文件

在nnunet的运行过程中遇到了设置环境变量的问题。之前没有接触过linux系统&#xff0c;但是autodl里面默认是linux系统。.bashrc 是一个在 Bash shell 启动时执行的脚本文件&#xff0c;常用于设置环境变量、定义别名、加载函数等&#xff0c;用户可以通过编辑这个文件来定制自…

实验3 知识表示与推理

实验3 知识表示与推理 一、实验目的 &#xff08;1&#xff09;掌握知识和知识表示的基本概念&#xff0c;理解其在AI中的深刻含义与意义&#xff1b; &#xff08;2&#xff09;熟悉AI中常用的知识表示方法的优缺点及其应用场景&#xff1b; &#xff08;3&#xff09;掌握产…

在 M1 Mac 上解锁 TensorFlow GPU 加速:从环境搭建到实战验证

在 M1 Mac 上解锁 TensorFlow GPU 加速&#xff1a;从环境搭建到实战验证 前言&#xff1a;苹果芯片的深度学习新纪元 随着 Apple Silicon 芯片的普及&#xff0c;M1/M2/M3 系列 Mac 已成为移动端深度学习开发的新选择。本文将以 TensorFlow 2.x 为例&#xff0c;手把手教你如…

Python 数据分析概述 ①

一文读懂Python数据分析&#xff1a;从基础到实践全攻略 在当今数字化浪潮中&#xff0c;数据分析已然成为解锁海量数据价值的关键钥匙&#xff0c;而Python凭借其独特优势&#xff0c;在数据分析领域大放异彩。今天&#xff0c;咱们就结合教学PPT内容&#xff0c;深入探索Pyt…

【Gin-Web】Bluebell社区项目梳理6:限流策略-漏桶与令牌桶

本文目录 一、限流二、漏桶三、令牌桶算法四、Gin框架中实现令牌桶限流 一、限流 限流又称为流量控制&#xff0c;也就是流控&#xff0c;通常是指限制到达系统的并发请求数。 限流虽然会影响部分用户的使用体验&#xff0c;但是能一定程度上保证系统的稳定性&#xff0c;不至…

Linux高并发服务器开发 第十九天(线程 进程)

目录 1.进程组和会话 2.守护进程 2.1守护进程daemon概念 2.2创建守护进程 3.线程 3.1线程的概念 3.2线程内核三级映射 3.3线程共享 3.4线程优缺点 4.线程控制原语 4.1获取线程id 4.2创建线程 4.3循环创建N个子线 4.4子线程传参地址&#xff0c;错误示例 4.5线程…

软件工程和系统分析与设计

软件工程 1、软件危机 2、软件过程模型 2.1 瀑布模型 2.2原型模型 2.3螺旋模型 2.4敏捷模型 2.5软件统一过程 3、软件能力成熟度模型 CMM 4、软件能力成熟度模型集成 CMMI 系统分析与设计 1、结构化方法SASD 1.1结构化分析 DFD 1.2结构化设计 SD-是一种面向数据流的设计…

Qt/C++面试【速通笔记一】

Qt 信号与槽机制 什么是信号&#xff08;Signal&#xff09;和槽&#xff08;Slot&#xff09;&#xff1f; 在Qt中&#xff0c;信号&#xff08;Signal&#xff09;和槽&#xff08;Slot&#xff09;是实现对象之间通信的一种机制。信号是对象在某些事件发生时发出的通知&…

LangChain大模型应用开发:构建Agent智能体

介绍 大家好&#xff0c;博主又来给大家分享知识了。今天要给大家分享的内容是使用LangChain进行大模型应用开发中的构建Agent智能体。 在LangChain中&#xff0c;Agent智能体是一种能够根据输入的任务或问题&#xff0c;动态地决定使用哪些工具(如搜索引擎、数据库查询等)来…

微服务架构概述及创建父子项目

目录 一&#xff0c;什么是单体架构 二&#xff0c;什么是集群和分布式架构 三&#xff0c;什么是微服务架构 四&#xff0c;解决微服务难题的方案Spring-cloud Spring Cloud Alibaba是阿里巴实现的方案&#xff0c;基于SpringCloud的规范。如果说Spring Cloud Netflix 是…

C/C++跳动的爱心

系列文章 序号直达链接1C/C李峋同款跳动的爱心2C/C跳动的爱心3C/C经典爱心4C/C满屏飘字5C/C大雪纷飞6C/C炫酷烟花7C/C黑客帝国同款字母雨8C/C樱花树9C/C奥特曼10C/C精美圣诞树11C/C俄罗斯方块小游戏12C/C贪吃蛇小游戏13C/C孤单又灿烂的神14C/C闪烁的爱心15C/C哆啦A梦16C/C简单…

量子计算的威胁,以及企业可以采取的措施

当谷歌、IBM、Honeywell和微软等科技巨头纷纷投身量子计算领域时&#xff0c;一场技术军备竞赛已然拉开帷幕。 量子计算虽能为全球数字经济带来巨大价值&#xff0c;但也有可能对相互关联的系统、设备和数据造成损害。这一潜在影响在全球网络安全领域引起了强烈关注。也正因如…

Unity制作游戏——前期准备:Unity2023和VS2022下载和安装配置——附安装包

1.Unity2023的下载和安装配置 &#xff08;1&#xff09;Unity官网下载地址&#xff08;国际如果进不去&#xff0c;进国内的官网&#xff0c;下面以国内官网流程为例子&#xff09; unity中国官网&#xff1a;Unity中国官网 - 实时内容开发平台 | 3D、2D、VR & AR可视化 …

23贪心算法

分发饼干 class Solution { public:int findContentChildren(vector<int>& g, vector<int>& s) {int i0,j0;int count0;sort(s.begin(),s.end());sort(g.begin(),g.end());while(i<g.size()&&j<s.size()){if(g[i]<s[j]){i;j;count;}else…

Spark 和 Flink

Spark 和 Flink 都是目前流行的大数据处理引擎&#xff0c;但它们在架构设计、应用场景、性能和生态方面有较大区别。以下是详细对比&#xff1a; 1. 架构与核心概念 方面Apache SparkApache Flink计算模型微批&#xff08;Micro-Batch&#xff09;为主&#xff0c;但支持结构…

Android 串口通信

引言 在iot项目中&#xff0c;Android 端总会有和硬件通信。 通信这里&#xff1a;串口通信&#xff0c;蓝牙通信或者局域网通信。 这里讲一下串口通信。 什么是串口&#xff1f; “串口”&#xff08;Serial Port&#xff09;通常是指一种用于与外部设备进行串行通信的接口。…

【计算机网络】OSI模型、TCP/IP模型、路由器、集线器、交换机

一、计算机网络分层结构 计算机网络分层结构 指将计算机网络的功能划分为多个层次&#xff0c;每个层次都有其特定的功能和协议&#xff0c;并且层次之间通过接口进行通信。 分层设计的优势&#xff1a; 模块化&#xff1a;各层独立发展&#xff08;如IPv4→IPv6&#xff0c…

从人机环境系统智能角度看传统IP的全球化二次创作法则

从人机环境系统智能的视角看&#xff0c;传统IP的全球化二次创作法则需结合技术、文化、伦理与环境的复杂协同。这一过程不仅是内容的本土化改编&#xff0c;更是人、机器与环境在动态交互中实现价值共创的体现。 一、人机环境系统智能的底层逻辑与IP二次创作的融合 1、感知层&…

实现 INFINI Console 与 GitHub 的单点登录集成:一站式身份验证解决方案

本文将为您详细解析如何通过 GitHub OAuth 2.0 协议&#xff0c;为 INFINI Console 实现高效、安全的单点登录&#xff08;Single Sign-On, SSO&#xff09;集成。通过此方案&#xff0c;用户可直接使用 GitHub 账户无缝登录 INFINI Console&#xff0c;简化身份验证流程&#…