Android组件之自定义ContentProvider

简介:

Android的数据存储有五种方式Shared Preferences、网络存储、文件存储、外储存储、SQLite,一般这些存储都只是在单独的一个应用程序之中达到一个数据的共享,有时候我们需要操作其他应用程序的一些数据,例如常见系统里的通讯录,短信,照片等等,所以云存储,通讯录,艳照门等等就孕育而生了。ContentProvider可以理解成内容提供者,也可以理解为一个接口,就是提供了一个供外部访问的接口,有的时候需要进行权限控制。

ContentProvider简介

ContentProvider向我们提供了我们在应用程序之前共享数据的一种机制,而我们知道每一个应用程序都是运行在不同的应用程序的,不同程序的之间数据共享是现实的需要,程序总不能使闭环的,Android中的ContentProvider外共享数据的好处是统一了数据的访问方式。简单总结说下:

  1. ContentProvider为存储和获取数据提供了统一的接口。ContentProvide对数据进行封装,不用关心数据存储的细节。使用表的形式来组织数据。
  2. 使用ContentProvider可以在不同的应用程序之间共享数据。 
  3. Android为常见的一些数据提供了默认的ContentProvider(包括音频、视频、图片和通讯录等)。 

说到了ContentProvider这么多好处,不能不说下Uri(Universal Resource Identifier)注意不是URL,通用资源标识符,看个简单的读取联系人的Uri,content://contacts/people,

  1. content://是前缀,固定的;
  2. contacts 主机名(或叫Authority)用于唯一标识这个ContentProvider,外部调用者可以根据这个标识来调用;
  3. people  路径(path)表示我们要操作的数据,路径的构建根据业务而定;

自定义ContentProvider

俗话说,欲善其事必先利其器,想要成为一个内容提供者,就先需要有数据,先建立一个SqlDbConncetion:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public  class  SqlDBConnection  extends  SQLiteOpenHelper {
 
     private  static  final  String DbName = "Book.db" ;
     private  static  int  version= 1 ;
 
     public  SqlDBConnection(Context context) {
         super (context, DbName,  null , version);
     }
 
     @Override
     public  void  onCreate(SQLiteDatabase db) {
         // TODO Auto-generated method stub
          String sqlString= "create table Book (id integer primary key autoincrement,Name nvarchar(200),Title nvarchar(200))" ;
          db.execSQL(sqlString);
     }
 
     @Override
     public  void  onUpgrade(SQLiteDatabase db,  int  oldVersion,  int  newVersion) {
         // TODO Auto-generated method stub
     }
}

  上篇文章讲的junit测试这个时候可以拿过来使用一下初始化下数据:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public  class  BookCase  extends  AndroidTestCase {
 
     public  void  Intial() {
         SqlDBConnection dbConnection =  new  SqlDBConnection(getContext());
         SQLiteDatabase sqlDataBase = dbConnection.getWritableDatabase();
         long  row =  0 ;
         for  ( int  i =  0 ; i <  5 ; i++) {
             ContentValues values =  new  ContentValues();
             values.put( "Name" "书籍" +i);
             values.put( "Title" "标题"  + i);
             row = sqlDataBase.insert( "Book" null , values);
             Log.i( "BookCase" "插入成功:"  + row);
         }
     }
}

 前面是基础工作,这个时候就可以建立一个自己的ContentProvider:

 主机名是需要自己去AndroidManifest.xml文件中自己配置的,要求是唯一的,最好是用包名就好:

1
2
<provider android:name= "com.example.googlecontentprovider.MyContentProvider"
          android:authorities= "com.example.googlecontentprovider.MyContentProvider" ></provider>

  如果觉得上面的那一串代码不是很好理解,下面调用的时候我会分别解释。

ContentResolver的使用

方法写在一个应用程序中调用属于正常,在另外一个程序中调用该程序的方法就是类似于接口了,下面先看原来初始化的数据:

重新新建一个Android测试项目,定义为BookCase,首先插入数据,定义一个Uri,这里面主机名就是上面定义的包名,book/insert与CONTENT_INSERT是对应的:

