Android设计模式系列(2)--SDK源码之观察者模式

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

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

2.结构


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


观察者分成了两个系列。

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

Java代码  

  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类:

Java代码  

  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. }

  和

Java代码  

  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类是一个抽象类:

Java代码  

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

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

Java代码  

  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).注意意外的更新,这也是观察者更新进行管理的原因之一。

时间: 2024-10-01 03:57:05

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

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

单例模式,可以说是GOF的23种设计模式中最简单的一个.这个模式相对于其他几个模式比较独立,它只负责控制自己的实例化数量单一(而不是考虑为用户产生什么样的实例),很有意思,是一个感觉上很干净的模式,本人很喜欢这个模式.android中很多地方都用到了单例模式,本文以输入法管理者InputMethodManager为例,展开分析.单例模式,Singleton Pattern,能够以其特有的优势,替代系统中全局变量,应用非常广泛. 1.意图保证一个类仅有一个实例,并提供一个访问它的全局访问点.热门词

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

Android中对组合模式的应用,可谓是泛滥成粥,随处可见,那就是View和ViewGroup类的使用.在android UI设计,几乎所有的widget和布局类都依靠这两个类.组合模式,Composite Pattern,是一个非常巧妙的模式.几乎所有的面向对象系统都应用到了组合模式. 1.意图将对象View和ViewGroup组合成树形结构以表示"部分-整体"的层次结构(View可以做为ViewGroup的一部分).组合模式使得用户对单个对象View和组合对象ViewGroup的使

android缓存系列:ASimpleCache源码分析

接触Acache是因为阅读oschina的开源android端代码,发现oschina采用了该框架缓存新闻分页数据.后来知道这是个杨福海的开源项目,他还开源过afinal框架,项目的地址如下: https://github.com/yangfuhai/ASimpleCache 一.官方介绍 ASimpleCache 是一个为android制定的 轻量级的 开源缓存框架.轻量到只有一个java文件(由十几个类精简而来). 1.它可以缓存什么东西? 普通的字符串.JsonObject.JsonArr

Android笔记: 查看SDK源码

Eclipse中设置查看JavaAndroid源码及文档的方法.pdf

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

eclipse导入java和android sdk源码,帮助文档 http://blog.csdn.net/ashelyhss/article/details/37993261 JavaDoc集成到Eclipse的帮助中 http://blog.chinaunix.net/uid-90129-id-132837.html android帮助文档打开慢的三种解决方法 set path=C:\Program Files\Git\bin; find . -name "*.html"|xarg

Android应用安全开发之源码安全

Android应用安全开发之源码安全 gh0stbo · 2016/01/21 10:24 0x00 简介 Android apk很容易通过逆向工程进行反编译,从而是其代码完全暴露给攻击者,使apk面临破解,软件逻辑修改,插入恶意代码,替换广告商ID等风险.我们可以采用以下方法对apk进行保护. 0x01 混淆保护 混淆是一种用来隐藏程序意图的技术,可以增加代码阅读的难度,使攻击者难以全面掌控app内部实现逻辑,从而增加逆向工程和破解的难度,防止知识产权被窃取. 代码混淆技术主要做了如下的工作:

Android # 4.0.x(1-3) 源码 下载 编译

Android 4.0源码下载方法:repo init -u https://android.googlesource.com/platform/manifest -b android-4.0.1_r1 官方下载页面:http://source.android.com/source/downloading.html Android SDK 4.0官方下载页面:http://developer.android.com/sdk/android-4.0.html android 4.0.3最新源码下载

Android 高级自定义Toast及源码解析

Toast概述 Toast的作用 不需要和用户交互的提示框. 更多参见官网:https://developer.android.com/guide/topics/ui/notifiers/toasts.html Toast的简单使用 Toast.makeText(MainActivity.this.getApplicationContext(),"沉迷学习,日渐消瘦",Toast.LENGTH_SHORT).show() 自定义Toast Toast customToast = new

Android应用Preference相关及源码浅析(Preference组件家族篇)

1 前言 前一篇(点我阅读前一篇<Android应用Preference相关及源码浅析(SharePreferences篇)>)我们讨论分析使用了Android的SharePreferences,相信看过的朋友都有了自己的感悟与理解,这一篇我们继续乘热打铁来说说SharePreferences的衍生品--Preference组件. 其实Preference组件大家一定不陌生,因为Android系统的Setting应用及我们市面上一些符合Android设计思想的应用的设置界面一般都会用它来实现,