当前位置: 首页 > 编程日记 > 正文

android专题-数据库Room

android专题-数据库Room

Room介绍

room是Google官方推荐的ORM数据库框架,抽象出sqlite访问的数据库。
包含三大组件:

  • Entity 定义 表结构,每个entity类对一个表,默认会把entity类所有字段都创建为表字段
  • Dao 定义entity类的CRUD(增删查改)功能
  • Database 数据库的访问入口,注解定义数据库的实体表,数据库版本控制,表结构导出、表结构修改等都在这里处理

步骤

1.添加依赖

 def room_version = '2.3.0-alpha04'implementation rootProject.ext.dependencies.room_runtimeannotationProcessor rootProject.ext.dependencies.room_compilerandroidTestImplementation rootProject.ext.dependencies.room_testing

2.定义Entity实体表

@Entity 注解并在@Database 注解中的 entities 属性所引用,Room就会在数据库中为这个被 @Entity 注解的类创建一张表
@PrimaryKey 每个entity至少要有一个主键(PrimaryKey),autoGenerate表示这个值自动生成
@Entity(primaryKeys = {“xxx”, “xxx”}) 如果是复合主键的话, 你需要使用注解@Entity的primaryKeys属性

@Entity(primaryKeys = {"firstName", "lastName"})
public class User {public String firstName;public String lastName;@IgnoreBitmap picture;
}

@ColumnInfo(name = “xxx”) 定义表字段,也可以不使用此注解,默认为所有属性生成表字段
@Ignore 如果Engity类中某个属性不想保存到数据库,可以使用@Ignore注解忽略

@Entity
public class User {@PrimaryKeypublic int id;public String firstName;public String lastName;@IgnoreBitmap picture;
}

@Entity(indices = {@Index(“name”), @Index(value = {“last_name”, “address”})}) 为了加速查询速度. 要给实体添加索引

@Entity(indices = {@Index("name"),@Index(value = {"last_name", "address"})})
public class User {@PrimaryKeypublic int id;public String firstName;public String address;@ColumnInfo(name = "last_name")public String lastName;@IgnoreBitmap picture;
}

@Entity(foreignKeys = @ForeignKey(entity = User.class,
parentColumns = “id”,
childColumns = “user_id”)) 设置外键

@Entity(foreignKeys = @ForeignKey(entity = User.class,parentColumns = "id",childColumns = "user_id"))
public class Book {@PrimaryKeypublic int bookId;public String title;@ColumnInfo(name = "user_id")public int userId;
}

@Embedded注解来表示一个对象

public class Address {public String street;public String state;public String city;@ColumnInfo(name = "post_code")public int postCode;
}
@Entity
public class User {@PrimaryKeypublic int id;public String firstName;@Embeddedpublic Address address;
}

demo:

package com.wrs.project.module.app.common.database.entity;
import androidx.room.Entity;
import androidx.room.PrimaryKey;
import androidx.annotation.NonNull;@Entity(tableName = "Article")
public class Article {@PrimaryKey(autoGenerate = true)private int id;private String author;private String title;private String content;private String token;private int comments;public String getAuthor() {return author;}public void setAuthor(String author) {this.author = author;}public String getTitle() {return title;}public void setTitle(String title) {this.title = title;}public String getContent() {return content;}public void setContent(String content) {this.content = content;}public int getId() {return id;}public void setId(int id) {this.id = id;}public String getToken() {return token;}public void setToken(String token) {this.token = token;}public int getComments() {return comments;}public void setComments(int comments) {this.comments = comments;}
}

3. 定义DAO

简单查询

@Dao
public interface MyDao {@Insert(onConflict = OnConflictStrategy.REPLACE)public void insertUsers(User... users);@Insertpublic void insertBothUsers(User user1, User user2);@Insertpublic void insertUsersAndFriends(User user, List<User> friends);
}

嵌套查询

