Lin*_*ngo 4 android kotlin dagger-2 android-room
我想向房间数据库添加回调以填充初始数据。
@Provides
@Singleton
fun provideRoom(context: Context): MyRoomDatabase {
return Room.databaseBuilder(context, MyRoomDatabase::class.java, "my_database")
.fallbackToDestructiveMigration()
.addCallback(object : RoomDatabase.Callback() {
@Override
override fun onCreate(db: SupportSQLiteDatabase) {
super.onCreate(db)
}
})
.build()
}
Run Code Online (Sandbox Code Playgroud)
为此,我需要回调中的数据库实例来访问 DAO 以插入数据。这是如何运作的?
编辑:
我想要达到的目标:
Create initial data for the room database at the app installation
我的Callback Class
:
class RoomCallback(
var myRoomDatabase : MyRoomDatabase
) : RoomDatabase.Callback() {
override fun onCreate(db: SupportSQLiteDatabase) {
myRoomDatabase.basicItemDao().insertList(
listOf(
BasicItem(),
BasicItem()
)
)
}
Run Code Online (Sandbox Code Playgroud)
}
我如何提供RoomDatabase
和RoomCallback
:
@Provides
@Singleton
fun provideRoom(context: Context, roomCallback: RoomCallback): MyRoomDatabase {
return Room.databaseBuilder(context, MyRoomDatabase::class.java, "my_database")
.fallbackToDestructiveMigration()
.addCallback(roomCallback)
.build()
}
@Provides
@Singleton
fun provideRoomCallback(myRoomDatabase: MyRoomDatabase): RoomCallback {
return RoomCallback(myRoomDatabase)
}
Run Code Online (Sandbox Code Playgroud)
问题:
-RoomCallback
和RoomDatabase
实例需要另一个实例。
更新:使用Kotlin Coroutine
和Dagger2
晚会但对于未来的读者来说,在创建时或打开时预填充数据库非常容易。确保您已经在gradle
文件中为Coroutine
. 首先创建您的数据库,如:
/**
* Main database.
*/
@Database(
entities = [
Login::class],
version = 1,
exportSchema = false
)
abstract class AppDatabase : RoomDatabase() {
abstract fun loginDao(): LoginDao
companion object {
@Volatile private var INSTANCE: AppDatabase? = null
fun getInstance(app: Application): AppDatabase = INSTANCE ?: synchronized(this) {
INSTANCE ?: buildDatabase(app).also { INSTANCE = it }
}
private fun buildDatabase(app: Application) =
Room.databaseBuilder(app,
AppDatabase::class.java,
"your database name")
.addCallback(object : Callback() {
// Pre-populate the database after onCreate has been called. If you want to prepopulate at opening time then override onOpen function instead of onCreate.
override fun onCreate(db: SupportSQLiteDatabase) {
super.onCreate(db)
// Do database operations through coroutine or any background thread
val handler = CoroutineExceptionHandler { _, exception ->
println("Caught during database creation --> $exception")
}
CoroutineScope(Dispatchers.IO).launch(handler) {
prePopulateAppDatabase(getInstance(app).loginDao())
}
}
})
.build()
suspend fun prePopulateAppDatabase(loginDao: LoginDao) {
val admin = Login(0, "Administrator", "1234", 1, isActive = true, isAdmin = true, isLogged = false)
loginDao.insertLoginData(admin)
}
}
}
Run Code Online (Sandbox Code Playgroud)
然后,您可以根据需要通过将以下代码放置到您的 daggerAppModule
或单独的数据库模块中来提供单例实例。
@Singleton
@Provides
fun provideDb(app: Application): AppDatabase {
return AppDatabase.getInstance(app)
}
@Singleton
@Provides
fun provideLoginDao(db: AppDatabase): LoginDao {
return db.loginDao()
}
Run Code Online (Sandbox Code Playgroud)
就是这样,你完成了。将单例数据库对象注入到任何地方,例如:
@Inject lateinit var loginDao: LoginDao
Run Code Online (Sandbox Code Playgroud)
然后使用它。
首先设置数据库
@Database(
entities = [User::class],
version = VERSION_CODE
)
abstract class DatabaseManager : RoomDatabase() {
abstract fun userDao(): UserDao
}
Run Code Online (Sandbox Code Playgroud)
现在创建一个 DatabaseModule
@Module
class DatabaseModule {
@Singleton
@Provides
fun provideRoomDatabase(@ApplicationContext context: Context): RoomDatabase {
return Room.databaseBuilder(context, RoomDatabase::class.java, "dbName")
.setJournalMode(JournalMode.TRUNCATE)
.build()
}
}
Run Code Online (Sandbox Code Playgroud)
您可以创建一个单独的模块或在DatabaseModule
其中添加一个自提供 dao 对象的方法。比如说我有一个UserDao
then
@Module
class UserModule {
@Singleton
@Provides
fun provideDao(database: DatabaseManager): UserDao {
return database.userDao()
}
}
Run Code Online (Sandbox Code Playgroud)
归档时间: |
|
查看次数: |
2884 次 |
最近记录: |