今天学习十分钟——Kotlin扩展

大家好,我是走川。只有十分钟,没时间废话了!我要发车了!!!!


Kotlin除了丰富语法糖之外,最让人喜欢就是扩展代理。因为篇幅有限,本文着重讲述扩展(Extension),而代理相关的内容另起一文。

kotlin允许扩展类的属性和方法,不需要继承或使用 Decorator 模式。扩展是一种静态行为,对被扩展的类代码本身不会造成任何影响,且只作用于类的实例,这意味着它并不扩展静态变量和静态方法。比起Java的一板一眼,Kotlin的扩展简直是给了开发者插上了想象的翅膀。通过扩展能实现各式各样的功能。


在介绍扩展,先介绍扩展的管理。为了方便管理全局性的扩展(即多个文件会引用到的扩展),我们需要单独建立一个kt文件,如下图。

入门篇

Kotlin分为属性扩展和方法扩展,以下将会一一介绍

属性扩展

范例


//扩展var需要实现 set/get 方法
var TextView.padding: Int
    set(value) = setPadding(value, value, value, value)
    get() = padding

//**val** 属性只需要实现  get 方法 
val TextView.wrapContent: Int
    get() = ViewGroup.LayoutParams.WRAP_CONTENT

//任意一个ViewGroup的子类的实例都可以引用这个扩展属性
TextView(context).apply {
    layoutParams.height = wrapContent
 }

实现原理分析
将上面的代码转成class后反编译成java文件,得到以下代码。

public static final void setPadding(@NotNull TextView $receiver, int value) {
      Intrinsics.checkParameterIsNotNull($receiver, "$receiver"); //参数判空
      $receiver.setPadding(value, value, value, value);
   }

public static final int getPadding(@NotNull TextView $receiver) {
      Intrinsics.checkParameterIsNotNull($receiver, "$receiver");//参数判空
      return getPadding($receiver);
   }

public static final int getWrapContent(@NotNull TextView $receiver) {
      Intrinsics.checkParameterIsNotNull($receiver, "$receiver");//参数判空
      return -2;
   }


 TextView this_$iv = new TextView(this.getBaseContext());
  this_$iv.getLayoutParams().height = ViewDSLKt.getWrapContent(this_$iv);

通过反编译后的代码,我们可以很明显的发现,kotlin的属性扩展本质是生成了对应的静态方法。而不是真正生成了字段。

方法扩展

```javascript
fun ImageView.loadImg(url: String?) {
    //加载图片
    ImageLoader.with(this.context).load(url).into(this)
}

//调用方式和正常方法一样
ImageView(context).loadImg("https://ss0.bdstatic.com/5aV1bjqh_Q23odCf/static/superman/img/logo/logo_white_fe6da1ec.png")

实现原理分析
将上面的代码转成class后反编译成java文件,得到以下代码。

  public static final void loadImg(@NotNull ImageView $receiver, @Nullable String url) {
      Intrinsics.checkParameterIsNotNull($receiver, "$receiver");//参数判空
      ImageLoader.with($receiver.getContext()).load(url).into($receiver);
   }

//类似我们的Utils方法
ImageViqw this_$iv = new ImageViqw(this.getBaseContext());;
ViewExtensionKt.loadImg(this_$iv, var2);

进阶篇


Q: 这就是你说的扩展???这尼玛也太简单了吧。我写一个Util类包一下也一样能实现啊!!!


A: 咳!咳!以上只是简单的入门,接下来来撸一个anko那样的DSL,为了怕你们点上面的关闭键,我们先看看最后的效果。

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        linearLayout {     //最外层布局为LinearLayout
            orientation = LinearLayout.VERTICAL    //设置布局方向
            //设置layoutParams
            lp {
                height = matchParent
                width = matchParent
            }
             //添加子View
            button {
                 text = "webview"    //设置文案
                 padding = 20        //设置内间距
                 //设置layoutParams
                 lp {
                     height = 200
                     width = wrapContent
                 }
                 //设置click事件
                 onClick { view -> startActivity(Intent(this@MainActivity, WebViewActivity::class.java)) }
             }
        }
    }
}

以上的代码布局,模仿了anko生成布局的能力,而针对其的扩展方法也很容易理解,代码如下

//设置最顶层view,传入在LinearLayout作用域内的block
fun Activity.linearLayout(block: LinearLayout.() -> Unit) {
    setContentView(LinearLayout(this).apply {
        block()
    })
}

//设置子View的Button,传入在Button作用域内的block
inline fun <reified T : ViewGroup> T.button(block: Button.() -> Unit) {
    addView(Button(context).apply {
        block()
    })
}

//设置View的监听
fun <T : View> T.onClick(block: (View) -> Unit) {
    setOnClickListener { block(this) }
}

//设置View的LayoutParams
inline fun <T : ViewGroup> T.lp(block: ViewGroup.MarginLayoutParams.() -> Unit): ViewGroup.MarginLayoutParams
        = ViewGroup.MarginLayoutParams(ViewGroup.MarginLayoutParams.WRAP_CONTENT, ViewGroup.MarginLayoutParams.WRAP_CONTENT)
        .apply {
            block()
        }

//扩展相对应的参数
inline val <T : ViewGroup> T.wrapContent: Int
    get() = ViewGroup.LayoutParams.WRAP_CONTENT

inline val <T : ViewGroup> T.matchParent: Int
    get() = ViewGroup.LayoutParams.MATCH_PARENT

inline var <T : View> T.padding: Int
    set(value) = setPadding(value, value, value, value)
    get() = ViewGroup.LayoutParams.WRAP_CONTENT

好了,只要照着以上的demo,很快你也能写一个属于自己的DSL,以上的代码没有很绕的内容,只需要照着写一个demo,马上就能够吸收啦!!~~~

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

推荐阅读更多精彩内容