@Dao
public interface MyDao {@Query("SELECT * FROM book "+ "INNER JOIN loan ON loan.book_id = book.id "+ "INNER JOIN user ON user.id = loan.user_id "+ "WHERE user.name LIKE :userName")public List<Book> findBooksBorrowedByNameSync(String userName);
}
@Dao
public interface MyDao {@Query("SELECT user.name AS userName, pet.name AS petName "+ "FROM user, pet "+ "WHERE user.id = pet.user_id")public LiveData<List<UserPet>> loadUserAndPetNames();// You can also define this class in a separate file, as long as you add the// "public" access modifier.static class UserPet {public String userName;public String petName;}
}

demo

package com.wrs.project.module.app.common.database.dao;import androidx.lifecycle.LiveData;
import androidx.room.Dao;
import androidx.room.Delete;
import androidx.room.Insert;
import androidx.room.OnConflictStrategy;
import androidx.room.Query;
import androidx.room.Update;import com.wrs.project.module.app.common.database.entity.Article;import java.util.List;@Dao
public interface ArticleDao {/*** 增* @param entities*/@Insertvoid insert(Article... entities);/*** 删* @param entities*/@Deletevoid delete(Article... entities);/*** 查* @param title*/@Query("SELECT * FROM Article WHERE title = :title")List<Article> findByTitle(String title);/*** 查* @param titles*/@Query("SELECT * FROM Article WHERE title IN (:titles)")List<Article> findInTitles(List<String> titles);/*** 改* @param entities*/@Updatevoid update(Article... entities);@Insert(onConflict = OnConflictStrategy.REPLACE)List<Long> insertAll(List<Article> entities);@Deletevoid delete(Article entity);@Query("DELETE FROM Article")void deleteAll();@Deletevoid deleteArticle(Article entity);@Query("SELECT * FROM Article")List<Article> findAll();@Query("SELECT * FROM Article")LiveData<List<Article>> loadSignList();
}

4. 定义Database

package com.wrs.project.module.app.common.database;import android.content.Context;import androidx.room.Room;
import androidx.room.RoomDatabase;
import androidx.room.TypeConverters;
import androidx.room.migration.Migration;
import androidx.sqlite.db.SupportSQLiteDatabase;import com.wrs.project.module.app.common.database.conversion.ConversionFactory;
import com.wrs.project.module.app.common.database.dao.ArticleDao;
import com.wrs.project.module.app.common.database.entity.Article;@androidx.room.Database(entities = { Article.class}, // 声明需要创建表的实体version = 2, // 数据库版本号exportSchema = true  // 表结构是否需要导出
)
@TypeConverters({ConversionFactory.class}) // 属性类型转换,这里值转换了Date类
public abstract class Database extends RoomDatabase {public abstract ArticleDao articleDao(); // 声明每个entity的dao类private static Database sInstance; // 创建访问数据库比较耗资源,生成单例方便访问public static Database getInstance(final Context context) {if (sInstance == null) {synchronized (Database.class) {if (sInstance == null) {sInstance = create(context.getApplicationContext());}}}return sInstance;}private static Database create(final Context context) {return Room.databaseBuilder(context,Database.class,"phonesign_db").addMigrations(MIGRATION_1_2) // 构建版本升级时表结构的变化.allowMainThreadQueries().build();}private static Migration MIGRATION_1_2 = new Migration(1,2) {@Overridepublic void migrate(SupportSQLiteDatabase database) {database.execSQL("ALTER TABLE Article "+ " ADD COLUMN token TEXT");database.execSQL("ALTER TABLE Article "+ " ADD COLUMN comments INTEGER NOT NULL DEFAULT 0");}};
}

5. 测试

