xamarin和mysql_Xamarin.Android 使用 SQLiteOpenHelper 进行数据库操作

一、前言

在手机中进行网络连接不仅是耗时也是耗电的,而耗电却是致命的。所以我们就需要数据库帮助我们存储离线数据,以便在用户未使用网络的情况下也可以能够使用应用的部分功能,而在需要网络连接的功能上采用提示方式,让用户决定是否打开网络。而本节我们将会学习如何访问数据库以及提供基本的增删改查功能,并且使他们尽量的解耦。

二、数据库

Xamarin.Android下创建本地数据库与在Java下的方式相同,而我们必须掌握使用SQLiteOpenHelper,因为这个类会简化我们创建数据的步骤,让我们只需要关注创建数据库中的表,并在数据库版本需要更新时进行操作。其中我们必须实现OnCreate方法和OnUpgrade方法,OnCreate方法仅会在数据库不存在的情况下才执行,所以不会重复执行。比如下面的代码。

48304ba5e6f9fe08f3fa1abda7d326ab.png

1 class LocationSqliteOpenHelper : SQLiteOpenHelper

2 {

3 public override void OnCreate(SQLiteDatabase db)

4 {

5 }

6

7 public override void OnUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)

8 {

9 }

10 }

48304ba5e6f9fe08f3fa1abda7d326ab.png

但是我们还需要使用父类的构造函数,指定数据库的名称以及初始版本。比如下面的代码我们将创建一个名为“test”的数据,并且初始版本为1.。

48304ba5e6f9fe08f3fa1abda7d326ab.png

1 class LocationSqliteOpenHelper : SQLiteOpenHelper

2 {

3 public LocationSqliteOpenHelper(Context context)

4 : base(context, “test”, null,1)

5 {

6 }

7 }

48304ba5e6f9fe08f3fa1abda7d326ab.png

学会了上面的操作,下面我们就可以创建一个名为Test的数据库,并且该数据库中含有一个USER表(SQLite数据库下的主键需要为INTEGER类型,并且是自增的)。

961ddebeb323a10fe0623af514929fc1.png

48304ba5e6f9fe08f3fa1abda7d326ab.png

1 public class TestSQLiteOpenHelper : SQLiteOpenHelper

2 {

3 public TestSQLiteOpenHelper(Context context)

4 : base(context, "Test", null, 1)

5 {

6 }

7

8 public override void OnCreate(SQLiteDatabase db)

9 {

10 db.ExecSQL("CREATE TABLE USER(id INTEGER PRIMARY KEY NOT NULL,uname TEXT NOT NULL,upwd TEXT NOT NULL)");

11 }

12

13 public override void OnUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)

14 {

15 db.ExecSQL("DROP TABLE IF EXISTS USER");

16 OnCreate(db);

17 }

18 }

48304ba5e6f9fe08f3fa1abda7d326ab.png

创建了数据库对象,下面我们就可以利用这个对象对数据库进行操作了,首先我们需要在MainActivity中的OnCreate方法中初始化该数据库对象。

TestSQLiteOpenHelper dbHelper = new TestSQLiteOpenHelper(this);

但是我们还不能直接使用dbHelper访问数据库,必须通过它的WritableDatabase属性或ReadableDatabase属性获取对应权限的数据库访问对象,WritableDataBase可以对数据库进行全部操作,ReadableDatabase可以对数据库进行读取操作。他们的返回类型都是SQLiteDataBase。所以我们还要根据需要获取他们的对象。

SQLiteDatabase db = dbHelper.WritableDatabase;

这样我们就可以通过db的Insert、Update、Query和Delete进行操作了,当然也可以使用ExecSQL直接执行我们SQL语句。下面我们将逐一介绍这些方法的使用。

1.添加(Insert)

首先是该方法的定义:

public virtual long Insert(string table, string nullColumnHack, ContentValues values);

其中参数的含义如下:

table:需要插入的表名。

nullColumnHack:当values为空或里面的值都为空时,数据库是不允许插入一个空行的,如果需要插入空行,则需要指定一个字段名称,这样当发生如上情况后将会将该字段设为NULL然后在尝试插入。

values:需要插入的数据。

关于前两个参数很简单不用过多介绍,如要介绍的是最后一个参数,它是一个ContentValues类型,通过它我们可以大大的简化自己拼接插入语句的繁琐,比如下面我们可以设置uname字段的值为yzf,upwd的值为123。

1 ContentValues cv = new ContentValues();

2 cv.Put("uname","yzf");

3 cv.Put("upwd","123");

关键就是Put方法,它拥有以下的重载方法。

48304ba5e6f9fe08f3fa1abda7d326ab.png

1 public void Put(string key, bool value);

2 public void Put(string key, byte[] value);

