SQLite 使用(针对Android)

2023-10-30

1. SQLite 介绍


在这里插入图片描述

SQLite的注意事项:

  • 标准的主键写法:_id。(不标准写法:id),此外,主键只能是Integer类型的。
  • 像varchar,char等最终内部都会转变成text格式。

android里面的SQLite数据库是由底层的sqlite.c的代码来动态生成的。

2. SQLite可视化工具 安装


官方下载地址:https://www.sqliteexpert.com/download.html

模拟器下面的db数据库查找:
在这里插入图片描述
在这里插入图片描述

3. SQLite的增删改查

3.1 SqliteOpenHelper 创建 库或表


android系统封装了一个SqliteOpenHelper抽象工具类,来对SQLite进行增删改查。

在这里插入图片描述

示例:

实现通过点击按钮,来创建一个数据库,并且数据库中带有表格的效果。

<?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"
    tools:context=".MainActivity"
    android:orientation="vertical">

    <Button
        android:text="生成DB文件"
        android:onClick="createDB"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

</LinearLayout>

MySqliteOpenHelper

package com.example.sqlitetest;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import androidx.annotation.Nullable;

/**
 *  MySqliteOpenHelper 工具类 单例模式(1.构造函数私有化,2.对外提供函数。)
 */
// 该类就可以理解为一个数据库了。
public class MySqliteOpenHelper extends SQLiteOpenHelper {

    // 单例模式
    private static SQLiteOpenHelper mInstance;

    // 提供对外提供的函数。
    public static synchronized SQLiteOpenHelper getInstance(Context context){
        if (mInstance == null){
            mInstance = new MySqliteOpenHelper(context,"itholmes.db",null,1);
        }
        return mInstance;
    }

    /**
     * 通过构造函数,将数据库名称 , 数据库版本号等定义出来。因此,必须要声明一个构造函数。
     * @param context context就是环境,在此环境下进行创建数据库。
     * @param name 数据库名称
     * @param factory 工厂
     * @param version 版本 数据库升级后需要不断加1操作!
     */
    public MySqliteOpenHelper(@Nullable Context context, @Nullable String name, @Nullable SQLiteDatabase.CursorFactory factory, int version) {
        super(context, name, factory, version);
    }

    /**
     * 数据库初始化用的:
     *      创建表 表数据初始化 数据库第一次创建的时候调用。第二次发现有该表了就不会重复创建了!
     *      此函数只会执行一次。 因此,添加了新的数据库或者修改了数据库,必须要对应项目目录将databases目录下的db等文件删除。
     * @param sqLiteDatabase
     */
    @Override
    public void onCreate(SQLiteDatabase sqLiteDatabase) {
        // 创建表:person表。

        // 主键: _id(标准) 或者 id(不标准) 必须唯一,也可以添加自动增长 。
        String sql = "create table persons(" +
                "_id integer primary key autoincrement," +
                "name text" +
                ")";

        // 执行sql
        sqLiteDatabase.execSQL(sql);
    }

    // 数据库升级用的
    @Override
    public void onUpgrade(SQLiteDatabase sqLiteDatabase, int i, int i1) {

    }

}

MainActivity

package com.example.sqlitetest;

import androidx.appcompat.app.AppCompatActivity;

import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import android.os.Bundle;
import android.view.View;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    // 生成DB文件
    public void createDB(View view){
        // 通过MySqliteOpenHelper来操作
        SQLiteOpenHelper helper = MySqliteOpenHelper.getInstance(this);

        // databases数据库文件的创建,靠下面这句话。
        // 读
        SQLiteDatabase readableDatabase = helper.getReadableDatabase();
        // 写
        // SQLiteDatabase writableDatabase = helper.getWritableDatabase();

    }

}

数据线连接到手机,开启开发者调试也可以找到:
在这里插入图片描述

注意:因为,SQLiteOpenHelper的onCreate只会加载一次,并且如果发现该数据库已经创建就不会在执行了。因此,修改了sqlite的数据库内容,再次加载前一定要删除对应databases目录下的内容。

3.2 SqliteOpenHelper 实现 增删改查


SqliteOpenHelper 实现 增删改查效果:

