用23种设计模式打造一个cocos creator的游戏框架----(十四)观察者模式

1、模式标准

模式名称:观察者模式

模式分类:行为型

模式意图:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。

结构图:

适用于:

1、当一个抽象模型有两个方面,其中一个方面依赖于另一个方面,将这两者封装在独立的对象中以使它们可以各自独立地改变和复用。

2、当对一个对象的改变需要同时改变其他对象,而不知道具体有多少对象有待改变时。

3、当一个对象必须通知其他对象,而它又不能假定其他对象是谁,即不希望这些对象是紧耦合的

2、分析与设计

观察者模式的主要使用场景是gui、网络服务器、发布订阅系统。在前面的设计模式中,我们采用了代理模式,通过"代理拦截修改"实现了数据层和视图层之间的响应式。虽然实现了响应式,但是其中的数据不是真实的数据源。真实数据源发生变化时,还需要通过xhgame.vm.gateVM.ps = 123来手动触发修改。我们希望的数据自动监听真实数据源的变化自动触发响应式。下面我们通过观察者模式来实现它,在游戏框架里我们统一使用modelComp中的数据源,先修改一下我们的意图

意图:定义对象(modelComp)间的一种一对多的依赖关系,当一个对象(modelComp)的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。

3、开始打造

主题接口

TypeScript 复制代码
export interface ISubject {
    observers: IObserver[]
    attach(observer: IObserver): void
    detach(observer: IObserver): void
    notify(): void
}

观察者接口

TypeScript 复制代码
export interface IObserver {
    update(subject: ISubject): void
}

这里新增一个modelComp 的抽象类继承之前的ecs中Comp

TypeScript 复制代码
export abstract class ModelComp<T> extends Comp implements ISubject {
    callback: Function = null
    reset(): void {

    }
    onAttach(entity: Entity) {

    }

    onDetach(entity: Entity) {

    }
    // 观察者模式
    observers: IObserver[] = []

    attach(observer: IObserver): void {
        const isExist = this.observers.includes(observer);
        if (isExist) {
            return console.log('已存在监听者');
        }
        this.observers.push(observer);
    }
    detach(observer: IObserver): void {
        const observerIndex = this.observers.indexOf(observer);
        if (observerIndex === -1) {
            return console.log('不存在监听者');
        }
        this.observers.splice(observerIndex, 1);
    }
    notify(): void {
        console.log('ModelSubject notify')
        for (const observer of this.observers) {
            observer.update(this);
        }
    }

}

接着是具体的主题,玩家的modelComp

TypeScript 复制代码
export class PlayerModelComp extends ModelComp<IPlayerInfo_JCQ> {

    callback: Function = null

    _playerInfo: IPlayerInfo_JCQ = {
        id: 0,
        openid: '',
        server_no: '',
        platform: '',
        ps: 0,
        gold: 0,
        diamond: 0,
        last_battle_id: 0
    }
    get playerInfo() {
        return this._playerInfo
    }
    set playerInfo(val) {
        this._playerInfo = val
        this.notify()
    }
   

    reset() {
        this.callback = null
        this._playerInfo = {
            id: 0,
            openid: '',
            server_no: '',
            platform: '',
            ps: 0,
            gold: 0,
            diamond: 0,
            last_battle_id: 0
        }
        // 
        this.observers = []

    }


    onAttach(entity: Entity) {
        this.callback && this.callback()
    }

    onDetach(entity: Entity) {

    }
}

设置一个玩家信息的监听者

TypeScript 复制代码
export class PlayerInfoObserver implements IObserver {
    update(subject: PlayerModelComp): void {
        const playerInfo = subject.playerInfo
        xhgame.vm.gateVM.ps = playerInfo.ps
        xhgame.vm.gateVM.gold = playerInfo.gold
        xhgame.vm.gateVM.diamond = playerInfo.diamond
    }
}

4、开始使用

TypeScript 复制代码
export class JCQPlayerEntity extends Entity {
    model: PlayerModelComp

    init() {
        this.model = this.attachComponent(PlayerModelComp)
    }

}

对playerModelComp添加多个监听者

TypeScript 复制代码
xhgame.game.playerEntity.model.attach(new PlayerInfoObserver())
xhgame.game.playerEntity.model.attach(new OtherObserver())

观察者内原本有自己的state或者info,现在用了vm来代替了

TypeScript 复制代码
export class PlayerInfoObserver implements IObserver {
    update(subject: PlayerModelComp): void {
        const playerInfo = subject.playerInfo
        xhgame.vm.gateVM.ps = playerInfo.ps
        xhgame.vm.gateVM.gold = playerInfo.gold
        xhgame.vm.gateVM.diamond = playerInfo.diamond
    }
}
相关推荐
许彰午8 分钟前
34_Java设计模式之单例模式
java·单例模式·设计模式
喵星人工作室1 小时前
C++火影忍者1.1.8
开发语言·c++·游戏
众乐乐_200814 小时前
用claude Fabel5一句话生成的游戏:三国天命(有资源包)
游戏
石一峰69917 小时前
C 语言函数设计模式实战经验
c语言·开发语言·设计模式
德迅--文琪18 小时前
守护数字游戏乐园:解析DDoS攻击与德迅云安全游戏盾防护方案
游戏·ddos
qq_2975746719 小时前
设计模式系列文章(基础篇第22篇):访问者模式——分离数据结构与操作,实现灵活扩展
数据结构·设计模式·访问者模式
jushi89991 天前
FB Neo 街机模拟器全游戏整合版 含25000+街机游戏怀旧复古街机游戏 解压即玩 热门怀旧街机游戏全集安卓+PC电脑版
android·游戏·电脑
串流游戏联盟1 天前
UU远程云电脑助力手机畅玩 Steam 新作 SpaceCraft!
游戏
Swift社区1 天前
AI + 鸿蒙游戏:下一代游戏架构正在形成吗?
人工智能·游戏·harmonyos