让对象保持消息灵通
一个WeatherData对象负责追踪目前的天气状况(温度,湿度,气压)。希望你们能建立一个应用,有三种布告板,分别显示目前的状况、气象统计及简单的预报。当WeatherObject对象获得最新的测量数据时,三种布告板必须实时更新。而且,这是一个可以扩展的气象站,Weather-O-Rama气象站希望公布一组API,好让其他开发人员可以写出自己的气象布告板,并插入此应用中,我们希望能提供这样的API。
class WeatherData{
public int getTemperature(){
}
public int getHumidity(){
}
public int getPressure(){
}
public void measurementsChanged(){
//一旦气象测量更新,此方法会被调用
}
}
public class WeatherData {
// 实例变量声明
public void measurementsChanged() {
float temp = getTemperature();
float humidity = getHumidity();
float pressure = getPressure();
currentConditionsDisplay.update(temp, humidity, pressure);
statisticsDisplay.update(temp, humidity, pressure);
forecastDisplay.update(temp, humidity, pressure);
}
// 这里是其他WeatherData方法
}
缺点:
针对实现编程
对于每个新的显示,都得修改这份代码
没有办法在运行时添加/移除显示元素
没有封装改变的部分
观察者模式:定义对象之间的一对多依赖,这样一来,当一个对象改变状态时,它的所有依赖者都会收到通知并自动更新;
有几种不同的实现方法,大多数围绕着包括主题和观察者接口的类设计
类图:

第四个设计原则:为了交互对象之间松耦合设计而努力。
主题只知道观察者实现了某个接口(也就是Observer接口)
任何时候我们都可以增加新的观察者。也可以在任何时候删除某些观察者。
有新类型的观察者出现时,主题的代码不需要修改,
独立复用主题和观察者
改变主题/观察者其中一方,不会影响另一方

