标签:
rx系列貌似在前一阶段火起来的,也是自己接触的少,一直没有去学习,今天就趁着周六,脑补一下。
关于理论的知识,网上的介绍的太多了,大家可以去看下,在文章的结尾,我也会附几篇好的文章。
public interface Watched {
//添加观察者
public void addWatcher(Watcher watcher);
//移除观察者
public void removeWatcher(Watcher watcher);
//通知观察者
public void notifyWathers(String str);
}
public interface Watcher {
//数据变化进行更新
public void update(String str);
}
public class ConcreteWathed implements Watched {
//观察者
List<Watcher> mList = new ArrayList<>();
@Override
public void addWatcher(Watcher watcher) {
mList.add(watcher);
}
@Override
public void removeWatcher(Watcher watcher) {
mList.remove(watcher);
}
@Override
public void notifyWathers(String str) {
for (Watcher w : mList) {
w.update(str);
}
}
}
public class ConcreteWather implements Watcher {
@Override
public void update(String str) {
System.out.println(str);
}
}
public static void main(String[] args){
Watched watched = new ConcreteWathed();
Watcher watcher1 = new ConcreteWather();
Watcher watcher2 = new ConcreteWather();
Watcher watcher3 = new ConcreteWather();
watched.addWatcher(watcher1);
watched.addWatcher(watcher2);
watched.addWatcher(watcher3);
watched.notifyWathers("I go");
}
I go
I go
I go
当然了,这只是简单的实现,只要晓得原理就行,除了自己实现,官方也给我们提供了观察者与被观察者接口。只要我们去实现接口就可以了。
public class XTObservable extends Observable {
private int data = 0;
public int getData(){
return data;
}
public void setData(int i){
if (this.data != i){
this.data = i;
setChanged();//发生改变
notifyObservers();//通知观察者
}
}
}
public class XTobserver implements Observer {
public XTobserver(XTObservable observable) {
observable.addObserver(this);
}
@Override
public void update(Observable observable, Object o) {
System.out.println("data is changed" + ((XTObservable) observable).getData());
}
}
public class Test {
public static void main(String[] args) {
XTObservable mObservable = new XTObservable();
XTobserver mXTobserver = new XTobserver(mObservable);
mObservable.setData(1);
mObservable.setData(2);
mObservable.setData(3);
}
}
data is changed1
data is changed2
data is changed3
上面已经手动实现观察者模式和通过系统提供类实现,当然这都不是重点,重点是Rx响应式编程
在项目工程的build.gradle文件添加这样的一句话(如果使用lambda)
classpath ‘me.tatarka:gradle-retrolambda:2.5.0‘(这一句在gradle版本下面紧接着)
在该module工程的build.gradle文件中添加
apply plugin: ‘me.tatarka.retrolambda‘(使用lambda)在文件的第二行
在buildTypes节点的下(不是节点内)添加下面一句
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
然后在依赖中添加下面几句(没有提示一定添加的可以根据自己选择性添加)
//rx一定添加
compile ‘io.reactivex:rxjava:1.1.0‘
compile ‘io.reactivex:rxandroid:1.1.0‘
compile ‘com.google.code.gson:gson:2.4‘
compile ‘com.jakewharton:butterknife:7.0.1‘
compile ‘com.squareup.picasso:picasso:2.5.2‘
//添加
compile ‘com.squareup.okhttp3:okhttp:3.+‘
至此,使用环境已经配置好了,接下来我们来简单的使用一下。
/**
* 使用create方式
*/
public static void createObserable() {
//定义被观察者
Observable<String> observable = Observable.create(new Observable.OnSubscribe<String>() {
@Override
public void call(Subscriber<? super String> subscriber) {
if (!subscriber.isUnsubscribed()) { //观察者和被观察者还有订阅消息
subscriber.onNext("hello"); //返回的数据
subscriber.onNext("hi");
subscriber.onNext(getUserName()); //因为是传入的是字符串泛型
subscriber.onCompleted(); //完成
}
}
});
//定义观察者
Subscriber showSub = new Subscriber() {
@Override
public void onCompleted() {
Log.i(TAG, "onCompleted"); //用于对话框消失
}
@Override
public void onError(Throwable e) {
Log.i(TAG, e.getMessage()); //错误处理
}
@Override
public void onNext(Object o) {
Log.i(TAG, o.toString());
}
};
observable.subscribe(showSub); //两者产生订阅
}
/**
* 可以用来写成我们的下载返回数据
*
* @return
*/
public static String getUserName() {
return "jsonName";
}
在主activity中调用,我们来看下控制台输出的结果:
/**
* 打印的功能 链式结构,更加易于代码的可毒性
*/
public static void createPrint() {
Observable.create(new Observable.OnSubscribe<Integer>() {
@Override
public void call(Subscriber<? super Integer> subscriber) {
if (!subscriber.isUnsubscribed()) {
for (int i = 0; i < 10; i++) {
subscriber.onNext(i);
}
subscriber.onCompleted();
}
}
}).subscribe(new Subscriber<Integer>() {
@Override
public void onCompleted() {
Log.i(TAG, "onCompleted");
}
@Override
public void onError(Throwable e) {
Log.i(TAG, e.getMessage());
}
@Override
public void onNext(Integer integer) {
Log.i(TAG, "result--->:" + integer);
}
});
}
看下控制台结果
/**
* 使用在被观察者,返回的对象一般都是数据类型
* 它接收一个集合作为输入,然后每次输出一个元素给subscriber
*/
public static void from() {
Integer[] items = {1, 2, 3, 4, 5, 6, 7, 8};
Observable onservable = Observable.from(items);
onservable.subscribe(new Action1() {
@Override
public void call(Object o) {
Log.i(TAG, o.toString());
}
});
}
控制台结果
/**
* 指定某一时刻进行数据发送
* interval()函数的两个参数:一个指定两次发射的时间间隔,另一个是用到的时间单位
*/
public static void interval() {
Integer[] items = {1, 2, 3, 4};
Observable observable = Observable.interval(1, 1, TimeUnit.SECONDS);
observable.subscribe(new Action1() {
@Override
public void call(Object o) {
Log.i(TAG, o.toString());
}
});
}
/**
* 假如我们只有3个独立的AppInfo对象并且我们想把他们转化为Observable并填充到RecyclerView的item中:
* 这里我们有两个数组,然后通过转化为Observable组成一个item
*/
public static void just() {
Integer[] items1 = {1, 2, 3, 4};
Integer[] items2 = {2, 4, 6, 8};
Observable observable = Observable.just(items1, items2);
observable.subscribe(new Subscriber<Integer[]>() {
@Override
public void onCompleted() {
Log.i(TAG, "onCompleted");
}
@Override
public void onError(Throwable e) {
Log.i(TAG, e.getMessage());
}
@Override
public void onNext(Integer[] integers) {
for (int i = 0; i < integers.length; i++) {
Log.i(TAG, "result--->" + i);
}
}
});
}
输出结果:
/**
* 指定输出数据的范围
*/
public static void range() {
Observable observable = Observable.range(1, 4);
observable.subscribe(new Subscriber<Integer>() {
@Override
public void onCompleted() {
Log.i(TAG, "onCompleted");
}
@Override
public void onError(Throwable e) {
Log.i(TAG, e.getMessage());
}
@Override
public void onNext(Integer o) {
Log.i(TAG, "next---->" + o);
}
});
}
输出结果:
/**
* 使用过滤功能 发送消息的时候,先过滤在发送
*/
public static void filter() {
Observable observable = Observable.just(1, 2, 3, 4, 5, 6);
observable.filter(new Func1<Integer, Boolean>() {
@Override
public Boolean call(Integer o) {
return o < 5;
}
}).observeOn(Schedulers.io()).subscribe(new Subscriber() {
@Override
public void onCompleted() {
Log.i(TAG, "onCompleted");
}
@Override
public void onError(Throwable e) {
Log.i(TAG, e.getMessage());
}
@Override
public void onNext(Object o) {
Log.i(TAG, o.toString());
}
});
}
输出结果:
好了,几个常用到的函数已经介绍完了,接下来就用几个例子来说验证一下吧。
/**
* 声明一个被观察者对象,作为结果返回
*/
public Observable<byte[]> downLoadImage(String path) {
return Observable.create(new Observable.OnSubscribe<byte[]>() {
@Override
public void call(Subscriber<? super byte[]> subscriber) {
if (!subscriber.isUnsubscribed()) { //存在订阅关系
//访问网络操作
//请求体
Request request = new Request.Builder().url(path).get().build();
//异步回调
mOkHttpClient.newCall(request).enqueue(new Callback() {
@Override
public void onFailure(Call call, IOException e) {
subscriber.onError(e);
}
@Override
public void onResponse(Call call, Response response) throws IOException {
if (response.isSuccessful()) {
byte[] bytes = response.body().bytes();
if (bytes != null) {
subscriber.onNext(bytes); //返回结果
}
}
subscriber.onCompleted(); //访问完成
}
});
}
}
});
}
//使用HTTP协议获取数据
mUtils.downLoadImageOne(url)
.subscribeOn(Schedulers.io()) //在子线程请求
.observeOn(AndroidSchedulers.mainThread()) //结果返回到主线程这一步很厉害啊,不用我们去用handler或者async切换线程了
// 主要我们去调用一下代码,就已经帮我们切换好了线程,是不是感觉有点很厉害啊
.subscribe(new Subscriber<byte[]>() {
@Override
public void onCompleted() {
Log.i(TAG,"onCompleted");//对话框消失
}
@Override
public void onError(Throwable e) {
Log.i(TAG,e.getMessage());
}
@Override
public void onNext(byte[] bytes) {
Bitmap bitmap = BitmapFactory.decodeByteArray(bytes,0,bytes.length);
mImageView.setImageBitmap(bitmap);
}
});
/**
*
* @param url 登录地址
* @param params 请求参数
* @return 后台返回的数据
*/
public Observable<String> login(String url, Map<String, String> params) {
return Observable.create((Observable.OnSubscribe<String>) subscriber -> {
if (!subscriber.isUnsubscribed()) {
//创建formbody
FormBody.Builder builder = new FormBody.Builder();
if (params != null && !params.isEmpty()) {
//循环获取body中的数据
for (Map.Entry<String, String> entry : params.entrySet()) {
builder.add(entry.getKey(), entry.getValue());
}
}
//请求体
RequestBody requestBody = builder.build();
Request request = new Request.Builder().url(url).post(requestBody).build();
mOkHttpClient.newCall(request).enqueue(new Callback() {
@Override
public void onFailure(Call call, IOException e) {
subscriber.onError(e);
}
@Override
public void onResponse(Call call, Response response) throws IOException {
if (response.isSuccessful()) {
//交给观察者处理数据
subscriber.onNext(response.body().string());
}
//完成的回调
subscriber.onCompleted();
}
});
}
});
}
Map<String, String> params = new HashMap<String, String>();
params.put("username", userName.getText().toString().trim());
params.put("password", passWord.getText().toString().trim());
mUtils.login(url, params).subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread()).subscribe(new Subscriber<String>() {
@Override
public void onCompleted() {
Log.i(TAG, "onCompleted");
}
@Override
public void onError(Throwable e) {
Log.i(TAG, e.getMessage());
}
@Override
public void onNext(String s) {
if (JsonUtils.parse(s)) {
Intent intent = new Intent(LoginActivity.this, ContentActivity.class);
startActivity(intent);
}
}
});
如果有想需要代码的,可以看这里,所有代码已经传至github。https://github.com/wuyinlei/RxAndroidDemo
好了,就先介绍到这里吧,这里在给大家推荐几篇比较好的博文还有。
Rx使用还是挺方便的,不过需要一定的学习成本,谨慎使用(嘿嘿)
标签:
原文地址:http://blog.csdn.net/wuyinlei/article/details/52004099