数据存储-SQLite

  一般使用到数据库存储,涉及到的数据量都较大,采用文件存储也能完成,但是文件操作复杂,效率低,大量结构化数据通常采用关系型数据库存储较为合适。Android中已经嵌入了轻量级的关系型数据库SQLite,直接按照数据库操作,实现增删改查即可。如果你想要完成一个简单增删改查作品,可以使用:UI设计+页面跳转+对话框+数据库技术架构。

  数据库操作一般分为两个步骤:创建数据库,接下来写SQL语句,执行并查看结果,常见操作为:增删改查。

第1步:配置权限(主要是因为API版本可能不同,配置一下权限较好,高版本都无需配置,我测试使用的是API34,所以我没有配置)

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

第2步:设计布局

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="用户名" />

        <EditText
            android:id="@+id/username"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="请输入用户名" />
    </LinearLayout>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="密    码" />

        <EditText
            android:id="@+id/pwd"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="请输入密码" />
    </LinearLayout>

    <Button
        android:id="@+id/add"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="增加" />

    <Button
        android:id="@+id/delete"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="删除" />

    <Button
        android:id="@+id/update"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="修改" />

    <Button
        android:id="@+id/query"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="查询" />

</LinearLayout>

设计图效果:

增加操作:输入用户名和密码,点击增加,往数据库中添加一条数据。

删除操作:根据用户输入的用户名,删除一条数据。(注:当然,如果不存在,删除肯定会失败,这些处理我就没有关注了,案例主要想表达的是能删除,如果需要做的精致或者实际开发中用到,请务必考虑周全)

修改操作:根据用户输入的用户名来更新密码,所以此时输入的密码,相当于是覆盖旧值。

查询操作:将表中的数据全部查出,并显示即可(实际开发中,结合ListView+ArrayAdapter效果更佳)

第3步:请求权限,并获得授权后,创建数据库。(如果你没有配置权限,则这一步都可以跳过了)

具体操作:在onCreate回调方法中动态请求权限,授权成功后,创建数据库

 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            requestPermissions(new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);
            File file = new File("data/data/com.yibinu.sqlitedemo/testdb.db3");
            Log.i("目录", "onCreate: " + file);
            if (file.exists() && !file.isDirectory()) {
                sqLiteDatabase = SQLiteDatabase.openOrCreateDatabase(file, null);
                String sql = "create table testdb(username varchar(10),pwd varchar(6))";
                sqLiteDatabase.execSQL(sql);
            } else {
                Log.i("===============", "onCreate: 文件不存在");
            }
        }

如果你在测试时,创建失败,还可以去指定目录位置手动创建文件testdb.db3

由于我的测试版本较高,就没有去写授权,直接使用

第4步:逻辑文件

package com.yibinu.sqlitedemo;

import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;

import android.Manifest;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.os.Build;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;