1.构建主题subject和观察者observer接口
//主题接口
public interface Subject{
//注册观察者
public void registerObserver(Observer o);
//删除观察者
public void removeObserver(Observer o);
//当主题状态改变时,这个方法会被调用,以通知所有的观察者
public void notifyObserver();
}
public interface Observer {
//当气象观测值改变时,主题会把这些状态值当作方法的参数,传送给观察者
public void update(float temp,float humidity,float pressure);
}
public interface DisplayElement{
//当需要显示时,调用此方法
public void display();
}
2.WeatherData类实现主题接口
public class WeatherData implements Subject{
private ArrayList<Observer> observers;
private float temperature;
private float humidity;
private float pressure;
public WeatherData(){
observers=new ArrayList<Observer>();
}
@Override
public void registerObserver(Observer o) {
observers.add(o);//一个观察者注册的时候,把它添加到list的末端
}
@Override
public void removeObserver(Observer o) {
int i=observers.indexOf(o);//获得对象索引
if(i>=0){
observers.remove(i);
}
}
@Override
public void notifyObserver() {
for(Observer observer:observers){
observer.update(temperature,humidity,pressure);
}
}
//当从气象站得到更新观测值时,我们通知观察者
public void measurementsChanged(){
notifyObserver();
}
//测试数据
public void setMeasurements(float temperature,float humidity,float pressure){
this.temperature=temperature;
this.humidity=humidity;
this.pressure=pressure;
measurementsChanged();
}
//WeatherData的其他方法
}
3.构造显示元素
显示类实现Observer接口,所以可以从WeatherData对象中获取变化
public class CurrentConditionDisplay implements Observer,DisplayElement{
private float temperature;
private float humidity;
private Subject weatherData;
//构造器被传入weatherData对象(Subject),用它来把 【显示】 注册为 【观察者】
public CurrentConditionDisplay(WeatherData weatherData){
this.weatherData=weatherData;
weatherData.registerObserver(this);
}
@Override
public void display() {
System.out.println("current conditions:" + temperature + "F degrees and " + humidity + "% humidity");
}
@Override
public void update(float temp, float humidity, float pressure) {
this.temperature=temp;
this.humidity=humidity;
display();
}
}
4.启动气象站
public class WeatherStation {
public static void main(String[] args) {
//创建对象
WeatherData weatherData = new WeatherData();
//创建显示
CurrentConditionsDisplay currentDisplay = new CurrentConditionsDisplay(weatherData);
//模拟气象测量
weatherData.setMeasurements(80,65,30.4f);
}
}
缺点:
每次更新信息都会全部通知到每个观察者,而观察者只想要有用的信息;
不能扩展更多的显示【因为观察者接口的update方法是固定参数】
我们可以让Observer按需要从Subject拉取。当Subiect的数据变化,我们马上通过调用update()传送数据,推送新的温度、湿度和气压值给Observer。
为了切换到拉取方式,我们需要对已有代码做一些小小的修改
Subject的发送通知.....
修改WeatherData中的notifyObservers()方法,不带参数地调用Observer中的update()方法:
public void notifyobservers() {
for(Observer observer:observers){
observer.update();
}
}
Observer的接收通知....
修改Observer接口,改变update()方法的签名,这样它就没有参数了:
public interface Observer {
public void update();
}
修改每个具体Observer,改变其各自的update0方法的签名,并使用WeatherData的getter方法从Subject获取气象数据。CurrentConditionsDisplay类的新代码如下:
public void update() {
this.temperature = weatherData.getTemperature();
this,humidity = weatherData.getHumidity();
display();
}
OO基础:抽象
OO原则:
封装变化。
组合优于继承
针对接口编程,而不是针对实现。
尽力达到交互对象之间的松耦合设计。【新增】
OO模式:
观察者模式:定义对象之间的一对多依赖这样,当一个对象改变状态时,它的所有依赖者会被通知并自动更新。
观察者模式定义对象之间的一对多关系。
主题使用通用接口更新观察者。
任何具体类型的观察者都可参与该模式,只要它们实现察者接口。
观察者是松耦合的,除了知道它们实现观察者接口之外,主题对它们的其他事情不知情
使用该模式时,你可以从主题推或拉数据(拉被认为更“正确”)
Swing大量使用观察者模式许多GUI框架也是这样。
你也会在其他很多地方发现该模式,包括RxJava、JavaBeans和RMI,以及其他语言的框架,像Cocoa、Swift和JavaScript事件。
观察者模式和出版/订阅模式相关。出版/订阅模式用于更复杂得多主题和/或多消息类型的情形。
观察者模式是一个常用的模式,当我们学习模型-视图-控制器时,还会看到它。
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende
给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最
//1.验证返回状态码是否是200pm.test("Statuscodeis200",function(){pm.response.to.have.status(200);});//2.验证返回body内是否含有某个值pm.test("Bodymatchesstring",function(){pm.expect(pm.response.text()).to.include("string_you_want_to_search");});//3.验证某个返回值是否是100pm.test("Yourtestname",function(){varjsonData=pm.response.json
1.在Python3中,下列关于数学运算结果正确的是:(B)a=10b=3print(a//b)print(a%b)print(a/b)A.3,3,3.3333...B.3,1,3.3333...C.3.3333...,3.3333...,3D.3.3333...,1,3.3333...解析: 在Python中,//表示地板除(向下取整),%表示取余,/表示除(Python2向下取整返回3)2.如下程序Python2会打印多少个数:(D)k=1000whilek>1: print(k)k=k/2A.1000 B.10C.11D.9解析: 按照题意每次循环K/2,直到K值小于等
了解Rails缓存如何工作的人可以真正帮助我。这是嵌套在Rails::Initializer.runblock中的代码:config.after_initializedoSomeClass.const_set'SOME_CONST','SOME_VAL'end现在,如果我运行script/server并发出请求,一切都很好。然而,在我的Rails应用程序的第二个请求中,一切都因单元化常量错误而变得糟糕。在生产模式下,我可以成功发出第二个请求,这意味着常量仍然存在。我已通过将以上内容更改为以下内容来解决问题:config.after_initializedorequire'some_cl
我经常迷上ruby的一件事是递归模式。例如,假设我有一个数组,它可能包含无限深度的数组作为元素。所以,例如:my_array=[1,[2,3,[4,5,[6,7]]]]我想创建一个方法,可以将数组展平为[1,2,3,4,5,6,7]。我知道.flatten可以完成这项工作,但这个问题是作为我经常遇到的递归问题的一个例子-因此我试图找到一个更可重用的解决方案。简而言之-我猜这种事情有一个标准模式,但我想不出任何特别优雅的东西。任何想法表示赞赏 最佳答案 递归是一种方法,它不依赖于语言。您在编写算法时要考虑两种情况:再次调用函数的情
这应该是一个简单的问题,但我找不到任何相关信息。给定一个Ruby中的正则表达式,对于每个匹配项,我需要检索匹配的模式$1、$2,但我还需要匹配位置。我知道=~运算符为我提供了第一个匹配项的位置,而string.scan(/regex/)为我提供了所有匹配模式。如果可能,我需要在同一步骤中获得两个结果。 最佳答案 MatchDatastring.scan(regex)do$1#Patternatfirstposition$2#Patternatsecondposition$~.offset(1)#Startingandendingpo
我想开始使用“Sinatra”框架进行编码,但我找不到该框架的“MVC”模式。是“MVC-Sinatra”模式或框架吗? 最佳答案 您可能想查看Padrino这是一个围绕Sinatra构建的框架,可为您的项目提供更“类似Rails”的感觉,但没有那么多隐藏的魔法。这是使用Sinatra可以做什么的一个很好的例子。虽然如果您需要开始使用这很好,但我个人建议您将它用作学习工具,以对您来说最有意义的方式使用Sinatra构建您自己的应用程序。写一些测试/期望,写一些代码,通过测试-重复:)至于ORM,你还应该结帐Sequel其中(imho