当前位置: 首页 > news >正文

网站信息登记表扫描件山东集团网站建设 中企动力

网站信息登记表扫描件,山东集团网站建设 中企动力,电商课程培训,网站 会员管理在Android笔记#xff08;二十二#xff09;#xff1a;Paging3分页加载库结合Compose的实现网络单一数据源访问一文中#xff0c;实现了单一数据源的访问。在实际运行中#xff0c;往往希望不是单纯地访问网络数据#xff0c;更希望将访问的网络数据保存到移动终端的SQL…在Android笔记二十二Paging3分页加载库结合Compose的实现网络单一数据源访问一文中实现了单一数据源的访问。在实际运行中往往希望不是单纯地访问网络数据更希望将访问的网络数据保存到移动终端的SQLite数据库中使得移动应用在离线的状态下也可以从数据库中获取数据进行访问。在本笔记中将讨论多层次数据的访问即结合网络资源本地SQLite数据库中的数据的处理。在本笔记中仍然采用Android笔记二十二中的网络资源 上列展示的json数组包含了多个json对象每个json对象的格式类似下列形式 {actors:演员, directors:导演, intro:电影简介, poster:http://localhost:5000/photo/s_ratio_poster/public/p2626067725.jpg, region:地区, release:发布年份, trailer_url:https://localhost:5000/trailer/268661/#content, video_url:https://localhost:5000/d04d3c0d2132a29410dceaeefa97e725/view/movie/M/402680661.mp4}一、分层次访问数据的架构 与单一数据源结构不同在于增加了RemoteMediator。当应用的已缓存数据用尽时RemoteMediator 会充当来自 Paging 库的信号。可以使用此信号从网络加载更多数据并将其存储在本地数据库中PagingSource 可以从本地数据库加载这些数据并将其提供给界面进行显示。 当需要更多数据时Paging 库从 RemoteMediator 实现调用 load() 方法。这是一项挂起功能因此可以放心地执行长时间运行的工作。此功能通常从网络源提取新数据并将其保存到本地存储空间。 此过程会处理新数据但长期存储在数据库中的数据需要进行失效处理例如当用户手动触发刷新时。这由传递到 load() 方法的 LoadType 属性表示。LoadType 会通知 RemoteMediator 是需要刷新现有数据还是提取需要附加或前置到现有列表的更多数据。 通过这种方式RemoteMediator 可确保应用以适当的顺序加载用户要查看的数据。 二、定义实体类 1.定义Film类 Entity(tableNamefilms) data class Film(PrimaryKey(autoGenerate false)SerializedName(name)val name:String,SerializedName(release)val release:String,SerializedName(region)val region:String,SerializedName(directors)val directors:String,SerializedName(actors)val actors:String,SerializedName(intro)val intro:String,SerializedName(poster)val poster:String,SerializedName(trailer_url)val trailer:String,SerializedName(video_url)val video:String )在上述代码中将Film类映射为数据库中的数据表films。对应的数据表结构如下所示 2.定义FilmRemoteKey类 因为从网络访问每一个条电影记录需要知道记录的上一页和下一页的内容因此定义FilmRemoteKey类代码如下 Entity(tableName filmRemoteKeys) data class FilmRemoteKey(PrimaryKey(autoGenerate false)val name:String,val prePage:Int?,val nextPage:Int? )FilmRemoteKey对应的数据表结构如下 name表示电影名也是关键字 prePage表示记录的上一页的页码因为第一页的所有记录没有上一页因此前5条记录的prePage均为空 nextPage表示记录的下一页的页面。 三、定义网络访问 1.网络访问服务接口 interface FilmApi {GET(film.json)suspend fun getData(Query(page) page:Int,Query(size) size:Int):ListFilm }2.Retrofit构建网络服务 object RetrofitBuilder {private const val BASE_URL http://10.0.2.2:5000/private fun getRetrofit(): Retrofit {return Retrofit.Builder().baseUrl(BASE_URL).addConverterFactory(GsonConverterFactory.create()).build()}val apiService:FilmApi getRetrofit().create(FilmApi::class.java) }四、定义数据库的访问 1.电影数据访问对象的接口 Dao interface FilmDao {/*** 插入数据列表* param films ListFilm*/Insert(onConflict OnConflictStrategy.REPLACE)suspend fun insertAll(films: ListFilm)/*** 检索所有的Film记录* return PagingSourceInt, Film*/Query(select * from films)fun queryAll(): PagingSourceInt, Film/*** Delete all* 删除表films中所有记录*/Query(DELETE FROM films)suspend fun deleteAll() }2.电影页码数据访问对象的接口 Dao interface FilmRemoteKeyDao {Query(SELECT * FROM filmRemoteKeys WHERE name :name)suspend fun findByName(name:String):FilmRemoteKeyInsert(onConflict OnConflictStrategy.REPLACE)suspend fun insertAllKeys(remoteKeys:ListFilmRemoteKey)Query(DELETE FROM filmRemoteKeys)suspend fun deleteAllKeys() }3.创建数据库 Database(entities [Film::class,FilmRemoteKey::class], version 1) abstract class FilmDatabase : RoomDatabase() {abstract fun filmDao(): FilmDaoabstract fun filmRemoteKeyDao():FilmRemoteKeyDaocompanion object{private var instance: FilmDatabase? null/*** 单例模式创建为一个FilmDB对象实例*/Synchronizedfun getInstance(context:Context FilmApp.context): FilmDatabase {instance?.let{return it}return Room.databaseBuilder(context,FilmDatabase::class.java,filmDB.db).build()}} }五、定义代码层 1.定义RemoteMediator类 OptIn(ExperimentalPagingApi::class) class FilmRemoteMediator(private val database:FilmDatabase,private val networkService:FilmApi ) : RemoteMediatorInt, Film() {private val filmDao database.filmDao()private val filmRemoteKeyDao database.filmRemoteKeyDao()override suspend fun load(loadType: LoadType,state: PagingStateInt, Film): MediatorResult {return try{/*** 从数据库获取缓存的当前页面*/val currentPage:Int when(loadType){//UI初始化刷新LoadType.REFRESH- {val remoteKey:FilmRemoteKey? getRemoteKeyToCurrentPosition(state)remoteKey?.nextPage?.minus(1)?:1}//在当前列表头添加数据使用LoadType.PREPEND- {val remoteKey getRemoteKeyForTop(state)val prevPage remoteKey?.prePage?:return MediatorResult.Success(remoteKey!null)prevPage}//尾部加载更多的记录LoadType.APPEND-{val remoteKey getRemoteKeyForTail(state)val nextPage remoteKey?.nextPage?:return MediatorResult.Success(remoteKey!null)nextPage}}/*** 联网状态下的处理* 获取网络资源* response*/val response networkService.getData(currentPage,5)val endOfPaginationReached response.isEmpty()val prePage if(currentPage 1) null else currentPage-1val nextPage if(endOfPaginationReached) null else currentPage1database.withTransaction{//刷新记录需要删除原有的记录if(loadType LoadType.REFRESH){filmDao.deleteAll()filmRemoteKeyDao.deleteAllKeys()}//获取的记录映射成对应的索引记录val keys:ListFilmRemoteKey response.map{film:Film-FilmRemoteKey(film.name,prePage,nextPage)}filmRemoteKeyDao.insertAllKeys(keys)filmDao.insertAll(response)}MediatorResult.Success(endOfPaginationReached)}catch(e:IOException){MediatorResult.Error(e)}catch(e:HttpException){MediatorResult.Error(e)}}/*** 获取当前位置对应的FilmRemoteKey* param state PagingStateInt, Film* return FilmRemoteKey?*/private suspend fun getRemoteKeyToCurrentPosition(state:PagingStateInt,Film):FilmRemoteKey?state.anchorPosition?.let{position:Int-state.closestItemToPosition(position)?.name?.let{name:String-filmRemoteKeyDao.findByName(name)}}/*** 获取当前页面从头部第一个位置对应的FilmRemoteKey* param state PagingStateInt, Film* return FilmRemoteKey?*/private suspend fun getRemoteKeyForTop(state:PagingStateInt,Film):FilmRemoteKey?state.pages.firstOrNull{ it:PagingSource.LoadResult.PageInt,Film-it.data.isNotEmpty()}?.data?.firstOrNull()?.let{film:Film-filmRemoteKeyDao.findByName(film.name)}/*** 获取当前尾部最后一个位置对应的FilmRemoteKey* param state PagingStateInt, Film* return FilmRemoteKey?*/private suspend fun getRemoteKeyForTail(state:PagingStateInt,Film):FilmRemoteKey?state.pages.lastOrNull{it:PagingSource.LoadResult.PageInt,Film-it.data.isNotEmpty()}?.data?.lastOrNull()?.let{film:Film-filmRemoteKeyDao.findByName(film.name)} }2.定义PagingSource数据源 ExperimentalPagingApi class FilmRepository(private val filmApi:FilmApi,private val filmDatabase:FilmDatabase ) {fun getAllFilms(): FlowPagingDataFilm {val pagingSourceFactory:()-PagingSourceInt, Film {filmDatabase.filmDao().queryAll()}return Pager(config PagingConfig(pageSize 5),initialKey null,remoteMediator FilmRemoteMediator(filmDatabase,filmApi),pagingSourceFactory pagingSourceFactory).flow} }六、定义视图模型层 OptIn(ExperimentalPagingApi::class) class MainViewModel(): ViewModel() {val filmRepository:FilmRepository FilmRepository(RetrofitBuilder.apiService,FilmDatabase.getInstance())fun getFilms()filmRepository.getAllFilms() }七、定义界面层 1.单独电影界面的定义 Composable fun FilmCard(film: Film?) {Card(modifier Modifier.fillMaxSize().padding(2.dp),elevation CardDefaults.cardElevation(5.dp),colors CardDefaults.cardColors(containerColor Color.DarkGray)){Column{Row(modifier Modifier.fillMaxSize()){AsyncImage(modifierModifier.width(180.dp).height(240.dp),model ${film?.poster},contentDescription ${film?.name})Column{Text(${film?.name},fontSize 18.sp,color Color.Green)Text(导演${film?.directors},fontSize 14.sp,color Color.White)Text(演员${film?.actors}, fontSize 14.sp,color Color.Green)}}Text(${film?.intro?.subSequence(0,60)} ...,fontSize 14.sp,color Color.White)Row(horizontalArrangement Arrangement.End,modifier Modifier.fillMaxSize()){Text(More,fontSize12.sp)IconButton(onClick {}){Icon(imageVector Icons.Default.MoreVert,tint Color.Green,contentDescription 更多...)}}}} }2.定义电影列表 Composable fun FilmScreen(mainViewmodel:MainViewModel){val films mainViewmodel.getFilms().collectAsLazyPagingItems()Column(horizontalAlignment Alignment.CenterHorizontally,modifier Modifier.background(Color.White)){LazyColumn{items(films.itemCount){FilmCard(films[it])}}} }八、定义主活动MainActivity class MainActivity : ComponentActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContent {val mainViewModel:MainViewModel viewModel()Ch11_DemoTheme {// A surface container using the background color from the themeSurface(modifier Modifier.fillMaxSize(),color MaterialTheme.colorScheme.background) {FilmScreen(mainViewmodel mainViewModel)}}}} }参考文献 Paging库概览 https://developer.android.google.cn/topic/libraries/architecture/paging/v3-overview?hlzh-cn
http://www.zqtcl.cn/news/109856/

