Android 账号登录

使用 Room 实现

效果图示例:

定义数据实体类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
@Entity(tableName = "Account")
data class AccountBean(
/**
* 系统会为使用 @Entity 标记的实体类创建对应的数据表
* 同时至少将一个属性设置为主键(使用 @PrimaryKey 注解)
* 这里设置 accountId 属性自增
*/
@PrimaryKey(autoGenerate = true) var accountId: Int = 0,
/**
* 系统默认生成的表名、字段名与类名、类属性名一致
* 如果想修改默认生成的表名和字段属性名,可更改 tableName 和 ColumnInfo 的属性值
*/
@ColumnInfo(name = "_loginAccount") var loginAccount: String,
@ColumnInfo(name = "_loginPassword") var loginPassword: String,
/**
* 修改数据表字段时记得必须修改数据库的版本号
* 否则会报错:
* Room cannot verify the data integrity. Looks like you've changed schema but forgot to update the version number. You can simply fix this by increasing the version number.
*/
@ColumnInfo(name = "_loginIpAddress") var loginIpAddress:String
)

定义数据库访问对象:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
@Dao
interface AccountDao {
/**
* 查询账号数据
* @param accountBean 数据实体
*/
@Insert
fun insertAccount(accountBean: AccountBean)

/**
* 查询并返回账号列表 [List<AccountBean>]
*/
@Query("select * from Account")
fun loadAccountList():List<AccountBean>

/**
* 根据登录账号 [loginAccount] 查询账号是否已经存在
*/
@Query("select * from Account where _loginAccount == :loginAccount")
fun findAccountByLoginAccount(loginAccount:String):AccountBean?

/**
* 通过账号数据 [accountBean] 去更新账号信息
*/
@Update
fun updateAccountBean(accountBean: AccountBean)

/**
* 删除账号 [accountBean] 数据信息
*/
@Delete
fun deleteAccount(accountBean: AccountBean)
}

创建 Database:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
/**
* FileName: RoomDatabase
* Founder: Jiang Houren
* Create Date: 2022/8/12 21:09
* Profile:
*
* 通过 @Database 注解声明对应实体类数组,指定数据库版本号为 1
*/
@Database(entities = [AccountBean::class], version = 2)
abstract class AccountDatabase : RoomDatabase() {
// 声明一个 AccountDao 类型的变量,
// 这样其他类就可以通过数据库轻松地访问 DAO 了
abstract val accountDao:AccountDao

/**
* 确保实例化遵循单例模式
*/
// var accountDb = Room.databaseBuilder(this,AccountDatabase::class.java,"account.db")
companion object{
// 使用 Room 提供的 databaseBuilder 构建数据库实例
// 第一个参数:上下文
// 第二个参数:对应的 DataBase 类
// 第三个参数:数据库文件的名称
val accountDb : AccountDatabase by lazy(mode = LazyThreadSafetyMode.SYNCHRONIZED) {
Room.databaseBuilder(MyApplication.applicationContext(),AccountDatabase::class.java,"account.db")
// 为了方便测试,这里添加了允许在主线程中操作的配置。 allowMainThreadQueries()
// 实际开发中,为了避免由于耗时操作导致的应用异常,数据库读写等必须放在子线程中操作。
// .allowMainThreadQueries()
// 不推荐
// 作用是允许破坏性地重新创建数据库,但是这样做会导致旧数据丢失。
// .fallbackToDestructiveMigration()
// 构建数据库时添加升级策略
// addMigrations 传入的是一个数组,如果之后数据库再升级,新增一个 Migration 策略即可。
.addMigrations(MIGRATION_1_2)
.build()
}
}
}

数据库升级:

1
2
3
4
5
6
7
8
9
10
val MIGRATION_1_2 = object : Migration(1, 2) {
/**
* 从数据库版本 1 升级到版本 2
* 新增了 _loginIpAddress 字段后,可通过 DataBase 执行对应的 SQL 语句。
* _loginIpAddress 是非空类型,并指定默认的值。
*/
override fun migrate(database: SupportSQLiteDatabase) {
database.execSQL("ALTER TABLE Account ADD COLUMN _loginIpAddress TEXT NOT NULL DEFAULT ''")
}
}

