观察者模式,是一种非常常见的设计模式,在很多系统中随处可见,尤其是涉及到数据状态发生变化需要通知的情况下。
本文以AbstractCursor为例子,展开分析。
观察者模式,Observer Pattern,是一个很实用的模式,本人曾经接触到的各种平台以及曾经参与项目中打印模板解释器中都用到了此模式。

1.意图
定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。
热门词汇:依赖 发布-订阅 事件 通知 更新 监听

2.结构


这是一个最简单的观察者模式,目标对象能够添加和删除观察者,当自己某种状态或者行为发生改变时,可通过notify通知注册的观察者进行更新操作。
分 析AbstractCursor的具体情况,我们发现实际工作有时需要对观察者进行统一管理,甚至观察者类型有很多种而又可以分成几个系列,这个时候是要 复杂的多,通过合理的分层这个问题很好解决。下面根据具体情况,我们画出android中abstractCurosr中用到的观察者模式结构图:


观察者分成了两个系列。

3.代码
列举其中相关核心代码如下:

  1. public abstract class AbstractCursor {
  2. //定义管理器
  3. DataSetObservable mDataSetObservable = new DataSetObservable();
  4. ContentObservable mContentObservable = new ContentObservable();
  5. //注册和卸载两类观察者
  6. public void registerContentObserver(ContentObserver observer) {
  7. mContentObservable.registerObserver(observer);
  8. }
  9. public void unregisterContentObserver(ContentObserver observer) {
  10. // cursor will unregister all observers when it close
  11. if (!mClosed) {
  12. mContentObservable.unregisterObserver(observer);
  13. }
  14. }
  15. public void registerDataSetObserver(DataSetObserver observer) {
  16. mDataSetObservable.registerObserver(observer);
  17. }
  18. public void unregisterDataSetObserver(DataSetObserver observer) {
  19. mDataSetObservable.unregisterObserver(observer);
  20. }
  21. //2类通知方法
  22. protected void onChange(boolean selfChange) {
  23. synchronized (mSelfObserverLock) {
  24. mContentObservable.dispatchChange(selfChange);
  25. if (mNotifyUri != null && selfChange) {
  26. mContentResolver.notifyChange(mNotifyUri, mSelfObserver);
  27. }
  28. }
  29. }
  30. protected void notifyDataSetChange() {
  31. mDataSetObservable.notifyChanged();
  32. }
  33. }
 再看看Observable<T>类和DataSetObservable类:
  1. public abstract class Observable<T> {
  2. /**
  3. * 观察者列表
  4. */
  5. protected final ArrayList<T> mObservers = new ArrayList<T>();
  6. public void registerObserver(T observer) {
  7. if (observer == null) {
  8. throw new IllegalArgumentException("The observer is null.");
  9. }
  10. synchronized(mObservers) {
  11. if (mObservers.contains(observer)) {
  12. throw new IllegalStateException("Observer " + observer + " is already registered.");
  13. }
  14. mObservers.add(observer);
  15. }
  16. }
  17. public void unregisterObserver(T observer) {
  18. if (observer == null) {
  19. throw new IllegalArgumentException("The observer is null.");
  20. }
  21. synchronized(mObservers) {
  22. int index = mObservers.indexOf(observer);
  23. if (index == -1) {
  24. throw new IllegalStateException("Observer " + observer + " was not registered.");
  25. }
  26. mObservers.remove(index);
  27. }
  28. }
  29. public void unregisterAll() {
  30. synchronized(mObservers) {
  31. mObservers.clear();
  32. }
  33. }
  34. }

  和

  1. public class DataSetObservable extends Observable<DataSetObserver> {
  2. /**
  3. * 数据发生变化时,通知所有的观察者
  4. */
  5. public void notifyChanged() {
  6. synchronized(mObservers) {
  7. for (DataSetObserver observer : mObservers) {
  8. observer.onChanged();
  9. }
  10. }
  11. }
  12. //... ... (其他方法)
  13. }

  观察者DataSetObserver类是一个抽象类:

  1. public abstract class DataSetObserver {
  2. public void onChanged() {
  3. // Do nothing
  4. }
  5. }

  所以我们具体看它的子类:

  1. public class AlphabetIndexer extends DataSetObserver{
  2. /*
  3. * @hide 被android系统隐藏起来了
  4. */
  5. @Override
  6. public void onChanged() {
  7. //观察到数据变化,观察者做自己该做的事情
  8. super.onChanged();
  9. mAlphaMap.clear();
  10. }
  11. }

  ContentObserver也是类似。

4.效果
(1).行为型模式
(2).目标和观察者间的抽象耦合(经典实现)。
(3).支持广播通信(相信这点android开发者看到后应该有启发吧)。
(4).注意意外的更新,这也是观察者更新进行管理的原因之一。

