真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

javascript設計模式之中介者模式的示例分析

這篇文章將為大家詳細講解有關(guān)javascript設計模式之中介者模式的示例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

成都創(chuàng)新互聯(lián)專注于企業(yè)網(wǎng)絡營銷推廣、網(wǎng)站重做改版、將樂網(wǎng)站定制設計、自適應品牌網(wǎng)站建設、H5開發(fā)、成都商城網(wǎng)站開發(fā)、集團公司官網(wǎng)建設、外貿(mào)網(wǎng)站制作、高端網(wǎng)站制作、響應式網(wǎng)頁設計等建站業(yè)務,價格優(yōu)惠性價比高,為將樂等各大城市提供網(wǎng)站開發(fā)制作服務。

Java可以用來干什么

Java主要應用于:1. web開發(fā);2. Android開發(fā);3. 客戶端開發(fā);4. 網(wǎng)頁開發(fā);5. 企業(yè)級應用開發(fā);6. Java大數(shù)據(jù)開發(fā);7.游戲開發(fā)等。

 先來理解這么一個問題,假如我們前端開發(fā)接的需求是需求方給我們需求,可能一個前端開發(fā)會和多個需求方打交道,所以會保持多個需求方的聯(lián)系,那么在程序里面就意味著保持多個對象的引用,當程序的規(guī)模越大,對象會越來越多,他們之間的關(guān)系會越來越復雜,那現(xiàn)在假如現(xiàn)在有一個中介者(假如就是我們的主管)來對接多個需求方的需求,那么需求方只需要把所有的需求給我們主管就可以,主管會依次看我們的工作量來給我們分配任務,這樣的話,我們前端開發(fā)就不需要和多個業(yè)務方聯(lián)系,我們只需要和我們主管(也就是中介)聯(lián)系即可,這樣的好處就弱化了對象之間的耦合。

日常生活中的列子:

中介者模式對于我們?nèi)粘I钪薪?jīng)常會碰到,比如我們?nèi)シ课葜薪槿プ夥?,房屋中介人在租房者和房東出租者之間形成一條中介;租房者并不關(guān)心租誰的房,房東出租者也并不關(guān)心它租給誰,因為有中介,所以需要中介來完成這場交易。

中介者模式的作用是解除對象與對象之間的耦合關(guān)系,增加一個中介對象后,所有的相關(guān)對象都通過中介者對象來通信,而不是相互引用,所以當一個對象發(fā)送改變時,只需要通知中介者對象即可。中介者使各個對象之間耦合松散,而且可以獨立地改變它們之間的交互。

實現(xiàn)中介者的列子如下:

不知道大家有沒有玩過英雄殺這個游戲,最早的時候,英雄殺有2個人(分別是敵人和自己);我們針對這個游戲先使用普通的函數(shù)來實現(xiàn)如下:

比如先定義一個函數(shù),該函數(shù)有三個方法,分別是win(贏), lose(輸),和die(敵人死亡)這三個函數(shù);只要一個玩家死亡該游戲就結(jié)束了,同時需要通知它的對手勝利了; 代碼需要編寫如下:

function Hero(name) {
  this.name = name;
  this.enemy = null; 
}
Hero.prototype.win = function(){
  console.log(this.name + 'Won');
}
Hero.prototype.lose = function(){
  console.log(this.name + 'lose');
}
Hero.prototype.die = function(){
  this.lose();
  this.enemy.win();
}
// 初始化2個對象
var h2 = new Hero("朱元璋");
var h3 = new Hero("劉伯溫");
// 給玩家設置敵人
h2.enemy = h3;
h3.enemy = h2;
// 朱元璋死了 也就輸了
h2.die(); // 輸出 朱元璋l(fā)ose 劉伯溫Won

現(xiàn)在我們再來為游戲添加隊友

比如現(xiàn)在我們來為游戲添加隊友,比如英雄殺有6人一組,那么這種情況下就有隊友,敵人也有3個;因此我們需要區(qū)分是敵人還是隊友需要隊的顏色這個字段,如果隊的顏色相同的話,那么就是同一個隊的,否則的話就是敵人;

