lateinit属性尚未初始化

问题描述 投票:16回答:6

我有一个自定义linearlayout类,当我想创建这个类的实例时,我得到一个错误lateinit属性尚未初始化我正在使用最新版本的butterknife库

这是我的kotlin课程

class MenuItemView : LinearLayout {

@BindView(R.id.menu_title_text_view_id)
lateinit var menuTitleTextView : CTextBasic

constructor(ctx: Context) : super(ctx) {
}

init {
    val view = LayoutInflater.from(context).inflate(R.layout.menu_item,this)
    ButterKnife.bind(this,view)
}

constructor(ctx: Context, attrs: AttributeSet) : super(ctx, attrs) {
    val menuAttrs = context.theme.obtainStyledAttributes(attrs, R.styleable.MenuItemView, 0, 0)
    try {
        val title: String = menuAttrs.getString(R.styleable.MenuItemView_menu_title)
        menuTitleTextView.text = title
    }catch (e : Exception){
        e.printStackTrace()
    }finally {
        menuAttrs.recycle()
    }
}
fun setTitle( title : String){
    menuTitleTextView.text = title
}
}

这是错误日志

    kotlin.UninitializedPropertyAccessException: lateinit property menuTitleTextView has not been initialized
at com.leavigstone.liberali.ui.custom.menu.MenuItemView.setTitle(MenuItemView.kt:48)
at com.leavigstone.liberali.ui.activities.MainActivity.onAddButtonClick(MainActivity.java:142)
at com.leavigstone.liberali.ui.activities.MainActivity_ViewBinding$3.doClick(MainActivity_ViewBinding.java:54)
at butterknife.internal.DebouncingOnClickListener.onClick(DebouncingOnClickListener.java:22)
at android.view.View.performClick(View.java:4780)
at android.view.View$PerformClick.run(View.java:19866)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:135)
at android.app.ActivityThread.main(ActivityThread.java:5254)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:698)
android kotlin butterknife
6个回答
6
投票

如果你不想使用任何第三方库,你可以添加这些扩展函数(我倾向于有一个ContextExtensions.ktViewExtensions.kt用于Context或View相关的扩展函数),然后放入它

inline fun <reified T : View> View.find(id: Int): T = findViewById(id) as T
inline fun <reified T : View> Activity.find(id: Int): T = findViewById(id) as T
inline fun <reified T : View> Fragment.find(id: Int): T = view?.findViewById(id) as T

这些让你从findActivityFragments中调用View。所以在你的班级而不是

@BindView(R.id.menu_title_text_view_id) lateinit var menuTitleTextView : CTextBasic

你可以有

val menuTitleTextView by lazy { find<CTextBasic>(R.id.menu_title_text_view_id) }

对于像UI这样的东西,当它们不需要改变时,最好是val而不是var。作为编程的一般规则,尽量使事物尽可能不变,你会得到更少的错误。


2
投票

使用Kotterknife作为Kotlin的Butter Knife-esque View Binding。

然后你可以绑定你的视图

val menuTitleTextView: CTextBasic by bindView(R.id.menu_title_text_view_id)

2
投票

我发现this适合我。

在项目build.gradle模块中更改app

dependencies {
    compile "com.jakewharton:butterknife:8.8.1"
    kapt "com.jakewharton:butterknife-compiler:8.8.1"
}

使用kapt而不是annotationProcessor

然后你可以这样做你熟悉的ButterKnife注释:

class MainActivity : AppCompatActivity() {

    @BindView(R.id.myButton)
    lateinit var myButton: Button

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        ButterKnife.bind(this)
        //...
    }
}

请享用。


2
投票

就我而言,我没有正确构建ButterKnife。确保您在模块的build.gradle中导入其编译器:

...
// Butter Knife
implementation "com.jakewharton:butterknife:$butterKnifeVersion"
kapt "com.jakewharton:butterknife-compiler:$butterKnifeVersion"
...

Jetbrain的样本线程中的discussion为我解释了这个问题。

另一个问题是您可能在创建容器之前访问视图。这是一个related question,讨论有针对kotlinx的合成属性,但同样的逻辑应该适用于Butterknife视图绑定


2
投票

apply plugin: 'kotlin-kapt'添加到app level build.gradle文件中

apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'
apply plugin: 'kotlin-kapt'

并在依赖部分

implementation "com.jakewharton:butterknife:8.8.1"
kapt "com.jakewharton:butterknife-compiler:8.8.1"

希望这可以帮助!


-2
投票

您的初始化程序块未被调用。只有在调用主构造函数时才会调用它。在您的情况下,当从xml布局创建视图对象时,将使用辅助构造函数。

将您的init{...}块更改为fun init(){...},并将其作为每个构造函数中的第一个语句

你忘了添加构造函数

constructor(context: Context, attrs: AttributeSet, defStyleAttr: Int) : super(context, attrs, defStyleAttr)

=>添加它并在其中调用init()

© www.soinside.com 2019 - 2024. All rights reserved.