activity_main.xml

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

    <Button
        android:text="生成DB文件"
        android:onClick="createDB"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

    <Button
        android:text="查询"
        android:onClick="query"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

    <Button
        android:text="插入"
        android:onClick="insert"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

    <Button
        android:text="修改"
        android:onClick="update"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

    <Button
        android:text="删除"
        android:onClick="delete"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

</LinearLayout>

MySqliteOpenHelper

package com.example.sqlitetest;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import androidx.annotation.Nullable;

/**
 *  MySqliteOpenHelper 工具类 单例模式(1.构造函数私有化,2.对外提供函数。)
 */
// 该类就可以理解为一个数据库了。
public class MySqliteOpenHelper extends SQLiteOpenHelper {

    // 单例模式
    private static SQLiteOpenHelper mInstance;

    // 提供对外提供的函数。
    public static synchronized SQLiteOpenHelper getInstance(Context context){
        if (mInstance == null){
            mInstance = new MySqliteOpenHelper(context,"itholmes.db",null,1);
        }
        return mInstance;
    }

    /**
     * 通过构造函数,将数据库名称 , 数据库版本号等定义出来。因此,必须要声明一个构造函数。
     * @param context context就是环境,在此环境下进行创建数据库。
     * @param name 数据库名称
     * @param factory 工厂
     * @param version 版本 数据库升级后需要不断加1操作!
     */
    public MySqliteOpenHelper(@Nullable Context context, @Nullable String name, @Nullable SQLiteDatabase.CursorFactory factory, int version) {
        super(context, name, factory, version);
    }

    /**
     * 数据库初始化用的:
     *      创建表 表数据初始化 数据库第一次创建的时候调用。第二次发现有该表了就不会重复创建了!
     *      此函数只会执行一次。 因此,添加了新的数据库或者修改了数据库,必须要对应项目目录将databases目录下的db等文件删除。
     * @param sqLiteDatabase
     */
    @Override
    public void onCreate(SQLiteDatabase sqLiteDatabase) {
        // 创建表:person表。

        // 主键: _id(标准) 或者 id(不标准) 必须唯一,也可以添加自动增长 。
        String sql = "create table persons(" +
                "_id integer primary key autoincrement," +
                "name text" +
                ")";

        // 执行sql
        sqLiteDatabase.execSQL(sql);
    }

    // 数据库升级用的
    @Override
    public void onUpgrade(SQLiteDatabase sqLiteDatabase, int i, int i1) {

    }

}

MainActivity

package com.example.sqlitetest;

import androidx.appcompat.app.AppCompatActivity;

import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import android.os.Bundle;
import android.util.Log;
import android.view.View;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    // 生成DB文件
    public void createDB(View view){
        // 通过MySqliteOpenHelper来操作
        SQLiteOpenHelper helper = MySqliteOpenHelper.getInstance(this);

        // databases数据库文件的创建,靠下面这句话。
        // 读
        SQLiteDatabase readableDatabase = helper.getReadableDatabase();
        // 写
        // SQLiteDatabase writableDatabase = helper.getWritableDatabase();
    }

    // sql 查询语句
    public void query(View view){
        // helper单例模式,因此随便调用。
        SQLiteOpenHelper helper = MySqliteOpenHelper.getInstance(this);
        // 查询调用getReadableDatabase就行。
        SQLiteDatabase db = helper.getReadableDatabase();
        // db.isOpen():如果db数据库是打开的,就返回true,反之返回false。
        if (db.isOpen()){
            // rawQuery方法进行查询,并且返回了一个Cursor游标。
            Cursor cursor = db.rawQuery("select * from persons", null);

            // 迭代游标
            // cursor.moveToNext()就是让游标下移,遍历数据。
            while (cursor.moveToNext()){
                // 可以直接通过索引获取。
                // int anInt = cursor.getInt(0);

                // 也可以通过cursor.getColumnIndex("_id")这种形式来获取,容错率大一点。
                int id = cursor.getInt(cursor.getColumnIndex("_id"));
                String name = cursor.getString(cursor.getColumnIndex("name"));

                Log.d("itholmes","_id:"+ id + ",name" + name);
            }

            // 一定记得关闭游标,否则耗费性能!
            cursor.close();
            // 数据库也要关闭!
            db.close();
        }
    }

    public void insert(View view) {
        // helper单例模式,因此随便调用。
        SQLiteOpenHelper helper = MySqliteOpenHelper.getInstance(this);

        // 增删改都需要调用getWritableDatabase
        SQLiteDatabase db = helper.getWritableDatabase();

        if (db.isOpen()){
            // 插入语句
            String sql = "insert into persons(name) values ('张三')";
            db.execSQL(sql);
        }

        // 关闭数据库
        db.close();
    }

    // 修改
    public void update(View view) {
        // helper单例模式,因此随便调用。
        SQLiteOpenHelper helper = MySqliteOpenHelper.getInstance(this);

        // 增删改都需要调用getWritableDatabase
        SQLiteDatabase db = helper.getWritableDatabase();

        if (db.isOpen()){
            String sql = "update persons set name = ? where _id = ?";
            // 添加参数
            db.execSQL(sql,new Object[]{"李四",1});
        }

        db.close();
    }

    // 删除
    public void delete(View view) {
        // helper单例模式,因此随便调用。
        SQLiteOpenHelper helper = MySqliteOpenHelper.getInstance(this);

        // 增删改都需要调用getWritableDatabase
        SQLiteDatabase db = helper.getWritableDatabase();

        if (db.isOpen()){
            String sql = "delete from persons where _id = ?";
            db.execSQL(sql,new Object[]{1});
        }
        db.close();
    }
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

