事情从这里开始变得复杂起来,由于我只是个入门者,所以有很多东西都解释不了,本节的内容有一大半应该都是从官方给的示例中改的,所以如果你知道是怎么回事的话麻烦也告诉一下我!
注意本篇内容与之前的内容有关联!
首先我们要明白我们的目的:现在我有界面和数据库,但是我需要在界面中进行交互,并保证我的数据库可以按照我的想法实现相关功能,所以简单说我们的目的就是将界面和数据库建立联系。
而这个联系就要通过ViewModel来实现!本篇就先实现创建小说名称可以保存到数据库这块功能。
1、新建viewmodel:FictionNameViewModel
直接上代码:
class FictionNameViewModel
(private val fictionNameRepository: FictionNameRepository) :ViewModel() {
private var bookId: Int = 0
private val numberId: Int = 0
private var systemTimeString: String =
SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(System.currentTimeMillis())
fun insertBook(fictionName: String) {
viewModelScope.launch {
val queryJob = async {
fictionNameRepository.getLastBookId()
}
val user = queryJob.await()
if (user == null) {
bookId = 1
fictionNameRepository.insertBook(Book(numberId, bookId, fictionName))
} else {
bookId = user.fictionId + 1
fictionNameRepository.insertBook(Book(numberId, bookId, fictionName))
}
fictionNameRepository.insertChapter(Chapter(numberId, bookId, 1, "", "", systemTimeString))
}
}
}
代码分析:
- 首先是FictionNameViewModel 的传入参数:FictionNameRepository,这个函数用来进行数据库操作,再深的意思我就不懂了。
- 然后FictionNameViewModel返回值类型为ViewModel,这个没啥好说的,之前用kotlin写的是AndroidViewModel。
- 接下来是变量定义,最后那个systemTimeString是系统时间,但是从我以前的app来看好像有点问题,后续修改,先不管。
- 创建了一个插入书籍的一个函数insertBook,这里你会看到一些协程的操作
- 首先是viewModelScope.launch 这个可以开启一个协程,那么协程是干嘛的?就是为了防止ui界面无响应,开辟的后台运行的线程(?好像是这么叫)。
- 接下来是async ,这是一个有返回值的异步协程,他会返回一个Job,需要调用.await()来获取其中的值。
- 如果你的操作不需要获得返回值,那么直接在launch里写就行了。
- 这段代码主要做的事是:先查询数据库,如果没有书,那么Id=1创建一本,如果有,那么Id+1,然后再创建一本书。创建这个动作在数据库中体现为增加了一条记录。
2、创建FictionNameRepository
这个是干嘛的?我猜是用来存放操作数据库的函数的。
interface FictionNameRepository {
suspend fun insertBook(book: Book)
suspend fun updateBook(book: Book)
suspend fun insertChapter(chapter: Chapter)
//获取id最大的书
suspend fun getLastBookId(): Book?
//获取一本书的id号,用于后续删除
suspend fun getBookOfId(idF: Int): Book?
}
从代码上能看出,它是一个接口函数,那么既然是接口函数,就会有地方调用或者重写。
当然,这个接口函数中,只是对FictionNameDao 中函数的一个映射?或者说是抽离之类的,反正对比上下两段代码,他们的差别除了函数名就是注解了。
@Dao
interface FictionNameDao {
@Insert
suspend fun insertBook(book: Book)
@Update
suspend fun updateBook(book: Book)
@Insert
suspend fun insertChapter(chapter: Chapter)
//获取id最大的书
@Query("SELECT * FROM fictionData ORDER BY fiction_id DESC LIMIT 1")
suspend fun getLastBookId(): Book?
//获取一本书的id号,用于后续删除
@Query("SELECT * FROM fictionData WHERE fiction_id = :idF")
suspend fun getBookOfId(idF: Int): Book?
}
3、FictionNameRepository与FictionNameDao关联
好,那么既然我们创建了这两个函数,那就要将他俩关联起来,这时候,就需要一个新的类函数来关联这两个接口了。
class OfflineFictionNameRepository
(private val fictionNameDao: FictionNameDao) : FictionNameRepository {
override suspend fun insertBook(book: Book) = fictionNameDao.insertBook(book)
override suspend fun updateBook(book: Book) = fictionNameDao.updateBook(book)
override suspend fun insertChapter(chapter: Chapter) = fictionNameDao.insertChapter(chapter)
//获取id最大的书
override suspend fun getLastBookId(): Book? = fictionNameDao.getLastBookId()
//获取一本书的id号,用于后续删除
override suspend fun getBookOfId(idF: Int): Book? = fictionNameDao.getBookOfId(idF)
}
可以看到,通过重写FictionNameRepository的接口函数,我们就能将两个接口的操作关联起来!
而绕了这么一大圈之后呢,我们总算是将FictionNameViewModel与数据库关联了起来,这时,我们就能在其他地方对数据库进行修改了!
注意,以上三步操作都是新的文件,别放在一个文件里,我不知道放一块还能不能运行