Python观察者模式

[python|高级篇|笔记|设计模式|观察者模式]


这两天读了[Head First设计模式][1]和[Python编程实战][2]两本书,对设计模式终于有了那么一丢丢的体会,在这里写下来,防止过两天忘了。两本一起读的主要原因是因为我对java一窍不通。
HF里面对设计模式讲的确实很通俗,按照他的说,观察者模式可以用报纸出版商和报纸订阅者之间的关系来描述:
报社出版报纸,可以将它看做一个生产者,报纸订阅者订阅报纸,可以将他看做一个个观察者。
报社出版商可以提供以下的服务:

  1. 注册一个报纸订阅者(Register)
  2. 注销一个报纸订阅者(Remove)
  3. 通知一个报纸订阅者(Notify)

报纸订阅者可以随意定制自己,你是要拿报纸看,还是铺桌子,还是擦屁股随你便

现在再来看观察者模式的定义
定义了对象之间一对多依赖,当一个对象改变状态时,这个对象的所有依赖者都会收到通知并按照自己的方式进行更新

按照HD上一个气象站的例子来看观察者模式
从气象站取得数据后要在三个布告牌显示这些数据,这三个布告牌显示的内容都不一样,一块实时显示气象数据,一块显示一段时间的统计数据,一块根据当前数据显示预测数据,当数据变化时,这三个布告牌要按照自己的方式实时更新数据,例如,实时显示布告牌就显示当前最新数据,统计数据布告牌将最大值最小值平均值显示出来

先定义一个Observed类,这个类的作用类似于报纸出版商,他提供注册,注销,通知等功能

class Observed(object):
    def __init__(self):
    #此处初始化一个列表,里面装满了每一个观察者
        self.__observers = []
    #注册一个观察者,注册一个就往列表里添一个,同时更新一下他的数据
    def registerObservers(self, observer):
        self.__observers.append(observer)
        observer.update(self)
    #注销一个观察者,谁不想看了,直接移走它,以后数据更新了,跟他毛关系都没了
    def romoveObserver(self, observer):
        self.__observers.remove(observer)
    #通知每一个观察者,数据更新了,你们去把数据整理整理在布告牌上发布吧
    def notifyObservers(self):
        for observer in self.__observers:
            observer.update(self)

定义一个WeatherModel类,这个类继承字Observed类,但是他提供采集数据的功能

#继承了Observed类,这样它具有Observed类的数据及方法
class WeatherModel(Observed):
    #初始化时,加入一些初始数据
    def __init__(self, temp, humidity, pressure):
        super(WeatherModel, self).__init__()
        self.temp = temp
        self.humidity = humidity
        self.pressure = pressure
    #增加一个valueChanged方法,当数据改变时,更新当前数据,然后通知每一个观察者数据变了,要更新了
    def valueChanged(self, temp, humidity, pressure):
        if self.temp != temp or \
        self.humidity != humidity or \
        self.pressure != pressure:
            self.temp = temp
            self.humidity = humidity
            self.pressure = pressure
            self.notifyObservers()

定义两个观察者类,CurrentCondition和StatisticCondition

#这个布告板显示实时的气象数据
class CurrentCondition(object):
    def __init__(self):
        self.currData = []
    #每一个观察者都有一个update方法,Observed类里面的notifyObservers方法就是
    #使用每个观察者各自的update方法更新数据,这个布告板里面的update方法是将三个气象参数打成元组然后
    #放到初始化的currData列表里
    def update(self, model):
        self.currData.append((model.temp, model.humidity, model.pressure))
        
#这个布告板对气象数据进行统计
class StaticsCondition(object):
    #初始化了三个列表,分别存放温度,湿度和压力,这样方便统计最大值,最小值和平均值
    def __init__(self):
        self.statDataHumidity = []
        self.statDataTemp = []
        self.statDataPressure = []
    #它的update方法是将每个参数放到各自的列表中去
    def update(self, model):
        self.statDataHumidity.append(model.humidity)
        self.statDataTemp.append(model.temp)
        self.statDataPressure.append(model.pressure)

之后,搞点数据测试一下

def main():
    #先创建两个观察者实例
    current = CurrentCondition()
    statistic = StaticsCondition()
    #创建一个模型实例,传入一些数
    model = WeatherModel(20.0, 55.0, 1013.11)
    #把这两个观察注册一下,之后就可以给他们传数据了
    model.registerObservers(current)
    model.registerObservers(statistic)
    #整点数据,这里只是随便写写,用于测试
    model.valueChanged(21.0, 52.3, 1013.12)
    model.valueChanged(21.2, 53.3, 1013.12)
    model.valueChanged(22.8, 56.1, 1013.18)
    #输出每个观察的数据列表
    print(current.currData)
    print(statistic.statDataTemp)
    print(statistic.statDataHumidity)
    print(statistic.statDataPressure)

输出的结果

#currData列表里面是每一次数据变化时,三个参数的实时变化列表,可以取出后再布告牌上显示,
#当然需要在CurrentCondition()里实现display方法才可以显示
[(20.0, 55.0, 1013.11), (21.0, 52.3, 1013.12), (21.2, 53.3, 1013.12), (22.8, 56.1, 1013.18)]
#三个列表分别存放温度,湿度,气压参数,可以在display方法里面按照时间统计一下
#平均值,最大值,最小值显示在布告牌上
[20.0, 21.0, 21.2, 22.8]
[55.0, 52.3, 53.3, 56.1]
[1013.11, 1013.12, 1013.12, 1013.18]

这就是观察者模式,说白了就是模型和视图没有关系,有点像MVC啊。
[1]: http://item.jd.com/10100236.html
[2]: http://item.jd.com/11518115.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,817评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,329评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,354评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,498评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,600评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,829评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,979评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,722评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,189评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,519评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,654评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,329评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,940评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,762评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,993评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,382评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,543评论 2 349

推荐阅读更多精彩内容