相关文章:

  • 备案期间怎么做网站微动漫怎么制作
  • 精美的php个人网站源码公司网站建设方案书
  • 站长如何做视频类网站如何诊断网站
  • 个人怎么做网站优化在线做印章的网站
  • 自己做的小网站市场营销互联网营销
  • 程序员会搭建非法网站吗永久免费生成app网页
  • 重庆渝北做网站哪里便宜浙江建设职业学校网站
  • 移动端网站开发介绍品牌宣传网站
  • 做文案图片上什么网站seo具体优化流程
  • 摄影做网站中国电商公司排名
  • 比较好的平面设计网站国际进出口贸易网站
  • 网站集约化建设困难seo技术网网
  • 咸宁网站建设报价成都网站设计 冠辰
  • 织梦网站首页目录在哪里自己编写的网站如何放到wordpress
  • 检测网站为什么打不开了推广网站有哪些方式
  • 建设银行官方网站下载安装python可以做网站吗
  • 邢台好蜘蛛网站建设郑州市网络设计公司
  • 忻府网站建设手机上做网站
  • 长宁长沙网站建设域名网站
  • 太原自助建站系统公司网站建设合同书
  • 购买深圳网站定制开发重庆学校网站建设
  • 站长素材音效下载重庆市建设工程信息网综合网
  • 网络营销专业就业方向wordpress插件dx-seo
  • 义乌有什么企业网站吗大良企业网站建设
  • 外链网站推荐几个html5标准网站建设
  • 博星卓越营销网站设计古镇镇建网站公司
  • 3. 是网站建设的重点网站后台添加关键词
  • 免费手机网站模板下载百度网站建设北京
  • 爱用建站建设网站的报告
  • 网站关键字被百度收录网站开发asp.net