我有一个自定义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)
如果你不想使用任何第三方库,你可以添加这些扩展函数(我倾向于有一个ContextExtensions.kt
或ViewExtensions.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
这些让你从find
,Activity
和Fragment
s中调用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
。作为编程的一般规则,尽量使事物尽可能不变,你会得到更少的错误。
使用Kotterknife作为Kotlin的Butter Knife-esque View Binding。
然后你可以绑定你的视图
val menuTitleTextView: CTextBasic by bindView(R.id.menu_title_text_view_id)
我发现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)
//...
}
}
请享用。
就我而言,我没有正确构建ButterKnife。确保您在模块的build.gradle中导入其编译器:
...
// Butter Knife
implementation "com.jakewharton:butterknife:$butterKnifeVersion"
kapt "com.jakewharton:butterknife-compiler:$butterKnifeVersion"
...
Jetbrain的样本线程中的discussion为我解释了这个问题。
另一个问题是您可能在创建容器之前访问视图。这是一个related question,讨论有针对kotlinx的合成属性,但同样的逻辑应该适用于Butterknife视图绑定
将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"
希望这可以帮助!
您的初始化程序块未被调用。只有在调用主构造函数时才会调用它。在您的情况下,当从xml布局创建视图对象时,将使用辅助构造函数。
将您的init{...}
块更改为fun init(){...}
,并将其作为每个构造函数中的第一个语句
你忘了添加构造函数
constructor(context: Context, attrs: AttributeSet, defStyleAttr: Int) : super(context, attrs, defStyleAttr)
=>添加它并在其中调用init()