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

2017-08-25  本文已影响179人  走川

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


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,马上就能够吸收啦!!~~~

上一篇下一篇

猜你喜欢

热点阅读