3 public void Put(string key, double value);

4 public void Put(string key, float value);

5 public void Put(string key, int value);

6 public void Put(string key, long value);

7 public void Put(string key, sbyte value);

8 public void Put(string key, short value);

9 public void Put(string key, string value);

48304ba5e6f9fe08f3fa1abda7d326ab.png

通过这些重载方法我们就可以插入不同类型的参数了,当然我们也可以通过Remove方法删除,如果我们需要为某个字段插入NULL值可以使用PutNull方法,判断某个字段是否存在可以用ContainsKey方法,最后就是对应的获取不同字段的值。

48304ba5e6f9fe08f3fa1abda7d326ab.png

1 public Object Get(string key);

2 public bool GetAsBoolean(string key);

3 public sbyte GetAsByte(string key);

4 public byte[] GetAsByteArray(string key);

5 public double GetAsDouble(string key);

6 public float GetAsFloat(string key);

7 public int GetAsInteger(string key);

8 public long GetAsLong(string key);

9 public short GetAsShort(string key);

10 public string GetAsString(string key);

48304ba5e6f9fe08f3fa1abda7d326ab.png

简单的介绍完ContentValues的使用,下面我们将使用它来添加一条数据,比如下面的代码将添加一条数据到User表中。

long id = db.Insert("User", null, cv);

返回值则为所插入数据的主键。

2.查询(Query)

首先是该方法的定义:

48304ba5e6f9fe08f3fa1abda7d326ab.png

1 public virtual ICursor Query(string table, string[] columns, string selection, string[] selectionArgs, string groupBy, string having, string orderBy);

2 public virtual ICursor Query(string table, string[] columns, string selection, string[] selectionArgs, string groupBy, string having, string orderBy, string limit);

3 public virtual ICursor Query(bool distinct, string table, string[] columns, string selection, string[] selectionArgs, string groupBy, string having, string orderBy, string limit);

48304ba5e6f9fe08f3fa1abda7d326ab.png

其中参数的含义如下:

table:需要查询的表名

columns:需要获取的字段,如果传入null则表示获取所有字段

selection:条件语句,其中我们可以实用”?”作为参数的占位符(不同于SQL SERVER中的@)

selectionArgs:条件参数,用于替换查询语句中的”?”

groupBy:分组语句

having:分组条件

orderBy:排序语句

limit:分页语句(如”1,3”表示获取第1到第3的数据共3条)

之前通过Insert插入的数据,此时我们可以通过Query方法从数据库中获取,比如下面的代码

ICursor ic = db.Query("User", new string[] { "id", "uname", "upwd" }, " id = ? ", new string[] { id.ToString() }, null, null, null);

该方法最后会返回一个实现了ICursor接口的对象,利用这个接口我们就可以从中获取数据了,下面我们获取其中的用户名和密码

1 ic.MoveToFirst();

2 string uname = ic.GetString(ic.GetColumnIndex("uname"));

3 string upwd = ic.GetString(ic.GetColumnIndex("upwd"));

因为ICursor是针对一个结果集的,所以我们需要先定位到第一条数据,所以采用MoveToFirst方法,然后通过GetString获取参数,但是还需要传递一个字段的位置,所以我们还需要使用GetColumnIndex获取指定字段名称的位置。

下面是关于ICursor方法的介绍

Count:获取多少条数据

IsAfterLast:当前是否在最后一条数据之后

IsBeforeFirst:当前是否在第一条数据之前

IsClosed:是否已关闭

IsFirst:是否是第一条数据

IsLast:是否是最后一条数据

Position:当前位置

GetColumnIndex:根据字段名获取位置,如果不存在该字段则返回-1

GetColumnName:根据位置获取字段名

MoveToFirst:移动到第一条数据

MoveToFirst:移动到最后一条数据

MoveToNext:移动到下一条数据

MoveToPosition:移动指定的位置

MoveToPrevious:移动到上一条数据

以下是根据位置获取对应类型的数据

GetDouble,GetFloat,GetInt,GetLong,GetShort,GetString

3.更新(Update)

首先是该方法的定义:

Update(string table, ContentValues values, string whereClause, string[] whereArgs);

其中参数的含义如下

table:需要更新的数据所在的表

values:更新后字段的值

whereClause:查询语句

whereArgs:查询语句中需要的参数

有了插入、查询数据的帮助下,我们可以在插入数据之后更新这条数据,然后再通过Query获取该数据,查看数据的是否变动。

1 ContentValues ncv = new ContentValues();

2 ncv.Put("uname", "zn");

3 ncv.Put("upwd", "456");

4 db.Update("User", ncv, " id = ? ", new string[] { id.ToString() });

这里的条件语句跟查询中的语句是类似的,然后我们查看获取的数据可以发觉的确发生了修改。

