• 观察者模式


    #include
    #include //for smart pointer
    #include <unordered_map> //for unordered_map
    #include

    using namespace std;

    class Widget
    {
    public:
        Widget(int id):ID(id){}
        
        int ID;
    };

    //1. 利用weak_ptr来缓存对象
    //模拟从数据库中加载,并创建shared_ptr指向widget对象
    shared_ptr loadWidget(int WidgetID)
    {
        return make_shared(WidgetID); 
    }

    //带缓存的工厂函数
    std::shared_ptr fastloadWidget(int WidgetID) //返回shared_ptr类型
    {
        //缓存:weak_ptr类型
        static std::unordered_map> cache;

        auto objPtr = cache[WidgetID].lock(); //objPtr的类型为shared_ptr,指向缓存的对象

        if (!objPtr) { //如果对象不在缓存中. 这里省略了缓存中因失效而不断累积std::weak_ptr的处理。
            objPtr = loadWidget(WidgetID);
            cache[WidgetID] = objPtr;
        }

        return objPtr;
    }

    //2. 观察者模式
    //2.1 观察者
    class WeatherObservers //抽象观察者
    {
    public:
        virtual void updateWeatherInfo(int num) = 0;
    };
    //机场:具体观察者
    class Airport : public WeatherObservers
    {
    public:
        void updateWeatherInfo(int num) override
        {
            std::cout <<"Airport: " << num << endl;
        }
    };
    //学校:具体观察者
    class School : public WeatherObservers
    {
    public:
        void updateWeatherInfo(int num) override
        {
            std::cout << "School: " << num << endl;
        }
    };

    //2.1 主题(气象站)
    class WeatherStation
    {
        using ObserverPtr = std::weak_ptr; //弱引用

        //set集合中保存观察者的弱引用(以ObserverPtr为关键字,基于ownership排序)
        using ObserverList = std::set>;

        ObserverList obs; //保存所有观察者
    public:
        //注册观察者
        void registerObserver(const ObserverPtr oPtr)
        {
            if (obs.find(oPtr) == obs.end()) {
                obs.insert(oPtr);
            }
        }
        //注销观察者
        void unregisterObserver(const ObserverPtr oPtr) //oPtr为weak_ptr类型
        {
            if (obs.find(oPtr) != obs.end())
            {
                obs.erase(oPtr);
            }
        }

        //通知各个观察者
        void notifyObservers(int num)
        {
            std::shared_ptr tempPtr;
            for (auto& ob : obs)
            {
                if ((tempPtr = ob.lock())) {
                    tempPtr->updateWeatherInfo(num);
                }
            }
        }
    };


    int main()
    {
        //观察者模式
        WeatherStation station;
        std::shared_ptr airport(new Airport());
        std::shared_ptr  school(new School());

        station.registerObserver(airport);
        station.registerObserver(school);

        station.notifyObservers(1);

        station.unregisterObserver(school);
        station.notifyObservers(2);

        return 0;
    }
    /*输出结果
    Airport: 1
    School: 1
    Airport: 2
    */

  • 相关阅读:
    数字钱包红海角逐,小程序生态快速引入可助力占领智慧设备入口
    双十二电容笔哪个品牌好?十大电容笔知名品牌
    聚观早报 | TCL召开电视新品发布会;OceanBase 4.3发布
    金融科技开题资源汇总
    软件测试/测试开发/人工智能丨 利用ChatGPT编写测试用例
    一些思考:腾讯股价为何持续都低
    Java 是否应该使用通配符导入( wildcard imports)
    Spring简介说明
    Redis入门到实战
    【前端】特效开发
  • 原文地址:https://blog.csdn.net/u012983289/article/details/132896750