美文网首页
今天学习十分钟——Kotlin扩展

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

作者: 走川 | 来源:发表于2017-08-25 01:46 被阅读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,马上就能够吸收啦!!~~~

    相关文章

      网友评论

          本文标题:今天学习十分钟——Kotlin扩展

          本文链接:https://www.haomeiwen.com/subject/gqnvcxtx.html