kotlin代理模式

koltin的特性大多不是空穴來風(fēng)土匀,而是為了解決一些固有問題呻疹。

kotlin代理模式官方文檔地址:http://kotlinlang.org/docs/reference/delegation.html

一.代理模式

代理是實現(xiàn)代碼復(fù)用的一種方法。
在面向?qū)ο蟮恼Z言中柏靶,代理模式是通過組合的方式來達(dá)到和繼承一樣的效果的。

代理模式定義:給某個對象提供一個代理對象,并由代理對象控制對于原對象的訪問灵妨,即客戶不直接操控原對象,而是通過代理對象間接地操控原對象落竹。

下面是代理模式UML圖:


代理模式三要素:1.RealSubject 原對象 2.Proxy 代理對象 3.Subject 接口

RealSubject和Proxy都實現(xiàn)了Subject接口泌霍,這樣兩者就具有了公共方法Request。

通過執(zhí)行Proxy中的Request方法述召,間接的執(zhí)行RealSubject中的Request方法朱转。

先來個??了解一下如何實現(xiàn):

interface Subject {
    void request();
}
class RealSubject implements Subject{

    @Override
    public void request() {
        System.out.println("RealSubject");
    }
}
class Proxy implements Subject{
    private RealSubject realSubject;

    public Proxy(RealSubject realSubject) {
        this.realSubject = realSubject;
    }

    @Override
    public void request() {
        System.out.println("Proxy start");
        realSubject.request();
        System.out.println("Proxy end");
    }
}
public static void main(String[] args){
    RealSubject realSubject = new RealSubject();
    Proxy proxy=new Proxy(realSubject);
    proxy.request();
}

以上代碼就是代理模式的實現(xiàn)原理。

通過代理模式:
功能1. 我們可以復(fù)用RealSubject類的代碼积暖。
功能2. 在執(zhí)行RealSubject的request方法執(zhí)行之前和執(zhí)行之后藤为,插入一段代碼(比如打印出來request方法的執(zhí)行時間)。

對于功能1 接下來讓我們思考一個問題:
假如Subject接口聲明了2個方法夺刑。而我們需要復(fù)用RealSubject其中的1個方法:

interface Subject {
    void request1();
    void request2();
}
class RealSubject implements Subject{
    @Override
    public void request1() {
        System.out.println("RealSubject request1");
    }

    @Override
    public void request2() {
        System.out.println("RealSubject request2");
    }
}
class Proxy implements Subject{
    private RealSubject realSubject;

    public Proxy(RealSubject realSubject) {
        this.realSubject = realSubject;
    }

    @Override
    public void request1() {
        realSubject.request1();
    }

    @Override
    public void request2() {
        System.out.println("Proxy request2");
    }
}

我們需要手動書寫Proxy類缅疟,然后重載request1和request2方法,我們可以很快的把代碼敲完遍愿。

如果Subject接口聲明了100個方法存淫,而我們想復(fù)用RealSubject類中的90個方法呢,這敲代碼花費(fèi)的時間不可忽視沼填。

kotlin成功的解決了這個問題桅咆。

二.kotlin代理模式的實現(xiàn)

kotlin實現(xiàn)代理模式非常簡單,看一個官網(wǎng)的??:

interface Base {
    fun print()
}

class BaseImpl(val x: Int) : Base {
    override fun print() { print(x) }
}

class Derived(b: Base) : Base by b

fun main(args: Array<String>) {
    val b = BaseImpl(10)
    Derived(b).print()
}

運(yùn)行結(jié)果是:10

轉(zhuǎn)為java代碼看一下廬山真面目:

// Base.java
import kotlin.Metadata;

public interface Base {
   void print();
}
// Derived.java
import kotlin.Metadata;
import kotlin.jvm.internal.Intrinsics;
import org.jetbrains.annotations.NotNull;

public final class Derived implements Base {
   private final Base $$delegate_0;

   public Derived(@NotNull Base b) {
      Intrinsics.checkParameterIsNotNull(b, "b");
      super();
      this.$$delegate_0 = b;
   }

   public void print() {
      this.$$delegate_0.print();
   }
}
// TestKt.java
import kotlin.Metadata;
import kotlin.jvm.internal.Intrinsics;
import org.jetbrains.annotations.NotNull;

public final class TestKt {
   public static final void main(@NotNull String[] args) {
      Intrinsics.checkParameterIsNotNull(args, "args");
      BaseImpl b = new BaseImpl(10);
      (new Derived((Base)b)).print();
   }
}
// BaseImpl.java
import kotlin.Metadata;

public final class BaseImpl implements Base {
   private final int x;

   public void print() {
      int var1 = this.x;
      System.out.print(var1);
   }

   public final int getX() {
      return this.x;
   }

   public BaseImpl(int x) {
      this.x = x;
   }
}

其實就是在編譯期自動生成了Derived類坞笙,解放了雙手岩饼。

我們可以按照需求復(fù)寫print()方法

interface Base {
    fun printMessage()
    fun printMessageLine()
}

class BaseImpl(val x: Int) : Base {
    override fun printMessage() { print(x) }
    override fun printMessageLine() { println(x) }
}

class Derived(b: Base) : Base by b {
    override fun printMessage() { print("abc") }
}

