• Android使用的设计模式1——观察者模式


    设计模式,对程序员来说是一个坎,想在程序员这条路走得更远,设计模式是你的必修课。从大学时代接触GoF到工作几年后重新看设计模式,每次感觉都不一样。这次想借着分析Android Framework源码的机会,顺道整理一下设计模式的知识。

      今天主要是先讲一下观察者模式,观察者模式对于做系统或者做公共库的朋友来说,应该很熟悉,基本上所有系统都会用到这个模式。整理的时候,主要是对模式进行进步讲解然后结合Android里面应用进行说明。设计模式的说明主要是参考《研磨设计模式》这本书,这本书讲解了GoF里面23种模式,而且比GoF更容易理解。如果对设计模式不了解的朋友,可以看看《研磨设计模式》这本书。

      结合Android源码的时候,主要是针对应用层的代码说明,而且尽量使用简单例子代码,所以不会出现太多Android系统级的代码,大部分是我们做APP开发时会编写的代码。

      这篇文章不会详细介绍设计模式,只是整理设计模式核心概念,结合Android应用说明。

      (PS:新建的QQ群,有兴趣可以加入一起讨论:Android群:322599434)

    1、观察者模式

      观察者模式:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。这种模式最常用在我们熟悉的事件驱动模型里面,像VC、Android界面里面的事件响应,就是基于观察者模式来实现。

      下面我们看看观察者模式的类图关系(图片是截取《研磨设计模式》里面插图):

     

     

    • Subject是我们定义目标对象,或者说被观察对象。
    • ConcreteSubject是实际的被观察对象。 
    • Observer是观察者的接口,定义了回调接口。
    • ConcreteObserver是实际的观察者对象。

     

    2、观察者模式含义

      下面讲解一下个人对观察者模式的理解,其实这个模式在我们平时工作中是很常用的模式,可能你已经使用过,缺没有意识到。

      观察者模式主要是针对一对多的数据更新。简单来说就是,系统里面某个元素更新了数据,然后有好几个元素是使用了这个元素的数据。此时更新了数据的对象,就要通知其他使用了它数据的对象,让他们都进行更新。

      标准的观察者对象是一对多的关系,不过我们都知道设计模式里面是很灵活,在我们使用的时候,经常需要进行变形。对现有的标准模式进行适当的修改来适应设计需求。

      在我们实际应用中,还会遇到一对一、或者多对一的情况。一对一就是,一个目标对应一个观察者。多对一是指多个目标对应一个观察者。

    3、观察者模式的简单实现

      下面给出一个最简单的观察者模式实现代码,把代码看懂了,再回头看文字讲解,会比较容易理解。下面是目标对象的接口和实现代码,只做了一个最简单的接口。

    复制代码
    //Edited by mythou
    //http://www.cnblogs.com/mythou/
       //目标对象接口
        public interface Subject
        {
            //调用这个方法注册一个新的观察者对象
            public void attach(Observer observer);
            
            //调用这个方法删除一个已注册的观察者对象
            public void detach(Observer observer);
            
            //调用这个方法通知所有注册的观察者对象
            void notifyObservers();
        }  
        
        //目标对象实现
        public class ConcreteSubject implements Subject
        {
        //向量容器,保存所有注册的观察者
    private Vector observersVector = new Vector(); public void attach(Observer observer) { observersVector.addElement(observer); } public void detach(Observer observer) { observersVector.removeElement(observer); } public void notifyObservers() { Enumeration enumeration = observers(); while(enumeration.hasMoreElements()) { ((Observer)enumeration.nextElement()).update(); } } }
    复制代码

    下面是观察者的实现:

    复制代码
    //Edited by mythou
    //http://www.cnblogs.com/mythou/
       //观察者接口
        public interface Observer
        {
            void update();
        }
        
        //观察者实现
        public class ConcreteObserver implements Observer
        {
            public void update()
            {
                System.out.println("The subject has changed! mythou notify");
            }
        }
    复制代码

    应用中使用的代码:

    复制代码
     //Edited by mythou
    //http://www.cnblogs.com/mythou/
      //客户端
        public class Client
        {
            private static ConcreteSubject subject;
            private static Observer observer;
            
            public static void main(String args[])
            {
                //创建目标对象
                subject = new ConcreteSubject();
                //创建观察者对象
                observer = new ConcreteObserver();
                subject.attach(observer);
                subject.change("mythou has changed data!");
            }
        }
    复制代码

    4、Android 应用开发中的观察者模式

      在我们做Android应用的时候,会大量使用观察者模式,因为Framework层里面的事件驱动都是基于观察者模式实现的。另外在Framework层里面的各种服务在数据变更的时候,也是通过观察者模式实现上层数据更新。下面会讲两个例子来说明:

     A)控件中Listener监听方式

      Android里面最典型的观察者就是我们使用的各种控件监听者。例如下面对某个按钮的监听:

    复制代码
    //Edited by mythou
    //http://www.cnblogs.com/mythou/
    Button baiduclickButton = (Button)findViewById(R.id.button1);
    //注册观察者 baiduclickButton.setOnClickListener(
    new OnClickListener() {
      //观察者实现 @Override
    public void onClick(View arg0) { Log.d("Mythou_Log", "Click the button "); } });
    复制代码

      例如上面的代码,注册了一个按钮的监听方法。这里实际上是应用了一对一的观察者模式,setOnClickListener()方法就是注册一个观察者,Button对象就是我们要观察的目标对象。而new出来的OnClickListener(),就是我们实际的观察者。

      每当我们的目标按钮对象被点击,状态发生变化的时候,就会通过回调注册的OnClickListener观察者的onClick方法会来通知观察者,Button状态发生变化。这里的onClick相当于前面例子里面的update方法。下面是Android源码里面对OnClickListener的定义,跟我们前面定义一个观察者接口类似。

    复制代码
    //Edited by mythou
    //http://www.cnblogs.com/mythou/
      public interface OnClickListener 
        {
            /**
             * Called when a view has been clicked.
             *
             * @param v The view that was clicked.
             */
            void onClick(View v);
        }
    复制代码

    B)GPS位置信息监听

      上面对按钮监听是属于一对一的观察者模式,当然你也可以用一个Listener监听多个按钮。下面我们再讲一个标准的一对多的观察者模式的应用。在我们使用GPS模块的时候,需要监听GPS模块位置变化,通常我们会编写下面代码:

    复制代码
    
    
    //Edited by mythou
    //http://www.cnblogs.com/mythou/
    // 通过GPS定位
    String LocateType= locationManager.GPS_PROVIDER;
    Location location = locationManager.getLastKnownLocation(LocateType);
    // 设置监听器,设置自动更新间隔这里设置1000ms,移动距离:0米。
    locationManager.requestLocationUpdates(provider, 1000, 0, locationListener);
    // 设置状态监听回调函数。statusListener是监听的回调函数。
    locationManager.addGpsStatusListener(statusListener); 
    
    //监听器实现
    private final GpsStatus.Listener statusListener = new GpsStatus.Listener() 
    {
        public void onGpsStatusChanged(int event) 
      {
          // GPS状态变化时的回调,获取当前状态
          GpsStatus status = locationManager.getGpsStatus(null);
        //自己编写的方法,获取卫星状态相关数据
           GetGPSStatus(event, status);
        }
    };
    复制代码

      GPS位置服务是Framework的一个系统层服务,整个系统只有一个运行的实例。但实际使用时,可能会出现好几个应用都在使用这个服务,因此会形成了一个一对多的观察者例子。这里不对代码进行深入对比讲解,只要对照上面的讲解,你就可以把目标对象、观察者接口、观察者实现、数据更新回调接口找出来。有兴趣的朋友还可以追查Android的源码,看看GpsStatus里面的具体实现。

    5、结语

      除了上面说的按钮响应、GPS服务外,还有一个例子比较经典,就是我们的BroadcastReceiver响应,也是基于观察者模式来实现,下面总结里面也会说到,观察者模式支持广播通信。有关BroadcastReceiver是怎么实现观察者模式的,有兴趣的朋友可以自己对比代码,和查看Android实现源码,自己动手才能加深印象。引入设计模式是为了对系统有更深入的理解。下面总结一下观察者模式的优缺点,方便我们在使用的时候,可以根据实际情况衡量。

    A)优点

    • 实现观察者和目标对象之间的抽象耦合。
    • 观察者模式支持广播通信。被观察者会所有的登记过的观察者发出通知。

    B)缺点

    • 可能会引起多余的数据通知。
    • 可能会导致通知数据循环,导致死锁产生。

    Edited by mythou

    原创博文,转载请标明出处:http://www.cnblogs.com/mythou/p/3370340.html 

  • 相关阅读:
    数据挖掘实践(23):实战-- 建筑能源得分预测报告(一)
    返回闭包
    函数指针
    Rust 中的 Closure
    Moves, copies and clones in Rust
    Rust的闭包类型(Fn, FnMut, FnOne的区别)
    Clone VS Copy
    rust socket
    A simple UNIX socket listener in Rust
    【firecracker】系统启动与epoll事件循环
  • 原文地址:https://www.cnblogs.com/wodehao0808/p/3386119.html
Copyright © 2020-2023  润新知