前言
首先先问一个问题,你在做Android开发的时候有没有被传值整的蒙圈?例如Fragment之间的传值,Activity之间的传值(ActivityForResult比较常用),Fragment与Activity之间的传值等等等等。
先列举几个简单的回传例子吧。
1.Activity之间的传值:Intent,startActivityForResult等等。
2.广播传值:BroadcastReceiver。
3.接口回调机制,也叫作代理模式,Android中经常用到。
4.线程间传值:Handler模式
5.Service与Activity之间的传值:ServiceConnection
如果你是Android的新手,那这几种东西都是很有必要了解的,其中比较难理解的应该就是接口回调机制了。反正我讲课的时候,基本上所有的学生在初次接触之后,肯定蒙圈~
一、机制
那么言归正传,我们接下来呢,要讲解的就是一种新的传值方式,也是一种比较简单的方式----观察者
首先我们来看一下观察者的机制一幅图,很简单就可以理解。
我们来简单的讲解一下上面的图片:
A:被观察者,信息的起点,用来包装信息
O:观察者,信息的中转站,用来传递信息
B:被改变者,信息的终点,用来处理信息
第一步:先把被改变者B,放入观察者O专门准备的一个表中,注册一下。
第二步:当被观察者A,触发观察者事件时,发送通知给观察者O。
第三步:观察者O,会扫描自己的注册表,如果发现有注册者,也就是被改变者B存在的话,则进行第四步。
第四步:观察者O,把从被观察者A处得到的信息,发送给被改变者B。
二、使用
说了这么多,咱们来简单的看一个例子:从Activity传值到Fragment。我觉得这是很多人都一直头疼的一个问题。
从上图可以看出,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()。
这就是观察者的使用步骤,是不是很简单?来试试吧。