SQLite 使用(针对Android) 的相关文章

  • 如何自定义菜单项的背景颜色?

    我正在尝试定制Toolbar的弹出菜单 现在我无法设置菜单项的背景颜色 我的 styles xml 如下所示
  • Xamarin Android Webview Javascript

    我正在尝试通过 Xamarin for Android 创建一个移动应用程序 它有一个显示网站的 WebView 问题是正常按钮会触发 但 javascript 事件不会触发 我已经启用了 Javascript 但没有运气 如何在 Andr
  • Android Studio:XML 布局中的“包装在容器中”

    编辑 XML 布局文件时 Eclipse 有一项称为 包裹在容器中 的功能 重新格式化 gt Android gt 可让您选择一个或多个视图并在其周围包裹您选择的布局 Android Studio中有类似的东西吗 目前正在实施中 问题 69
  • 按下按钮时应用不同的样式

    有没有办法在按下按钮时将样式应用于按钮 如果我有一种风格样式 xml
  • Android Q:file.mkdirs() 返回 false

    我们有一个应用程序 使用外部存储来存储一些临时文件 图像 二进制数据 该代码已经运行了几年 直到最近才发生重大变化 在 Android Q 上它不起作用 File f new File Environment getExternalStor
  • fetchUuidsWithSdp 的奇怪 UUID 逆转

    我有一个在树莓派上运行的 python 蓝牙服务器 使用 PyBluez 我在服务器中使用的uuid是 8f86d132 4ab8 4c15 b8df 0b70cf10ea56 我正在打电话device fetchUuidsWithSdp
  • Android 初学者:Android gridview 中的触摸事件

    我正在使用以下代码来使用 gridview 执行操作 稍作修改http developer android com resources tutorials views hello gridview html http developer a
  • sqlite 列名称引用(功能或错误)

    我遇到了一些对我来说看起来很奇怪的东西 但可能是我滥用了 sqlite3 create table t v 0 text insert into t values aa select from t v aa pragma table inf
  • 为什么是 javascript:history.go(-1);无法在移动设备上工作?

    首先 一些背景 我有一个向用户呈现搜索页面 html 表单 的应用程序 填写标准并单击 搜索 按钮后 结果将显示在标准部分下方 在结果列表中 您可以通过单击将您带到新页面的链接来查看单个结果的详细信息 在详细信息页面中 我添加了一个 返回结
  • 从 BroadcastReceiver 类调用活动方法

    我知道我可以做一个内部接收器类来调用接收器中的任何方法 但我的主要活动太大了 要做的事情也很多 因此 我需要一个扩展广播接收器的类 但它不是内部类 并且可以从我的主要活动中调用一种方法 我不知道是否可能 但我的活动是家庭活动和 single
  • 您使用什么物理 Android 设备进行测试?

    有什么好的推荐用于测试目的的物理 Android 设备吗 我正在苹果阵营寻找像 iPod touch 这样的设备 可以帮助 iOS 开发人员测试他们的东西 我知道有 Nexus One 但那东西相当昂贵 而且我并不真正关心手机的东西 而是可
  • 在 Samsung Galaxy S5 Android 5.0 上使用 MediaPlayer 加载音频流需要超过 10 秒

    由于更新至 Android 5 0 MediaPlayer 在 Samsung Galaxy S5 上无法正常工作 启动音频流后加载时间超过 10 秒 示例代码 MediaPlayer mPlayer new MediaPlayer Str
  • 上网本上可以进行Android开发吗? [关闭]

    Closed 这个问题是无关 help closed questions 目前不接受答案 我想使用我的上网本进行 Android 开发 但是当我尝试使用 Eclipse 运行 SDK 时 没有加载任何内容 上网本对于 Android 开发来
  • Android - 以编程方式选择菜单选项

    有没有办法以编程方式选择菜单选项 基本上 我希望视图中的按钮能够执行与按特定菜单选项相同的操作 我正在考虑尝试调用 onOptionsItemSelected MenuItem item 但我不知道要为菜单项添加什么 是的 有一种方法可以选
  • 没有用于警告的设置器/字段 Firebase 数据库检索数据填充列表视图

    我只是想将 Firebase 数据库中的数据填充到我的列表视图中 日志显示正在检索数据 但适配器不会将值设置为列表中单个列表项中的文本 它只说 没有二传手 场地插入值 这让我觉得我的设置器没有正确制作 但 Android Studio 自动
  • 没有支持 FEATURE_CAMERA_EXTERNAL 的 Android 设备

    根据this doc https source android com devices camera external usb cameras一些 Android 设备允许使用 Camera2 API 访问外部 USB 摄像头 我检查了大约
  • 在virtualenv中下载sqlite3

    我正在尝试使用命令创建应用程序python3 manage py startapp webapp但我收到一条错误消息 django core exceptions ImproperlyConfigured 加载时出错 pysqlite2 或
  • Android:如何从网络异步获取搜索建议?

    我创建了一个可搜索的活动 现在 我想添加从网络服务获取的搜索建议 我想异步获取这些建议 根据添加自定义建议 http developer android com guide topics search adding custom sugge
  • 如何使用 AccessibilityService 在 Android 中模拟按键

    我正在编写一个辅助服务 我一直在尝试在应用程序上进行一些自动搜索 我使用accessibilityservice action paste来填充EditText 然后我需要模拟软键盘上的按键 但我不知道如何做 你们能帮我一下吗 你可以尝试A
  • 发布的 Android apk 出现错误“包文件未正确签名”

    我最近将我的应用程序上传到 Android 市场 但是由于错误 下载时它拒绝运行 包文件未正确签名 我首先使用 eclipse 发布了数据包 右键单击导出 创建密钥库然后发布 但它拒绝工作 然后我下载了 keytool 和 jarsigne

