日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

依赖注入 这样的坑游戏编程要谨慎

發(fā)布時間:2024/8/26 编程问答 45 豆豆
生活随笔 收集整理的這篇文章主要介紹了 依赖注入 这样的坑游戏编程要谨慎 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

1 IGame游戲公司的故事

1.1 討論會

話說有一個叫IGame的游戲公司,正在開發(fā)一款A(yù)RPG游戲(動作&角色扮演類游戲,如魔獸世界、夢幻西游這一類的游戲)。一般這類游戲都有一個基本的功能,就是打怪(玩家攻擊怪物,借此獲得經(jīng)驗、虛擬貨幣和虛擬裝備),并且根據(jù)玩家角色所裝備的武器不同,攻擊效果也不同。這天,IGame公司的開發(fā)小組正在開會對打怪功能中的某一個功能點如何實現(xiàn)進行討論,他們面前的大屏幕上是這樣一份需求描述的ppt:
?

圖1.1 需求描述ppt


各個開發(fā)人員,面對這份需求,展開了熱烈的討論,下面我們看看討論會上都發(fā)生了什么。

1.2 實習(xí)生小李的實現(xiàn)方式

在經(jīng)過一番討論后,項目組長Peter覺得有必要整理一下各方的意見,他首先詢問小李的看法。小李是某學(xué)校計算機系大三學(xué)生,對游戲開發(fā)特別感興趣,目前是IGame公司的一名實習(xí)生。

經(jīng)過短暫的思考,小李闡述了自己的意見:

“我認為,這個需求可以這么實現(xiàn)。HP當然是怪物的一個屬性成員,而武器是角色的一個屬性成員,類型可以使字符串,用于描述目前角色所裝備的武器。角色類有一個攻擊方法,以被攻擊怪物為參數(shù),當實施一次攻擊時,攻擊方法被調(diào)用,而這個方法首先判斷當前角色裝備了什么武器,然后據(jù)此對被攻擊怪物的HP進行操作,以產(chǎn)生不同效果。”

