发布订阅模式的应用:解决react中复杂层级的数据交互

以自定义事件为调度中心,创建一个EventCenter类,默认导出一个EventCenter实例

复制代码
// 调度中心
class EventCenter {
    constructor() {
        // 创建一个事件中心,数据模型:{ event : [fn, fn] }
        this.eventCenter = {};
    }

    /**
     * 订阅事件
     * eventName {string} 订阅事件名称
     * backFn {Function} 收到通知的回调函数
     **/
    listen(eventName, backFn) {
        // 确定是否有订阅过该事件
        if (!this.eventCenter[eventName]) {
            this.eventCenter[eventName] = [];
        }
        // 控制backFn的唯一性,防止出现重复的backFn
        const backFnList = this.eventCenter[eventName];
        const hasFn = backFnList.some((fn) => fn === backFn);
        if (hasFn) {
            return;
        }
        backFnList.push(backFn);
    }
    
    /**
     * 取消订阅函数
     * eventName {string} 取消订阅的事件名称
     * backFn {Function} 取消订阅的回调函数
     **/
    unListen(eventName, backFn) {
        // 判断是否有该事件,及该事件是否有订阅者
        let backFnList = this.eventCenter[eventName];
        if (!backFnList || backFnList.length === 0) {
            return;
        }
        backFnList = backFnList.filter((fn) => fn !== backFn); // 过滤掉取消订阅的函数
    }
    
    /**
     * 发布函数
     * eventName {string} 发布通知的事件名称
     * ...args {any} 发布通知的相关参数
     **/
    publish(eventName, ...args) {
        // 判断是否有该事件,及该事件是否有订阅者
        const backFnList = this.eventCenter[eventName];
        if (!backFnList || backFnList.length === 0) {
            return;
        }
        backFnList.forEach((backFn) => {
            backFn.apply(null, args); // 调用回调函数,并传入发布时的相关参数
        });
    }
}

export default new EventCenter(); // 默认导出一个实例

publish中使用了apply是为了把参数进行转换,具体请参考此文章:《javascript 关于bind、apply、call函数改变this指向》
运用场景一: 添加商品与购物车的数量变化 ,这种情况往往是不知道跨越多少不同层级的情况。
发布者 :商品加入购物车的操作
订阅者:购物车的数量

复制代码
// 商品列表 ProductList.js
import EventCenter from './EventCenter';
const ProductList = ()=>{
	// 模拟商品数据
	const productList = [
		{id:1, name: '商品1'}
	];
	const addBuyCar = (product)=>{
		EventCenter.publish('addBuyCar', product); // 发布通知,添加商品到购物车
	}
	return (
		<ul>
			{productList.map(product => {
				return (<li key={product.id}>
					商品名称:{product.name} 
					<button onClick={()=>addBuyCar(product)}>加入购物车</button>
				</li>)
			})}
		</ul>
	);
}
export default ProductList;

// 购物车商品种类数量 BuyCarCount.js
import {useEffect, useState} from 'react';
import EventCenter from './EventCenter';
const BuyCarCount = ()=>{

	const [productList, setProductList] = useState([]); // 产品种类列表
	
	useEffect(()=>{
		// 订阅添加商品时的回调函数
		const backFn = (product)=>{
			setProductList(pList => {
				// 判断购物车是否已存在此商品,没有才添加进去
				const hasProduct = pList.some(pro => pro.id === product.id);
				if(!hasProduct){
					return [...pList, product];
				}
				return pList;
			});
		}
		
		EventCenter.listen('addBuyCar', backFn); // 订阅addBuyCar事件
		return ()=> EventCenter.unListen('addBuyCar', backFn); // 组件销毁后,取消订阅
	}, []);
	
	return <div>购物车商品种类:{productList.map(item=> item.name)}</div>;
}
export default BuyCarCount;
相关推荐
兆子龙17 分钟前
当「多应用共享组件」成了刚需:我们从需求到模块联邦的落地小史
前端·架构
StarkCoder18 分钟前
SwiftUI路由管理架构揭秘:从混乱到优雅的蜕变
前端框架
Qinana19 分钟前
从代码到智能体:MCP 协议如何重塑 AI Agent 的边界
前端·javascript·mcp
Wect28 分钟前
LeetCode 130. 被围绕的区域:两种解法详解(BFS/DFS)
前端·算法·typescript
不会敲代码132 分钟前
从入门到进阶:手写React自定义Hooks,让你的组件更简洁
前端·react.js
用户54330814419434 分钟前
拆完 Upwork 前端我沉默了:你天天卷的那些技术,人家根本没用
前端
洋洋技术笔记34 分钟前
Vue实例与数据绑定
前端·vue.js
Marshall15135 分钟前
zzy-scroll-timer:一个跨框架的滚动定时器插件
前端·javascript
明月_清风2 小时前
打字机效果优化:用 requestAnimationFrame 缓冲高频文字更新
前端·javascript
明月_清风2 小时前
Markdown 预解析:别等全文完了再渲染,如何流式增量渲染代码块和公式?
前端·javascript