1
2
3
4
5
6
7
8
9
10
11
12
public  void  bookInsert() {
     Uri uri = Uri
             .parse( "content://com.example.googlecontentprovider.MyContentProvider/book/insert" );
     ContentResolver resolver = getContext().getContentResolver();
     ContentValues values =  new  ContentValues();
     values.put( "Name" "书籍5" );
     values.put( "Title" "标题5" );
     uri = resolver.insert(uri, values);
     Log.i( "BookCase" "Uri"  + uri);
     long  id = ContentUris.parseId(uri);
     Log.i( "BookCase" "测试成功"  + id);
}

  显示结果如下:

然后更新刚才插入的数据,同样的更具Code给Uri赋值,然后初始化一个ContentResolver,调用update方法:

1
2
3
4
5
6
7
8
9
public  void  bookUpdate() {
     Uri uri = Uri
             .parse( "content://com.example.googlecontentprovider.MyContentProvider/book/update" );
     ContentResolver resolver = getContext().getContentResolver();
     ContentValues values= new  ContentValues();
     values.put( "Name" "修改" );
     int  count = resolver.update(uri, values,  " id=?" , new  String[]{ "10" });
     Log.i( "BookCase" "更新了"  + count +  "行" );
}

  结果如下:

删除插入的数据:

1
2
3
4
5
6
7
8
9
public  void  bookDelete() {
         Uri uri = Uri
                 .parse( "content://com.example.googlecontentprovider.MyContentProvider/book/delete" );
         ContentResolver resolver = getContext().getContentResolver();
         String where = " id=?" ;
         String[] argString = { "10" };
         int  count = resolver.delete(uri, where, argString);
         Log.i( "BookCase" "删除了"  + count +  "行" );
     }

结果如下:

查询所有的数据:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public  void  bookQuery() {
     Uri uri = Uri
             .parse( "content://com.example.googlecontentprovider.MyContentProvider/book/query" );
     ContentResolver resolver = getContext().getContentResolver();
     Cursor  cursor=resolver.query(uri,  new  String[]{ "id" , "Name" , "Title" },  null null null );
     if  (cursor.getCount()> 0 ) {
         while  (cursor.moveToNext()) {
             int  id=cursor.getInt(cursor.getColumnIndex( "Id" ));
             String nameString=cursor.getString(cursor.getColumnIndex( "Name" ));
             String titleString=cursor.getString(cursor.getColumnIndex( "Title" ));
             Log.i( "BookCase" , id+ "---" +nameString+ "---" +titleString);
         }
     }
     
}

  Log下的记录:

查询单条记录:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public  void  bookQuerySingle() {
     Uri uri = Uri
             .parse( "content://com.example.googlecontentprovider.MyContentProvider/book/query" );
     ContentResolver resolver = getContext().getContentResolver();
     uri=ContentUris.withAppendedId(uri, 1 );
     Cursor  cursor=resolver.query(uri,  new  String[]{ "id" , "Name" , "Title" },  null null null );
     if  (cursor.getCount()> 0 ) {
         while  (cursor.moveToNext()) {
             int  id=cursor.getInt(cursor.getColumnIndex( "Id" ));
             String nameString=cursor.getString(cursor.getColumnIndex( "Name" ));
             String titleString=cursor.getString(cursor.getColumnIndex( "Title" ));
             Log.i( "BookCase" , id+ "---" +nameString+ "---" +titleString);
         }
     }
}

 结果如图:

本文转自Fly_Elephant博客园博客,原文链接:http://www.cnblogs.com/xiaofeixiang/p/4072486.html,如需转载请自行联系原作者