我們可以先定義一個數(shù)組players來保存所有的玩家,在創(chuàng)建玩家之后,循環(huán)players來給每個玩家設置隊友或者敵人;

var players = [];

接著我們再來編寫Hero這個函數(shù);代碼如下:

var players = []; // 定義一個數(shù)組 保存所有的玩家
function Hero(name,teamColor) {
  this.friends = [];  //保存隊友列表
  this.enemies = [];  // 保存敵人列表
  this.state = 'live'; // 玩家狀態(tài)
  this.name = name;   // 角色名字
  this.teamColor = teamColor; // 隊伍的顏色
}
Hero.prototype.win = function(){
  // 贏了
  console.log("win:" + this.name);
};
Hero.prototype.lose = function(){
  // 輸了
  console.log("lose:" + this.name);
};
Hero.prototype.die = function(){
  // 所有隊友死亡情況 默認都是活著的
  var all_dead = true;
  this.state = 'dead'; // 設置玩家狀態(tài)為死亡
  for(var i = 0,ilen = this.friends.length; i < ilen; i+=1) {
    // 遍歷,如果還有一個隊友沒有死亡的話,則游戲還未結(jié)束
    if(this.friends[i].state !== 'dead') {
      all_dead = false; 
      break;
    }
  }
  if(all_dead) {
    this.lose(); // 隊友全部死亡,游戲結(jié)束
    // 循環(huán) 通知所有的玩家 游戲失敗
    for(var j = 0,jlen = this.friends.length; j < jlen; j+=1) {
      this.friends[j].lose();
    }
    // 通知所有敵人游戲勝利
    for(var j = 0,jlen = this.enemies.length; j < jlen; j+=1) {
      this.enemies[j].win();
    }
  }
}
// 定義一個工廠類來創(chuàng)建玩家 
var heroFactory = function(name,teamColor) {
  var newPlayer = new Hero(name,teamColor);
  for(var i = 0,ilen = players.length; i < ilen; i+=1) {
    // 如果是同一隊的玩家
    if(players[i].teamColor === newPlayer.teamColor) {
      // 相互添加隊友列表
      players[i].friends.push(newPlayer);
      newPlayer.friends.push(players[i]);
    }else {
      // 相互添加到敵人列表
      players[i].enemies.push(newPlayer);
      newPlayer.enemies.push(players[i]);
    }
  }
  players.push(newPlayer);
  return newPlayer;
};
    // 紅隊
var p1 = heroFactory("aa",'red'),
  p2 = heroFactory("bb",'red'),
  p3 = heroFactory("cc",'red'),
  p4 = heroFactory("dd",'red');
    
// 藍隊
var p5 = heroFactory("ee",'blue'),
  p6 = heroFactory("ff",'blue'),
  p7 = heroFactory("gg",'blue'),
  p8 = heroFactory("hh",'blue');
// 讓紅隊玩家全部死亡
p1.die();
p2.die();
p3.die();
p4.die();
// lose:dd lose:aa lose:bb lose:cc
// win:ee win:ff win:gg win:hh

如上代碼:Hero函數(shù)有2個參數(shù),分別是name(玩家名字)和teamColor(隊顏色),

首先我們可以根據(jù)隊顏色來判斷是隊友還是敵人;同樣也有三個方法win(贏),lose(輸),和die(死亡);如果每次死亡一個人的時候,循環(huán)下該死亡的隊友有沒有全部死亡,如果全部死亡了的話,就輸了,因此需要循環(huán)他們的隊友,分別告訴每個隊友中的成員他們輸了,同時需要循環(huán)他們的敵人,分別告訴他們的敵人他們贏了;因此每次死了一個人的時候,都需要循環(huán)一次判斷他的隊友是否都死亡了;因此每個玩家和其他的玩家都是緊緊耦合在一起了。

下面我們可以使用中介者模式來改善上面的demo;

首先我們?nèi)匀欢xHero構(gòu)造函數(shù)和Hero對象原型的方法,在Hero對象的這些原型方法中,不再負責具體的執(zhí)行的邏輯,而是把操作轉(zhuǎn)交給中介者對象,中介者對象來負責做具體的事情,我們可以把中介者對象命名為playerDirector;