4.删除(Delete)

首先是该方法的定义:

public virtual int Delete(string table, string whereClause, string[] whereArgs);

关于参数的说明跟Update是相同的,所以实用方式这里就不做介绍了。

全部实例的全部代码如下所示:

TestSQLiteOpenHelper.cs

48304ba5e6f9fe08f3fa1abda7d326ab.png

1 public class TestSQLiteOpenHelper : SQLiteOpenHelper

2 {

3 public TestSQLiteOpenHelper(Context context)

4 : base(context, "Test", null, 1)

5 {

6 }

7

8 public override void OnCreate(SQLiteDatabase db)

9 {

10 db.ExecSQL("CREATE TABLE USER(id INTEGER PRIMARY KEY NOT NULL,uname TEXT NOT NULL,upwd TEXT NOT NULL)");

11 }

12

13 public override void OnUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)

14 {

15 db.ExecSQL("DROP TABLE IF EXISTS USER");

16 OnCreate(db);

17 }

18 }

48304ba5e6f9fe08f3fa1abda7d326ab.png

MainActivity的OnCreate中

48304ba5e6f9fe08f3fa1abda7d326ab.png

1 TestSQLiteOpenHelper dbHelper = new TestSQLiteOpenHelper(this);

2 SQLiteDatabase db = dbHelper.WritableDatabase;

3

4 ContentValues cv = new ContentValues();

5 cv.Put("uname","yzf");

6 cv.Put("upwd","123");

7 long id = db.Insert("User", null, cv);

8

9 ContentValues ncv = new ContentValues();

10 ncv.Put("uname", "zn");

11 ncv.Put("upwd", "456");

12 db.Update("User", ncv, " id = ? ", new string[] { id.ToString() });

13

14 ICursor ic = db.Query("User", new string[] { "id", "uname", "upwd" }, " id = ? ", new string[] { id.ToString() }, null, null, null);

15 ic.MoveToFirst();

16 string uname = ic.GetString(ic.GetColumnIndex("uname"));

17 string upwd = ic.GetString(ic.GetColumnIndex("upwd"));

48304ba5e6f9fe08f3fa1abda7d326ab.png

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python 绝对值误差小于10-6_Python 被低估了的 10 个小技巧

hi,各位朋友们,小帅b回来啦,几日不见,想我了么?今天给大家分享几个我认为不错的 Python 小技巧,有些可能被你低估了哟,get 起来!那么接下来就是:学习 Python 的正确姿势俗…

java bean验证_javaBean--登录验证