Android设计模式系列(2)--SDK源码之观察者模式的更多相关文章

  1. Android设计模式系列(3)--SDK源码之单例模式

    单例模式,可以说是GOF的23种设计模式中最简单的一个.这个模式相对于其他几个模式比较独立,它只负责控制自己的实例化数量单一(而不是考虑为用户产生什么样的实例),很有意思,是一个感觉上很干净的模式,本 ...

  2. Android设计模式系列(1)--SDK源码之组合模式

    Android中对组合模式的应用,可谓是泛滥成粥,随处可见,那就是View和ViewGroup类的使用.在android UI设计,几乎所有的widget和布局类都依靠这两个类.组合模式,Compos ...

  3. Android开发系列之系统源码目录

    相信大家对于Google给出的那副经典Android架构图非常的熟悉,从下往上依次是Linux内核层(主要是负责硬件管理调度),HAL层(主要是硬件抽象层),libs层+Runtime,Framewo ...

  4. Android笔记: 查看SDK源码

    Eclipse中设置查看JavaAndroid源码及文档的方法.pdf 经验证方法可行

  5. Android设计模式系列

    http://www.cnblogs.com/qianxudetianxia/category/312863.html Android设计模式系列(12)--SDK源码之生成器模式(建造者模式) 摘要 ...

  6. Android基础之在Eclipes中关联SDK源码和查看SDK源码

    在进行Android应用开发的时候,我们有时候需要查看某个类或接口的源码从而了解如何去使用一个类或者实现一个接口,查看源码有助于我们的学习某个封装的类的底层是如何实现的,这样可以帮助我们掌握类或者接口 ...

  7. 怎样将Android SDK源码 导入到Eclipse中?

    在Eclipse中导入android sdk源码 http://blog.csdn.net/hahahacff/article/details/8590649

  8. eclipse导入java和android sdk源码,帮助文档

    eclipse导入java和android sdk源码,帮助文档 http://blog.csdn.net/ashelyhss/article/details/37993261 JavaDoc集成到E ...

  9. Android 7.1.1系统源码下载、编译、刷机-Nexus 6实战

    想成为一位合格的Android程序员或者一位Android高级工程师是十分有必要知道Android的框架层的工作原理,要知道其工作原理那么就需要阅读Android的源代码. 想要阅读Android的源 ...

随机推荐

  1. mybatis源码分析(3)-----SqlSessionHolder作用

    1. sqlSessionHolder 是位于mybatis-spring 包下面,他的作用是对于sqlSession和事务的控制 sqlSessionHolder 继承了spring的Resourc ...

  2. Make a printer-port EEPROM programmer and dongle

    You can easily use a PC's printer port for serial-EEPROM programming. You can use a device-programme ...

  3. CAS3.5.x(x>1)支持OAuth2 server

    国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...

  4. 关于Mantis变更日志(Changelog)和路线图(Roadmap)的说明

    变更日志(Changelog):是已经修改好了问题的日志,需要给项目添加版本号,并且在添加/解决问题时都指定了相应的版本号,才会显示. 路线图(Roadmap):是计划在某个版本修改某些问题的日志,需 ...

  5. 数字锁相环Octave仿真

    clc; clear all; % 仿真数据长度 SimLens = 1000; % 载波信号 Fs = 2400; Ts = 1 / Fs; Fsig = 60; % 随机初相 Delta_Phas ...

  6. 浅谈ASP.NET的Postback

    说道ASP.NET的Postback,就得说Web Page的生命周期,但是Web Page的生命周期却不是三言两语就能够说得清楚的,所以在这里单纯站的编程的角度,撇开Web Page 的生命周期浅谈 ...

  7. 一道有序洗牌的笔试题,阿里\UC等都用过

    题目:给定一个已经降序排好序的正数数组,要求按「最小.最大.次小.次大……」的顺序重新排序.期望的时间复杂度为O(n),空间复杂度为O(1),即不能申请额外的数组. 例如:输入[7,6,5,4,3,2 ...

  8. 如何更换Office 2013的product key?

    第一步 第二步 第三步 ... ... ... ... ^_^   参考资料 ======================== Change Product Key Office 2013 Home ...

  9. 监听内容变化 TextWatcher @功能

    监听多个TextView的内容变化 使用示例 TextWatcherUtils.addTextChangedListener(isAllNotEmpty -> btnLogin.setEnabl ...

  10. 模拟jQuery中的ready方法及实现按需加载css,js实例代码

    这篇文章介绍了模拟jQuery中的ready方法及实现按需加载css,js实例代码,有需要的朋友可以参考一下     一.ready函数的实现经常用jQuery类库或其他类库中的ready方法,有时候 ...