在 Activity 中调用应用程序上下文

问题描述

我试图从 Android 上的活动调用我的数据库(用 Room 制作),但它需要应用程序上下文,如果我在我的活动的构造函数中传递“应用程序:应用程序”,构建崩溃并告诉我:

java.lang.class 没有零参数构造函数

这是我的代码

class PhotoActivity(application: Application) : AppCompatActivity() {


    private val pictureDao = PictureDatabase.getDatabase(app)

//Some code ....

我需要一个上下文,我试图传递“this”,但出现另一个错误

有人能给我一些帮助吗,我是这项技术的初学者

编辑:

这是我的数据库类,只是为了向您展示我为什么需要应用程序上下文


@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 的间接子代。

您可以通过以下方式进行,

  1. private val pictureDao by lazy{ PictureDatabase.getDatabase(this) }
  2. 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)