- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
这是一个自我回答的问题,因为我在我的项目中遇到了代表曝光问题,但在经过几个小时的研究后最终解决了这个问题。我没有保持沉默,而是认为这可能会在 future 帮助某人。本教程演示了如何创建 Room 数据库并在 Activity/fragment 中使用它。此处给出的示例用例是查询数据库的大小并更新 fragment 中的 View 。
注意:下面的代码中有一些 Dagger-Hilt 依赖注入(inject),但如果您手动执行自己的依赖注入(inject),则应该应用相同的方法。我也希望你对 MVVM 架构有一些基本的了解。如果您对涉及 LiveData 的其他方法感兴趣,您可以在这里找到有用的 Java 相关问题:resource 1 , resource 2 ;然而,重点是 Kotlin,这个解决方案不需要 LiveData。
最佳答案
您必须将项目中的 kotlin 文件关联起来,因为项目的包是结构化的,但导入应该保持不变。在这种情况下,我使用 Dagger-Hilt 进行依赖注入(inject)以避免样板代码。
ItemsYouAreStoringInDB.kt
import androidx.room.Entity
import androidx.room.PrimaryKey
@Entity(tableName = "items")
data class ItemsYouAreStoringInDB(/*Parameter of Item entry*/) {
@PrimaryKey(autoGenerate = true)
var id: Int? = null
}
YourDao.kt
import androidx.room.*
@Dao
interface YourDAO {
// Other insertion/deletion/query operations
@Query("SELECT count(id) FROM items") // items is the table in the @Entity tag of ItemsYouAreStoringInDB.kt, id is a primary key which ensures each entry in DB is unique
suspend fun numberOfItemsInDB() : Int // suspend keyword to run in coroutine
}
你的数据库.kt
import androidx.room.Database
import androidx.room.RoomDatabase
@Database(
entities = [ItemsYouAreStoringInDB::class], // Tell the database the entries will hold data of this type
version = 1
)
abstract class YourDatabase : RoomDatabase() {
abstract fun getYourDao(): YourDAO
}
使用 Dagger-Hilt 进行依赖注入(inject),可以创建 YourRepository,因为 Dagger-Hilt 在后台通过 YourDatabase 的抽象乐趣 getYourDao() 提供 notificationDao
import path.to.ItemsYouAreStoringInDB
import path.to.YourDAO
import javax.inject.Inject // Dagger-Hilt to allow @Inject constructor
class YourRepository @Inject constructor(
private val yourDAO: YourDAO
){
// Other functions from YourDao.kt
suspend fun numberOfItemsInDB() = yourDAO.numberOfItemsInDB()
}
这不是关于如何使用 Dagger-Hilt 的演示,但需要以下两个文件:
import android.content.Context
import androidx.room.Room
import dagger.Module
import dagger.Provides
import dagger.hilt.InstallIn
import dagger.hilt.components.SingletonComponent
import dagger.hilt.android.qualifiers.ApplicationContext
import path.to.YourDatabase
import javax.inject.Singleton
@Module
@InstallIn(SingletonComponent::class)
object AppModule {
@Singleton // Tell Dagger-Hilt to create a singleton accessible everywhere in ApplicationCompenent (i.e. everywhere in the application)
@Provides
fun provideYourDatabase(
@ApplicationContext app: Context
) = Room.databaseBuilder(
app,
YourDatabase::class.java,
"your_db_name"
).build() // The reason we can construct a database for the repo
@Singleton
@Provides
fun provideYourDao(db: YourDatabase) = db.getYourDao() // The reason we can implement a Dao for the database
BaseApplication.kt
import android.app.Application
import dagger.hilt.android.HiltAndroidApp
@HiltAndroidApp
class BaseApplication : Application() {}
您还需要更新 AndroidManifest 文件并选择 BaseApplication 作为应用程序入口点
<application android:name="path.to.BaseApplication" ...
允许 Android 利用 Dagger-Hilt。
import dagger.hilt.android.lifecycle.HiltViewModel
import androidx.lifecycle.ViewModel
import path.to.YourRepository
@HiltViewModel
class MainViewModel @Inject constructor(
private val repository: YourRepository
): ViewModel() {
suspend fun databaseSize() : Int {
return repository.numberOfItemsInDB()
}
}
现在您的 View 模型可以创建并且可以在整个应用程序中作为单例访问(不能存在两个实例),您可以在 fragment/Activity 中使用它。 viewmodel 可以访问存储库,该存储库可以通过查询 Room 数据库来接收信息。这是一个如何在 fragment 中使用它的示例:
@AndroidEntryPoint // Dagger-Hilt requirement
class YourFragment : Fragment(R.layout.fragment_yourFragmentName) {
private val viewModel: MainViewModel by viewModels()
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
setViewsBasedOnRepo() // You might want to call this in onResume()
}
private fun setViewsBasedOnRepo() {
GlobalScope.launch(Dispatchers.Main) { // Dispatchers.Main because only the Main thread can touch UI elements. Otherwise you may wish to use Dispatchers.IO instead!
val size =
withContext(Dispatchers.Default) { viewModel.databaseSize() }
if (size == 0) { // Do stuff based on an empty database
btnAddItemsToDB.visibility = View.VISIBLE
textViewWarnNoItemsInDB.visibility = View.VISIBLE
recyclerViewItems.visibility = View.INVISIBLE
} else { // Do other stuff when database has entries of type ItemsYouAreStoringInDB
btnAddItemsToDB.visibility = View.INVISIBLE
textViewWarnNoItemsInDB.visibility = View.INVISIBLE
rvNotifications.visibility = View.VISIBLE
}
}
}
}
关于android - 如何在 Kotlin [Dagger-Hilt] 中创建和使用房间数据库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63146318/
我在使用 Room 或 idk 时遇到问题,实际上问题出在哪里,我需要帮助找出问题出在哪里,我正在使用 Hilt DI,创建数据库实例的那一刻它崩溃了这是我的代码 错误 E/AndroidRuntim
我有一个关于数据结构和类设计的问题(抱歉太长了)。为了简单起见,假设这是一个游戏,我想在房间之间导航(想象一系列 2D 非滚动屏幕,例如早期的银河战士/恶魔城)。每个房间可以有许多导出(例如上、下、左
我使用当前日期(1-25)作为父ID并使用房间(08-00_11-00_karpet1-)作为 child ID。该数据库中包含在该日期(父 ID)订购该房间(子 id)的用户信息。 问题1 使用此布
在我的 Android 项目中,我使用 Room 库来处理 SQLite 数据库。我使用我的数据库来保存国家电话代码。我的数据库预装了两个国家(观看 populateDatabaseWithCount
我正在尝试将 Room 持久性库添加到 Android 应用程序项目中。在 build.gradle 文件中,我添加了以下依赖项: implementation 'android.arch.persi
人们可以提前从 25 场讲座中选择最多 5 场。所有这些讲座都在五个房间的五个时间段内在一天内进行。听众可以参加的每个(首选)讲座都让她更快乐,他选择但不能参加的每个讲座(因为另一个首选讲座在同一时间
我在 Android 上使用 OrmLite 而不是 SQLite 和 SQLCipher 来加密数据库。有没有办法加密 Room 数据库? 最佳答案 默认情况下,Room 将数据存储在应用程序的内部
使用 Room ORM,我使用 @Entity 注释声明了一个实体 EQPreset。该实体包含一个数组 int[]。它给出以下错误: 错误:无法确定如何将此字段 (int[] arr) 保存到数据库
我正在尝试构建一个管理 child 托儿所的应用程序,特别是管理哪个 child 在哪个时间点在托儿所的哪个房间里。 Nursery 链式店有多个分支机构。每个分店有几个房间,每个房间对应一个年龄段,
我在生产环境中遇到了“android.database.sqlite.SQLiteDatabaseLockedException”异常。错误分析时出现异常。我的项目数据库有空间。项目中没有使用多进程。
我想实现 Android Room 持久性。 这是我的 DAO 界面。 @Dao interface FoodDao { /** * Returns all data in tabl
我正在尝试使用 Room 数据库和 LiveData。我有 ViewModels,它保存从 dao 获得的 LiveData。如果我更新Transaction ,然后LiveData>观察正常,但是
在 Firebase ,创建“房间”(例如用于聊天)很容易,正如其各种示例中所记录的那样。 对于聊天的数据结构,我会使用这样的东西: rooms room1 member_co
我试图从 Activity 中将一行插入 SQLITE 数据库,然后返回要存储在 Activity 中的变量中的 rowId。请参阅下面我使用的方法和逻辑。 private void insert
我正在使用 XMPPFramework 开发聊天应用程序 加入现有房间后如何接收消息历史记录? 现在我像这样加入房间: XMPPJID *roomJid = [XMPPJID jidWithStrin
我在我的应用程序中使用 Room 并将数据插入到我的数据库中时 ConcurrentModificationException有时会被抛出。为什么会这样? 我使用分页 api,在每次 api 调用后,
我想为 pb 添加值,由于将 pb_value 包含到实体中,应用程序崩溃了。我是学习室的新手,我不确定将额外项目合并到数据库中的正确方法。 E/AndroidRuntime: FATAL EXCEP
我想在当前 pb 中添加值、日期和详细信息。我在 pbInfo 的数据库中收到错误“冲突声明”。我应该如何修复此错误? @Entity(tableName = "pb_table") data cla
我正在尝试制作一个聊天应用程序,用户可以在其中聊天。我想将两个用户 uid 字符串插入函数并返回一个连接的字符串。但我希望以某种方式组织 uid,以便返回的值始终相同。 func (id1, id2)
我不断收到以下错误: Cannot figure out how to save this field into database. You can consider adding a type co
我是一名优秀的程序员,十分优秀!