kotlin代理模式

koltin的特性大多不是空穴来风,而是为了解决一些固有问题。

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

一.代理模式

代理是实现代码复用的一种方法。
在面向对象的语言中,代理模式是通过组合的方式来达到和继承一样的效果的。

代理模式定义:给某个对象提供一个代理对象,并由代理对象控制对于原对象的访问,即客户不直接操控原对象,而是通过代理对象间接地操控原对象。

下面是代理模式UML图:


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

RealSubject和Proxy都实现了Subject接口,这样两者就具有了公共方法Request。

通过执行Proxy中的Request方法,间接的执行RealSubject中的Request方法。

先来个🌰了解一下如何实现:

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();
}

以上代码就是代理模式的实现原理。

通过代理模式:
功能1. 我们可以复用RealSubject类的代码。
功能2. 在执行RealSubject的request方法执行之前和执行之后,插入一段代码(比如打印出来request方法的执行时间)。

对于功能1 接下来让我们思考一个问题:
假如Subject接口声明了2个方法。而我们需要复用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个方法,而我们想复用RealSubject类中的90个方法呢,这敲代码花费的时间不可忽视。

kotlin成功的解决了这个问题。

二.kotlin代理模式的实现

kotlin实现代理模式非常简单,看一个官网的🌰:

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()
}

运行结果是:10

转为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类,解放了双手。

我们可以按照需求复写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除了可以实现Base接口,还可以继承其他父类,方法名字遇到冲突怎么办

比如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代理模式的总结

1.只能实现对接口方法的代理,即Base类不能为抽象类。
2.不方便对所有的代理方法进行统一处理。比如说在执行每个方法前都执行相同的逻辑,而java动态代理可以方便的实现这个功能。
3.方法名称有冲突时,代理类方法优先级较高。
4.编译期自动生成代理模式。不会影响运行效率。

四.继承和代理的选择

如果仅仅是代码复用和方法重写,继承能达到和代理一样的效果。

继承和代理的使用都存在条件限制:

如果使用继承的话,父类必须为可继承的,并且想要覆盖的方法也必须为可重写的,即java中类和方法都不能存在final修饰符,kotlin中明确使用open修饰符。

使用代理的话,两者需要存在公共接口,比如上面例子中类DerivedParent都需要实现Base接口。

由于kotlin、java存在单继承的约束(每个类只能存在一个父类),在使用继承或者代理均可的情况下,推荐使用代理。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352

推荐阅读更多精彩内容

  • https://blog.csdn.net/luanlouis/article/details/24589193 ...
    小陈阿飞阅读 859评论 1 1
  • 本篇文章继续介绍Java反射机制,不同的是侧重于介绍动态代理。动态代理是代理模式中的一种,是通过Java反射机制来...
    Android进阶与总结阅读 599评论 0 0
  • 【非常日,平常心】 前几天三个姓朱的家庭刚聚餐,今天三个姓朱的家庭又聚餐了,每家一瓶红酒,对不喝酒的人来说已经超...
    竹童阅读 254评论 0 0
  • /Lauren 我们曾经的爱 不过是 误入藕花深处 一场意外 一场错误罢了
    林屿真阅读 336评论 1 6
  • 2017年7月5日我们14人一行如期从合肥出发了,两个多小时的动车到达第一站武汉,辗转摆渡车进入国际机场第一航站楼...
    潘海原阅读 135评论 1 0