在playerDirector開放一個對外暴露的接口ReceiveMessage,負責接收player對象發(fā)送的消息,而player對象發(fā)送消息的時候,總是把自身的this作為參數(shù)發(fā)送給playerDirector,以便playerDirector 識別消息來自于那個玩家對象。

代碼如下:

var players = []; // 定義一個數(shù)組 保存所有的玩家
function Hero(name,teamColor) {
  this.state = 'live'; // 玩家狀態(tài)
  this.name = name;   // 角色名字
  this.teamColor = teamColor; // 隊伍的顏色
}
Hero.prototype.win = function(){
  // 贏了
  console.log("win:" + this.name);
};
Hero.prototype.lose = function(){
  // 輸了
  console.log("lose:" + this.name);
};
// 死亡
Hero.prototype.die = function(){
  this.state = 'dead';
  // 給中介者發(fā)送消息,玩家死亡
  playerDirector.ReceiveMessage('playerDead',this);
}
// 移除玩家
Hero.prototype.remove = function(){
  // 給中介者發(fā)送一個消息,移除一個玩家
  playerDirector.ReceiveMessage('removePlayer',this);
};
// 玩家換隊
Hero.prototype.changeTeam = function(color) {
  // 給中介者發(fā)送一個消息,玩家換隊
  playerDirector.ReceiveMessage('changeTeam',this,color);
};
// 定義一個工廠類來創(chuàng)建玩家 
var heroFactory = function(name,teamColor) {
  // 創(chuàng)建一個新的玩家對象
  var newHero = new Hero(name,teamColor);
  // 給中介者發(fā)送消息,新增玩家
  playerDirector.ReceiveMessage('addPlayer',newHero);
  return newHero;
};
var playerDirector = (function(){
  var players = {}, // 保存所有的玩家
    operations = {}; // 中介者可以執(zhí)行的操作
  // 新增一個玩家操作
  operations.addPlayer = function(player) {
    // 獲取玩家隊友的顏色
    var teamColor = player.teamColor;
    // 如果該顏色的玩家還沒有隊伍的話,則新成立一個隊伍
    players[teamColor] = players[teamColor] || [];
    // 添加玩家進隊伍
    players[teamColor].push(player);
   };
  // 移除一個玩家
  operations.removePlayer = function(player){
    // 獲取隊伍的顏色
    var teamColor = player.teamColor,
    // 獲取該隊伍的所有成員
    teamPlayers = players[teamColor] || [];
    // 遍歷
    for(var i = teamPlayers.length - 1; i>=0; i--) {
      if(teamPlayers[i] === player) {
        teamPlayers.splice(i,1);
      }
    }
  };
  // 玩家換隊
  operations.changeTeam = function(player,newTeamColor){
    // 首先從原隊伍中刪除
    operations.removePlayer(player);
    // 然后改變隊伍的顏色
    player.teamColor = newTeamColor;
    // 增加到隊伍中
    operations.addPlayer(player);
  };
  // 玩家死亡
operations.playerDead = function(player) {
  var teamColor = player.teamColor,
  // 玩家所在的隊伍
  teamPlayers = players[teamColor];

  var all_dead = true;
  //遍歷 
  for(var i = 0,player; player = teamPlayers[i++]; ) {
    if(player.state !== 'dead') {
      all_dead = false;
      break;
    }
  }
  // 如果all_dead 為true的話 說明全部死亡
  if(all_dead) {
    for(var i = 0, player; player = teamPlayers[i++]; ) {
      // 本隊所有玩家lose
      player.lose();
    }
    for(var color in players) {
      if(color !== teamColor) {
        // 說明這是另外一組隊伍
        // 獲取該隊伍的玩家
        var teamPlayers = players[color];
        for(var i = 0,player; player = teamPlayers[i++]; ) {
          player.win(); // 遍歷通知其他玩家win了
        }
      }
    }
  }
};
var ReceiveMessage = function(){
  // arguments的第一個參數(shù)為消息名稱 獲取第一個參數(shù)
  var message = Array.prototype.shift.call(arguments);
  operations[message].apply(this,arguments);
};
return {
  ReceiveMessage : ReceiveMessage
};
})();
// 紅隊
var p1 = heroFactory("aa",'red'),
  p2 = heroFactory("bb",'red'),
  p3 = heroFactory("cc",'red'),
    p4 = heroFactory("dd",'red');
    
  // 藍隊
  var p5 = heroFactory("ee",'blue'),
    p6 = heroFactory("ff",'blue'),
    p7 = heroFactory("gg",'blue'),
    p8 = heroFactory("hh",'blue');
  // 讓紅隊玩家全部死亡
  p1.die();
  p2.die();
  p3.die();
  p4.die();
  // lose:aa lose:bb lose:cc lose:dd 
  // win:ee win:ff win:gg win:hh