import java.io.File;

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

    Button add;
    Button delete;
    Button update;
    Button query;

    EditText username;
    EditText pwd;

    SQLiteDatabase sqLiteDatabase = null;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        add = findViewById(R.id.add);
        delete = findViewById(R.id.delete);
        update = findViewById(R.id.update);
        query = findViewById(R.id.query);
        username = findViewById(R.id.username);
        pwd = findViewById(R.id.pwd);

        delete.setOnClickListener(this);
        add.setOnClickListener(this);
        update.setOnClickListener(this);
        query.setOnClickListener(this);

        File file = new File("data/data/com.yibinu.sqlitedemo/testdb.db3");
        Log.i("目录", "onCreate: " + file);
        if (file.exists() && !file.isDirectory()) {
            sqLiteDatabase = SQLiteDatabase.openOrCreateDatabase(file, null);
            String sql = "create table testdb(username varchar(10),pwd varchar(6))";
            sqLiteDatabase.execSQL(sql);
        } else {
            Log.i("===============", "onCreate: 文件不存在");
        }
    }
    @Override
    public void onClick(View v) {
        int id = v.getId();
        if (id == R.id.add) {
            String userName = username.getText().toString().trim();
            String pWD = pwd.getText().toString().trim();
            String addSql = "insert into testdb(username,pwd) values(?,?);";
            sqLiteDatabase.execSQL(addSql, new String[]{userName, pWD});
            Toast.makeText(this, "插入数据成功", Toast.LENGTH_SHORT).show();
        } else if (id == R.id.delete) {
            String delUserName = username.getText().toString().trim();
            String deleteSql = "delete from testdb where username=?";
            sqLiteDatabase.execSQL(deleteSql, new String[]{delUserName});
            Toast.makeText(this, "删除数据成功", Toast.LENGTH_SHORT).show();
        } else if (id == R.id.update) {
            String newUserName = username.getText().toString().trim();
            String pWDD = pwd.getText().toString().trim();
            String updateSql = "update testdb set pwd=? where username = ?";
            sqLiteDatabase.execSQL(updateSql, new String[]{pWDD, newUserName});
            Toast.makeText(this, "更新数据成功", Toast.LENGTH_SHORT).show();
        } else if (id == R.id.query) {
            String queryDateSql = "select * from testdb";
            Cursor cursor = sqLiteDatabase.rawQuery(queryDateSql, null);
            while (cursor.moveToNext()) {
                String myname = cursor.getString(0);
                String mypwd = cursor.getString(1);
                Log.i("【查询结果】", "姓名:" + myname + "密码:" + mypwd);
            }
            Toast.makeText(this, "查询成功,请查看日志", Toast.LENGTH_SHORT).show();
        }
    }
}

效果

运行成功后,出现主界面

接下来看动态演示

SQLite效果

核心技术:

SQLiteDatabase类,通过该类对象,可以执行SQL语句,具体操作,通过openOrCreateDatabase方法,创建一个数据库对象,需要两个参数

public static SQLiteDatabase openOrCreateDatabase(@NonNull File file,
            @Nullable CursorFactory factory) 
  1. 数据库名
  2. 游标工厂

当然,一般方法和构造方法都可以重载,所以如果你看到有多个参数,也可以查看一下原型,就可以直接使用了。

接下来就通过数据库对象,执行SQL语句,核心方法是:

public void execSQL(String sql, Object[] bindArgs)
  1. SQL语句
  2. 参数

看个添加一行数据的案例

String userName = username.getText().toString().trim();
String pWD = pwd.getText().toString().trim();
String addSql = "insert into testdb(username,pwd) values(?,?);";
sqLiteDatabase.execSQL(addSql, new String[]{userName, pWD});

前两行,获取用户输入的用户名和密码,第3行,构造一个SQL语句,如果有参数,就使用?占位置,第4行,执行SQL语句,由于有参数,所以要把真实的数据去替换?所占据的位置。如果没有参数,就写为null,执行结束后,数据库中就已经添加上该条数据了。

修改和删除是一样的道理,查询要复杂一点,它的复杂在于,数据库的主要功能就是查询,查询结果通常是多行数据,就需要使用到游标(如果你会Oracle数据库操作,游标就非常简单了)

查询可以是带参的,也可以是不带参数的

String queryDateSql = "select * from testdb";
            Cursor cursor = sqLiteDatabase.rawQuery(queryDateSql, null);
            while (cursor.moveToNext()) {
                String myname = cursor.getString(0);
                String mypwd = cursor.getString(1);
                Log.i("【查询结果】", "姓名:" + myname + "密码:" + mypwd);
            }

游标可以简单理解为指向这个结果集,第一次移动,则来到第一行,再移动一次,就来到第二行

所以只要移动后,数据不为空,则表示移动后的位置是有一条数据的,就可以获取到对应的值

移动主要依靠moveToNext方法,读取对应的值,则使用getXXX方法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/594720.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数据库复习2