相关文章
|
2月前
|
缓存 前端开发 Android开发
安卓开发中的自定义视图:从零到英雄
【10月更文挑战第42天】 在安卓的世界里,自定义视图是一块画布,让开发者能够绘制出独一无二的界面体验。本文将带你走进自定义视图的大门,通过深入浅出的方式,让你从零基础到能够独立设计并实现复杂的自定义组件。我们将探索自定义视图的核心概念、实现步骤,以及如何优化你的视图以提高性能和兼容性。准备好了吗?让我们开始这段创造性的旅程吧!
30 1
|
2月前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
|
3月前
|
Android开发 开发者
安卓应用开发中的自定义视图
【9月更文挑战第37天】在安卓开发的海洋中,自定义视图犹如一座座小岛,等待着勇敢的探索者去发现其独特之处。本文将带领你踏上这段旅程,从浅滩走向深海,逐步揭开自定义视图的神秘面纱。
45 3
|
3月前
|
存储 Android开发 开发者
深入理解安卓应用开发的核心组件
【10月更文挑战第8天】探索Android应用开发的精髓,本文带你了解安卓核心组件的奥秘,包括Activity、Service、BroadcastReceiver和ContentProvider。我们将通过代码示例,揭示这些组件如何协同工作,构建出功能强大且响应迅速的应用程序。无论你是初学者还是资深开发者,这篇文章都将为你提供新的视角和深度知识。
|
3月前
|
数据可视化 Android开发 开发者
安卓应用开发中的自定义View组件
【10月更文挑战第5天】在安卓应用开发中,自定义View组件是提升用户交互体验的利器。本篇将深入探讨如何从零开始创建自定义View,包括设计理念、实现步骤以及性能优化技巧,帮助开发者打造流畅且富有创意的用户界面。
110 0
|
30天前
|
XML 搜索推荐 前端开发
安卓开发中的自定义视图:打造个性化UI组件
在安卓应用开发中,自定义视图是一种强大的工具,它允许开发者创造独一无二的用户界面元素,从而提升应用的外观和用户体验。本文将通过一个简单的自定义视图示例,引导你了解如何在安卓项目中实现自定义组件,并探讨其背后的技术原理。我们将从基础的View类讲起,逐步深入到绘图、事件处理以及性能优化等方面。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧。
|
2月前
|
搜索推荐 前端开发 Android开发
安卓应用开发中的自定义视图实现
【10月更文挑战第30天】在安卓开发的海洋中,自定义视图是那抹不可或缺的亮色,它为应用界面的个性化和交互体验的提升提供了无限可能。本文将深入探讨如何在安卓平台创建自定义视图,并展示如何通过代码实现这一过程。我们将从基础出发,逐步引导你理解自定义视图的核心概念,然后通过一个实际的代码示例,详细讲解如何将理论应用于实践,最终实现一个美观且具有良好用户体验的自定义控件。无论你是想提高自己的开发技能,还是仅仅出于对安卓开发的兴趣,这篇文章都将为你提供价值。
|
2月前
|
Android开发 开发者 UED
安卓开发中自定义View的实现与性能优化
【10月更文挑战第28天】在安卓开发领域,自定义View是提升应用界面独特性和用户体验的重要手段。本文将深入探讨如何高效地创建和管理自定义View,以及如何通过代码和性能调优来确保流畅的交互体验。我们将一起学习自定义View的生命周期、绘图基础和事件处理,进而探索内存和布局优化技巧,最终实现既美观又高效的安卓界面。
44 5
|
3月前
|
XML 前端开发 Java
安卓应用开发中的自定义View组件
【10月更文挑战第5天】自定义View是安卓应用开发的一块基石,它为开发者提供了无限的可能。通过掌握其原理和实现方法,可以创造出既美观又实用的用户界面。本文将引导你了解自定义View的创建过程,包括绘制技巧、事件处理以及性能优化等关键步骤。
|
3月前
|
测试技术 数据库 Android开发
深入解析Android架构组件——Jetpack的使用与实践
本文旨在探讨谷歌推出的Android架构组件——Jetpack,在现代Android开发中的应用。Jetpack作为一系列库和工具的集合,旨在帮助开发者更轻松地编写出健壮、可维护且性能优异的应用。通过详细解析各个组件如Lifecycle、ViewModel、LiveData等,我们将了解其原理和使用场景,并结合实例展示如何在实际项目中应用这些组件,提升开发效率和应用质量。
55 6