我們可以看到如上代碼;玩家與玩家之間的耦合代碼已經(jīng)解除了,而把所有的邏輯操作放在中介者對象里面進去處理,某個玩家的任何操作不需要去遍歷去通知其他玩家,而只是需要給中介者發(fā)送一個消息即可,中介者接受到該消息后進行處理,處理完消息之后會把處理結(jié)果反饋給其他的玩家對象。使用中介者模式解除了對象與對象之間的耦合代碼; 使程序更加的靈活.

中介者模式實現(xiàn)購買商品的列子

下面的列子是書上的列子,比如在淘寶或者天貓的列子不是這樣實現(xiàn)的,也沒有關(guān)系,我們可以改動下即可,我們最主要來學習下使用中介者模式來實現(xiàn)的思路。

首先先介紹一下業(yè)務:在購買流程中,可以選擇手機的顏色以及輸入購買的數(shù)量,同時頁面中有2個展示區(qū)域,分別顯示用戶剛剛選擇好的顏色和數(shù)量。還有一個按鈕動態(tài)顯示下一步的操作,我們需要查詢該顏色手機對應的庫存,如果庫存數(shù)量小于這次的購買數(shù)量,按鈕則被禁用并且顯示庫存不足的文案,反之按鈕高亮且可以點擊并且顯示假如購物車。

HTML代碼如下:

選擇顏色:

  
    請選擇
    紅色
    藍色
  
  

輸入購買的數(shù)量: 

  你選擇了的顏色:
  

你輸入的數(shù)量: 

 

  請選擇手機顏色和購買數(shù)量

首先頁面上有一個select選擇框,然后有輸入的購買數(shù)量輸入框,還有2個展示區(qū)域,分別是選擇的顏色和輸入的數(shù)量的顯示的區(qū)域,還有下一步的按鈕操作;

我們先定義一下:

假設我們提前從后臺獲取到所有顏色手機的庫存量

var goods = {
  // 手機庫存
  "red": 6,
  "blue": 8
};

接著 我們下面分別來監(jiān)聽colorSelect的下拉框的onchange事件和numberInput輸入框的oninput的事件,然后在這兩個事件中作出相應的處理

常規(guī)的JS代碼如下:

// 假設我們提前從后臺獲取到所有顏色手機的庫存量
var goods = {
  // 手機庫存
  "red": 6,
  "blue": 8
};
/*
我們下面分別來監(jiān)聽colorSelect的下拉框的onchange事件和numberInput輸入框的oninput的事件,
然后在這兩個事件中作出相應的處理
*/
var colorSelect = document.getElementById("colorSelect"),
  numberInput = document.getElementById("numberInput"),
  colorInfo = document.getElementById("colorInfo"),
  numberInfo = document.getElementById("numberInfo"),
  nextBtn = document.getElementById("nextBtn");
    