package com.wrs.project.module.app.common;import android.content.Context;
import android.util.Log;import androidx.test.ext.junit.runners.AndroidJUnit4;
import androidx.test.platform.app.InstrumentationRegistry;import com.wrs.project.module.app.common.database.DBService;
import com.wrs.project.module.app.common.database.entity.Article;import org.junit.Test;
import org.junit.runner.RunWith;import java.util.ArrayList;
import java.util.Date;
import java.util.List;import static org.junit.Assert.assertEquals;@RunWith(AndroidJUnit4.class)
public class DatabaseInstrumentedTest {@Testpublic void useAppContext() {// Context of the app under test.Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext();AppMgr.context = appContext;AppMgr.debug = true;List<Article> list = new ArrayList<>();for (int i = 0; i < 6; i++) {Article entity = new Article();entity.setTitle("title" + i);entity.setDate(new Date());list.add(entity);}DBService.insertAll(list);List<Article> resultList = DBService.findAll();if (null != resultList && resultList.size() > 0) {for (int i = 0; i < resultList.size(); i++) {Article entity = resultList.get(i);Log.e("article", entity.getTitle());}}List<Article>  entityList = DBService.findByTitle("title5");Log.e("article", entityList.toString());}
}

项目源码:https://codechina.csdn.net/android1/projectbasic
上篇:无 目录 下篇:android专题-蓝牙扫描、连接、读写

如果觉得可以就点个👍吧,欢迎粉丝收藏,土豪打赏,您的关注就是我们创作的动力!

读者有什么想看的相关技术篇章,欢迎评论留言!

QQ交流群:908058499

相关文章:

CentOS 6.7快速搭建lamp环境

安装前要关闭防火墙&#xff0c;防止外网不能访问&#xff0c;这一点很重要&#xff0c;要不然外网访问不了&#xff1b; ①关闭防火墙&#xff1a;service iptables stop ②永久关闭防火墙&#xff1a;chkconfig iptables off ③查看防火墙状态&#xff1a;service iptables …

1005 生日礼物

1005 生日礼物 时间限制: 1 s空间限制: 128000 KB题目等级 : 黄金 Gold题解题目描述 Description9月12日是小松的朋友小寒的生日。小松知道小寒特别喜欢蝴蝶&#xff0c;所以决定折蝴蝶作为给小寒的生日礼物。他来到了PK大学最大的一家地下超市&#xff0c;在超市里&#xff0c…

css多行省略号兼容ie,css超出2行部分省略号……,兼容ie

html>Page Title.wrap {height: 40px;line-height: 20px;overflow: hidden;}.wrap .text {float: right;margin-left: -5px;width: 100%;word-break: break-all;}.wrap::before {float: left;width: 5px;content: ‘‘;height: 40px;}.text {position: relative;line-height…

机器学习01-定义、线性回归、梯度下降

目录 一、定义 What is Machine Learning 二、建模 Model Representation 三、一元线性回归 Linear Regression with One Variable 3.1 一元线性归回的符号约定 Notation 3.2 一元线性回归 Linear Regression with One Variable 3.3 代价函数 Cost Function 3.4 梯度下降…

android专题-蓝牙扫描、连接、读写

android专题-蓝牙扫描、连接、读写 概念 外围设备 可以被其他蓝牙设备连接的外部蓝牙设备&#xff0c;不断广播自身的蓝牙名及其数据&#xff0c;如小米手环、共享单车、蓝牙体重秤 中央设备 可以搜索并连接周边的外围设备&#xff0c;并与之进行数据读写通讯&#xff0c;…

2 并发编程--开启进程的两种方式

multiprocessing 英 /mʌltɪprəʊsesɪŋ/ n. [计][通信] 多重处理 1、multiprocessing 模块介绍 python中的多线程无法利用多核优势&#xff0c;如果想要充分地使用多核CPU的资源&#xff08;os.cpu\_count\(\)查看&#xff09;&#xff0c;在python中大部分情况需要使用多…

POJ 2112 Optimal Milking(二分+最大流)

POJ 2112 Optimal Milking 题目链接 题意&#xff1a;给定一些机器和奶牛&#xff0c;在给定距离矩阵&#xff0c;&#xff08;不在对角线上为0的值代表不可达&#xff09;&#xff0c;每一个机器能容纳m个奶牛。问全部奶牛都能挤上奶&#xff0c;那么走的距离最大的奶牛的最小…

ajax的loading方法,Ajax加载中显示loading的方法

使用ajaxStart方法定义一个全局的“加载中。。。”提示$(function(){$("#loading").ajaxStart(function(){$(this).html.("");});$("#loading").ajaxSuccess(function(){$(this).html.("");// $(this).empty(); // 或者直接清除});});…

机器学习02-分类、逻辑回归

目录 一、分类问题 Classification 二、分类问题的估值 Hypothesis Representation 三、分类问题的决策边界 Decision Boundary 四、分类问题的代价函数 Cost Function 五、简化的代价函数与梯度下降Simplified Cost Function & Gradient Descent 5.1 简化代价函数 …

python绘制盖尔圆并做特征值的隔离

本程序并非智能到直接运行隔离出所有特征值&#xff0c;而是需要高抬贵手&#xff0c;手动调节变换矩阵D的参数&#xff0c;以实现特征值的隔离。若期待直接找到能特征值隔离的D矩阵参数变化范围&#xff0c;怕足下要失望了&#xff0c;鄙人暂没有做到那一步&#xff0c;一是因…

mysql 电商项目(一)

mysql 电商项目 - MySQL数据库开发规范 1、数据库基本设计规范  2、索引设计规范 3、数据库字段设计规范 4、数据库SQL开发规范 5、数据库操作行为规范 转载于:https://www.cnblogs.com/Eric15/articles/9719814.html

Android专题-常用第三方框架

Android专题-常用第三方框架 HTTP网络请求 带*号的是个人推荐比较好用的 HTTP网络请求 okhttp * :https://github.com/square/okhttp retrofit:https://github.com/square/retrofit Volley:https://github.com/google/volley Android Async HTTP:https://github.com/andr…

WPF显示经常使用的几个显示文字控件TextBox, TextBlock, Lable

WPF显示经常使用的几个显示文字控件TextBox&#xff0c; TextBlock&#xff0c; Lable TextBox&#xff0c; TextBlock。 Lable 当中TextBox 和Lable均继承了Control类 能够对其进行模板编辑。而TextBlock没有继承Control所以不能对其进行模板编辑 我的程序中须要做一个二级菜单…

机器学习03-神经网络

目录 一、非线性估值Non-Linear Hypothesis 二、神经网络建模 Neural Network 三、复习逻辑回归问题矩阵式 3.1 没有进行正则化 3.2 进行正则化 四、神经网络的代价函数 4.1 符号约定Notation 4.2 代价函数 五、反向传播算法 Backpropagation Alg 5.1 任务 5.2 一个…

python 打包

一、下载 pip install Pyinstaller 二、使用Pyinstaller 1、使用下载安装的方式安装的Pyinstaller打包方式 将需要打包的文件放在解压得到的Pyinstaller文件夹中&#xff0c;打开cmd窗口&#xff0c;把路径切换到当前路径打开命令提示行&#xff0c;输入以下内容&#xff08;最…

iOS架构篇-3 网络接口封装

iOS架构篇-3 网络接口封装 关键字:iOS,网络接口封装,Alamofire,swift 网络接口API通常都需要自己封装一套管理,这里以swift版的Alamofire为例. 实现功能: 1.暴露参数请求地址url、请求方法method、请求参数params、请求头header、请求响应response(响应数据、响应头resp…

coursera 《现代操作系统》 -- 第十一周 IO系统

本周要求 错题 下列I/O控制方式中&#xff0c;哪一个不需要硬件支持&#xff1f; 中断方式 轮询方式 DMA方式 I/O处理机方式 中断方式&#xff1a;中断控制器 轮询方式&#xff1a;CPU不断查询设备以了解其是否就绪 DMA:使用到了 DMA 控制器 4。 在设备管理中&#xff0c;缓冲…

matlab图形绘制基础(东北大学MOOC笔记)

%% 二维图形绘制 % 多纵轴曲线绘制 figure(1); t 0:0.01:2*pi; y1 sin(t); y2 10*cos(t); % plotyy(t, y1, t, y2); yyaxis left plot(t, y1); ylim([min(y1), max(y1)]); yyaxis right plot(t, y2); ylim([min(y2), max(y2)]);% 绘制极坐标图 figure(2); theta 0 : 0.01 :…

【转载】pycharm远程调试配置

pycharm远程调试配置https://www.cnblogs.com/liangjiongyao/p/8794324.html

Tornado 类与类组合降低耦合

转载于:https://www.cnblogs.com/shiluoliming/p/6760548.html

matlab生成多组多维高斯分布数据

matlab生成多组多维高斯分布数据 之所以写这么一个函数&#xff0c;是因为在练习用matlab实现聚类分析&#xff0c;用matlab生成的高斯分布数据可以作为很好的数据。当然&#xff0c;直接load进鸢尾花数据集也可以拿来练手&#xff0c;到后边再对鸢尾花数据集进行分析。 代码…

Android架构篇-5 CI/CD(持续集成、持续交付、持续部署)

Android架构篇-5 CI/CD(持续集成、持续交付、持续部署) CI CI是指持续集成,代码的更新会定期自动构建、测试并合并到公共仓库中,方便多分支时解决冲突问题 CD CD是指持续交付和/或持续部署,开发人员改动代码会自动测试提交到仓库,运维实施人员将其部署到生产环境中,方…

OpenCV中图像以Mat类型保存时各通道数据在内存中的组织形式及python代码访问各通道数据的简要方式...

OpenCV中图像以Mat类型保存时各通道数据在内存中的组织形式及python代码访问各通道数据的简要方式 以最简单的4 x 5三通道图像为例&#xff0c;其在内存中Mat类型的数据组织形式如下&#xff1a; 每一行的每一列像素的三个通道数据组成一个一维数组&#xff0c;一行像素组成一个…

CV01-语义分割笔记和两个模型VGG ResNet的笔记

目录 一、语义分割 二、VGG模型 2.1 VGG特征提取部分 2.2 VGG图像分类部分 三、ResNet模型 3.1 为什么是ResNet 3.2 11卷积调整channel维度大小 3.3 ResNet里的BottleNeck 3.4 Global Average Pooling 全局平均池化 3.5 Batch Normalization 学习语义分割理论&#x…

matlab编程实现k_means聚类(k均值聚类)

1. 聚类的定义 以下内容摘抄自周志华《机器学习》 根据训练数据是否拥有标记信息&#xff0c;机器学习任务可以大致分为两大类&#xff1a;“监督学习”&#xff08;supervised learning&#xff09;和“无监督学习”&#xff08;unsupervised learning&#xff09;。分类和回…

一目了然了解JAVA集合体系

在编程中&#xff0c;常常需要集中存放多个数据。从传统意义上讲&#xff0c;数组是我们的一个很好的选择&#xff0c;前提是我们事先已经明确知道我们将要保存的对象的数量。一旦在数组初始化时指定了这个数组长度&#xff0c;这个数组长度就是不可变的&#xff0c;如果我们需…

杭电1175简单搜索 连连看

连连看 Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 34807 Accepted Submission(s): 8657 Problem Description “连连看”相信很多人都玩过。没玩过也没关系&#xff0c;下面我给大家介绍一下游戏规则&#…

IOS专栏目录

IOS 专栏目录 iOS基础篇 iOS高级篇 ios架构篇-1 项目组织架构 ios架构篇-2 国际化多语言 iOS架构篇-3 网络接口封装 iOS架构篇-4 架构模式MVVM iOS架构篇-5 CI/CD(持续集成、持续交付、持续部署) iOS专题1-蓝牙扫描、连接、读写 iOS 直播专题1-直播流程原理 iOS 直播专题2-…

CV03-双线性差值pytorch实现

一、双线性差值 1.1 公式 在理解双线性差值&#xff08;Bilinear Interpolation&#xff09;的含义基础上&#xff0c;参考pytorch差值的官方实现注释&#xff0c;自己实现了一遍。 差值就是利用已知点来估计未知点的值。一维上&#xff0c;可以用两点求出斜率&#xff0c;再…

matlab编程实现基于密度的聚类(DBSCAN)

1. DBSCAN聚类的基本原理 详细原理可以参考链接&#xff1a; https://www.cnblogs.com/pinard/p/6208966.html 这是找到的相对很详细的介绍了&#xff0c;此链接基本仍是周志华《机器学习》中的内容&#xff0c;不过这个链接更通俗一点&#xff0c;且算法流程感觉比《机器学习…