随机推荐

  • (附源码)vue3.0+.NET6实现聊天室(实时聊天SignalR)

    参考文章 搭建文章 gitte源码 在线体验 可以注册两个号来测试 演示图 一 整体介绍 介绍SignalR一种通讯模型Hub 中心模型 或者叫集线器模型 调用这个模型写好的方法 去发送消息 内容有 Hub模型的方法介绍 服务器端代码介绍
  • Webpack详解

    零 文章目录 Webpack详解 1 webpack基本概念 1 webpack是什么 静态模块打包工具 官网 https webpack docschina org 官网文档 https webpack docschina org con
  • selenium 淘宝爬虫(需要扫码登录一下)

    from selenium import webdriver import time import re def gethtml url 定义函数获取html源代码 由于淘宝是动态网页无法用requests库获取所以使用selenium模拟
  • Signature expired

    django 中使用itsdangerous加密时会出现 Signature expired 签名过期的问题 解决办法 expires in设置时间稍微长一点 然后重启pycharm 重新打开项目 比如设置expires in 3600 单
  • AI+林业重塑未来,科技力量守护森林生态宝

    引言 在科技日新月异的时代 人工智能 AI 作为一项崭新的技术 正在革新着我们社会的方方面面 素有 地球之肺 美誉的森林作为人类生存发展中不可或缺的一部分 其管理与保护也开始融入AI技术 催生出了AI 林业这一新兴领域 在AI 林业的浪潮中
  • RT_thread(二)线程的操作

    文章目录 重点 一 线程是什么 二 线程的工作机制 1 线程控制块 2 线程API 1 思维图 2 创建静态线程 3启动线程 4 静态线程脱离 5 创建动态线程 6 动态线程删除 7 使线程让出处理器资源 8 使线程休眠几个节拍 9 线程节
  • Log4j日志记录

    Log4j主要由Loggers 日志记录器 Appenders 输出器 和Layout 日志格式化器 组成 其中Loggers控制日志的输出级别与日志是否输出 Appenders指定日志的输出方式 输出到控制台 文件等 Loyout控制日志
  • 计算机网络03之三种可靠传输机制

    1 停止 等待协议 为什么要有停止 等待协议 答 除了比特出现差错 底层信道还会出现丢包问题 而且为了流量控制 停止 等待协议的原理 答 停止 等待 就是每发完一个分组就停止发送 等待接收端确认 在收到确认后才发送下一个分组 停止 等待协议
  • 网页基本常用标签

    一 标签 元素 1 表示网页的开始 2 网页的头部 3
  • Docker Hub

    Docker Hub是docker的中心仓库 Docker Hub里存储了公共的 Docker 镜像 并且提供服务来帮助你构建和管理你的 Docker 环境 Docker Hub和docker及其组件一起工作 Docker Hub会帮助你和
  • 石头扫地机器人离线了怎么办_关于激光头故障,石头扫地机器人无限次复活记!...

    关于激光头故障 石头扫地机器人无限次复活记 2019 08 17 19 15 41 61点赞 174收藏 82评论 小编注 此篇文章来自即可瓜分10万金币 周边好礼达标就有 邀新任务奖励无上限 点击查看活动详情 创作立场声明 本文记录的是个
  • 【Shell牛客刷题系列】SHELL17 将字段逆序输出文件的每行:巩固awk命令与学习反向列示文件内容的tac命令

    该系列是基于牛客Shell题库 针对具体题目进行查漏补缺 学习相应的命令 刷题链接 牛客题霸 Shell篇 该系列文章都放到专栏下 专栏链接为 专栏 Linux 欢迎关注专栏 本文知识预告 首先学习了反向列示文件内容的命令 tac 这个命令
  • cocos2dx lua 读取json数据

    这里读取的数据是从excel转成json格式的数据 local sData cc FileUtils getInstance getStringFromFile filename local var json decode sData if
  • 基于STM32F103ZET6的RX8025SA时钟电路设计

    1 现在网上大多例程都是关于RX8025T的 带温度补偿的时钟芯片手册 RX8025SA与RX8025T的区别 RX8025T内置高稳定性32 768kHz的DTCXO 数字温度补偿晶体振荡器 可保证在全温度范围内的精度保持在 5ppm R
  • #program once 和 #ifndef

    program once 和 ifndef 在写小demo的时候 注意到vs中会自动生成 program once 看别人写的代码的时候见到比较多的反而是 ifndef define endif 从字面上看两种方式会产生相同的效果 避免同一
  • C语言--__attribute__((aligned(n)))

    include
  • 用户态和内核态的转换

    用户态切换到内核态的3种方式 a 系统调用 这是用户态进程主动要求切换到内核态的一种方式 用户态进程通过系统调用申请使用操作系统提供的服务程序完成工作 比如前例中fork 实际上就是执行了一个创建新进程的系统调用 而系统调用的机制其核心还是
  • 网络打印机因IP地址变动导致脱机的解决方法

    PC版本win10 某一天因为种种原因导致网络打印机的IP变动了 那么我们在自己PC机上添加的打印机就无法使用了 来看作者是怎么解决的 假设目标IP由192 168 1 16改为192 168 2 195 打开我的打印机和扫面仪菜单
  • CSDN站外文章记录收藏

    序号 c opencv相关 1 OpenCV模板匹配算法详解 2 无参考图像质量评价算法Blind Image Quality Evaluation Using Perception Based Features 3 mtf calcula
  • SQLite 使用(针对Android)

    文章目录 1 SQLite 介绍 2 SQLite可视化工具 安装 3 SQLite的增删改查 3 1 SqliteOpenHelper 创建 库或表 3 2 SqliteOpenHelper 实现 增删改查 1 SQLite 介绍 SQL