var pubsub = {};
(function (q) {
var topics = {}, // 回調函數存放的數組
subUid = -1;
// 發佈方法
q.publish = function (topic, args) {
if (!topics[topic]) {
return false;
}
setTimeout(function () {
var subscribers = topics[topic],
len = subscribers ? subscribers.length : 0;
while (len--) {
subscribers[len].func(topic, args);
}
}, 0);
return true;
};
//訂閱方法
q.subscribe = function (topic, func) {
if (!topics[topic]) {
topics[topic] = [];
}
var token = (++subUid).toString();
topics[topic].push({
token: token,
func: func
});
return token;
};
//退訂方法
q.unsubscribe = function (token) {
for (var m in topics) {
if (topics[m]) {
for (var i = 0, j = topics[m].length; i < j; i++) {
if (topics[m][i].token === token) {
topics[m].splice(i, 1);
return token;
}
}
}
}
return false;
};
} (pubsub));
將訂閱者放在了字典值,發佈時通過控制鍵來控制訂閱者。
----------------------------------------------------------------
function Observer() {
this.fns = [];
}
Observer.prototype = {
subscribe: function (fn) {
this.fns.push(fn);
},
unsubscribe: function (fn) {
this.fns = this.fns.filter(
function (el) {
if (el !== fn) {
return el;
}
}
);
},
update: function (o, thisObj) {
var scope = thisObj || window;
this.fns.forEach(
function (el) {
el.call(scope, o);
}
);
}
};
//測試
var o = new Observer;
var f1 = function (data) {
console.log('Robbin: ' + data + ', 趕緊幹活了!');
};
var f2 = function (data) {
console.log('Randall: ' + data + ', 找他加點工資去!');
};
o.subscribe(f1);
o.subscribe(f2);
o.update("Tom回來了!")
//退訂f1
o.unsubscribe(f1);
//再來驗證
o.update("Tom回來了!");
和上相同, 把所有的訂閱者放在了數組裏面了。
---------------------------------------------------------------------
//通用代碼
var observer = {
//訂閱
addSubscriber: function (callback) {
this.subscribers[this.subscribers.length] = callback;
},
//退訂
removeSubscriber: function (callback) {
for (var i = 0; i < this.subscribers.length; i++) {
if (this.subscribers[i] === callback) {
delete (this.subscribers[i]);
}
}
},
//發佈
publish: function (what) {
for (var i = 0; i < this.subscribers.length; i++) {
if (typeof this.subscribers[i] === 'function') {
this.subscribers[i](what);
}
}
},
// 將對象o具有觀察者功能 [ 將 o 對象賦予此對象的功能 ]
make: function (o) {
for (var i in this) {
o[i] = this[i];
o.subscribers = [];
}
}
};
var blogger = {
recommend: function (id) {
var msg = 'dudu 推薦了的帖子:' + id;
this.publish(msg);
}
};
var user = {
vote: function (id) {
var msg = '有人投票了!ID=' + id;
this.publish(msg);
}
};
observer.make(blogger);
observer.make(user);
var tom = {
read: function (what) {
console.log('Tom看到了如下信息:' + what)
}
};
var mm = {
show: function (what) {
console.log('mm看到了如下信息:' + what)
}
};
// 訂閱
blogger.addSubscriber(tom.read);
blogger.addSubscriber(mm.show);
blogger.recommend(123); //調用發佈
//退訂
blogger.removeSubscriber(mm.show);
blogger.recommend(456); //調用發佈
//另外一個對象的訂閱
user.addSubscriber(mm.show);
user.vote(789); //調用發佈
比較重要的是上面那個make函數 ,將對象擴充。賦予了參數對象職能。
-----------------------------------------------------------
jQuery 版本:
(function ($) {
var o = $({});
$.subscribe = function () {
o.on.apply(o, arguments);
};
$.unsubscribe = function () {
o.off.apply(o, arguments);
};
$.publish = function () {
o.trigger.apply(o, arguments);
};
} (jQuery));
調用方法比上面3個版本都簡單:
//回調函數
function handle(e, a, b, c) {
// `e`是事件對象,不需要關注
console.log(a + b + c);
};
//訂閱
$.subscribe("/some/topic", handle);
//發佈
$.publish("/some/topic", ["a", "b", "c"]); // 輸出abc
$.unsubscribe("/some/topic", handle); // 退訂
//訂閱
$.subscribe("/some/topic", function (e, a, b, c) {
console.log(a + b + c);
});
$.publish("/some/topic", ["a", "b", "c"]); // 輸出abc
//退訂(退訂使用的是/some/topic名稱,而不是回調函數哦,和版本一的例子不一樣
$.unsubscribe("/some/topic");
我是發佈者 你是訂閱者。。 我是主動, 你是被動。 一切都是我主動來做事情。。