试述SQL的特点 有两个关系 S(A,B,C, D)和 T(C,D,E,F)&#xff0c;写出与下列查询等价的 SQL 表达式: 用SQL语句建立第2章习题6中的4个表&#xff1b;针对建立的4个表用SQL完成第2章习题6中的查询 针对习题4中的4个表试用SQL完成以下各项操作 (1)找出所有供应商的姓名和所在城市…

【高阶数据结构(一)】并查集详解

&#x1f493;博主CSDN主页:杭电码农-NEO&#x1f493;   ⏩专栏分类:高阶数据结构专栏⏪   &#x1f69a;代码仓库:NEO的学习日记&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学习更多Go语言知识   &#x1f51d;&#x1f51d; 高阶数据结构 1. 前言2. 并查集…

上传文件到 linux

一、mac 法一&#xff1a;scp 先进入mac的 Node_exporter文件&#xff08;要上传的文件&#xff09;目录下 输入scp -P 端口号 文件名 rootIP:/存放路径 scp -P 22 node_exporter-1.8.0.linux-amd64.tar.gz root192.***.2:/root 法二、 rz mac 安装 lrzsz&#xff0c;然后…

社交媒体数据恢复:batchat

蝙蝠聊天数据恢复方法 1. 数据恢复的基本原理 蝙蝠聊天的聊天记录一旦删除是不能够恢复的。这是因为蝙蝠聊天的聊天记录是保存于本地的&#xff0c;一旦删除&#xff0c;就如同在电脑或手机上删除文件一样&#xff0c;数据不会存储在服务器端。这意味着&#xff0c;如果你删除…

如何远程连接办公室电脑?

远程办公成为了现代工作生活的一部分&#xff0c;特别是在面对突如其来的疫情时&#xff0c;远程连接办公室电脑成为了一种常见的解决方案。通过远程连接&#xff0c;员工可以在不在办公室的情况下&#xff0c;直接访问办公室电脑上的文件和应用程序&#xff0c;实现远程协作和…

UE5(射线检测)学习笔记

这一篇会讲解射线检测点击事件、离开悬停、进入悬停事件的检测&#xff0c;以及关闭射线检测的事件&#xff0c;和射线检测蓝图的基础讲解。 创建一个简单的第三人称模板 创建一个射线检测的文件夹RadiationInspection&#xff0c;并且右键蓝图-场景组件-命名为BPC_Radiation…

阅读欣赏推荐之(六)——纪录片《阿基米德的秘密》

阿基米德是古希腊物理学家、数学家&#xff0c;静力学和流体静力学的奠基人。有人评价说除了伟大的牛顿和伟大的爱因斯坦&#xff0c;再没有一个人像阿基米德那样为人类的进步做出过这样大的贡献。即使是牛顿和爱因斯坦&#xff0c;也都曾从他身上汲取过智慧和灵感。他是“理论…

[暂未实现]APP签名不同保留数据覆盖安装记录

APP签名不同无法直接覆盖安装 使用adb可以卸载应用同时保留数据&#xff0c;但签名不同也无法覆盖安装&#xff08;安装原来签名的应用打开和卸载前一样&#xff09; 使用adb导出应用数据&#xff08;QQ&#xff09;db文件只有1kb&#xff0c;显然此方法也行不通

FreeBSD下安装Linux兼容系统Ubuntu

FreeBSD有个很神奇的功能&#xff0c;就是跟Linux二进制兼容&#xff0c;也就是可以直接运行linux的bin文件。还有个更神奇的功能&#xff0c;就是能运行出一套Linux系统&#xff0c;完全是linux的用户&#xff0c;linux的目录系统&#xff0c;而且还可以选是Centos系统还是Ubu…

在离线环境中将运行 Oracle DB 12c 的 CentOS 7.5 原地升级并迁移至 RHEL 7.9