// 監(jiān)聽change事件
colorSelect.onchange = function(e){
  select();
};
numberInput.oninput = function(){
  select();
};
function select(){
  var color = colorSelect.value,  // 顏色
    number = numberInput.value, // 數(shù)量
    stock = goods[color]; // 該顏色手機對應的當前庫存
      
  colorInfo.innerHTML = color;
  numberInfo.innerHTML = number;

  // 如果用戶沒有選擇顏色的話,禁用按鈕
  if(!color) {
    nextBtn.disabled = true;
    nextBtn.innerHTML = "請選擇手機顏色";
      return;
  }
  // 判斷用戶輸入的購買數(shù)量是否是正整數(shù)
  var reg = /^\d+$/g;
  if(!reg.test(number)) {
    nextBtn.disabled = true;
    nextBtn.innerHTML = "請輸入正確的購買數(shù)量";
    return;
  }
  // 如果當前選擇的數(shù)量大于當前的庫存的數(shù)量的話,顯示庫存不足
  if(number > stock) {
    nextBtn.disabled = true;
    nextBtn.innerHTML = "庫存不足";
    return;
  }
  nextBtn.disabled = false;
  nextBtn.innerHTML = "放入購物車";
}

上面的代碼雖然是完成了頁面上的需求,但是我們的代碼都耦合在一起了,目前雖然問題不是很多,假如隨著以后需求的改變,SKU屬性越來越多的話,比如頁面增加一個或者多個下拉框的時候,代表選擇手機內(nèi)存,現(xiàn)在我們需要計算顏色,內(nèi)存和購買數(shù)量,來判斷nextBtn是顯示庫存不足還是放入購物車;代碼如下:

HTML代碼如下:

選擇顏色:
  
    請選擇
    紅色
    藍色
  
  
  
  選擇內(nèi)存:        請選擇     32G     64G      

輸入購買的數(shù)量: 

  你選擇了的顏色:
  你選擇了內(nèi)存:   

你輸入的數(shù)量:  

  請選擇手機顏色和購買數(shù)量

JS代碼變?yōu)槿缦拢?/p>

// 假設我們提前從后臺獲取到所有顏色手機的庫存量
var goods = {
  // 手機庫存
  "red|32G": 6,
  "red|64G": 16,
  "blue|32G": 8,
  "blue|64G": 18
};
/*
我們下面分別來監(jiān)聽colorSelect的下拉框的onchange事件和numberInput輸入框的oninput的事件,
然后在這兩個事件中作出相應的處理
 */
var colorSelect = document.getElementById("colorSelect"),
  memorySelect = document.getElementById("memorySelect"),
  numberInput = document.getElementById("numberInput"),
  colorInfo = document.getElementById("colorInfo"),
  numberInfo = document.getElementById("numberInfo"),
  memoryInfo = document.getElementById("memoryInfo"),
  nextBtn = document.getElementById("nextBtn");
    
// 監(jiān)聽change事件
colorSelect.onchange = function(){
  select();
};
numberInput.oninput = function(){
  select();
};
memorySelect.onchange = function(){
  select();  
};
function select(){
  var color = colorSelect.value,  // 顏色
    number = numberInput.value, // 數(shù)量
    memory = memorySelect.value, // 內(nèi)存
    stock = goods[color + '|' +memory]; // 該顏色手機對應的當前庫存
      
  colorInfo.innerHTML = color;
  numberInfo.innerHTML = number;
  memoryInfo.innerHTML = memory;
  // 如果用戶沒有選擇顏色的話,禁用按鈕
  if(!color) {
    nextBtn.disabled = true;
    nextBtn.innerHTML = "請選擇手機顏色";
      return;
    }
    // 判斷用戶輸入的購買數(shù)量是否是正整數(shù)
    var reg = /^\d+$/g;
    if(!reg.test(number)) {
      nextBtn.disabled = true;
      nextBtn.innerHTML = "請輸入正確的購買數(shù)量";
      return;
    }
    // 如果當前選擇的數(shù)量大于當前的庫存的數(shù)量的話,顯示庫存不足
    if(number > stock) {
      nextBtn.disabled = true;
      nextBtn.innerHTML = "庫存不足";
      return;
    }
    nextBtn.disabled = false;
    nextBtn.innerHTML = "放入購物車";
  }

關(guān)于“javascript設計模式之中介者模式的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。


文章名稱:javascript設計模式之中介者模式的示例分析
URL網(wǎng)址:http://weahome.cn/article/gopdhs.html

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部