问题描述
我试图从 Android 上的活动调用我的数据库(用 Room 制作),但它需要应用程序上下文,如果我在我的活动的构造函数中传递“应用程序:应用程序”,构建崩溃并告诉我:
java.lang.class
这是我的代码:
class PhotoActivity(application: Application) : AppCompatActivity() {
private val pictureDao = PictureDatabase.getDatabase(app)
//Some code ....
有人能给我一些帮助吗,我是这项技术的初学者
编辑:
这是我的数据库类,只是为了向您展示我为什么需要应用程序上下文
@Database(entities = [Picture::class],version = 1,exportSchema = false)
abstract class PictureDatabase : RoomDatabase(){
abstract fun pictureDao() : PictureDao
companion object{
@Volatile
private var INSTANCE : PictureDatabase? = null
fun getDatabase(context: Context): PictureDatabase {
if(INSTANCE == null){
synchronized(this){
INSTANCE = Room.databaseBuilder(
context.applicationContext,PictureDatabase::class.java,"pictures.db"
).build()
}
}
return INSTANCE!!
}
}
}
解决方法
首先创建一个ViewModel,在ViewModel里面你可以访问你的Dao,pictureDao。
class PictureViewModel(application: Application) :
AndroidViewModel(application) {
val pictureDao: PictureDao
init {
pictureDao =
PictureDatabase.getDatabase(application).pictureDao()
}
}
然后在您的活动中,初始化 ViewModel。并访问您的 Dao。
class PhotoActivity : AppCompatActivity() {
private lateinit var pictureViewModel: PictureViewModel
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_photo)
pictureViewModel =
ViewModelProviders.of(this).get(PictureViewModel::class.java)
//You can now access your Dao class here:
val pictureDao = pictureViewModel.pictureDao
}
}
,
Activity 是我们在清单中声明的东西,然后使用 intent
启动它们,但是,Activity 实例的创建是由 system
完成的,而不是由我们完成的。一个实例是使用 constructor
创建的,但如果是我们,那么我们可以有任意数量的重载构造函数。但是系统只需要一个构造函数,它应该是一个零参数构造函数,并且它应该是公共的。
所以你的活动签名
class PhotoActivity(application: Application) : AppCompatActivity() {
应该改为
class PhotoActivity() : AppCompatActivity() {
要调用 fun getDatabase(context: Context): PictureDatabase
,您可以从 this
传递 activity
。 Activity 是 Context
的间接子代。
您可以通过以下方式进行,
private val pictureDao by lazy{ PictureDatabase.getDatabase(this) }
-
private lateinit var pictureDao:PictureDatabase
然后在onCreate()
中初始化它
final override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.your_layout)
PictureDatabase.getDatabase(this)
//your logic goes here
}
,
您不应将 Application
传递给构造函数。您应该将 applicationContext
传递给 getDatabase()
,例如 private val pictureDao = PictureDatabase.getDatabase(applicationContext)