《OpenShift / RHEL / DevSecOps 汇总目录》 说明 本文只是说明如何在 CentOS 7.5 上准备 Oracle DB 12c 验证环境&#xff0c;而将该环境升级并迁移至 RHEL 7.9 的操作过程请参见&#xff1a;《在离线环境中将 CentOS 7.5 原地升级并迁移至 RHEL 7.9》一文。 另外&#xff…

DEM(高程)数据下载及计算可见性

数据下载 下载链接: 地理空间数据云 (gscloud.cn) 数据部分介绍 ASTER是美国宇航局Terra航天器(1999年发射)上的五台仪器之一,在日本为经济产业省(METI)建造。美国/日本联合科学团队负责仪器设计、校准和数据验证。 高级星载热发射和反射辐射计(ASTER)全球数字高程…

Android BINDER是干嘛的?

1.系统架构 2.binder 源码位置&#xff1a; 与LINUX传统IPC对比

【c++设计模式15】结构型7:代理模式(Proxy Pattern)

【c设计模式15】结构型7&#xff1a;代理模式&#xff08;Proxy Pattern&#xff09; 一、定义二、适用场景三、过程四、代理模式类图五、C示例代码六、使用注意事项七、结论 类型序号设计模式描述结构型1适配器模式&#xff08;Adapter Pattern&#xff09;它用于在不修改已有…

一次完整的 http 请求是怎样的?

一次完整的 http 请求是怎样的&#xff1f; &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&#x1f496; 域名解析 --> 发起 TCP 的 3 次握手 --> 建立 TCP 连接后发起 http 请求 --> 服务器响应 http 请求&#xff0c;浏览器得到 html 代码 --…

Activating More Pixels in Image Super-Resolution Transformer

cvpr2023https://github.com/XPixelGroup/HAT?tabreadme-ov-file问题引入&#xff1a; – 现在的transformer based的SR模型“感受野”不够&#xff1b; – 分析&#xff1a;原本认为transformer-based的方法优于CNN-based的方法是因为可以利用更加long-range的信息&#xff0…

MySql数据库(概念篇)

数据库概念 什么是数据库 数据库见名之意&#xff0c;就是用来存储数据的仓库&#xff0c;是一个长期存储在计算机内的、有组织的、可共享的、统一管理的大量数据的集合。 没接触数据库之前&#xff0c;一般都是将数据存储在文件中。比如execl文件&#xff0c;word文件中。但是…

基于 Dockerfile 部署 LNMP 架构

目录 前言 1、任务要求 2、Nginx 镜像创建 2.1 建立工作目录并上传相关安装包 2.2 编写 Nginx Dockerfile 脚本 2.3 准备 nginx.conf 配置文件 2.4 生成镜像 2.5 创建 Nginx 镜像的容器 2.6 验证nginx 3、Mysql 镜像创建 3.1 建立工作目录并上传相关安装包 3.2 编写…

flink sql 优化

文章目录 一、参数方面二、资源方面三、总结 提示&#xff1a;实时flink sql 参考很多网上方法与自己实践方法汇总(版本:flink1.13) 一、参数方面 flink sql参数配置 //关闭详细算子链(默认为true),true后job性能会略微有提升。false则可以展示更详细的DAG图方便地位性能结点…

4. HBuilderX中的插件商城

前言 在HBuilderX中有一个插件市场&#xff0c;这个和VSCode的插件库不太像&#xff0c;硬要做个简单类比的话&#xff0c;个人认为HBuilderX中的插件市场更像是npm库&#xff0c;它里面有许多其他开发者开发的插件&#xff0c;这些插件更多的是为uniapp服务的&#xff0c;比如…

第23章 微内核架构软件测试(下午题)

一、微内核架构概述 &#xff08;一&#xff09;概念 1、微内核架构 微内核&#xff1a;精简的内核 宏内核&#xff1a;中央集权控制中心 核心系统 能运行的最小模块插件模块 专业处理&#xff0c;额外特性的独立组件增加/扩展核心系统的业务逻辑能力连接方式 OSGI、消息机…