大家好,我是走川。只有十分钟,没时间废话了!我要发车了!!!!
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,马上就能够吸收啦!!~~~