Xamarin.Android之ContentProvider
一、前言
掌握了如何使用SQLiteOpenHelper之后,我们就可以进行下一步的学习。本章我们将会学习如何使用ContentProvider来将数据库方面的操作封装起来,同时它还可以供其他应用访问并操作数据库。
二、概念
首先我们不会急于写代码,而是要搞懂如何利用ContentProvider对数据库进行操作,因为我们不会直接操作数据库对象,而是通过URI来操作数据库。这就好比你要获取User表的全部内容,那么这个URI就是content://base/user其中base是自己命名的,最好是能够唯一。因为我们需要依靠这个区分数据库,然后就是user是用来区分操作的是哪个表,当然你也可以不用命名为user可以是其他的名称,最终反正要依靠代码去判断的。这样我们就可以避免在活动中直接对数据库对象操作,也方便对数据库进行统一的维护。
三、实际操作
1、搭建基本框架
新建一个LocationContentProvider类,并且继承自ContentProvider,还要重写该类的OnCreate、Delete、GetType、Insert、Query和Update方法。
2、设计数据库以及URI
下面是笔者设计好的结构:
public static string PROVIDER_NAME = "xamarin-cn.location"; private const string DATABASE_NAME = "location";
private const int DATABASE_VERSION = ; private const string USERTABLE_NAME = "tuser";
private const int USER = ;
private const int USER_ID = ;
private static IDictionary<string, string> userProjectionMap;
public class UserTable
{
public static Android.Net.Uri CONTENT_URI = Android.Net.Uri.Parse("content://" + PROVIDER_NAME + "/user");
public const string _ID = "_id";
public const string UserName = "username";
public const string UserPwd = "userpwd";
public const string Age = "age";
}
其中 PROVIDER_NAME 是URI的基址,DATABASE_NAME和DATABASE_VERSION是数据库的命名的和版本号,下面就是tuser表的信息,分别是表名、URI标识1、URI标识2、表结构键值对。UserTable类中的就是该表公开的属性,其中包含表的字段以及查询该表的URI路径。我们可以看到该表的URI路径为content://xamarin-cn.location/user。
3、初始化UriMatcher
我们需要通过UriMatcher这个类来判断URI操作的是哪个数据库的哪个表,这样就不需要我们自己通过字符串进行判断,具体的初始化可以见如下代码:
private static UriMatcher uriMatcher;
static LocationContentProvider()
{
userProjectionMap = new Dictionary<string, string>();
userProjectionMap.Add(UserTable._ID, UserTable._ID);
userProjectionMap.Add(UserTable.UserName, UserTable.UserName);
userProjectionMap.Add(UserTable.UserPwd, UserTable.UserPwd);
userProjectionMap.Add(UserTable.Age, UserTable.Age);
uriMatcher = new UriMatcher(UriMatcher.NoMatch);
uriMatcher.AddURI(PROVIDER_NAME, "user", USER);
uriMatcher.AddURI(PROVIDER_NAME, "user/#", USER_ID);
}
这里我们通过UriMatcher的AddURI方法添加的不同类型URI,其中有针对整张表的,还有针对表中某条数据的。
4、建立数据库
这里我就不一一介绍了直接放出代码:
private LocationSqliteOpenHelper dbHelper;
class LocationSqliteOpenHelper : SQLiteOpenHelper
{
public LocationSqliteOpenHelper(Context context)
: base(context, DATABASE_NAME, null, DATABASE_VERSION)
{
} public override void OnCreate(SQLiteDatabase db)
{
StringBuilder strSql = new StringBuilder();
strSql.AppendFormat("CREATE TABLE {0} (", USERTABLE_NAME);
strSql.AppendFormat("{0} INTEGER PRIMARY KEY NOT NULL,", UserTable._ID);
strSql.AppendFormat("{0} TEXT NOT NULL,", UserTable.UserName);
strSql.AppendFormat("{0} TEXT NOT NULL,", UserTable.UserPwd);
strSql.AppendFormat("{0} INTEGER NOT NULL)", UserTable.Age);
db.ExecSQL(strSql.ToString());
} public override void OnUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)
{
db.ExecSQL("DROP TABLE IF EXISTS " + USERTABLE_NAME);
OnCreate(db);
}
}
5.初始化ContentProvider
首先我们需要在OnCreate方法中将dbHelper初始化:
public override bool OnCreate()
{
dbHelper = new LocationSqliteOpenHelper(Context);
return true;
}
6.完善Query方法
为了能够方便的组织查询语句这里笔者使用了SQLiteQueryBuilder对象来组织,下面就是查询的代码:
public override Android.Database.ICursor Query(Android.Net.Uri uri, string[] projection, string selection, string[] selectionArgs, string sortOrder)
{
SQLiteQueryBuilder qb = new SQLiteQueryBuilder();
qb.Tables = USERTABLE_NAME;
//根据uri判断查询的是某条数据还是针对整个表,从而决定条件语句
switch (uriMatcher.Match(uri))
{
case USER:
{
//设置需要获取的字段
//userProjectionMap默认包含的所有字段
qb.SetProjectionMap(userProjectionMap);
}
break;
case USER_ID:
{
qb.SetProjectionMap(userProjectionMap);
//拼接条件语句
//其中uri.PathSegments.ElementAt(1) 将会获取第二个片段,
//就是第二个“/”后台的内容,如果后面还存在“/”则获取他们之间的内容
qb.AppendWhere(UserTable._ID + "=" + uri.PathSegments.ElementAt());
}
break;
}
SQLiteDatabase db = dbHelper.ReadableDatabase;
ICursor c = qb.Query(db, projection, selection, selectionArgs, null, null, " _id desc");
c.SetNotificationUri(Context.ContentResolver, uri);
return c;
}
代码中的注释已经将重点部分都介绍了,关于Query的参数可以跟数据库对象的Query进行比较,都是一样的只是少了一部分参数。
7.完善Insert
因为SQLite规定了id只能是数据库自动生成的,所以在插入数据库这块只需要判断操作的是哪个表,介于笔者这里只有一个表所以没有该项操作,下面是具体的代码:
public override Android.Net.Uri Insert(Android.Net.Uri uri, ContentValues values)
{
SQLiteDatabase db = dbHelper.WritableDatabase;
long rowId = db.Insert(USERTABLE_NAME, null, values);
//拼接最终形成的URI
Android.Net.Uri result = ContentUris.WithAppendedId(UserTable.CONTENT_URI, rowId);
Context.ContentResolver.NotifyChange(result, null);
return result;
}
唯一要说明的就是在添加完数据之后要将这条数据组成的uri返回,这样就可以方便以后的查询。
8.完善Update
public override int Update(Android.Net.Uri uri, ContentValues values, string selection, string[] selectionArgs)
{
SQLiteDatabase db = dbHelper.WritableDatabase;
int count = ;
switch (uriMatcher.Match(uri))
{
case USER:
{
count = db.Update(USERTABLE_NAME, values, selection, selectionArgs);
}
break;
case USER_ID:
{
String userid = uri.PathSegments.ElementAt();
string select = "";
//如果还有附加的查询语句则拼接上去
if (selection != null)
select = " AND(" + selection + ")";
count = db.Update(USERTABLE_NAME, values, UserTable._ID + "=" + userid + select, selectionArgs);
}
break;
}
Context.ContentResolver.NotifyChange(uri, null);
return count;
}
这里跟查询一样,需要判断是针对某条数据还是整个表。
9.完善Delete
理解了Update,删除就简单了,只是将db.Update方法改写成Delete即可,代码如下所示:
public override int Delete(Android.Net.Uri uri, string selection, string[] selectionArgs)
{
SQLiteDatabase db = dbHelper.WritableDatabase;
int count = ;
switch (uriMatcher.Match(uri))
{
case USER:
{
count = db.Delete(USERTABLE_NAME, selection, selectionArgs);
}
break;
case USER_ID:
{
String userid = uri.PathSegments.ElementAt();
string select = "";
if (selection != null)
select = " AND(" + selection + ")";
count = db.Delete(USERTABLE_NAME,
UserTable._ID + "=" + userid + select,
selectionArgs);
}
break;
}
Context.ContentResolver.NotifyChange(uri, null);
return count;
}
最后就是GetType方法,只要返回空字符串即可。
四、操作ContentProvider
现在我们回到MainActivity中使用ContentProvider对数据库进行操作,其中最关键的是ContentResolver是不是跟ContentProvider是配对的?通过ContentResolver我们就可以通过URI来操作数据库,而不需要关注具体的数据库对象,比如下面的代码我们进行了插入、查询、更新和删除操作,代码量要比使用SQLiteOpenHelper更少,同时也便于后期的维护:
ContentValues values = new ContentValues();
values.Put(LocationContentProvider.UserTable.UserName, "yzf");
values.Put(LocationContentProvider.UserTable.UserPwd, "");
values.Put(LocationContentProvider.UserTable.Age, );
Android.Net.Uri uri = ContentResolver.Insert(LocationContentProvider.UserTable.CONTENT_URI, values); var c = ContentResolver.Query(uri, null, null, null, null);
c.MoveToFirst();
string username = c.GetString(c.GetColumnIndex(LocationContentProvider.UserTable.UserName)); values.Clear();
values.Put(LocationContentProvider.UserTable.UserName, "zn");
ContentResolver.Update(uri, values, null, null); c = ContentResolver.Query(uri, null, null, null, null);
c.MoveToFirst();
username = c.GetString(c.GetColumnIndex(LocationContentProvider.UserTable.UserName)); ContentResolver.Delete(uri, null, null);
Xamarin.Android之ContentProvider的更多相关文章
- Xamarin.Android开发实践(十四)
Xamarin.Android之ListView和Adapter 一.前言 如今不管任何应用都能够看到列表的存在,而本章我们将学习如何使用Xamarin去实现它,以及如何使用适配器和自定义适配器(本文 ...
- Xamarin.Android开发实践(十三)
Xamarin.Android之SQLite.NET ORM 一.前言 通过<Xamarin.Android之SQLiteOpenHelper>和<Xamarin.Android之C ...
- Xamarin.Android开发实践(十二)
Xamarin.Android之ContentProvider 一.前言 掌握了如何使用SQLiteOpenHelper之后,我们就可以进行下一步的学习.本章我们将会学习如何使用ContentProv ...
- Xamarin.Android之ListView和Adapter
一.前言 如今不管任何应用都能够看到列表的存在,而本章我们将学习如何使用Xamarin去实现它,以及如何使用适配器和自定义适配器(本文中的适配器的主要内容就是将原始的数据转换成了能够供列表控件显示的项 ...
- Xamarin.Android之SQLite.NET ORM
一.前言 通过<Xamarin.Android之SQLiteOpenHelper>和<Xamarin.Android之ContentProvider>的学习,我们已经掌握了如何 ...
- XAMARIN.ANDROID SIGNALR 实时消息接收发送示例
SignalR 是一个开发实时 Web 应用的 .NET 类库,使用 SignalR 可以很容易的构建基于 ASP.NET 的实时 Web 应用.SignalR 支持多种服务器和客户端,可以 Host ...
- Android之ContentProvider数据存储
一.ContentProvider保存数据介绍 一个程序可以通过实现一个ContentProvider的抽象接口将自己的数据完全暴露出去,而且ContentProvider是以类似数据库中表的方式将数 ...
- XAMARIN ANDROID 二维码扫描示例
现在二维码的应用越来越普及,二维码扫描也成为手机应用程序的必备功能了.本文将基于 Xamarin.Android 平台使用 ZXing.Net.Mobile 做一个简单的 Android 条码扫描示 ...
- 我正在使用Xamarin的跨平台框架—Xamarin.Android回忆录
一.缘起 在自己给别家公司做兼职外包的时候,已经明确知道外包的活不是那么好干的,一般在经历了初期热血澎湃的激情后,逐渐冷淡,愤怒,再冷淡,再愤怒…,听上去好像高潮迭起,但令人尴尬的是,这高潮迭起我们都 ...
随机推荐
- ABP文档 - EntityFramework 集成
文档目录 本节内容: Nuget 包 DbContext 仓储 默认仓储 自定义仓储 特定的仓储基类 自定义仓储示例 仓储最佳实践 ABP可使用任何ORM框架,它已经内置了EntityFrame(以下 ...
- $.extend()的实现源码 --(源码学习1)
目标: $.extend({ add:function(a,b){ return a + b; } }) console.log($.a ...
- jQuery学习之路(7)- 用原生JavaScript实现jQuery的某些简单功能
▓▓▓▓▓▓ 大致介绍 学习了妙味,用原生的JavaScript实现jQuery中的某些部分功能 定义自己的函数库lQuery ▓▓▓▓▓▓ $()选择器的实现 jQuery是面向对象的,所以自己编写 ...
- 用WebRequest +HtmlAgilityPack 从外网抓取数据到本地
相信大家对于WebRequest 并不陌生,我们在C#中发请求的方式,就是创建一个WebRequest .那么如果我们想发一个请求到外网,比如国内上不了的一些网站,那么该怎么做呢? 其实WebRequ ...
- css元素水平居中和垂直居中的方式
关于居中的问题,一直处于疑惑不解的状态,知道的几种方法好像也不是每一次都会起到作用,所以更加迷惑.主要是不清楚该 在什么情况下采用哪种解决方法,所以,整理了一些方法,梳理一下思路,做一个总结. 1. ...
- Win10 UWP开发系列——开源控件库:UWPCommunityToolkit
在开发应用的过程中,不可避免的会使用第三方类库.之前用过一个WinRTXamlToolkit.UWP,现在微软官方发布了一个新的开源控件库—— UWPCommunityToolkit 项目代码托管在G ...
- 在你的ASP.NET MVC中使用查找功能
在程序中,使用查找功能是少之不了.今天在ASP.NET环境下演示一回. 在cshtml视图中,有三个文本框,让用户输入关键词,然后点击最右连的“搜索”铵钮,如果有结果将显示于下面. Html: 表格放 ...
- css3更改input单选和多选的样式
在项目开发中我们经常会遇到需要更改input单选和多选样式的情况,今天就给大家介绍一种简单改变input单选和多选样式的办法. 在这之前先简单介绍一下:before伪类 :before 选择器向选定的 ...
- BPM合同管理解决方案分享
一.方案概述合同是组织与组织间所订协议的法律 表现形式,体现着双方对于合作在法律和道德上的承诺.然而,大多数企业的合同管理都或多或少存在合同审批过程不规范.签订草率.审批权责不清.合同执行跟踪难.合同 ...
- 满堂红CIO邓劲翔:房屋中介突围
人脸识别.客户关系管理进度监控.业务流程实时监控.网站访问人数及流量实时监控等实际企业应用场景淋漓尽致.羽羽如生的以大屏幕上图表形式展现在人们面前,如果你不去继续询问,你不会知道这是一家才刚刚在房地产 ...