fun main(args: Array<String>) {
    val b = BaseImpl(10)
    Derived(b).printMessage()
    Derived(b).printMessageLine()
}

輸出:abc10

Derived除了可以實現(xiàn)Base接口荚虚,還可以繼承其他父類,方法名字遇到?jīng)_突怎么辦

比如Derived繼承了父類Parent籍茧,而父類同樣擁有print方法:

interface Base {
    fun print()
}

class BaseImpl(val x: Int) : Base {
    override fun print() {
        print(x)
    }
}

open class Parent {
    open fun print() {
        println("Parent print")
    }
}

class Derived(b: Base) : Parent(),Base by b

fun main(args: Array<String>) {
    val b = BaseImpl(10)
    Derived(b).print()
}

輸出:10
可以看到父類print方法會被覆蓋版述。

三.kotlin代理模式的總結(jié)

1.只能實現(xiàn)對接口方法的代理,即Base類不能為抽象類硕糊。
2.不方便對所有的代理方法進(jìn)行統(tǒng)一處理院水。比如說在執(zhí)行每個方法前都執(zhí)行相同的邏輯,而java動態(tài)代理可以方便的實現(xiàn)這個功能简十。
3.方法名稱有沖突時檬某,代理類方法優(yōu)先級較高。
4.編譯期自動生成代理模式螟蝙。不會影響運(yùn)行效率恢恼。

四.繼承和代理的選擇

如果僅僅是代碼復(fù)用和方法重寫,繼承能達(dá)到和代理一樣的效果胰默。

繼承和代理的使用都存在條件限制:

如果使用繼承的話场斑,父類必須為可繼承的,并且想要覆蓋的方法也必須為可重寫的牵署,即java中類和方法都不能存在final修飾符漏隐,kotlin中明確使用open修飾符。

使用代理的話奴迅,兩者需要存在公共接口青责,比如上面例子中類DerivedParent都需要實現(xiàn)Base接口。

由于kotlin取具、java存在單繼承的約束(每個類只能存在一個父類)脖隶,在使用繼承或者代理均可的情況下,推薦使用代理暇检。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末产阱,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子块仆,更是在濱河造成了極大的恐慌构蹬,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件悔据,死亡現(xiàn)場離奇詭異庄敛,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)蜜暑,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進(jìn)店門铐姚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來策肝,“玉大人肛捍,你說我怎么就攤上這事隐绵。” “怎么了拙毫?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵依许,是天一觀的道長。 經(jīng)常有香客問我缀蹄,道長峭跳,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任缺前,我火速辦了婚禮蛀醉,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘衅码。我一直安慰自己拯刁,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布逝段。 她就那樣靜靜地躺著垛玻,像睡著了一般。 火紅的嫁衣襯著肌膚如雪奶躯。 梳的紋絲不亂的頭發(fā)上帚桩,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天,我揣著相機(jī)與錄音嘹黔,去河邊找鬼账嚎。 笑死,一個胖子當(dāng)著我的面吹牛参淹,可吹牛的內(nèi)容都是我干的醉锄。 我是一名探鬼主播,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼浙值,長吁一口氣:“原來是場噩夢啊……” “哼恳不!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起开呐,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤烟勋,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后筐付,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體卵惦,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年瓦戚,在試婚紗的時候發(fā)現(xiàn)自己被綠了沮尿。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,137評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情审胸,我是刑警寧澤,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布姥敛,位于F島的核電站,受9級特大地震影響瞎暑,放射性物質(zhì)發(fā)生泄漏彤敛。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一了赌、第九天 我趴在偏房一處隱蔽的房頂上張望墨榄。 院中可真熱鬧,春花似錦勿她、人聲如沸渠概。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽播揪。三九已至,卻和暖如春筒狠,著一層夾襖步出監(jiān)牢的瞬間猪狈,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工辩恼, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留雇庙,地道東北人。 一個月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓灶伊,卻偏偏與公主長得像疆前,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子聘萨,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,901評論 2 345

推薦閱讀更多精彩內(nèi)容

  • https://blog.csdn.net/luanlouis/article/details/24589193 ...
    小陳阿飛閱讀 850評論 1 1
  • 本篇文章繼續(xù)介紹Java反射機(jī)制竹椒,不同的是側(cè)重于介紹動態(tài)代理。動態(tài)代理是代理模式中的一種米辐,是通過Java反射機(jī)制來...
    Android進(jìn)階與總結(jié)閱讀 591評論 0 0
  • 【非常日翘贮,平常心】 前幾天三個姓朱的家庭剛聚餐赊窥,今天三個姓朱的家庭又聚餐了,每家一瓶紅酒狸页,對不喝酒的人來說已經(jīng)超...
    竹童閱讀 250評論 0 0
  • /Lauren 我們曾經(jīng)的愛 不過是 誤入藕花深處 一場意外 一場錯誤罷了
    林嶼真閱讀 334評論 1 6
  • 2017年7月5日我們14人一行如期從合肥出發(fā)了锨能,兩個多小時的動車到達(dá)第一站武漢,輾轉(zhuǎn)擺渡車進(jìn)入國際機(jī)場第一航站樓...
    潘海原閱讀 130評論 1 0