[Android-Java]使用观察者的正确姿势

作者: 飞翔的逗比娟 | 来源:发表于2016-03-21 15:46 被阅读712次

    前言

    首先先问一个问题,你在做Android开发的时候有没有被传值整的蒙圈?例如Fragment之间的传值,Activity之间的传值(ActivityForResult比较常用),Fragment与Activity之间的传值等等等等。
    先列举几个简单的回传例子吧。

    1.Activity之间的传值:Intent,startActivityForResult等等。
    2.广播传值:BroadcastReceiver。
    3.接口回调机制,也叫作代理模式,Android中经常用到。
    4.线程间传值:Handler模式
    5.Service与Activity之间的传值:ServiceConnection

    如果你是Android的新手,那这几种东西都是很有必要了解的,其中比较难理解的应该就是接口回调机制了。反正我讲课的时候,基本上所有的学生在初次接触之后,肯定蒙圈~

    一、机制

    那么言归正传,我们接下来呢,要讲解的就是一种新的传值方式,也是一种比较简单的方式----观察者
    首先我们来看一下观察者的机制一幅图,很简单就可以理解。

    观察者机制.png

    我们来简单的讲解一下上面的图片:
    A:被观察者,信息的起点,用来包装信息
    O:观察者,信息的中转站,用来传递信息
    B:被改变者,信息的终点,用来处理信息

    第一步:先把被改变者B,放入观察者O专门准备的一个表中,注册一下。
    第二步:当被观察者A,触发观察者事件时,发送通知给观察者O。
    第三步:观察者O,会扫描自己的注册表,如果发现有注册者,也就是被改变者B存在的话,则进行第四步。
    第四步:观察者O,把从被观察者A处得到的信息,发送给被改变者B。

    二、使用

    说了这么多,咱们来简单的看一个例子:从Activity传值到Fragment。我觉得这是很多人都一直头疼的一个问题。

    9834458E-C584-41D8-A8D6-59278836E0DF.png

    从上图可以看出,Activity中有一个输入框和按钮。Fragment中就很简单了,就只有一个TextView,我们现在想做的就是要在Activity的EditText输入框中输入内容,当点击提交的时候,在Fragment的TextView中显示Activity的EditText中输入的内容。
    在看代码之前呢,我们先来理清一下思路。确定一下谁是被观察者A,谁是被改变者B,当然,观察者O是一个单独的类,他们两个肯定都不是。

    被改变者:Fragment

    被观察者:Activity

    好的,我们已经确定了被观察者和被改变者。下面我们先来把我们的观察者类写一下。

    public class MyObserverable extends Observable {
    
        public static MyObserverable observerable;
    
        // 单例:在所有内存中,只存在一个此对象
        public static MyObserverable getObserverable() {
            if (observerable == null) {
                observerable = new MyObserverable();
            }
            return observerable;
        }
    
        public void setMessage(Object msg) {
            // 告诉观察者对象,所观察的内容发生改变
            observerable.setChanged();
            // 刷新信息,或者信息传递
            observerable.notifyObservers(msg);
        }
    
    }
    

    这里我用的是一个单例,简单明确,并且很好用的哦~最基本的东西观察者已经做好了,在这里我们可以看到,观察者在Java中是一个单独的类,Observable。

    下面就要使用这个观察者类了,就遵循我们之前所说的观察者机制就好。

    第一步:先把被改变者B,放入观察者O专门准备的一个表中,注册一下。
    public class Fragment01 extends Fragment implements Observer {
        TextView tv;
    
        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container,
                Bundle savedInstanceState) {
            // TODO Auto-generated method stub
            View v = inflater.inflate(R.layout.f1, null);
            tv = (TextView) v.findViewById(R.id.textView1);
            // 注册观察者
            MyObserverable.getObserverable().addObserver(this);
            return v;
        }
    
        // 当我们告诉观察者,信息发生改变的时候,调用此方法
        @Override
        public void update(Observable observable, Object data) {
            tv.setText(data.toString());
        }
    }
    

    因为我这里面是使用的单例类,所以我直接通过调用MyOserverable的静态方法获取的观察者对象。

    第二步:当被观察者A,触发观察者事件时,发送通知给观察者O。
    
    public class MainActivity extends FragmentActivity {
        EditText et;
        Button btn;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            et = (EditText) findViewById(R.id.et);
            btn = (Button) findViewById(R.id.btn);
            btn.setOnClickListener(new OnClickListener() {
                @Override
                public void onClick(View v) {
    
                    String str = et.getText().toString();
                    // 当事件触发,通知观察者
                    MyObserverable.getObserverable().setMessage(str);
    
                }
            });
        }
    
    }
    
    第三步:观察者O,会扫描自己的注册表,如果发现有注册者,也就是被改变者B存在的话,则进行第四步。
    public class MyObserverable extends Observable {
    
        ...
            // 扫描注册表
        public void setMessage(Object msg) {
            // 告诉观察者对象,所观察的内容发生改变
            observerable.setChanged();
            // 刷新信息,或者信息传递
            observerable.notifyObservers(msg);
        }
    
    }
    

    我们在写观察者类的时候,已经完成了这一步。

    第四步:观察者O,把从被观察者A处得到的信息,发送给被改变者B。
    public class Fragment01 extends Fragment implements Observer {
        ...
    
        // 当我们告诉观察者,信息发生改变的时候,调用此方法
        @Override
        public void update(Observable observable, Object data) {
            tv.setText(data.toString());
        }
    }
    

    在第一步的代码中其实我们已经实现了处理方法:即实现Observer接口,并实现接口中的抽象方法update()。

    这就是观察者的使用步骤,是不是很简单?来试试吧。

    相关文章

      网友评论

      • 带心情去旅行:整理的不错
        飞翔的逗比娟:@zhouyouxi 谢谢支持~
      • qing的世界:eventbus可破之
        qing的世界:@路遥遥遥呗 是滴lz自己实现了一套!学习价值很大:sunglasses::sunglasses::sunglasses:
        飞翔的逗比娟:那个是别人编写的,这个是系统提供的~那个的内部是反射实现的,个人爱好~都行~
        誓词倾城:@qing的世界 好用的不止一点,哈哈
      • 7fb2d224837d:观察者模式的意义是什么呢~直接写回调貌似也可以实现 或者直接获取AC的信息
        7fb2d224837d:@路遥遥遥呗 你说的有道理。注册太多被观察者会不会效率很低呢…每次都要遍历一遍
        飞翔的逗比娟:首先呢,接口回调对于初学者来说比较难,其次呢,接口回调有一定的耦合度。所以在一定情况下,观察者还是比较好的啊
        qing的世界:@爱睡的猫用来解耦啊
      • 吃人的锅:简单明了
        飞翔的逗比娟:@痞子郭 谢谢~

      本文标题:[Android-Java]使用观察者的正确姿势

      本文链接:https://www.haomeiwen.com/subject/rppklttx.html