官术网_书友最值得收藏!

The Observer design pattern

Another similarity that the keen reader may have noticed is with the Observer design pattern. It is mainly used in object-oriented applications as a way for objects to communicate with each other without having any knowledge of who depends on its changes.

In Clojure, a simple version of the Observer pattern can be implemented using watches:

(def numbers (atom [])) 
 
(defn adder [key ref old-state new-state] 
  (print "Current sum is " (reduce + new-state))) 
 
(add-watch numbers :adder adder) 

We will start by creating our program state, which in this case is an atom holding an empty vector. Next, we will create a watch function that knows how to sum all numbers in numbers. Finally, we will add our watch function to the numbers atom under the :adder key (useful for removing watches).

The adder key conforms with the API contract required by add-watch and receives four arguments. In this example, we only care about new-state.

Now, whenever we update the value of numbers, its watch will be executed, as demonstrated in the following code:

(swap! numbers conj 1) 
;; Current sum is  1 
 
(swap! numbers conj 2) 
;; Current sum is  3 
 
(swap! numbers conj 7) 
;; Current sum is  10 

The highlighted lines indicate the result that is printed on the screen each time we update the atom.

Though useful, the Observer pattern still requires some amount of work in setting up the dependencies and the required program state, in addition to being hard to compose.

That being said, this pattern has been extended and is at the core of one of the Reactive Programming frameworks we will look at later in this book, Microsoft's Reactive Extensions (Rx).

主站蜘蛛池模板: 延边| 临西县| 安远县| 平泉县| 祁东县| 九龙坡区| 三江| 二手房| 大港区| 巴马| 无极县| 长丰县| 富裕县| 夏津县| 朝阳县| 拜泉县| 霍林郭勒市| 扶绥县| 和田县| 南雄市| 兴义市| 桓台县| 昌黎县| 营山县| 安泽县| 永宁县| 宁阳县| 饶河县| 水富县| 芜湖县| 黄龙县| 洞头县| 华宁县| 乌拉特前旗| 武山县| 徐州市| 铜川市| 包头市| 明光市| 潜江市| 西乌珠穆沁旗|