packagecom.JAVABean;importjava.util.HashMap;importjava.util.Map;publiccla***egister{privateStringname;privateStringage;privateStringemail;privateMaperrorsnull;//声明一个保存全部错误信息的map集合publicRegister(){//在构造方法中初始化属性this.name""…

java读取src xml文件路径_Java获取路径方法相对路径读取xml文件方法

(1)、request.getRealPath("/");//不推荐使用获取工程的根路径(2)、request.getRealPath(request.getRequestURI());//获取jsp的路径,这个方法比较好用,可以直接在servlet和jsp中使用(3)、request.getSession().getServletContext().getRealPa…

释放tcp连接的命令是_最实用的6个网络命令,网络故障不求人

很多弱电工程师朋友在项目中经常遇到一些网络故障,需要通过一些一些命令去检测、定位故障点,通过使用网络命令,故障解决的工作取得了事半功倍的效果。下面就一起温故而知新吧!一、ping命令(因特网包探索器)…

airpods2怎么查正品 ios11系统_拼多多AirPods2开箱评测,4种办法教你验真假,10个AirPods技巧教你玩...

大家好,Apple今天给大家分享一下拼多多上车AirPods 2无线充电盒版的经验,顺便整理了一波AirPods使用技巧,希望你用得上。入手理由自从去年10月份入手了iPhone XR,其实就挺想入款无线耳机的,所以一直在等AirPods升级换代…

java中for break的用法_java break语句的使用方法

在switch语中,break语句用来终止switch语句的执行。使程序 switch语句后的第一个语句 开始执行。在Java中,可以为每个代码块加一个括号,一个代码块通常 用大括号{}括起来的一段 代码。加标号的格式break语句有两种形式:无标签和有标签。无标签的break语句用来跳出单…

windows文件保护_Windows系统下媲美时间机器的系统备份工具,统统免费

Windows和macOS系统谁更美?不同的人有不同的见解。但体验过macOS之后很多电脑玩家会感叹,TimeMachine时间机器太好用了,Windows下有没有同类功能呢?TimeMachine提供了全盘完整备份、增量备份、文件历史版本等功能。它们在Windows …

JAVA结课_一点心情,写java结课考试之前

突然发现,已经好久没有上来写blog了,本来还以为能够天天写,后来发现,确是心有余力而不足啊。学期进入中段,课业慢慢多了,各种各样的事情也接踵而来了。本学期的java课程也已经结课了,8周32个学时…

sql怎么撤回update_腾讯SQL“现役运动员”给你的实践小技巧

引言SQL的全称是Structured Query Language(结构化查询语言),是一种古老而简洁的程序设计语言。看似平平无奇,一直被各种吐槽,但却有着众多语言所难得的漫长寿命,并展现出极好的拓展性,在不同时期衍生出不同的子语言。…

mysql 同一帐号多次登录_freeradius2.1.3 防止用户帐号重复登录

freeradius2.1.3 防止用户帐号重复登录一、修改 etc/raddb/sites-enabled 目录中的default 及inner-tunnel 这两个文件中的# Session database, used for checking Simultaneous-Use. Either the radutmp# or rlm_sql module can handle this.# The rlm_sql module is *much…

小程序input wxss_19. 教你零基础搭建小程序:wxss-尺寸单位

这章以后的四章都是介绍小程序样式文件——wxss 的使用,分为以下三个部分一、尺寸方案二、样式导入三、选择器这章先来讲wxss的尺寸单位—— rpxwxss的定义:WXSS( WeiXin Style Sheets )是⼀套样式语言,用于描述 WXML 的组件样式。与 CSS 相比…

java 最优算法_java 问题 求个最优算法

不知道是不是你要的package test;import java.util.Scanner;public class Number {/*** param args*/public static void main(String[] args) {int count 15;int val 5;Scanner input new Scanner(System.in);System.out.print("请输入开始数:");int …

某一个接口403 其他接口可以调通_Neo的务实外设指南 篇三十六:一个就够,65W快充+C口混插+最多6个设备 - 飞利浦65W摩天轮插座_插座...

2020-10-26 15:29:0623点赞23收藏2评论嗨,大家好!我是沈少!之前晒雷电3扩展坞的时候,已经有小伙伴注意到我用来提供PD充电的是一个很小巧的魔方插座。也有朋友私下提醒我,这类产品虽然支持PD快充协议,但一般…

linux java 获取路径怎么写_linux中java获取路径怎么写?

linux中java获取路径怎么写?在Unix/Linux中,路径的分隔采用正斜"/",比如"cd /home/java"。在java的代码开发中 是代表转义字符。相对路径和绝对路径. 指的是当前目录.. 指的是当前目录的上一级目录./book表示当前目录下的…

layerconfirm 自动关闭问题 没有阻塞问题_微信新版本自动更新?赶紧关闭这个功能...

前不久安卓用户也迎来了微信新版本的更新不少伙伴惊呼“猝不及防,一觉醒来发现微信自动更新了”一时间还冲上了话题的热搜榜究竟是怎么肥事?小移了解到:原来是因为部分用户设置了“微信自动更新”那么问题来了,如何关闭微信自动更…

java 动态生成getset_通过get、set方法,动态生成对象

最近在看Java的反射,把学习的东西整理一下,大家共同研究,有需要改进的地方,请大家指正。import java.beans.PropertyDescriptor;import java.lang.reflect.Field;import java.lang.reflect.Method;public class ReflectionTest {p…

list steam_在 Steam 中国版上玩单机游戏也会受到防沉迷系统管控

今天看到一个消息,Steam 中国版,也就是所谓的“蒸汽平台”,最近正在测试。重点在于,哪怕你玩“理论上无需联网”的单机游戏,游玩时间也会受到著名的防沉迷系统的限制。以下是我在动点科技编写的新闻全文(原文链接是界面…

java的默认访问权限_java类的访问权限

1.解析Java有四种访问权限, 其中三种有访问权限修饰符,分别为private,public和protected,还有一种不带任何修饰符。private: Java语言中对访问权限限制的最窄的修饰符,一般称之为“私有的”。被其修饰的类、属性以及方…

github流程图_逆天插件,VSCode里也能画流程图了?Visio可以淘汰了?

vscode-drawio简介网络之大,人才百出,在开源背景下,一些功能只有你想不到,没有做不到。这不,对于写代码的程序员来说,竟然也可以在VSCode IDE里边写代码,边画逻辑流程图了。最近,在g…

CentOS+ISCSI

九、配置iSCSI 添加1块大小为10G的虚拟硬盘; 安装iSCSI服务端targetcli; 使用新增加的硬盘创建卷组,名称为iscsivg,再创建iSCSI共享逻辑卷,逻辑 卷名称为iscsistore,大小为5G; 使用上述逻辑卷创建后端存储,名称为serverc.iscsistore; 定义iSCSI的IQN为iqn.2022-…