定義:是保證一個類只有一個實例,而且提供一個訪問它的全局訪問點。html
需求:一些對象咱們每每只須要一個,好比線程池、全局緩存、瀏覽器中的window對象、登陸浮窗等。算法
優勢:瀏覽器
一、能夠用來劃分命名空間,減小全局變量的數量 緩存
二、能夠被實例化,且實例化一次,再次實例化生成的也是第一個實例
bash
class CreateUser{
constructor(name){
this.name = name;
this.getName();
}
getName(){
return this.name;
}
}
var ProxyMode = (function(){
var instance = null;
return function(name){
if(!instance){
instance = new CreateUser(name);
}
return instance;
}
})();
var a= new ProxyMode('aaa');//CreateUser {name: "aaa"}
var b= new ProxyMode('bbb');//CreateUser {name: "aaa"}結果都是第一個對象複製代碼
定義:對象間的一種一對多的依賴關係。app
需求:當一個對象的狀態發生變化時,全部依賴於他的對象都將獲得通知。函數
優勢:時間上的解耦,對象之間的解耦。ui
var targetObj = {
name:'aa'
}
var targetObj2 = {
name:'aa'
}
// 定義值改變時的處理函數(觀察者)
function observer(oldVal, newVal) {
// 其餘處理邏輯...
targetObj2.name = newVal
console.info(newVal);
}
// 定義name屬性及其set和get方法(name屬性爲被觀察者)
Object.defineProperty(targetObj, 'name', {
enumerable: true,
configurable: true,
get: function() {
return name;
},
set: function(val) {
observer(name, val)
name = val
}
});
targetObj.name = '11';
targetObj.name = '22';
console.log(targetObj2.name)複製代碼
在網上看到一個對發佈者/訂閱者模式與觀察者模式,比較精闢的總結:this
有中間商賺差價的是發佈者/訂閱者模式,無中間商賺差價的是觀察者模式。spa
function Publish(){
let s =[];
this.add= function(sMember){
let isExist = s.some(function(item){
if(item == sMember){
return true;
}else{
return false;
}
})
if(!isExist){
s.push(sMember);
}
}
this.send = function(data){
s.forEach(function(fn){
fn(data);
})
return this;
}
}
let a = function(data){
console.log(data);
}
let a = function(data){
console.log(data);
}
let p = new Publish();
p.add(a);
p.add(b);
p.send('sadasdsa');複製代碼
定義:將其成員對象的實例化推遲到子類來實現的類。
需求:建立對象的流程賦值的時候,好比依賴於不少設置文件等 ;處理大量具備相同屬性的小對象;注:不能濫用
優勢:不暴露建立對象的具體邏輯,而是將將邏輯封裝在一個函數中。
function createObjet(name ,age){
var obj = new Object();
obj.name = name;
obj.age = age;
obj.run = function (){
return this.name + this.age;
}
return obj;
}
var a = createObjet('lao',22);複製代碼
var Person = (function() {
var name = 'xxx'
function sayName() {
console.log(name)
}
return{
name: name,
sayName: sayName
}
})()複製代碼
使用ES6中的proxy實現
const handler = {
get: function() {
console.log('doing');
}
}
const initialObj = {
id: 1,
name: 'cc'
}
const proxiedObj = new Proxy(initialObj, handler);
console.log(proxiedObj.name);複製代碼
策略模式的本意將算法的使用與算法的實現分離開來,避免多重判斷調用哪些算法。適用於有多個判斷分支的場景,如解決表單驗證的問題。
// 對於vip客戶
function vipPrice() {
this.discount = 0.5;
}
vipPrice.prototype.getPrice = function(price) {
return price * this.discount;
}
// 對於老客戶
function oldPrice() {
this.discount = 0.3;
}
oldPrice.prototype.getPrice = function(price) {
return price * this.discount;
}
// 對於普通客戶
function Price() {
this.discount = 1;
}
Price.prototype.getPrice = function(price) {
return price ;
}
// 上下文,對於客戶端的使用
function Context() {
this.name = '';
this.strategy = null;
this.price = 0;
}
Context.prototype.set = function(name, strategy, price) {
this.name = name;
this.strategy = strategy;
this.price = price;
}
Context.prototype.getResult = function() {
console.log(this.name + ' 的結帳價爲: ' + this.strategy.getPrice(this.price));
}
var context = new Context();
var vip = new vipPrice();
context.set ('vip客戶', vip, 200);
context.getResult(); // vip客戶 的結帳價爲: 100
var old = new oldPrice();
context.set ('老客戶', old, 200);
context.getResult(); // 老客戶 的結帳價爲: 60
var Price = new Price();
context.set ('普通客戶', Price, 200);
context.getResult(); // 普通客戶 的結帳價爲: 200複製代碼
接下去的幾種模式設計到繼承這方面
function Car(name, year, country){
this.name = name;
this.year = year;
this.country = country;
}
Car.prototype.sayName = function(){
return this.name;
}
var mycar = new Car("BMW", 2017, "Germany");
console.log(mycar);複製代碼
function Parent(name){
this.name = name;
}
Parent.protoType.sayName = function(){
console.log('parent name:',this.name);
}
Parent.protoType.doSomeThing = function(){
console.log('parent doSomeThing');
}
function Child(name,fatherName){
Parent.call(this,faterName);
this.name = name;
}
Child.protoType = Parent.protoType;
Child.protoType.constructor = Child;
Child.protoType.sayNmae = function(){
console.log('child name:',this.name);
}
var child = new Child('son');
child.sayName();
child.doSomeThing();
//經過原型鏈實現對象屬性和方法的繼承,而經過構造函數來實現實例屬性的繼承。複製代碼
function Parent(name){
this.name = name;
}
Parent.protoType.sayName = function(){
console.log('parent name:',this.name);
}
Parent.protoType.doSomeThing = function(){
console.log('parent doSomeThing');
}
function Child(name,fatherName){
Parent.call(this,faterName);
this.name = name;
}
function create(proto){
function F(){}
F.proto=proto;
return new F();
}
Child.protoType = create(Parent.protoType);
Child.protoType.sayNmae = function(){
console.log('child name:',this.name);
}
Child.protoType.constructor = Child;
//用F(){}去替代父類執行構造函數複製代碼