詳細的可以看這一篇文章: 《Do not always trust @JvmOverloads》
@JvmOverloads 注解是用來干撒的?
/**
* Instructs the Kotlin compiler to generate overloads for this function that substitute default parameter values.
*
* If a method has N parameters and M of which have default values, M overloads are generated: the first one
* takes N-1 parameters (all but the last one that takes a default value), the second takes N-2 parameters, and so on.
*/
@Target(AnnotationTarget.FUNCTION, AnnotationTarget.CONSTRUCTOR)
@Retention(AnnotationRetention.BINARY)
@MustBeDocumented
public annotation class JvmOverloads
也就是說,如果方法參數中有默認值的,那么這個注解可以幫我們生成多個重載方法,這樣方便我們調用,可以省掉一些不關心的參數。
比如這樣:
@JvmOverloads
fun test(name: String = "hello", code: Int = 0){
}
最終會幫我們重載三個方法出來
@JvmOverloads
public final void test(@NotNull String name, int code) {
Intrinsics.checkParameterIsNotNull(name, "name");
}
@JvmOverloads
public final void test(@NotNull String name) {
test$default(this, name, 0, 2, (Object)null);
}
@JvmOverloads
public final void test() {
test$default(this, (String)null, 0, 3, (Object)null);
}
@JvmOverloads 自定義View
class EmoticonView @JvmOverloads constructor(
context: Context, attrs: AttributeSet? = null, defStyleAttr: Int = 0
) : RelativeLayout(context, attrs, defStyleAttr)
按照上面的介紹,會幫忙自動生成多個重載方法,那么坑來了的,注意這個 defStyleAttr = 0 。。。
我們先看看 EditText 的構造函數, 如果是XML來寫的布局,那么會調用第二個構造函數,也就是 public EditText(Context context, AttributeSet attrs)
, 那么默認傳入的 樣式 defStyleAttr
就是 com.android.internal.R.attr.editTextStyle
public class EditText extends TextView {
public EditText(Context context) {
this(context, null);
}
public EditText(Context context, AttributeSet attrs) {
this(context, attrs, com.android.internal.R.attr.editTextStyle);
}
public EditText(Context context, AttributeSet attrs, int defStyleAttr) {
this(context, attrs, defStyleAttr, 0);
}
public EditText(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);
}
而我們如果使用 IDE 自動提示的 @JvmOverload
image.png
寫出來的代碼長這樣:
class InputVIew @JvmOverloads constructor(
context: Context, attrs: AttributeSet? = null, defStyleAttr: Int = 0
) : EditText(context, attrs, defStyleAttr) {
}
看到了么,你的代碼 defStyleAttr
默認是 0
而不是 com.android.internal.R.attr.editTextStyle
。
那么說了這么多,這樣會導致什么問題呢?
親測,焦點沒有了的,點擊之后鍵盤也無法彈起。
那么這個坑 如何填呢?
修改默認值即可:
class InputVIew @JvmOverloads constructor(
context: Context, attrs: AttributeSet? = null, defStyleAttr: Int = android.support.design.R.attr.editTextStyle
) : EditText(context, attrs, defStyleAttr)
注意
像我在開頭提到的文章 《Do not always trust @JvmOverloads》 ,在最后寫到: Button, EditText, RadioButton, Switch 都會有類似的問題,請大家平時開發過程中要留意。