而在闡述完后,小李也飛快的在自己的電腦上寫了一個Demo,來演示他的想法,Demo代碼如下。
?

  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • namespace IGameLi
  • {
  • ? ?
  • /// <summary>
  • ? ?
  • /// 怪物
  • ? ?
  • /// </summary>
  • ? ? internal sealed class Monster
  • ? ? {
  • ? ?? ???
  • /// <summary>
  • ? ?? ???
  • /// 怪物的名字
  • ? ?? ???
  • /// </summary>
  • ? ?? ???public String Name { get; set; }
  • ? ?? ???
  • /// <summary>
  • ? ?? ???
  • /// 怪物的生命值
  • ? ?? ???
  • /// </summary>
  • ? ?? ???public Int32 HP { get; set; }
  • ? ?? ???public Monster(String name,Int32 hp)
  • ? ?? ???{
  • ? ?? ?? ?? ?this.Name = name;
  • ? ?? ?? ?? ?this.HP = hp;
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace IGameLi
  • {
  • ? ?
  • /// <summary>
  • ? ?
  • /// 角色
  • ? ?
  • /// </summary>
  • ? ? internal sealed class Role
  • ? ? {
  • ? ?? ???private Random _random = new Random();
  • ? ?
  • ? ?? ???
  • /// <summary>
  • ? ?? ???
  • /// 表示角色目前所持武器的字符串
  • ? ?? ???
  • /// </summary>
  • ? ?? ???public String WeaponTag { get; set; }
  • ? ?
  • ? ?? ???
  • /// <summary>
  • ? ?? ???
  • /// 攻擊怪物
  • ? ?? ???
  • /// </summary>
  • ? ?? ???
  • /// <param name="monster">被攻擊的怪物</param>
  • ? ?? ???public void Attack(Monster monster)
  • ? ?? ???{
  • ? ?? ?? ?? ?if (monster.HP <= 0)
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? Console.WriteLine("此怪物已死");
  • ? ?? ?? ?? ?? ? return;
  • ? ?? ?? ?? ?}
  • ? ?
  • ? ?? ?? ?? ?if ("WoodSword" == this.WeaponTag)
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? monster.HP -= 20;
  • ? ?? ?? ?? ?? ? if (monster.HP <= 0)
  • ? ?? ?? ?? ?? ? {
  • ? ?? ?? ?? ?? ?? ???Console.WriteLine("攻擊成功!怪物" + monster.Name + "已死亡");
  • ? ?? ?? ?? ?? ? }
  • ? ?? ?? ?? ?? ? else
  • ? ?? ?? ?? ?? ? {
  • ? ?? ?? ?? ?? ?? ???Console.WriteLine("攻擊成功!怪物" + monster.Name + "損失20HP");
  • ? ?? ?? ?? ?? ? }
  • ? ?? ?? ?? ?}
  • ? ?? ?? ?? ?else if ("IronSword" == this.WeaponTag)
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? monster.HP -= 50;
  • ? ?? ?? ?? ?? ? if (monster.HP <= 0)
  • ? ?? ?? ?? ?? ? {
  • ? ?? ?? ?? ?? ?? ???Console.WriteLine("攻擊成功!怪物" + monster.Name + "已死亡");
  • ? ?? ?? ?? ?? ? }
  • ? ?? ?? ?? ?? ? else
  • ? ?? ?? ?? ?? ? {
  • ? ?? ?? ?? ?? ?? ???Console.WriteLine("攻擊成功!怪物" + monster.Name + "損失50HP");
  • ? ?? ?? ?? ?? ? }
  • ? ?? ?? ?? ?}
  • ? ?? ?? ?? ?else if ("MagicSword" == this.WeaponTag)
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? Int32 loss = (_random.NextDouble() < 0.5) ? 100 : 200;
  • ? ?? ?? ?? ?? ? monster.HP -= loss;
  • ? ?? ?? ?? ?? ? if (200 == loss)
  • ? ?? ?? ?? ?? ? {
  • ? ?? ?? ?? ?? ?? ???Console.WriteLine("出現(xiàn)暴擊!!!");
  • ? ?? ?? ?? ?? ? }
  • ? ?
  • ? ?? ?? ?? ?? ? if (monster.HP <= 0)
  • ? ?? ?? ?? ?? ? {
  • ? ?? ?? ?? ?? ?? ???Console.WriteLine("攻擊成功!怪物" + monster.Name + "已死亡");
  • ? ?? ?? ?? ?? ? }
  • ? ?? ?? ?? ?? ? else
  • ? ?? ?? ?? ?? ? {
  • ? ?? ?? ?? ?? ?? ???Console.WriteLine("攻擊成功!怪物" + monster.Name + "損失" + loss + "HP");
  • ? ?? ?? ?? ?? ? }
  • ? ?? ?? ?? ?}
  • ? ?? ?? ?? ?else
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? Console.WriteLine("角色手里沒有武器,無法攻擊!");
  • ? ?? ?? ?? ?}
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace IGameLi
  • {
  • ? ? class Program
  • ? ? {
  • ? ?? ???static void Main(string[] args)
  • ? ?? ???{
  • ? ?? ?? ?? ?
  • //生成怪物
  • ? ?? ?? ?? ?Monster monster1 = new Monster("小怪A", 50);
  • ? ?? ?? ?? ?Monster monster2 = new Monster("小怪B", 50);
  • ? ?? ?? ?? ?Monster monster3 = new Monster("關(guān)主", 200);
  • ? ?? ?? ?? ?Monster monster4 = new Monster("最終Boss", 1000);
  • ? ?
  • ? ?? ?? ?? ?
  • //生成角色
  • ? ?? ?? ?? ?Role role = new Role();
  • ? ?
  • ? ?? ?? ?? ?
  • //木劍攻擊
  • ? ?? ?? ?? ?role.WeaponTag = "WoodSword";
  • ? ?? ?? ?? ?role.Attack(monster1);
  • ? ?
  • ? ?? ?? ?? ?
  • //鐵劍攻擊
  • ? ?? ?? ?? ?role.WeaponTag = "IronSword";
  • ? ?? ?? ?? ?role.Attack(monster2);
  • ? ?? ?? ?? ?role.Attack(monster3);
  • ? ?
  • ? ?? ?? ?? ?
  • //魔劍攻擊
  • ? ?? ?? ?? ?role.WeaponTag = "MagicSword";
  • ? ?? ?? ?? ?role.Attack(monster3);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?
  • ? ?? ?? ?? ?Console.ReadLine();
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼


    程序運行結(jié)果如下:
    ?

    圖1.2 小李程序的運行結(jié)果


    1.3 架構(gòu)師的建議

    小李闡述完自己的想法并演示了Demo后,項目組長Peter首先肯定了小李的思考能力、編程能力以及初步的面向?qū)ο蠓治雠c設(shè)計的思想,并承認小李的程序正確完成了需求中的功能。但同時,Peter也指出小李的設(shè)計存在一些問題,他請小于講一下自己的看法。

    小于是一名有五年軟件架構(gòu)經(jīng)驗的架構(gòu)師,對軟件架構(gòu)、設(shè)計模式和面向?qū)ο笏枷胗休^深入的認識。他向Peter點了點頭,發(fā)表了自己的看法:

    “小李的思考能力是不錯的,有著基本的面向?qū)ο蠓治鲈O(shè)計能力,并且程序正確完成了所需要的功能。不過,這里我想從架構(gòu)角度,簡要說一下我認為這個設(shè)計中存在的問題。

    首先,小李設(shè)計的Role類的Attack方法很長,并且方法中有一個冗長的if…else結(jié)構(gòu),且每個分支的代碼的業(yè)務(wù)邏輯很相似,只是很少的地方不同。

    再者,我認為這個設(shè)計比較大的一個問題是,違反了OCP原則。在這個設(shè)計中,如果以后我們增加一個新的武器,如倚天劍,每次攻擊損失500HP,那么,我們就要打開Role,修改Attack方法。而我們的代碼應(yīng)該是對修改關(guān)閉的,當有新武器加入的時候,應(yīng)該使用擴展完成,避免修改已有代碼。

    一般來說,當一個方法里面出現(xiàn)冗長的if…else或switch…case結(jié)構(gòu),且每個分支代碼業(yè)務(wù)相似時,往往預(yù)示這里應(yīng)該引入多態(tài)性來解決問題。而這里,如果把不同武器攻擊看成一個策略,那么引入策略模式(Strategy Pattern)是明智的選擇。

    最后說一個小的問題,被攻擊后,減HP、死亡判斷等都是怪物的職責,這里放在Role中有些不當。”

    Tip:OCP原則,即開放關(guān)閉原則,指設(shè)計應(yīng)該對擴展開放,對修改關(guān)閉。

    Tip:策略模式,英文名Strategy Pattern,指定義算法族,分別封裝起來,讓他們之間可以相互替換,此模式使得算法的變化獨立于客戶。

    小于邊說,邊畫了一幅UML類圖,用于直觀表示他的思想。
    ?

    圖1.3 小于的設(shè)計


    Peter讓小李按照小于的設(shè)計重構(gòu)Demo,小李看了看小于的設(shè)計圖,很快完成。相關(guān)代碼如下:
    ?

  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace IGameLiAdv
  • {
  • ? ? internal interface IAttackStrategy
  • ? ? {
  • ? ?? ???void AttackTarget(Monster monster);
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace IGameLiAdv
  • {
  • ? ? internal sealed class WoodSword : IAttackStrategy
  • ? ? {
  • ? ?? ???public void AttackTarget(Monster monster)
  • ? ?? ???{
  • ? ?? ?? ?? ?monster.Notify(20);
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace IGameLiAdv
  • {
  • ? ? internal sealed class IronSword : IAttackStrategy
  • ? ? {
  • ? ?? ???public void AttackTarget(Monster monster)
  • ? ?? ???{
  • ? ?? ?? ?? ?monster.Notify(50);
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace IGameLiAdv
  • {
  • ? ? internal sealed class MagicSword : IAttackStrategy
  • ? ? {
  • ? ?? ???private Random _random = new Random();
  • ? ?
  • ? ?? ???public void AttackTarget(Monster monster)
  • ? ?? ???{
  • ? ?? ?? ?? ?Int32 loss = (_random.NextDouble() < 0.5) ? 100 : 200;
  • ? ?? ?? ?? ?if (200 == loss)
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? Console.WriteLine("出現(xiàn)暴擊!!!");
  • ? ?? ?? ?? ?}
  • ? ?? ?? ?? ?monster.Notify(loss);
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace IGameLiAdv
  • {
  • ? ?
  • /// <summary>
  • ? ?
  • /// 怪物
  • ? ?
  • /// </summary>
  • ? ? internal sealed class Monster
  • ? ? {
  • ? ?? ???
  • /// <summary>
  • ? ?? ???
  • /// 怪物的名字
  • ? ?? ???
  • /// </summary>
  • ? ?? ???public String Name { get; set; }
  • ? ?
  • ? ?? ???
  • /// <summary>
  • ? ?? ???
  • /// 怪物的生命值
  • ? ?? ???
  • /// </summary>
  • ? ?? ???private Int32 HP { get; set; }
  • ? ?
  • ? ?? ???public Monster(String name,Int32 hp)
  • ? ?? ???{
  • ? ?? ?? ?? ?this.Name = name;
  • ? ?? ?? ?? ?this.HP = hp;
  • ? ?? ???}
  • ? ?
  • ? ?? ???
  • /// <summary>
  • ? ?? ???
  • /// 怪物被攻擊時,被調(diào)用的方法,用來處理被攻擊后的狀態(tài)更改
  • ? ?? ???
  • /// </summary>
  • ? ?? ???
  • /// <param name="loss">此次攻擊損失的HP</param>
  • ? ?? ???public void Notify(Int32 loss)
  • ? ?? ???{
  • ? ?? ?? ?? ?if (this.HP <= 0)
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? Console.WriteLine("此怪物已死");
  • ? ?? ?? ?? ?? ? return;
  • ? ?? ?? ?? ?}
  • ? ?
  • ? ?? ?? ?? ?this.HP -= loss;
  • ? ?? ?? ?? ?if (this.HP <= 0)
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? Console.WriteLine("怪物" + this.Name + "被打死");
  • ? ?? ?? ?? ?}
  • ? ?? ?? ?? ?else
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? Console.WriteLine("怪物" + this.Name + "損失" + loss + "HP");
  • ? ?? ?? ?? ?}
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace IGameLiAdv
  • {
  • ? ?
  • /// <summary>
  • ? ?
  • /// 角色
  • ? ?
  • /// </summary>
  • ? ? internal sealed class Role
  • ? ? {
  • ? ?? ???
  • /// <summary>
  • ? ?? ???
  • /// 表示角色目前所持武器
  • ? ?? ???
  • /// </summary>
  • ? ?? ???public IAttackStrategy Weapon { get; set; }
  • ? ?
  • ? ?? ???
  • /// <summary>
  • ? ?? ???
  • /// 攻擊怪物
  • ? ?? ???
  • /// </summary>
  • ? ?? ???
  • /// <param name="monster">被攻擊的怪物</param>
  • ? ?? ???public void Attack(Monster monster)
  • ? ?? ???{
  • ? ?? ?? ?? ?this.Weapon.AttackTarget(monster);
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace IGameLiAdv
  • {
  • ? ? class Program
  • ? ? {
  • ? ?? ???static void Main(string[] args)
  • ? ?? ???{
  • ? ?? ?? ?? ?
  • //生成怪物
  • ? ?? ?? ?? ?Monster monster1 = new Monster("小怪A", 50);
  • ? ?? ?? ?? ?Monster monster2 = new Monster("小怪B", 50);
  • ? ?? ?? ?? ?Monster monster3 = new Monster("關(guān)主", 200);
  • ? ?? ?? ?? ?Monster monster4 = new Monster("最終Boss", 1000);
  • ? ?
  • ? ?? ?? ?? ?
  • //生成角色
  • ? ?? ?? ?? ?Role role = new Role();
  • ? ?
  • ? ?? ?? ?? ?
  • //木劍攻擊
  • ? ?? ?? ?? ?role.Weapon = new WoodSword();
  • ? ?? ?? ?? ?role.Attack(monster1);
  • ? ?
  • ? ?? ?? ?? ?
  • //鐵劍攻擊
  • ? ?? ?? ?? ?role.Weapon = new IronSword();
  • ? ?? ?? ?? ?role.Attack(monster2);
  • ? ?? ?? ?? ?role.Attack(monster3);
  • ? ?
  • ? ?? ?? ?? ?
  • //魔劍攻擊
  • ? ?? ?? ?? ?role.Weapon = new MagicSword();
  • ? ?? ?? ?? ?role.Attack(monster3);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?? ?? ?? ?role.Attack(monster4);
  • ? ?
  • ? ?? ?? ?? ?Console.ReadLine();
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼


    編譯運行以上代碼,得到的運行結(jié)果與上一版本代碼基本一致。

    1.4 小李的小結(jié)

    Peter顯然對改進后的代碼比較滿意,他讓小李對照兩份設(shè)計和代碼,進行一個小結(jié)。小李簡略思考了一下,并結(jié)合小于對一次設(shè)計指出的不足,說道:

    “我認為,改進后的代碼有如下優(yōu)點:

    第一,雖然類的數(shù)量增加了,但是每個類中方法的代碼都非常短,沒有了以前Attack方法那種很長的方法,也沒有了冗長的if…else,代碼結(jié)構(gòu)變得很清晰。

    第二,類的職責更明確了。在第一個設(shè)計中,Role不但負責攻擊,還負責給怪物減少HP和判斷怪物是否已死。這明顯不應(yīng)該是Role的職責,改進后的代碼將這兩個職責移入Monster內(nèi),使得職責明確,提高了類的內(nèi)聚性。

    第三,引入Strategy模式后,不但消除了重復(fù)性代碼,更重要的是,使得設(shè)計符合了OCP。如果以后要加一個新武器,只要新建一個類,實現(xiàn)IAttackStrategy接口,當角色需要裝備這個新武器時,客戶代碼只要實例化一個新武器類,并賦給Role的Weapon成員就可以了,已有的Role和Monster代碼都不用改動。這樣就實現(xiàn)了對擴展開發(fā),對修改關(guān)閉。”

    Peter和小于聽后都很滿意,認為小李總結(jié)的非常出色。

    IGame公司的討論會還在進行著,內(nèi)容是非常精彩,不過我們先聽到這里,因為,接下來,我們要對其中某些問題進行一點探討。別忘了,本文的主題可是依賴注入,這個主角還沒登場呢!讓主角等太久可不好。

    2 探究依賴注入

    2.1 故事的啟迪

    我們現(xiàn)在靜下心來,再回味一下剛才的故事。因為,這個故事里面隱藏著依賴注入的出現(xiàn)原因。我說過不只一次,想真正認清一個事物,不能只看“它是什么?什么樣子?”,而應(yīng)該先弄清楚“它是怎么來的?是什么樣的需求和背景促使了它的誕生?它被創(chuàng)造出來是做什么用的?”。

    回想上面的故事。剛開始,主要需求是一個打怪的功能。小李做了一個初步面向?qū)ο蟮脑O(shè)計:抽取領(lǐng)域場景中的實體(怪物、角色等),封裝成類,并為各個類賦予屬性與方法,最后通過類的交互完成打怪功能,這應(yīng)該算是面向?qū)ο笤O(shè)計的初級階段。

    在小李的設(shè)計基礎(chǔ)上,架構(gòu)師小于指出了幾點不足,如不符合OCP,職責劃分不明確等等,并根據(jù)情況引入策略模式。這是更高層次的面向?qū)ο笤O(shè)計。其實就核心來說,小于只做了一件事:利用多態(tài)性,隔離變化。它清楚認識到,這個打怪功能中,有些業(yè)務(wù)邏輯是不變的,如角色攻擊怪物,怪物減少HP,減到0怪物就會死;而變化的僅僅是不同的角色持有不同武器時,每次攻擊的效用不一樣。于是他的架構(gòu),本質(zhì)就是把變化的部分和不變的部分隔離開,使得變化部分發(fā)生變化時,不變部分不受影響。

    我們再仔細看看小于的設(shè)計圖,這樣設(shè)計后,有個基本的問題需要解決:現(xiàn)在Role不依賴具體武器,而僅僅依賴一個IAttackStrategy接口,接口是不能實例化的,雖然Role的Weapon成員類型定義為IAttackStrategy,但最終還是會被賦予一個實現(xiàn)了IAttackStrategy接口的具體武器,并且隨著程序進展,一個角色會裝備不同的武器,從而產(chǎn)生不同的效用。賦予武器的職責,在Demo中是放在了測試代碼里。

    這里,測試代碼實例化一個具體的武器,并賦給Role的Weapon成員的過程,就是依賴注入!這里要清楚,依賴注入其實是一個過程的稱謂!

    2.2 正式定義依賴注入

    下面,用稍微正式一點的語言,定義依賴注入產(chǎn)生的背景緣由和依賴注入的含義。在讀的過程中,讀者可以結(jié)合上面的例子進行理解。

    依賴注入產(chǎn)生的背景:

    隨著面向?qū)ο蠓治雠c設(shè)計的發(fā)展,一個良好的設(shè)計,核心原則之一就是將變化隔離,使得變化部分發(fā)生變化時,不變部分不受影響(這也是OCP的目的)。為了做到這一點,要利用面向?qū)ο笾械亩鄳B(tài)性,使用多態(tài)性后,客戶類不再直接依賴服務(wù)類,而是依賴于一個抽象的接口,這樣,客戶類就不能在內(nèi)部直接實例化具體的服務(wù)類。但是,客戶類在運作中又客觀需要具體的服務(wù)類提供服務(wù),因為接口是不能實例化去提供服務(wù)的。就產(chǎn)生了“客戶類不準實例化具體服務(wù)類”和“客戶類需要具體服務(wù)類”這樣一對矛盾。為了解決這個矛盾,開發(fā)人員提出了一種模式:客戶類(如上例中的Role)定義一個注入點(Public成員Weapon),用于服務(wù)類(實現(xiàn)IAttackStrategy的具體類,如WoodSword、IronSword和MagicSword,也包括以后加進來的所有實現(xiàn)IAttackStrategy的新類)的注入,而客戶類的客戶類(Program,即測試代碼)負責根據(jù)情況,實例化服務(wù)類,注入到客戶類中,從而解決了這個矛盾。

    依賴注入的正式定義:

    依賴注入(Dependency Injection),賣手機游戲是這樣一個過程:由于某客戶類只依賴于服務(wù)類的一個接口,而不依賴于具體服務(wù)類,所以客戶類只定義一個注入點。在程序運行過程中,客戶類不直接實例化具體服務(wù)類實例,而是客戶類的運行上下文環(huán)境或?qū)iT組件負責實例化服務(wù)類,然后將其注入到客戶類中,保證客戶類的正常運行。

    3 依賴注入那些事兒

    上面我們從需求背景的角度,講述了依賴注入的來源和定義。但是,如果依賴注入僅僅就只有這么點東西,那也沒有什么值得討論的了。但是,上面討論的僅僅是依賴注入的內(nèi)涵,其外延還是非常廣泛的,從依賴注入衍生出了很多相關(guān)的概念與技術(shù),下面我們討論一下依賴注入的“那些事兒”。

    3.1 依賴注入的類別

    依賴注入有很多種方法,上面看到的例子中,只是其中的一種,下面分別討論不同的依賴注入類型。

    3.1.1 Setter注入

    第一種依賴注入的方式,就是Setter注入,上面的例子中,將武器注入Role就是Setter注入。正式點說:

    Setter注入(Setter Injection)是指在客戶類中,設(shè)置一個服務(wù)類接口類型的數(shù)據(jù)成員,并設(shè)置一個Set方法作為注入點,這個Set方法接受一個具體的服務(wù)類實例為參數(shù),并將它賦給服務(wù)類接口類型的數(shù)據(jù)成員。

    圖3.1 Setter注入示意


    上圖展示了Setter注入的結(jié)構(gòu)示意圖,客戶類ClientClass設(shè)置IServiceClass類型成員_serviceImpl,并設(shè)置Set_ServiceImpl方法作為注入點。Context會負責實例化一個具體的ServiceClass,然后注入到ClientClass里。

    下面給出Setter注入的示例代碼。
    ?

  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace SetterInjection
  • {
  • ? ? internal interface IServiceClass
  • ? ? {
  • ? ?? ???String ServiceInfo();
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace SetterInjection
  • {
  • ? ? internal class ServiceClassA : IServiceClass
  • ? ? {
  • ? ?? ???public String ServiceInfo()
  • ? ?? ???{
  • ? ?? ?? ?? ?return "我是ServceClassA";
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace SetterInjection
  • {
  • ? ? internal class ServiceClassB : IServiceClass
  • ? ? {
  • ? ?? ???public String ServiceInfo()
  • ? ?? ???{
  • ? ?? ?? ?? ?return "我是ServceClassB";
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace SetterInjection
  • {
  • ? ? internal class ClientClass
  • ? ? {
  • ? ?? ???private IServiceClass _serviceImpl;
  • ? ?
  • ? ?? ???public void Set_ServiceImpl(IServiceClass serviceImpl)
  • ? ?? ???{
  • ? ?? ?? ?? ?this._serviceImpl = serviceImpl;
  • ? ?? ???}
  • ? ?
  • ? ?? ???public void ShowInfo()
  • ? ?? ???{
  • ? ?? ?? ?? ?Console.WriteLine(_serviceImpl.ServiceInfo());
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace SetterInjection
  • {
  • ? ? class Program
  • ? ? {
  • ? ?? ???static void Main(string[] args)
  • ? ?? ???{
  • ? ?? ?? ?? ?IServiceClass serviceA = new ServiceClassA();
  • ? ?? ?? ?? ?IServiceClass serviceB = new ServiceClassB();
  • ? ?? ?? ?? ?ClientClass client = new ClientClass();
  • ? ?
  • ? ?? ?? ?? ?client.Set_ServiceImpl(serviceA);
  • ? ?? ?? ?? ?client.ShowInfo();
  • ? ?? ?? ?? ?client.Set_ServiceImpl(serviceB);
  • ? ?? ?? ?? ?client.ShowInfo();
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼


    運行結(jié)果如下:
    ?

    圖3.2 Setter注入運行結(jié)果


    3.1.2 構(gòu)造注入

    另外一種依賴注入方式,是通過客戶類的構(gòu)造函數(shù),向客戶類注入服務(wù)類實例。

    構(gòu)造注入(Constructor Injection)是指在客戶類中,設(shè)置一個服務(wù)類接口類型的數(shù)據(jù)成員,并以構(gòu)造函數(shù)為注入點,這個構(gòu)造函數(shù)接受一個具體的服務(wù)類實例為參數(shù),并將它賦給服務(wù)類接口類型的數(shù)據(jù)成員。
    ?

    圖3.3 構(gòu)造注入示意


    圖3.3是構(gòu)造注入的示意圖,可以看出,與Setter注入很類似,只是注入點由Setter方法變成了構(gòu)造方法。這里要注意,由于構(gòu)造注入只能在實例化客戶類時注入一次,所以一點注入,程序運行期間是沒法改變一個客戶類對象內(nèi)的服務(wù)類實例的。

    由于構(gòu)造注入和Setter注入的IServiceClass,ServiceClassA和ServiceClassB是一樣的,所以這里給出另外ClientClass類的示例代碼。
    ?

  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace ConstructorInjection
  • {
  • ? ? internal class ClientClass
  • ? ? {
  • ? ?? ???private IServiceClass _serviceImpl;
  • ? ?
  • ? ?? ???public ClientClass(IServiceClass serviceImpl)
  • ? ?? ???{
  • ? ?? ?? ?? ?this._serviceImpl = serviceImpl;
  • ? ?? ???}
  • ? ?
  • ? ?? ???public void ShowInfo()
  • ? ?? ???{
  • ? ?? ?? ?? ?Console.WriteLine(_serviceImpl.ServiceInfo());
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼


    可以看到,唯一的變化就是構(gòu)造函數(shù)取代了Set_ServiceImpl方法,成為了注入點。

    3.1.3 依賴獲取

    上面提到的注入方式,都是客戶類被動接受所依賴的服務(wù)類,這也符合“注入”這個詞。不過還有一種方法,可以和依賴注入達到相同的目的,就是依賴獲取。

    依賴獲取(Dependency Locate)是指在系統(tǒng)中提供一個獲取點,客戶類仍然依賴服務(wù)類的接口。當客戶類需要服務(wù)類時,從獲取點主動取得指定的服務(wù)類,具體的服務(wù)類類型由獲取點的配置決定。

    可以看到,這種方法變被動為主動,使得客戶類在需要時主動獲取服務(wù)類,而將多態(tài)性的實現(xiàn)封裝到獲取點里面。獲取點可以有很多種實現(xiàn),也許最容易想到的就是建立一個Simple Factory作為獲取點,客戶類傳入一個指定字符串,以獲取相應(yīng)服務(wù)類實例。如果所依賴的服務(wù)類是一系列類,那么依賴獲取一般利用Abstract Factory模式構(gòu)建獲取點,然后,將服務(wù)類多態(tài)性轉(zhuǎn)移到工廠的多態(tài)性上,而工廠的類型依賴一個外部配置,如XML文件。

    不過,不論使用Simple Factory還是Abstract Factory,都避免不了判斷服務(wù)類類型或工廠類型,這樣系統(tǒng)中總要有一個地方存在不符合OCP的if…else或switch…case結(jié)構(gòu),這種缺陷是Simple Factory和Abstract Factory以及依賴獲取本身無法消除的,而在某些支持反射的語言中(如C#),通過將反射機制的引入徹底解決了這個問題(后面討論)。

    下面給一個具體的例子,現(xiàn)在我們假設(shè)有個程序,既可以使用Windows風(fēng)格外觀,又可以使用Mac風(fēng)格外觀,而內(nèi)部業(yè)務(wù)是一樣的。
    ?

    圖3.4 依賴獲取示意


    上圖乍看有點復(fù)雜,不過如果讀者熟悉Abstract Factory模式,應(yīng)該能很容易看懂,這就是Abstract Factory在實際中的一個應(yīng)用。這里的Factory Container作為獲取點,是一個靜態(tài)類,它的“Type構(gòu)造函數(shù)”依據(jù)外部的XML配置文件,決定實例化哪個工廠。下面還是來看示例代碼。由于不同組件的代碼是相似的,這里只給出Button組件的示例代碼,完整代碼請參考文末附上的完整源程序。
    ?

  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace DependencyLocate
  • {
  • ? ? internal interface IButton
  • ? ? {
  • ? ?? ???String ShowInfo();
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace DependencyLocate
  • {
  • ? ? internal sealed class WindowsButton : IButton
  • ? ? {
  • ? ?? ???public String Description { get; private set; }
  • ? ?
  • ? ?? ???public WindowsButton()
  • ? ?? ???{
  • ? ?? ?? ?? ?this.Description = "Windows風(fēng)格按鈕";
  • ? ?? ???}
  • ? ?
  • ? ?? ???public String ShowInfo()
  • ? ?? ???{
  • ? ?? ?? ?? ?return this.Description;
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace DependencyLocate
  • {
  • ? ? internal sealed class MacButton : IButton
  • ? ? {
  • ? ?? ???public String Description { get; private set; }
  • ? ?
  • ? ?? ???public MacButton()
  • ? ?? ???{
  • ? ?? ?? ?? ?this.Description = " Mac風(fēng)格按鈕";
  • ? ?? ???}
  • ? ?
  • ? ?? ???public String ShowInfo()
  • ? ?? ???{
  • ? ?? ?? ?? ?return this.Description;
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace DependencyLocate
  • {
  • ? ? internal interface IFactory
  • ? ? {
  • ? ?? ???IWindow MakeWindow();
  • ? ?
  • ? ?? ???IButton MakeButton();
  • ? ?
  • ? ?? ???ITextBox MakeTextBox();
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace DependencyLocate
  • {
  • ? ? internal sealed class WindowsFactory : IFactory
  • ? ? {
  • ? ?? ???public IWindow MakeWindow()
  • ? ?? ???{
  • ? ?? ?? ?? ?return new WindowsWindow();
  • ? ?? ???}
  • ? ?
  • ? ?? ???public IButton MakeButton()
  • ? ?? ???{
  • ? ?? ?? ?? ?return new WindowsButton();
  • ? ?? ???}
  • ? ?
  • ? ?? ???public ITextBox MakeTextBox()
  • ? ?? ???{
  • ? ?? ?? ?? ?return new WindowsTextBox();
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace DependencyLocate
  • {
  • ? ? internal sealed class MacFactory : IFactory
  • ? ? {
  • ? ?? ???public IWindow MakeWindow()
  • ? ?? ???{
  • ? ?? ?? ?? ?return new MacWindow();
  • ? ?? ???}
  • ? ?
  • ? ?? ???public IButton MakeButton()
  • ? ?? ???{
  • ? ?? ?? ?? ?return new MacButton();
  • ? ?? ???}
  • ? ?
  • ? ?? ???public ITextBox MakeTextBox()
  • ? ?? ???{
  • ? ?? ?? ?? ?return new MacTextBox();
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • using System.Xml;
  • ? ?
  • namespace DependencyLocate
  • {
  • ? ? internal static class FactoryContainer
  • ? ? {
  • ? ?? ???public static IFactory factory { get; private set; }
  • ? ?
  • ? ?? ???static FactoryContainer()
  • ? ?? ???{
  • ? ?? ?? ?? ?XmlDocument xmlDoc = new XmlDocument();
  • ? ?? ?? ?? ?xmlDoc.Load("http://www.cnblogs.com/Config.xml");
  • ? ?? ?? ?? ?XmlNode xmlNode = xmlDoc.ChildNodes[1].ChildNodes[0].ChildNodes[0];
  • ? ?
  • ? ?? ?? ?? ?if ("Windows" == xmlNode.Value)
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? factory = new WindowsFactory();
  • ? ?? ?? ?? ?}
  • ? ?? ?? ?? ?else if ("Mac" == xmlNode.Value)
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? factory = new MacFactory();
  • ? ?? ?? ?? ?}
  • ? ?? ?? ?? ?else
  • ? ?? ?? ?? ?{
  • ? ?? ?? ?? ?? ? throw new Exception("Factory Init Error");
  • ? ?? ?? ?? ?}
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼
  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • ? ?
  • namespace DependencyLocate
  • {
  • ? ? class Program
  • ? ? {
  • ? ?? ???static void Main(string[] args)
  • ? ?? ???{
  • ? ?? ?? ?? ?IFactory factory = FactoryContainer.factory;
  • ? ?? ?? ?? ?IWindow window = factory.MakeWindow();
  • ? ?? ?? ?? ?Console.WriteLine("創(chuàng)建 " + window.ShowInfo());
  • ? ?? ?? ?? ?IButton button = factory.MakeButton();
  • ? ?? ?? ?? ?Console.WriteLine("創(chuàng)建 " + button.ShowInfo());
  • ? ?? ?? ?? ?ITextBox textBox = factory.MakeTextBox();
  • ? ?? ?? ?? ?Console.WriteLine("創(chuàng)建 " + textBox.ShowInfo());
  • ? ?
  • ? ?? ?? ?? ?Console.ReadLine();
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼


    這里我們用XML作為配置文件。配置文件Config.xml如下:
    ?

  • <?xml version="1.0" encoding="utf-8" ?>
  • <config>
  • ? ? <factory>Mac</factory>
  • </config>
  • 復(fù)制代碼


    可以看到,這里我們將配置設(shè)置為Mac風(fēng)格,編譯運行上述代碼,運行結(jié)果如下:
    ?

    圖3.5 配置Mac風(fēng)格后的運行結(jié)果


    現(xiàn)在,我們不動程序,僅僅將配置文件中的“Mac”改為Windows,運行后結(jié)果如下:
    ?

    圖3.6 配置為Windows風(fēng)格后的運行結(jié)果


    從運行結(jié)果看出,我們僅僅通過修改配置文件,就改變了整個程序的行為(我們甚至沒有重新編譯程序),這就是多態(tài)性的威力,也是依賴注入效果。


    3.2 反射與依賴注入

    回想上面Dependency Locate的例子,我們雖然使用了多態(tài)性和Abstract Factory,但對OCP貫徹的不夠徹底。在理解這點前,朋友們一定要注意潛在擴展在哪里,潛在會出現(xiàn)擴展的地方是“新的組件系列”而不是“組件種類”,也就是說,這里我們假設(shè)組件就三種,不會增加新的組件,但可能出現(xiàn)新的外觀系列,如需要加一套Ubuntu風(fēng)格的組件,我們可以新增UbuntuWindow、UbuntuButton、UbuntuTextBox和UbuntuFactory,并分別實現(xiàn)相應(yīng)接口,這是符合OCP的,因為這是擴展。但我們除了修改配置文件,還要無可避免的修改FactoryContainer,需要加一個分支條件,這個地方破壞了OCP。依賴注入本身是沒有能力解決這個問題的,但如果語言支持反射機制(Reflection),則這個問題就迎刃而解。

    我們想想,現(xiàn)在的難點是出在這里:對象最終還是要通過“new”來實例化,而“new”只能實例化當前已有的類,如果未來有新類添加進來,必須修改代碼。如果,我們能有一種方法,不是通過“new”,而是通過類的名字來實例化對象,那么我們只要將類的名字作為配置項,就可以實現(xiàn)在不修改代碼的情況下,加載未來才出現(xiàn)的類。所以,反射給了語言“預(yù)見未來”的能力,使得多態(tài)性和依賴注入的威力大增。

    下面是引入反射機制后,對上面例子的改進:
    ?

    圖3.7 引入反射機制的Dependency Locate


    可以看出,引入反射機制后,結(jié)構(gòu)簡單了很多,一個反射工廠代替了以前的一堆工廠,Factory Container也不需要了。而且以后有新組件系列加入時,反射工廠是不用改變的,只需改變配置文件就可以完成。下面給出反射工廠和配置文件的代碼。
    ?

  • using System;
  • using System.Collections.Generic;
  • using System.Linq;
  • using System.Text;
  • using System.Reflection;
  • using System.Xml;
  • ? ?
  • namespace DependencyLocate
  • {
  • ? ? internal static class ReflectionFactory
  • ? ? {
  • ? ?? ???private static String _windowType;
  • ? ?? ???private static String _buttonType;
  • ? ?? ???private static String _textBoxType;
  • ? ?
  • ? ?? ???static ReflectionFactory()
  • ? ?? ???{
  • ? ?? ?? ?? ?XmlDocument xmlDoc = new XmlDocument();
  • ? ?? ?? ?? ?xmlDoc.Load("http://www.cnblogs.com/Config.xml");
  • ? ?? ?? ?? ?XmlNode xmlNode = xmlDoc.ChildNodes[1].ChildNodes[0];
  • ? ?
  • ? ?? ?? ?? ?_windowType = xmlNode.ChildNodes[0].Value;
  • ? ?? ?? ?? ?_buttonType = xmlNode.ChildNodes[1].Value;
  • ? ?? ?? ?? ?_textBoxType = xmlNode.ChildNodes[2].Value;
  • ? ?? ???}
  • ? ?
  • ? ?? ???public static IWindow MakeWindow()
  • ? ?? ???{
  • ? ?? ?? ?? ?return Assembly.Load("DependencyLocate").CreateInstance("DependencyLocate." + _windowType) as IWindow;
  • ? ?? ???}
  • ? ?
  • ? ?? ???public static IButton MakeButton()
  • ? ?? ???{
  • ? ?? ?? ?? ?return Assembly.Load("DependencyLocate").CreateInstance("DependencyLocate." + _buttonType) as IButton;
  • ? ?? ???}
  • ? ?
  • ? ?? ???public static ITextBox MakeTextBox()
  • ? ?? ???{
  • ? ?? ?? ?? ?return Assembly.Load("DependencyLocate").CreateInstance("DependencyLocate." + _textBoxType) as ITextBox;
  • ? ?? ???}
  • ? ? }
  • }
  • 復(fù)制代碼


    配置文件如下:
    ?

  • <?xml version="1.0" encoding="utf-8" ?>
  • <config>
  • ? ? <window>MacWindow</window>
  • ? ? <button>MacButton</button>
  • ? ? <textBox>MacTextBox</textBox>
  • </config>
  • 復(fù)制代碼


    反射不僅可以與Dependency Locate結(jié)合,也可以與Setter Injection與Construtor Injection結(jié)合。反射機制的引入,降低了依賴注入結(jié)構(gòu)的復(fù)雜度,使得依賴注入徹底符合OCP,并為通用依賴注入框架(如Spring.NET中的IoC部分、Unity等)的設(shè)計提供了可能性。

    3.3 多態(tài)的活性與依賴注入

    3.3.1 多態(tài)性的活性

    這一節(jié)我們討論多態(tài)的活性及其與依賴注入類型選擇間密切的關(guān)系。

    首先說明,“多態(tài)的活性”這個術(shù)語是我個人定義的,因為我沒有找到既有的概念名詞可以表達我的意思,所以就自己造了一個詞。這里,某多態(tài)的活性是指被此多態(tài)隔離的變化所發(fā)生變化的頻繁程度,頻繁程度越高,則活性越強,反之亦然。

    上文說過,多態(tài)性可以隔離變化,但是,不同的變化,發(fā)生的頻率是不一樣的,這就使得多態(tài)的活性有所差別,這種差別影響了依賴注入的類型選擇。

    舉例來說,本文最開始提到的武器多態(tài)性,其活性非常高,因為在那個程序中,Role在一次運行中可能更換多次武器。而現(xiàn)在我們假設(shè)Role也實現(xiàn)了多態(tài)性,這是很可能的,因為在游戲中,不同類型的角色(如暗夜精

    靈、牛頭人、矮人等)很多屬性和業(yè)務(wù)是想通的,所以很可能通過一個IRole或AbstractRole抽象類實現(xiàn)多態(tài)性,不過,Role在實例化后(一般在用戶登錄成功后),是不會變化的,很少有游戲允許同一個玩家在運行中變換Role類型,所以Role應(yīng)該是一但實例化,就不會變化,但如果再實例化一個(如另一個玩家登錄),則可能就變化了。最后,還有一種多態(tài)性是活性非常低的,如我們熟悉的數(shù)據(jù)訪問層多態(tài)性,即使我們實現(xiàn)了SQL Server、Oracle和Access等多種數(shù)據(jù)庫的訪問層,并實現(xiàn)了依賴注入,但幾乎遇不到程序運行著就改數(shù)據(jù)庫或短期內(nèi)數(shù)據(jù)庫頻繁變動的情況。

    以上不同的多態(tài)性,不但特征不同,其目的一般也不同,總結(jié)如下:

    高活多態(tài)性——指在客戶類實例運行期間,服務(wù)類可能會改變的多態(tài)性。

    中活多態(tài)性——指在客戶類實例化后,服務(wù)類不會改變,但同一時間內(nèi)存在的不同實例可能擁有不同類型的服務(wù)類。

    低活多態(tài)性——指在客戶類實例化后,服務(wù)類不會改變,且同一時間內(nèi)所有客戶類都擁有相同類型的服務(wù)類。

    以上三種多態(tài)性,比較好的例子就是上文提到的武器多態(tài)性(高活)、角色多態(tài)性(中活)和數(shù)據(jù)訪問層多態(tài)性(低活)。另外,我們說一種多態(tài)性是空間穩(wěn)定的,如果同一客戶類在同一時間內(nèi)的所有實例都依賴相同類型的服務(wù)類,反之則叫做空間不穩(wěn)定多態(tài)性。我們說一種多態(tài)性是時間穩(wěn)定的,如果一個客戶類在實例化后,所以來的服務(wù)類不能再次更改,反之則叫做時間不穩(wěn)定多態(tài)性。顯然,高活多態(tài)性時間和空間均不穩(wěn)定;中活多態(tài)性是時間穩(wěn)定的,但空間不穩(wěn)定;低活多態(tài)性時間空間均穩(wěn)定。

    3.3.2 不同活性多態(tài)的依賴注入選擇

    一般來說,高活多態(tài)性適合使用Setter注入。因為Setter注入最靈活,也是唯一允許在同一客戶類實例運行期間更改服務(wù)類的注入方式。并且這種注入一般由上下文環(huán)境通過Setter的參數(shù)指定服務(wù)類類型,方便靈活,適合頻繁變化的高活多態(tài)性。

    對于中活多態(tài)性,則適合使用Constructor注入。因為Constructor注入也是由上下文環(huán)境通過Construtor的參數(shù)指定服務(wù)類類型,但一點客戶類實例化后,就不能進行再次注入,保證了其時間穩(wěn)定性。

    而對于低活多態(tài)性,則適合使用Dependency Locate并配合文件配置進行依賴注入,或Setter、Constructor配合配置文件注入,因為依賴源來自文件,如果要更改服務(wù)類,則需要更改配置文件,一則確保了低活多態(tài)性的時間和空間穩(wěn)定性,二是更改配置文件的方式方便于大規(guī)模服務(wù)類替換。(因為低活多態(tài)性一旦改變行為,往往規(guī)模很大,如替換整個數(shù)據(jù)訪問層,如果使用Setter和Construtor傳參,程序中需要改變的地方不計其數(shù))

    本質(zhì)上,這種選擇是因為不同的依賴注入類型有著不同的穩(wěn)定性,大家可以細細體會“活性”、“穩(wěn)定性”和“依賴注入類型”之間密切的關(guān)系。

    4 IoC Container

    4.1 IoC Container出現(xiàn)的必然性

    上面討論了諸多依賴注入的話題。說道依賴注入,就不能不說IoC Container(IoC容器),那么到底什么是IoC容器?我們還是先來看看它的出現(xiàn)背景。

    我們知道,軟件開發(fā)領(lǐng)域有句著名的論斷:不要重復(fù)發(fā)明輪子!因為軟件開發(fā)講求復(fù)用,所以,對于應(yīng)用頻繁的需求,總是有人設(shè)計各種通用框架和類庫以減輕人們的開發(fā)負擔。例如,數(shù)據(jù)持久化是非常頻繁的需求,于是各種ORM框架應(yīng)運而生;再如,對MVC的需求催生了Struts等一批用來實現(xiàn)MVC的框架。

    隨著面向?qū)ο蠓治雠c設(shè)計的發(fā)展和成熟,OOA&D被越來越廣泛應(yīng)用于各種項目中,然而,我們知道,用OO就不可能不用多態(tài)性,用多態(tài)性就不可能不用依賴注入,所以,依賴注入變成了非常頻繁的需求,而如果全部手工完成,不但負擔太重,而且還容易出錯。再加上反射機制的發(fā)明,于是,自然有人開始設(shè)計開發(fā)各種用于依賴注入的專用框架。這些專門用于實現(xiàn)依賴注入功能的組件或框架,就是IoC Container。

    從這點看,IoC Container的出現(xiàn)有其歷史必然性。目前,最著名的IoC也許就是Java平臺上的Spring框架的IoC組件,而.NET平臺上也有Spring.NET和Unity等。

    4.2 IoC Container的分類

    前面曾經(jīng)討論了三種依賴注入方式,但是,想通過方式對IoC Container進行分類很困難,因為現(xiàn)在IoC Container都設(shè)計很完善,幾乎支持所有依賴注入方式。不過,根據(jù)不同框架的特性和慣用法,還是可以講IoC Container分為兩個大類。

    4.2.1 重量級IoC Container

    所謂重量級IoC Container,是指一般用外部配置文件(一般是XML)作為依賴源,并托管整個系統(tǒng)各個類的實例化的IoC Container。這種IoC Container,一般是承接了整個系統(tǒng)幾乎所有多態(tài)性的依賴注入工作,并承接了所有服務(wù)類的實例化工作,而且這些實例化依賴于一個外部配置文件,這種IoC Container,很像通過一個文件,定義整個系統(tǒng)多態(tài)結(jié)構(gòu),視野宏大,想要很好駕馭這種IoC Container,需要一定的架構(gòu)設(shè)計能力和豐富的實踐經(jīng)驗。

    Spring和Spring.NET是重量級IoC Container的例子。一般來說,這種IoC Container穩(wěn)定性有余而活性不足,適合進行低活多態(tài)性的依賴注入。

    4.2.2 輕量級IoC Container

    還有一種IoC Container,一般不依賴外部配置文件,而主要使用傳參的Setter或Construtor注入,這種IoC Container叫做輕量級IoC Container。這種框架很靈活,使用方便,但往往不穩(wěn)定,而且依賴點都是程序中的字符串參數(shù),所以,不適合需要大規(guī)模替換和相對穩(wěn)定的低活多態(tài)性,而對于高活多態(tài)性,有很好的效果。

    Unity是一個典型的輕量級IoC Container。

    4.3 .NET平臺上典型IoC Container推介

    4.3.1 Spring.NET
    ?


    Spring.NET是Java平臺上Spring對.NET平臺的移植,使用方法和Spring很像,并且功能強大,是.NET平臺上大中型開發(fā)IoC Container的首選之一。除了DI外,Spring.NET也包括AOP等諸多功能。

    4.3.2 Unity
    ?


    對于小型項目和講求敏捷的團隊,Spring.NET可能有點太重量級,那么可以選擇輕量級的Unity。Unity是微軟patterns & practices團隊推出的輕量級框架,非常好用,目前最新版本是1.2。

    總結(jié)

    以上是生活随笔為你收集整理的依赖注入 这样的坑游戏编程要谨慎的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

    欧美久久久久久久久中文字幕 | 天天草天天 | 日日操夜夜操狠狠操 | 天天搞天天 | 日韩在线中文字幕视频 | 天天操天天干天天摸 | 美国人与动物xxxx | 狠狠色伊人亚洲综合网站野外 | 一区二区三区在线观看 | 国产黄 | 在线看片视频 | 久久精品首页 | 婷婷电影在线观看 | 超碰在线个人 | 一本一本久久aa综合精品 | 久久久亚洲网站 | 日本中文字幕在线播放 | 在线观看国产高清视频 | 在线精品视频免费播放 | 久久在线观看 | 久久久久免费精品国产 | 911精品视频 | 国产污视频在线观看 | 99精品在线看 | 久久经典国产视频 | 九色视频网址 | 国产日韩在线一区 | 久久avav | 国产理论在线 | 国产精品 日韩 欧美 | 日韩电影一区二区三区 | 成人国产网站 | 人人插人人射 | 成人资源网 | 久操伊人 | 欧美精品久久久久a | 亚洲免费av在线播放 | 亚色视频在线观看 | 在线中文字幕播放 | 精品国产诱惑 | 狠狠的干| 日韩欧美视频在线免费观看 | 国产一级视频在线观看 | 婷婷丁香九月 | japanese黑人亚洲人4k | 日韩91av | 亚洲涩涩涩 | 久久99久久99精品免观看软件 | 成人av观看 | 亚洲日本va午夜在线电影 | 日精品在线观看 | 免费网站黄 | 国内99视频 | 精品国产亚洲日本 | 91丨九色丨国产丨porny精品 | 欧美日韩国产mv | 91精品区 | 五月婷婷综合激情 | 在线观看成人网 | 四虎成人精品永久免费av | 综合伊人久久 | 欧洲精品视频一区二区 | 欧美在线视频精品 | 91精品黄色 | 二区三区中文字幕 | 欧美午夜精品久久久久久孕妇 | 成人在线观看你懂的 | 日韩系列在线观看 | 在线成人av | japanesexxxhd奶水| 亚洲国产网址 | 国产高清黄色 | 色姑娘综合| 中文字幕av一区二区三区四区 | 亚洲高清色综合 | 亚洲毛片视频 | 超碰国产人人 | 亚洲一级国产 | 国偷自产视频一区二区久 | 天堂av免费看 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 黄色99视频 | 国产高清视频在线 | 日韩大片在线免费观看 | 精品国偷自产在线 | 亚洲三级网站 | 亚洲视频免费在线观看 | 亚洲国产手机在线 | 午夜免费福利片 | 亚洲最大激情中文字幕 | 成人一级片在线观看 | 久久精品亚洲一区二区三区观看模式 | 操操操干干干 | 色爱区综合激月婷婷 | 毛片精品免费在线观看 | 久草视频在线播放 | 日韩午夜在线观看 | 久久久久久久久影院 | 久久99热精品这里久久精品 | 丁香影院在线 | 精品国产一区二区三区蜜臀 | 狠狠综合久久av | 国产伦理一区二区三区 | 日韩在线观看中文 | 国产成人av在线影院 | 亚洲天天在线 | 91在线影视 | 激情综合国产 | 日韩欧美高清视频在线观看 | 天天看天天操 | 91成人在线观看喷潮 | 国产精品久久久久久久免费大片 | 一区二区精品在线视频 | 一区二区在线不卡 | 亚洲一二三久久 | 欧美日韩大片在线观看 | 国产精品免费成人 | ,午夜性刺激免费看视频 | a v在线视频 | 成人精品国产免费网站 | 九九视频网 | 天天摸日日摸人人看 | 欧美日韩一区二区在线观看 | 日韩精品在线观看av | 美女久久网站 | 激情婷婷网 | 日韩视频在线不卡 | 99在线精品视频 | 国产特级毛片aaaaaa高清 | 成人小视频免费在线观看 | 久久综合九色综合久久久精品综合 | 美女网站视频久久 | 亚洲自拍偷拍色图 | 国产又粗又猛又色 | www.超碰| 日本在线h | 日韩av片免费在线观看 | 亚洲 欧美变态 另类 综合 | 精品国产精品国产偷麻豆 | 91大神精品视频在线观看 | 国产精品视频永久免费播放 | 欧美一级在线看 | 色多多污污| 成人一区在线观看 | 手机在线永久免费观看av片 | 91香蕉视频黄 | 成年人在线免费看视频 | 十八岁以下禁止观看的1000个网站 | 黄网在线免费观看 | 日韩电影黄色 | www在线免费观看 | 国产成人久久精品一区二区三区 | 日日爽天天爽 | 丝袜美腿亚洲综合 | 午夜精品成人一区二区三区 | 天天干,天天射,天天操,天天摸 | 久久午夜色播影院免费高清 | 国产爽视频 | 九色最新网址 | av在线影视 | 亚洲日日夜夜 | 国产丝袜网站 | 99精品久久99久久久久 | 中文字幕中文字幕在线中文字幕三区 | 97在线观看 | 精品国产99国产精品 | 在线国产专区 | 亚洲黄色区 | 国产免费av一区二区三区 | 欧美色综合天天久久综合精品 | 日韩aⅴ视频 | 丁香六月婷婷综合 | 国产精品麻豆一区二区三区 | 国产一级电影免费观看 | 国产一区二区在线播放 | 亚洲黄色一级大片 | 国产精品一区二区三区久久久 | 亚州性色| 中文字幕电影一区 | 精品成人久久 | 国产精品成人久久久久久久 | 国产精品美女在线观看 | 久久精品视频播放 | 国产成人精品一区二区三区福利 | 久久久久久高潮国产精品视 | 婷婷色五| 韩日精品中文字幕 | 亚洲永久精品国产 | 国产亚洲精品久久久久久网站 | 亚洲精品国产精品久久99 | 日产av在线播放 | 97视频免费播放 | 美女久久久久 | 超碰97免费 | 国产成人三级一区二区在线观看一 | 摸bbb搡bbb搡bbbb| 国产成人一区二区三区在线观看 | 精品在线播放 | 国产婷婷色 | 天天爱综合 | 国产一区二区三区午夜 | 97免费中文视频在线观看 | 久久久久久久久久久综合 | 国产视频亚洲精品 | 91国内在线 | 九九爱免费视频在线观看 | 国产免费一区二区三区最新6 | 国产精品免费成人 | 日韩精品无 | 中文字幕在线观看第一区 | 91在线麻豆 | 久草久草视频 | 中文字幕电影一区 | 999久久久欧美日韩黑人 | 黄色网大全 | av免费看在线 | 一区二区 精品 | 99在线看| 日韩免费在线观看 | 国产精品女主播一区二区三区 | 国产精品99久久99久久久二8 | 日韩午夜av电影 | 一级黄毛片 | 日韩欧美视频一区二区 | 亚洲精品99久久久久久 | 91禁看片| 亚洲涩涩一区 | 国产麻豆成人传媒免费观看 | av福利第一导航 | 看黄色91| 91.精品高清在线观看 | 日本特黄一级 | 亚洲国产大片 | 国产精品成人av电影 | 四虎成人精品在永久免费 | 91成人在线视频 | 热久久免费视频 | 国产精品情侣视频 | 日韩在线观看精品 | 91视频免费看 | 日韩二区在线观看 | 国内精品久久久久久中文字幕 | 麻豆成人精品 | 中文字幕视频在线播放 | 久久久免费国产 | 91最新视频 | 在线观看 国产 | 日本性动态图 | 国产精品a久久久久 | 亚洲 欧美 另类人妖 | 久久艹免费 | 97在线精品视频 | 中文字幕 国产专区 | 亚洲精品无 | 欧美日韩国产网站 | 91久久黄色 | 色综合久久久久网 | 成人黄色大片在线观看 | 亚洲成人一二三 | 最新中文字幕在线播放 | 久久首页 | 国产91勾搭技师精品 | 国产麻豆成人传媒免费观看 | 欧美久久久久久久久中文字幕 | 亚洲97在线| 麻豆免费视频 | 激情欧美日韩一区二区 | 欧美aaa视频 | 午夜少妇一区二区三区 | 国产精品九九九九九九 | 亚洲欧洲在线视频 | 国产黄色大片 | 国产一级一片免费播放放a 一区二区三区国产欧美 | 色视频成人在线观看免 | 色资源网免费观看视频 | 国产精品黄色在线观看 | 国产一区二区三精品久久久无广告 | 91在线看黄 | 亚洲综合网 | 超碰av在线播放 | 久久精品看| 久久桃花网 | 色视频在线免费观看 | 亚洲午夜久久久久 | 在线观看亚洲精品视频 | 日韩黄色免费电影 | 91精品在线视频观看 | 91久久精| 不卡的av在线 | 少妇bbw搡bbbb搡bbb | 中日韩男男gay无套 日韩精品一区二区三区高清免费 | 人人爱夜夜操 | av久久久 | 91av大全 | 在线免费观看国产精品 | 欧美另类亚洲 | 高清一区二区三区av | 亚洲国产日韩一区 | 手机在线看永久av片免费 | 黄色日视频 | 免费在线观看视频一区 | 少妇bbw搡bbbb搡bbbb | 97超级碰碰碰碰久久久久 | 欧美91成人网 | 久久综合给合久久狠狠色 | 日韩三级视频在线看 | 91九色porn在线资源 | 婷婷六月网 | 国产在线一卡 | 久久99精品波多结衣一区 | 亚洲成人精品在线 | 有码中文在线 | 色综合久久久久综合体 | 青青河边草免费直播 | 在线精品视频在线观看高清 | 成年人黄色大全 | 91亚洲国产 | 免费视频久久 | 亚洲一区二区麻豆 | 又黄又刺激 | 99久久网站 | av7777777| 99精品欧美一区二区三区 | 香蕉视频18| 午夜精品麻豆 | 91精品国产电影 | 日本中文在线播放 | 中文字幕999 | 日韩三级免费观看 | 国产一区二区电影在线观看 | 91精品免费在线观看 | 欧美日韩免费一区二区三区 | 在线免费观看av网站 | 国产 一区二区三区 在线 | 五月天综合在线 | 亚洲视频999 | 成年人电影免费看 | 成人aⅴ视频 | 国产成人精品一区在线 | 欧美一区二区日韩一区二区 | 丁香一区二区 | av不卡免费在线观看 | 超碰公开在线观看 | 国产精品嫩草影院9 | 精品国产诱惑 | 国产精品成人久久 | 日韩免费看视频 | 91黄色视屏 | 国产日韩精品欧美 | 啪啪凸凸 | 国产一区二区精品久久 | 日本特黄特色aaa大片免费 | 亚洲综合爱 | 伊人久久在线观看 | 不卡av免费在线观看 | 91在线91拍拍在线91 | 色婷婷午夜 | 在线看片日韩 | 久久电影网站中文字幕 | 中文字幕在线播放视频 | 日本精品一区二区在线观看 | 国产一区在线视频 | 91看国产| 最近最新最好看中文视频 | 亚洲日本成人网 | 欧美日一级片 | 久草在线免费看视频 | 国产精品久久久久影视 | 亚洲男人天堂a | 色婷婷狠狠操 | avwww在线观看 | 91香蕉视频黄 | 97综合视频 | 欧美激情视频一二三区 | 色综合久久久久综合体桃花网 | 欧美精品一区二区免费 | 精品久久亚洲 | 夜夜干天天操 | 97超碰成人 | 日韩电影一区二区在线 | 国内精品久久久久久久影视麻豆 | 少妇bbb好爽| 日韩成人在线免费观看 | 欧美一级在线观看视频 | 久久玖 | 亚洲一区精品人人爽人人躁 | 天天人人综合 | 国产在线视频导航 | 国产精品福利在线 | 久久av免费观看 | 免费色视频在线 | 欧美一级激情 | 国产成人福利在线 | 18国产精品福利片久久婷 | 国产精品12 | 国产精品自在线 | 亚洲黄色片在线 | 91麻豆精品国产91久久久无限制版 | 天天干天天射天天操 | 欧洲精品视频一区二区 | 五月天亚洲综合 | 亚洲国产精品500在线观看 | 91av视频在线播放 | av免费福利 | 中文字幕视频在线播放 | 精品国产乱码一区二区三区在线 | 国产九色视频在线观看 | 久久国产午夜精品理论片最新版本 | 久久久久久久久久久久国产精品 | 国产精品国产三级国产不产一地 | 久久不射影院 | 免费在线观看91 | 伊人宗合| 91成人久久| 成人免费在线观看av | 97视频久久久 | 少妇视频一区 | 天天干天天天天 | 九草在线视频 | 国产美女无遮挡永久免费 | 综合色综合 | 日韩av免费一区二区 | 成人在线观看资源 | 国产精品久久久久久久久久东京 | 久久综合亚洲鲁鲁五月久久 | 精品在线二区 | 日韩中文字幕a | 国产日韩欧美在线一区 | 国产精品久久久久久久婷婷 | 91自拍视频在线观看 | 天天干天天摸天天操 | www.亚洲激情.com | 欧美日韩3p | 亚洲精品视频网站在线观看 | 久草久| 99久久99久久精品国产片 | 欧美性生活免费看 | 中文字幕在线免费观看 | av观看免费在线 | 99精品国产在热久久下载 | 亚州精品国产 | 天堂av在线网 | 久久dvd | 国产美女网站在线观看 | 久久精品国产一区 | 国产大尺度视频 | 久久亚洲成人网 | 三级大片网站 | 国产区 在线 | 天天爽综合网 | 国产又黄又爽又猛视频日本 | 天天干夜夜爱 | 亚洲无在线 | 精品一二三四五区 | 色网站免费在线观看 | 在线观看视频三级 | 日韩免费看 | 91探花在线视频 | 色婷婷国产精品一区在线观看 | 日韩免费在线看 | 久久久人人人 | 久久久久久久久精 | 超碰在线9| 在线91av | 91视频 - 114av| 成人免费观看视频网站 | av免费网站观看 | aaawww| 久久精品一区二区三区视频 | 看av免费网站| 亚洲欧美婷婷六月色综合 | 免费高清影视 | 天天色天天色 | 波多野结衣视频一区二区三区 | 色综合久久五月天 | 欧美一级久久久久 | 四虎影视精品永久在线观看 | 久久综合精品国产一区二区三区 | 日韩一区二区三免费高清在线观看 | 九九涩涩av台湾日本热热 | 国产精品ⅴa有声小说 | a'aaa级片在线观看 | 久久免费看 | 精品96久久久久久中文字幕无 | 欧美日韩精品在线 | 99精品视频免费观看 | ,久久福利影视 | 亚洲成年人在线播放 | 欧美性黑人 | 涩涩网站在线播放 | 国产又粗又猛又黄又爽视频 | 国产男女无遮挡猛进猛出在线观看 | 成人久久久久久久久 | 成人h视频在线 | 亚洲理论在线观看 | 亚洲 欧美 日韩 综合 | 国产又粗又猛又黄又爽的视频 | 免费看黄的视频 | 亚洲精品视频久久 | 国产另类xxxxhd高清 | 黄色三级网站在线观看 | 国产在线视频资源 | 韩国一区二区三区在线观看 | 精品视频网站 | 日韩欧美国产激情在线播放 | 在线观看国产www | 欧美性网站 | www.精选视频.com | 激情网站免费观看 | 亚洲日韩精品欧美一区二区 | 天天干夜夜爽 | 天天操天天摸天天爽 | 国产夫妻av在线 | www五月天婷婷| 久草视频网 | www.久久免费 | 97超碰色 | 国产精品免费久久久 | 97在线免费视频观看 | 欧美黄色特级片 | 欧美日在线 | 99在线免费视频观看 | 国产女v资源在线观看 | 久久99精品久久久久久清纯直播 | 久久欧美精品 | 国产精品网站 | 91亚洲国产 | 欧美激情精品久久久久久免费印度 | 天天艹天天 | 97看片吧 | 911免费视频 | 中文字幕有码在线观看 | 亚洲人成人天堂h久久 | www.xxxx变态.com | 国产精品初高中精品久久 | 日韩精品在线看 | 成人免费视频网址 | 国产九九在线 | 国产亚洲精品久久19p | 国产在线超碰 | 亚洲精品电影在线 | 激情婷婷在线观看 | 成人h视频| 很黄很污的视频网站 | 国产一区二区三精品久久久无广告 | 99精品在线观看视频 | 五月婷婷六月丁香 | 久久婷婷色综合 | 久久午夜精品 | 国产美女久久久 | 日本xxxx.com | 久久一区二 | 黄色一级大片在线免费看国产一 | 亚洲高清不卡av | 97视频在线免费 | 美女国产网站 | 亚洲丝袜一区二区 | 国产黄a三级 | 久久激情视频 | 六月丁香综合网 | 在线视频日韩一区 | 精品国产免费观看 | 正在播放国产一区 | 久久久久麻豆v国产 | 免费高清在线视频一区· | 亚洲精品视频在线免费播放 | 欧洲精品久久久久毛片完整版 | 丁香婷婷久久久综合精品国产 | 毛片3 | 在线播放一区二区三区 | 日韩 在线观看 | 日韩美av在线 | 日本久久久久久久久久久 | 狠狠干夜夜爽 | 亚洲国产一区av | 97热在线观看 | 久色小说| 一级成人在线 | 欧美日韩一区二区三区不卡 | 一级片视频在线 | 日韩视频图片 | 69国产成人综合久久精品欧美 | 97av影院| 午夜在线免费视频 | 免费高清男女打扑克视频 | 久久高清| 日韩在线国产精品 | 精品久久久久久久久久久久久久久久久久 | 中文字幕精品三级久久久 | 草草草影院 | 成人av中文字幕在线观看 | 99热最新地址 | 国产一级淫片在线观看 | 国产视频丨精品|在线观看 国产精品久久久久久久久久久久午夜 | 91日韩在线 | 人人射人人 | 香蕉视频4aa| 黄色不卡av | 色国产精品一区在线观看 | 五月激情站| 中文在线www| 中文字幕中文字幕在线中文字幕三区 | 天天干天天操 | 在线看小早川怜子av | 夜夜爽夜夜操 | 精品在线亚洲视频 | 精品国产一区二区三区四区在线观看 | 色婷婷在线观看视频 | 日韩精品视频在线观看免费 | 久久精品99国产精品酒店日本 | 久久精品理论 | www.com.黄 | 国产尤物在线视频 | 日韩成人精品一区二区 | www.天天综合| 五月婷婷在线观看视频 | 91最新中文字幕 | 美女国产在线 | 综合激情av | 黄色成人av在线 | 播五月综合| 欧美午夜精品久久久久 | 国产一区二区网址 | 亚洲人成影院在线 | 久久首页 | 免费的黄色av | 国产第一页福利影院 | 日批视频在线观看免费 | 91麻豆精品国产午夜天堂 | 免费97视频 | 国产一级免费观看视频 | 色www免费视频 | 91精品免费在线视频 | 亚洲国产高清在线 | 日本成人黄色片 | 日韩免费一二三区 | 国产精品日韩 | 91av在线播放视频 | av线上看 | 国产精品久久久久免费a∨ 欧美一级性生活片 | www五月| 国产视 | 久久久久久免费视频 | 夜又临在线观看 | 天天综合91 | 国产免费久久av | 色综合久久久久 | 免费精品视频在线 | 不卡的av在线播放 | 在线免费看黄网站 | 97av.com | 成年人免费在线看 | 久久久国产精品久久久 | 婷婷视频在线观看 | 亚州国产精品视频 | 久久精选 | 亚洲乱码中文字幕综合 | 91 在线视频 | 亚洲国产精品va在线 | 在线日韩中文字幕 | 国产精品久久一区二区无卡 | 在线免费观看欧美日韩 | 最近中文字幕免费视频 | 日批视频在线播放 | 免费日韩一区二区三区 | 麻豆国产在线播放 | 国内精品在线一区 | 91av大全| 成人网在线免费视频 | 亚洲国产小视频在线观看 | 精品久久精品久久 | 久久午夜精品影院一区 | 91香蕉国产在线观看软件 | 国产精品永久免费在线 | 中文字幕免费中文 | 国产精品九九九 | 久久久国产高清 | 国产精品久久久久久久免费 | 亚洲黄色av一区 | 亚洲一区二区三区91 | 在线视频免费观看 | 日韩成人看片 | 激情深爱五月 | 国产高清免费在线播放 | 丁香激情五月婷婷 | 91av社区 | 色欧美视频 | 色婷婷狠狠 | 三级黄色片在线观看 | 日韩一区二区久久 | 欧美91片 | 午夜黄色一级片 | 久久伊人爱 | 青青河边草手机免费 | 精品视频在线观看 | 色综合婷婷 | 亚洲激情 欧美激情 | 国产亚洲免费的视频看 | 激情综合网五月激情 | 免费的黄色的网站 | 中文字幕视频播放 | www操操| 激情网在线视频 | 国产一区免费看 | 欧美一级日韩三级 | 精品国产电影一区 | 久久久成人精品 | 99草视频在线观看 | 国产午夜麻豆影院在线观看 | 亚洲精品88欧美一区二区 | 91精品视频在线看 | 国产精品一级在线 | 国产又粗又硬又长又爽的视频 | 天天色视频| 久久毛片高清国产 | 尤物一区二区三区 | 91人人爱 | 亚洲成aⅴ人片久久青草影院 | 久久夜色电影 | 西西44人体做爰大胆视频 | 五月婷婷开心 | 亚洲综合小说电影qvod | 视频在线观看亚洲 | 2019免费中文字幕 | 婷婷久久精品 | 99热这里只有精品久久 | 久久久久亚洲天堂 | a级国产乱理伦片在线观看 亚洲3级 | 午夜视频久久久 | 国产另类av | 白丝av免费观看 | 久久国产影视 | 在线观看不卡视频 | 2019中文最近的2019中文在线 | 欧美日韩国产精品一区二区亚洲 | 免费高清看电视网站 | 亚洲国产三级 | 欧美日韩成人一区 | 免费观看高清 | 久久麻豆精品 | 99久久影视 | 人人爱人人射 | 欧美日韩国产高清视频 | 99情趣网视频 | 一区av在线播放 | 久热免费 | 日日操天天操狠狠操 | 久久夜色精品亚洲噜噜国4 午夜视频在线观看欧美 | 成人黄色在线观看视频 | 视频在线一区二区三区 | 亚洲天堂社区 | 色综合天天色综合 | 国产在线观看a | 奇米影视在线99精品 | 免费电影一区二区三区 | 亚洲男人天堂2018 | 亚洲激情 在线 | 99r国产精品 | 亚洲电影影音先锋 | 亚洲精品综合一二三区在线观看 | 97国产精品| 中文字幕在线网址 | 婷婷婷国产在线视频 | 日韩在线观看精品 | 免费在线国产黄色 | 91精品国产乱码 | 婷婷五天天在线视频 | 在线观看的a站 | 又黄又爽又刺激 | 狠狠色丁香婷婷综合最新地址 | 免费看特级毛片 | 九九九国产 | 黄色aa久久 | 在线免费观看不卡av | 久久久精品午夜 | 国产精品亚洲精品 | 日韩精品一区二区三区在线播放 | 久久成人国产精品一区二区 | 久久伊人八月婷婷综合激情 | 在线播放第一页 | 久久一精品| 久久国产精品影片 | 日韩免费观看一区二区 | 黄色精品久久久 | 九九九九精品九九九九 | 国产999视频在线观看 | 欧美一区二区在线看 | 四虎国产精品成人免费影视 | 黄色一级大片免费看 | 国产精品9999 | av7777777| 国产色拍拍拍拍在线精品 | 日韩欧美视频免费看 | 欧美一区日韩精品 | 午夜影院先 | 免费三级黄 | 日韩在线看片 | 韩国av电影在线观看 | 毛片888| 欧美日韩伦理在线 | 成人黄色资源 | 日韩av片免费在线观看 | 亚洲综合少妇 | 人人爱天天操 | 免费观看一级一片 | 日本黄色免费在线 | 国产最新在线观看 | 国产高清视频色在线www | 91激情在线视频 | av夜夜操| 国产在线精品一区二区 | 69国产盗摄一区二区三区五区 | 色五丁香 | 视频 国产区 | 在线免费色 | 欧美一区二区免费在线观看 | 国产精品夜夜夜一区二区三区尤 | 五月婷婷狠狠 | 人人干天天干 | 在线观看视频精品 | 波多野结衣亚洲一区二区 | 久久精品中文字幕一区二区三区 | 中文字幕第一页在线视频 | 国产女人18毛片水真多18精品 | 成人一区不卡 | 日韩亚洲精品电影 | 成人日批视频 | 中文字幕在线观看你懂的 | 在线免费观看视频a | 四虎8848免费高清在线观看 | 久久国产精品成人免费浪潮 | 国产免费一区二区三区最新 | 亚洲一区二区视频 | 国产精品99久久免费黑人 | 欧美成人精品欧美一级乱黄 | 久久影视一区二区 | 免费福利视频网站 | 亚州国产视频 | 久久亚洲免费 | 中文字幕中文字幕中文字幕 | 亚洲一级黄色大片 | 色综合天天天天做夜夜夜夜做 | 97久久久免费福利网址 | 国产精品欧美久久久久天天影视 | 少妇超碰在线 | 在线精品视频在线观看高清 | 国产精品伦一区二区三区视频 | 日日噜噜噜噜夜夜爽亚洲精品 | 在线看日韩| 免费观看av | 看毛片网站 | 99精品视频免费观看视频 | 五月婷婷av | 欧美另类xxx | 久草在线最新免费 | 亚洲另类人人澡 | 人人超碰人人 | 免费成人在线电影 | 99这里只有精品视频 | 色综合久久88色综合天天6 | 欧美男同视频网站 | 国产精品剧情 | 日韩高清免费在线观看 | 伊人黄 | 福利网在线 | 中文字幕免费观看视频 | 四虎影视成人永久免费观看视频 | 中文字幕精品久久 | 亚洲午夜在线视频 | 国产精品网红直播 | 在线观看国产麻豆 | 91欧美国产 | 日韩中文在线视频 | 夜色资源站国产www在线视频 | 青青河边草观看完整版高清 | 免费亚洲一区二区 | www.狠狠操.com | 久久久久久久免费观看 | 91成人免费在线 | 天天干天天插 | a级成人毛片 | 美女国产网站 | 亚洲不卡123 | 国产黄色av影视 | 久久99国产精品久久 | 国产成人久久精品77777综合 | 国产99久久久精品 | 免费在线国产精品 | 久久久精品视频成人 | 日韩电影中文,亚洲精品乱码 | 日韩在线视频观看免费 | 尤物97国产精品久久精品国产 | av大片免费| 免费日韩电影 | 国产精品1区2区 | 日韩美女av在线 | 91精品国自产拍天天拍 | 探花国产在线 | 亚洲va在线va天堂va偷拍 | 国产午夜在线观看 | 奇米影视在线99精品 | 在线视频精品 | 国产精品美女免费视频 | 精品99在线视频 | 91看片在线看片 | 亚洲国产片色 | 国产粉嫩在线观看 | 久热香蕉视频 | 国产一区二区在线免费播放 | 国产精品久久久久国产精品日日 | 成年人免费电影在线观看 | 狠狠狠狠狠狠狠干 | 91重口视频 | 久久艹艹 | 国产精品一区二区久久久久 | 美女网站在线 | 91av视频在线播放 | 超级碰碰碰免费视频 | 亚洲成人午夜在线 | 808电影| 91av视频在线观看免费 | 日韩欧美一区二区三区视频 | 中文字幕色站 | 日本资源中文字幕在线 | 91日韩在线 | 午夜精品久久久久久99热明星 | h动漫中文字幕 | 国产一区视频在线播放 | 日本久久久久 | 激情综合国产 | 中文字幕资源网在线观看 | 成人黄色小说视频 | 国产亚洲视频系列 | 亚洲国产视频直播 | 免费在线播放黄色 | 天天干天天草 | 黄色大全免费网站 | 在线a人片免费观看视频 | 欧美久久成人 | 99精品在线免费在线观看 | 在线99| av在线免费网 | 欧美日韩精品区 | 欧美一级日韩免费不卡 | 97在线精品国自产拍中文 | 免费视频久久久久久久 | 日日干干| 欧美一区二区日韩一区二区 | 国产精品自产拍在线观看 | 伊人av综合 | 中文字幕一区二区三区久久 | 99精品视频免费全部在线 | 精品久久久久久国产91 | 91视频免费视频 | 欧美男同网站 | 久久久99精品免费观看乱色 | 亚洲黄色免费在线看 | 有码视频在线观看 | 日韩中文字幕a | 欧美少妇xxx| 国产 在线观看 | 99精品免费在线观看 | 最近能播放的中文字幕 | 亚洲日本一区二区在线 | 欧美三级免费 | 五月天中文字幕mv在线 | 亚洲在线观看av | 亚洲极色| 在线a视频 | 天天爽网站 | 久久论理| 免费在线国产精品 | 国产男女免费完整视频 | 九九久久久久久久久激情 | 日韩欧美网站 | 国产网站av | 国产亚洲精品成人av久久影院 | 午夜a区 | 最近在线中文字幕 | 天堂久色| 国产精品久久久久一区二区三区共 | 99热 精品在线 | 麻豆 free xxxx movies hd | 最近高清中文字幕 | 日韩精品最新在线观看 | 久久国语 | 天天干天天色2020 | 99精品国产福利在线观看免费 | 午夜久久网 | 91在线网站 | 国产福利电影网址 | 在线影院中文字幕 | 奇米先锋 | 精品国产诱惑 | 成人在线观看av | 国产高清免费视频 | 91视频在线播放视频 | 国产aa精品 |