迪米特法則(Law of emeter)
定義:一個對象應該對其他對象了解最少
迪米特法則的核心觀念就是類間解耦,弱耦合,只有弱耦合了以后,類的復用性才可以提高。
形象一點的比喻類似於:監獄內的犯人是不應該跟外面的人接觸的,當然或許會有探親的。這里的監獄就是類,里面的犯人就是類內部的信息,而監獄里的獄警就相當於迪米特法則的執行者
舉個例子
家人探望犯人
家人:家人只與犯人是親人,但是不認識他的獄友
package com.loulijun.chapter5; public class Family { //家人探望犯人 public void visitPrisoner(Prisoners prisoners) { //家人希望犯人與獄友互幫互助 Inmates inmates = prisoners.helpEachOther(); //獄友說,我們是盟友 inmates.weAreFriend(); } }
犯人:犯人與家人是親人,犯人與獄友是朋友
package com.loulijun.chapter5; public class Prisoners { private Inmates inmates = new Inmates(); public Inmates helpEachOther() { System.out.println("家人說:你和獄友之間應該互相幫助..."); return inmates; } }
獄友:犯人與獄友是朋友,但是不認識他的家人
package com.loulijun.chapter5; //Inmates是獄友的意思 public class Inmates { public void weAreFriend() { System.out.println("獄友說:我們是獄友..."); } }
場景類:發生在監獄里
package com.loulijun.chapter5; public class Prison { public static void main(String args[]) { Family family = new Family(); family.visitPrisoner(new Prisoners()); } }
運行結果:
家人說:你和獄友之間應該互相幫助...
獄友說:我們是獄友...
看到這樣的結果,是不是有些別扭,家人告訴犯人要與獄友好好相處,而獄友確冒出來說話。這顯然越界了,因為監獄只允許家人探望犯人,而不是隨便誰都可以見的
這里的家人和獄友有了溝通是違背迪米特法則的,所以我們需要將家人和獄友隔離開,對其進行重構
家人
package com.loulijun.chapter5; public class Family { //家人探望犯人 public void visitPrisoner(Prisoners prisoners) { System.out.print("家人說:"); prisoners.helpEachOther(); } }
犯人
package com.loulijun.chapter5; public class Prisoners { private Inmates inmates = new Inmates(); public Inmates helpEachOther() { System.out.println("犯人和獄友之間應該互相幫助..."); System.out.print("犯人說:"); inmates.weAreFriend(); return inmates; } }
獄友
package com.loulijun.chapter5; //Inmates是獄友的意思 public class Inmates { public void weAreFriend() { System.out.println("我們是獄友..."); } }
監獄
package com.loulijun.chapter5; public class Prison { public static void main(String args[]) { Family family = new Family(); family.visitPrisoner(new Prisoners()); } }
運行結果
家人說:犯人和獄友之間應該互相幫助...
犯人說:我們是獄友...
這樣家人和獄友就分開了,但是也表達了家人希望獄友能跟犯人互相幫助的意願。也就是兩個類通過第三個類實現信息傳遞
網上還有如下一些關於應用迪米特法則的注意事項:
① 在類的划分上,應該創建有弱耦合的類;
② 在類的結構設計上,每一個類都應當盡量降低成員的訪問權限;
③ 在類的設計上,只要有可能,一個類應當設計成不變類;
④ 在對其他類的引用上,一個對象對其它對象的引用應當降到最低;
⑤ 盡量降低類的訪問權限;
⑥ 謹慎使用序列化功能;
⑦ 不要暴露類成員,而應該提供相應的訪問器(屬性)。