Activity:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
class AccountActivity : BaseActivity<ActivityAccountBinding>() {
private val TAG = "TAG_AccountActivity"

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)

initView()
}

private fun initView() {
mViewBinding.btnSave.setOnClickListener {
// 数据库操作是耗时操作,因此不能在主线程中进行。
// java.lang.IllegalStateException: Cannot access database on the main thread since it may potentially lock the UI for a long period of time.
thread {
val account =
AccountDatabase.accountDb.accountDao.findAccountByLoginAccount(getLoginAccount())
account?.let {
Log.d(TAG, "账号:${getLoginAccount()} 已经存在了")
if (it.loginPassword == getLoginPassword()) {
Log.d(TAG, "密码相同不用操作")
} else {
Log.d(TAG, "密码不同更新密码")
it.loginPassword = getLoginPassword()
AccountDatabase.accountDb.accountDao.updateAccountBean(it)
}
} ?: let {
Log.d(TAG, "账号:${getLoginAccount()} 不存在")
val accounBean = AccountBean(
loginAccount = getLoginAccount(),
loginPassword = getLoginPassword(),
loginIpAddress = "..."
)
AccountDatabase.accountDb.accountDao.insertAccount(accounBean)
Log.d(TAG, "账号:${getLoginAccount()} 已保存")
}
}
}

mViewBinding.btnQuery.setOnClickListener {
thread {
// loadAccountList 方法返回的是一个 List 数组,通过数组遍历将查询结果显示在 TextView 上。
val list = AccountDatabase.accountDb.accountDao.loadAccountList()
list.let {
runOnUiThread {
mViewBinding.tvLoginResult.text = ""
for (i in it.indices) {
mViewBinding.tvLoginResult.append(" id:${it[i].accountId}")
mViewBinding.tvLoginResult.append(" 账号:${it[i].loginAccount}")
mViewBinding.tvLoginResult.append(" 密码:${it[i].loginPassword}\n")
}
}
}
}
}

mViewBinding.btnDelete.setOnClickListener {
thread {
val list = AccountDatabase.accountDb.accountDao.loadAccountList()
list.let {
// 删除所有数据,
// 但保存数据时,id 还是在自增
for (i in it.indices) {
AccountDatabase.accountDb.accountDao.deleteAccount(it[i])
}
}
}
}
}

private fun getLoginPassword(): String {
return mViewBinding.etLoginPassword.text.toString()
}

private fun getLoginAccount(): String {
return mViewBinding.etLoginAccount.text.toString()
}

override fun getViewBinding(): ActivityAccountBinding {
return ActivityAccountBinding.inflate(layoutInflater)
}

/**
* 界面跳转
*/
companion object {
fun actionStart(context: Context) {
val intent = Intent(context, AccountActivity::class.java)
context.startActivity(intent)
}
}
}
activity_account.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
xmlns:app="http://schemas.android.com/apk/res-auto">

<EditText
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/et_login_account"
android:hint="请输入登录账号"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"/>
<EditText
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/et_login_password"
android:hint="请输入登录密码"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/et_login_account" />

<Button
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/btn_save"
android:text="保存"
app:layout_constraintTop_toBottomOf="@id/et_login_password"
app:layout_constraintStart_toStartOf="parent"/>
<Button
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/btn_query"
android:text="查询"
app:layout_constraintTop_toBottomOf="@id/btn_save"
app:layout_constraintStart_toStartOf="parent"/>
<Button
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/btn_delete"
android:text="删除"
app:layout_constraintTop_toBottomOf="@id/btn_query"
app:layout_constraintStart_toStartOf="parent"/>
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/tv_login_result"
app:layout_constraintTop_toBottomOf="@id/btn_delete"
app:layout_constraintStart_toStartOf="parent"/>
</androidx.constraintlayout.widget.ConstraintLayout>

备注

参考资料:

《Android Jetpack开发 原理解析与应用实战》

欢迎关注微信公众号:非也缘也