- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我正在使用 lateinit 属性以避免使用 ?运算符(operator)。我有很多 View 属性是第一次在 getViews() 函数中分配的。如果该功能不存在,我的应用程序将与来自 Kotlin 代码的 NPE 一起崩溃。
在我看来,lateinit 属性基本上破坏了该语言良好的 null 安全特性。我知道它们是在 M13 中引入的,因为它有更好的框架支持,但值得吗?
或者我在这里遗漏了什么?
代码如下:
package com.attilapalfi.exceptional.ui
import android.os.Bundle
import android.support.v7.app.AppCompatActivity
import android.view.View
import android.widget.Button
import android.widget.ImageView
import android.widget.TextView
import com.attilapalfi.exceptional.R
import com.attilapalfi.exceptional.dependency_injection.Injector
import com.attilapalfi.exceptional.model.Exception
import com.attilapalfi.exceptional.model.ExceptionType
import com.attilapalfi.exceptional.model.Friend
import com.attilapalfi.exceptional.persistence.*
import com.attilapalfi.exceptional.rest.ExceptionRestConnector
import com.attilapalfi.exceptional.ui.helpers.ViewHelper
import com.attilapalfi.exceptional.ui.question_views.QuestionYesNoClickListener
import com.google.android.gms.maps.MapView
import java.math.BigInteger
import javax.inject.Inject
public class ShowNotificationActivity : AppCompatActivity(), QuestionChangeListener {
@Inject
lateinit val exceptionTypeStore: ExceptionTypeStore
@Inject
lateinit val friendStore: FriendStore
@Inject
lateinit val imageCache: ImageCache
@Inject
lateinit val metadataStore: MetadataStore
@Inject
lateinit val viewHelper: ViewHelper
@Inject
lateinit val exceptionInstanceStore: ExceptionInstanceStore
@Inject
lateinit val exceptionRestConnector: ExceptionRestConnector
@Inject
lateinit val questionStore: QuestionStore
private lateinit var sender: Friend
private lateinit var exception: Exception
private lateinit var exceptionType: ExceptionType
private lateinit var exceptionNameView: TextView
private lateinit var exceptionDescView: TextView
private lateinit var senderImageView: ImageView
private lateinit var senderNameView: TextView
private lateinit var sendDateView: TextView
private lateinit var mapView: MapView
private lateinit var questionText: TextView
private lateinit var noButton: Button
private lateinit var yesButton: Button
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_show_notification)
Injector.INSTANCE.applicationComponent.inject(this)
questionStore.addChangeListener(this)
getModelFromBundle()
getViews()
loadViewsWithData()
}
override fun onDestroy() {
super.onDestroy()
questionStore.removeChangeListener(this)
}
private fun getModelFromBundle() {
val bundle = intent.extras
val instanceId = BigInteger(bundle.getString("instanceId"))
exception = exceptionInstanceStore.findById(instanceId)
sender = friendStore.findById(exception.fromWho)
exceptionType = exceptionTypeStore.findById(exception.exceptionTypeId)
}
private fun getViews() {
exceptionNameView = findViewById(R.id.notif_full_exc_name) as TextView
exceptionDescView = findViewById(R.id.notif_exc_desc) as TextView
senderImageView = findViewById(R.id.notif_sender_image) as ImageView
senderNameView = findViewById(R.id.notif_sender_name) as TextView
sendDateView = findViewById(R.id.notif_sent_date) as TextView
mapView = findViewById(R.id.notif_map) as MapView
questionText = findViewById(R.id.notif_question_text) as TextView
noButton = findViewById(R.id.notif_question_no) as Button
yesButton = findViewById(R.id.notif_question_yes) as Button
}
private fun loadViewsWithData() {
exceptionNameView.text = exceptionType.prefix + "\n" + exceptionType.shortName
exceptionDescView.text = exceptionType.description
imageCache.setImageToView(sender, senderImageView)
senderNameView.text = viewHelper.getNameAndCity(exception, sender)
sendDateView.text = exception.date.toString()
loadQuestionToViews()
}
private fun loadQuestionToViews() {
if (exception.question.hasQuestion) {
showQuestionViews()
} else {
hideQuestionViews()
}
}
private fun showQuestionViews() {
questionText.text = exception.question.text
val listener = QuestionYesNoClickListener(exception, exceptionRestConnector, noButton, yesButton)
noButton.setOnClickListener(listener)
yesButton.setOnClickListener(listener)
}
private fun hideQuestionViews() {
questionText.visibility = View.INVISIBLE
noButton.visibility = View.INVISIBLE
yesButton.visibility = View.INVISIBLE
}
override fun onQuestionsChanged() {
onBackPressed()
}
}
最佳答案
lateinit 的相同基本功能实际上可以在 M13 之前的 Delegates.notNull 中实现。
还有其他功能也允许您绕过可空性强制执行。 !! 运算符会将可空值转换为非空值。
重点不是严格要求可空性约束,而是要使可空性成为语言中非常明确的一部分。每次您使用 lateinit 或 !! 时,您都会有意识地决定离开可空性约束的安全性,希望有充分的理由。
根据经验,最好尽可能避免lateinit、!!,甚至?(可为空)。
很多时候你可以使用 lazy 委托(delegate)以避免 lateinit 可以让你处于非空值的领域(M14 现在禁止将 val 与 lateinit 一起使用)。
您链接到的代码包含许多 lateinit View 。您可以通过执行以下操作将它们保留为非空值:
private val mapView: MapView by lazy { findViewById(R.id.notif_map) as MapView }
这将在第一次使用 mapView 时初始化该值,然后使用先前初始化的值。需要注意的是,如果您在调用 setContentView 之前尝试使用 mapView,这可能会中断。但是,这似乎没什么大不了的,您已经获得了初始化就在声明旁边的好处。
这就是 kotterknife library用于实现 View 注入(inject)。
关于android - Kotlin lateinit 属性,NPE 危险?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33023443/
我在 init block 中初始化响应但是抛出异常:/为什么? lateinit var response: MutableLiveData> init { response.valu
我想知道如何在 C# 中使用 后期初始化的 类字段和 可为空引用类型 。 想象一下下面的类: public class PdfCreator { private PdfDoc doc; p
我正在尝试使用注释@MockBean 初始化(通过模拟)两个对象 它似乎只有在我调用方法 mock(className) 时才有效,但由于我想在多个方法上使用模拟类,我不想在我的测试方法中重复相同的代
使用数据绑定(bind)、ViewModel、LiveData ( MVVM )。 在我的布局中有一个表单可以添加员工详细信息。 布局:
使用 Retrofit2 和 rxjava2 未在 Recyclerview 中设置 Gson 可转换数据,然后通过其订阅给出错误: UninitializedPropertyAccessExcept
我的 Kotlin Activity 中有 lateinit 属性,这是它的简化版本: class CreateNewListOfQuestions : AppCompatActivity() {
我有一个自定义的线性布局类,当我想创建这个类的实例时,我收到以下错误: lateinit 属性尚未初始化 我正在使用最新版本的 butterknife 库。 以下是我的 Kotlin 类: class
以下两种方法有什么区别或没有区别: 1. private lateinit var binding: ResultProfileBinding override fun onCreate(savedI
所以我想要实现的是稍后在主函数中设置顶级变量,但我不想让它成为一个肯定会破坏 Extension 的 lateinit var可变功能。 例如,此代码不起作用,因为扩展变量不支持 lateinit 修
如果我有一个 lateinit 变量,我可以检查它是否使用 (this::lateInitVar.isInitialized) 初始化,如 https://stackoverflow.com/a/47
关于这个问题已经有几个 QA,但对我来说,他们似乎在谈论不同的事情(大多数情况下是 Android 的 Kotlin + Dagger2),而不是应用我的具体案例。 我通过阅读 this blog p
成员 lateinit 变量初始化可以通过以下方式检查: class MyClass { lateinit var foo: Any ... fun doSomething()
以下 Kotlin 代码有效 val (x, y) = getSomeXYPair() 但是你不能用 lateinit 全局变量来做到这一点 class Foo { private latei
当我运行从 Firebase github 存储库下载的 quickstart-Android Firebase 数据库入门应用程序时,无论我选择(通过对话框)运行 Java 版本还是 Kotlin
您好,当我使用 Kotlin 编写 Android 程序时,我在代码中看到了 lateinit。 Java 中的等价物是什么?如何将此代码从 Kotlin 更改为 Java? public class
我想知道是否有办法检查 lateinit 变量是否已初始化。例如: class Foo() { private lateinit var myFile: File fun bar(pa
我是 Android 的新手,我正在尝试使用 Dagger2 和 MVVM 模式以及 kotlin 创建一个应用程序,但我无法让它运行。 我的应用类代码。 class PruebaDaggerApp:
我在 BreakingNewsFragment 中使用这一行 viewModel = (activity as NewsActivity).viewModel 初始化了 viewModel,但是收到以
我知道有类似的问题,但我只是找不到类似的东西,我一直在学习新的东西来学习,同时将 kotlin 合成转换为 viewvbinding模式,我遇到了这个错误 kotlin.UninitializedPr
我在尝试将环境变量值分配给 lateinit 变量时遇到错误。错误是“原始类型的属性不允许'lateinit'修饰符”。 我的application.properties(读取环境变量) my.pro
我是一名优秀的程序员,十分优秀!