观察者模式----在RecycleView中使用

参考文档:

http://www.jianshu.com/p/d55ee6e83d66
文章中的例子来源于参考文献,更详细的讲解在参考文章中,本文是自己的一个理解

观察者模式使用场景:

A对象(观察者)对B对象(被观察者)的某种变化高度敏感,需要在B变化的一瞬间做出反应。在代码上实现上,通常是被观察者主动去注册或者订阅一个观察者,

构建观察者模式

1.被观察者持有观察者对象,这一步骤通常是通过被观察者注册(Register方法)或者订阅(Subscribe方法)观察者实现的。
2.观察者同样持有被观察者对象,同时调用被观察者的订阅方法完成订阅。(被观察者拿到观察者对象这一步通常在观察者类中去做。)
3.被观察者具有一个"通知观察者的"方法,该方法通过调用被观察者内部持有的观察者对象的方法来达到通知观察者的目的。
(其实所谓收到通知就是被观察者做某些操作时,同时调用了观察者的特定方法。)

Paste_Image.png
Paste_Image.png
Paste_Image.png
实现过程

被观察者接口

/** * 主题(发布者、被观察者) */
public interface Subject { 
/** * 注册观察者 */
 void registerObserver(Observer observer);
 /** * 移除观察者 */ 
void removeObserver(Observer observer);
 /** * 通知观察者 */ 
void notifyObservers(); }

观察者接口

/** * 观察者 */
public interface Observer { 
void update();
}

具体实现
被观察者:
1.被观察者持有观察者对象。
2.被观察者持有的观察者对象通过订阅(注册方法拿到)
3.被观察者数据改变的时候通知观察者(通过调用观察者的update方法)

public class WeatherData implements Subject { 

private List<Observer> observers;//1.被观察者持有观察者对象。

private float temperature;
private float humidity;
private float pressure;
private List<Float> forecastTemperatures;
 //未来几天的温度
 public WeatherData() { this.observers = new ArrayList<Observer>(); 

 //被观察者持有的观察者对象通过订阅
 @Override
 public void registerObserver(Observer observer) { this.observers.add(observer); } 

 @Override 
 public void removeObserver(Observer observer) { 
   this.observers.remove(observer); 
 } 
通知观察者
@Override 
public void notifyObservers() { 
   for (Observer observer : observers) { 
     observer.update();
   } 
}

在数据更新的时候通知观察者
  public void changeWeatherData(float temperature, float humidity, float pressure, List<Float> forecastTemperatures) { 
   this.temperature = temperature;
   this.humidity = humidity;
   this.pressure = pressure; 
   this.forecastTemperatures = forecastTemperatures;
   notifyObservers();
 } 
   public float getTemperature() { return temperature; }
   public float getHumidity() { return humidity; }
   public float getPressure() { return pressure; } 
   public List<Float> getForecastTemperatures() { return forecastTemperatures; }}

观察者:
1.观察者持有被观察者对象(WeatherData)
2.将自身传递给被观察者(使用被观察者的注册方法)
3.提供一个方法,负责在被观察者改变的时候刷新自身。(持有update方法,update方法实际做事情,该方法在被观察者中被调用)

public class CurrentConditionsDisplay implements Observer, DisplayElement { 
        private WeatherData weatherData; //持有被观察者对象
        private float temperature;
        private float humidity;
        private float pressure;
     public CurrentConditionsDisplay(WeatherData weatherData) {
        this.weatherData = weatherData;
        this.weatherData.registerObserver(this);//被观察者对象拿到观察者
    }

 @Override
 public void display() { 
        System.out.println("当前温度为:" + this.temperature + "℃");
        System.out.println("当前湿度为:" + this.humidity); 
        System.out.println("当前气压为:" + this.pressure); } 

@Override 
public void update() { 
      //数据都是通过被观察者拿到的
      this.temperature = this.weatherData.getTemperature(); 
      this.humidity = this.weatherData.getHumidity();
      this.pressure = this.weatherData.getPressure(); 
      display(); 
    }
}

使用例子:

 //被观察者
 WeatherData weatherData = new WeatherData(); 
 //观察者
CurrentConditionsDisplay currentConditionsDisplay = new CurrentConditionsDisplay(weatherData);
//被观察者数据改变
 List<Float> forecastTemperatures = new ArrayList<Float>();
 forecastTemperatures.add(22f);
 forecastTemperatures.add(-1f);
 forecastTemperatures.add(9f);
 weatherData.changeWeatherData(22f, 0.8f, 1.2f, forecastTemperatures);
//被观察者数据改变的时候(即调用changeWeatherData时,实际上内部调用了观察者的update方法,主动通知观察者)

笔者使用实例:

场景:在RecycleView中需要一个测试每一个展示出来的ip的延时,并且展示在Recycleview上。延时的展示和其他数据的展示是不同步的。测量延时采用的策略是 ping 地址5秒后拿到一个结果。所以采用观察者模式,当获取到延时数据(即数据改变的时候)更新UI。

Screenshot_2016-11-28-11-02-18-518_guo.com.busine.png

被观察者RobotBean对象
1.持有观察者对象:registerOberver方法
2.被观察者改变的时候通知观察者:notifyIpChange方法

Paste_Image.png

观察者ViewHolder对象
1.持有被观察对象

Paste_Image.png

2.传递观察者:

Paste_Image.png

3.提供一个更新方法

Paste_Image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容

  • 1 场景问题# 1.1 订阅报纸的过程## 来考虑实际生活中订阅报纸的过程,这里简单总结了一下,订阅报纸的基本流程...
    七寸知架构阅读 4,569评论 5 57
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,594评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,421评论 25 707
  • 工厂模式类似于现实生活中的工厂可以产生大量相似的商品,去做同样的事情,实现同样的效果;这时候需要使用工厂模式。简单...
    舟渔行舟阅读 7,716评论 2 17
  • 当你看到这些细碎文字时,我正在开往福州的火车上,向南,向南。 想象过无数次自己的毕业季, 毕业旅行也好, 重回大一...
    文刀Van_Sword阅读 326评论 0 1