ContentResolver contentResolver = getContentResolver(); //getContentResolver();是上下文中携带的方法,用于处理内容提供者提供的访问方式。
寻找哪个内容提供者需要用通过uri,uri路径分文 "主机名"+"具体要操作的数据" 其中"主机名"在提供数据库程序的清单文件中定义,"具体要操作的数据"在这个程序的源码中(一般文件名有provide字样),
我们可以通过查找"UriMatcher"函数搜索相关的信息。
提供权限 读写短信 操作SD卡:
<uses-permission android:name="android.permission.READ_SMS"/>
<uses-permission android:name="android.permission.WRITE_SMS"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
public void click(View v)
{
ContentResolver contentResolver = getContentResolver();
//通过查找TelephonyProvider程序的清单文件和源码可知,主机名为"sms" 后面为空则对应SMS_ALL
Uri uri = Uri.parse("content://sms/");
//使用ContentResolver提供的查询方法,查找短信相关信息
Cursor cursor = contentResolver.query(uri, new String[]{"address", "body"}, null, null, null);
//将cursor指向的内容封装到xml文件中
OutputStream os;
try {
os = new FileOutputStream(Environment.getExternalStorageDirectory().getPath()+"/msmInfo.xml");
XmlSerializer serializer = Xml.newSerializer();
serializer.setOutput(os, "utf-8");
serializer.startDocument("utf-8", true);
serializer.startTag(null, "info");
while(cursor.moveToNext())
{
serializer.startTag(null, "sms");
serializer.startTag(null, "address");
String address = cursor.getString(0);
serializer.text(address);
serializer.endTag(null, "address");
serializer.startTag(null, "body");
String body = cursor.getString(1);
serializer.text(body);
serializer.endTag(null, "body");
serializer.endTag(null, "sms");
}
serializer.endTag(null, "info");
serializer.endDocument();
} catch (Exception e) {
e.printStackTrace();
}
}
}
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
public void click(View v)
{
ContentResolver contentResolver = getContentResolver();
Uri uri = Uri.parse("content://sms/");
//写入列名及其对应的值
ContentValues values = new ContentValues();
values.put("address", "110");
values.put("body", "hello chenchen");
contentResolver.insert(uri, values);
}
}
1.data表中存储联系人的所有信息
2.通过mimetype 来区分信息对应的类型
3.通过raw_contact_id(即联系人的id)来区分不同信息对应的联系人
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
public void click(View v)
{
ContentResolver contentResolver = getContentResolver();
//用于处理raw_contacts表格的uri
Uri uri_raw = Uri.parse("content://com.android.contacts/raw_contacts");
//用于处理data表格的uri
Uri uri_data = Uri.parse("content://com.android.contacts/data");
//查出raw_contacts表格中所有id号,也就是用户id
Cursor cursor = contentResolver.query(uri_raw, new String[]{"contact_id"}, null, null, null);
while(cursor.moveToNext())
{
//获得一个contact_id号
String contact_id = cursor.getString(0);
System.out.println(" contact_id:"+contact_id);
//查找data1表中所有contact_id为当前id的内容 ************注意此处为"mimetype"而不是直接写表中的"mimetype_id",此处是一个视图
Cursor cursor_data1 = contentResolver.query(uri_data, new String[]{"mimetype", "data1"}, "raw_contact_id=?", new String[]{contact_id}, null);
while(cursor_data1.moveToNext())
{
//查出当前行的mime类型,从而确定当前data1列的数据的类型
String mimetype = cursor_data1.getString(0);
System.out.println("*************mimetype*"+mimetype+"*****************");
String data = cursor_data1.getString(1);
System.out.println("*************data1*"+data+"*****************");
if("vnd.android.cursor.item/name".equals(mimetype)){
System.out.println("******name:"+data+"********");
}else if("vnd.android.cursor.item/phone_v2".equals(mimetype)){
System.out.println("******phone:"+data+"********");
}else if("vnd.android.cursor.item/email_v2".equals(mimetype)){
System.out.println("******email:"+data+"********");
}
}
}
}
}
public class ReadContactUtils {
/** * 获取本机通讯录信息,返回通讯录中每个条目的List * @param context 传入上下文,用于获取ContentResolver内容解析者 * @return */
public static List<Contact> getContacts(Context context)
{
List<Contact> list = new ArrayList<Contact>();
Contact contact=null;
ContentResolver contentResolver = context.getContentResolver();
Uri uri_raw = Uri.parse("content://com.android.contacts/raw_contacts");
Uri uri_data1 = Uri.parse("content://com.android.contacts/data");
Cursor cursor = contentResolver.query(uri_raw, new String[]{"contact_id"}, null, null, null);//由于要返回所有的条目,所以不需要写后面的参数
while(cursor.moveToNext())
{
//从raw_contact_id表中获取contact_id
String contact_id = cursor.getString(0);
//System.out.println("*******"+contact_id+"********");
if(contact_id != null)
{
contact = new Contact();
//使用contact_id查出data表中对应的所有对应项 ,这些对应项就是一个联系人的所有信息
Cursor cursor_data1 = contentResolver.query(uri_data1, new String[]{"mimetype", "data1"}, "raw_contact_id=?" ,new String[]{contact_id}, null);
while(cursor_data1.moveToNext())
{
//获取表的数据的类型
String mimetype = cursor_data1.getString(0);
String data1 = cursor_data1.getString(1);
//System.out.println("****mimetype*"+mimetype+"******data1*"+data1+"****");
//根据不同类型找到data1相应的位置
if("vnd.android.cursor.item/name".equals(mimetype)){
contact.setName(data1);
}else if("vnd.android.cursor.item/phone_v2".equals(mimetype)){
contact.setPhone(data1);
}else if("vnd.android.cursor.item/email_v2".equals(mimetype)){
contact.setEmail(data1);
}
}
}
list.add(contact);
}
return list;
}
}
注意应当先在raw_contact_id表中加入新的contact_id,然后再添加data2中的各项信息。否则data2中不能插入信息。
public void click(View v)
{
String name = et_name.getText().toString().trim();
String email = et_email.getText().toString().trim();
String phone = et_phone.getText().toString().trim();
ContentResolver contentResolver = getContentResolver();
Uri uri_raw = Uri.parse("content://com.android.contacts/raw_contacts");
Uri uri_data1 = Uri.parse("content://com.android.contacts/data");
//看看raw_contact_id中的contact_id到第几条了
Cursor cursor = contentResolver.query(uri_raw, null, null, null, null);
int count = cursor.getCount();
int id = count+1;
//在raw_contact_id表中加入新信息的contact_id
ContentValues idValues = new ContentValues();
idValues.put("contact_id", id);
contentResolver.insert(uri_raw, idValues);
//分别加入信息中的每个行
ContentValues contentValues = new ContentValues();
contentValues.put("raw_contact_id", id);
contentValues.put("mimetype", "vnd.android.cursor.item/name");
contentValues.put("data1", name);
contentResolver.insert(uri_data1, contentValues);
contentValues.put("raw_contact_id", id);
contentValues.put("mimetype", "vnd.android.cursor.item/phone_v2");
contentValues.put("data1", phone);
contentResolver.insert(uri_data1, contentValues);
contentValues.put("raw_contact_id", id);
contentValues.put("mimetype", "vnd.android.cursor.item/email_v2");
contentValues.put("data1", email);
contentResolver.insert(uri_data1, contentValues);
}
public class MainActivity extends Activity {
private Uri uri;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
uri = Uri.parse("content://sms");
//第二个参数的解释 为true可以是不精确的路径 为false必须是精确路径
//notifyForDescendents If true changes to URIs beginning with uri will also cause
//notifications to be sent. If false only changes to the exact URI specified by uri
//will cause notifications to be sent. If true, than any URI values at or below the
//specified URI will also trigger a match.
getContentResolver().registerContentObserver(uri, true, new MyObeserver(new Handler()));
}
class MyObeserver extends ContentObserver {
public MyObeserver(Handler handler) {
super(handler);
}
//发现监视路径下的数据库发生变化就会调用此方法
@Override
public void onChange(boolean selfChange) {
//把短信的address 和 body内容取出来
Cursor cursor = getContentResolver().query(uri, new String[]{"address","body"}, null, null, "date desc");
cursor.moveToFirst();
String address = cursor.getString(0);
String body = cursor.getString(1);
System.out.println("body:"+body+"address:"+address);
super.onChange(selfChange);
}
}
}
前几个条目主要说了我们如何获取其他数据的数据库信息,使用的是内容解析者提供的方法来对其他程序的数据库进行操作。
下面我们看一下数据库提供方程序是如何工作的。
1.同四大组件中其他的三个一样,也是继承一个类(ContentProvider),并为这个类在清单文件中配置一下。我们需要实现其中的增删改查方法。
public class MyContentProvider extends ContentProvider {
@Override
public boolean onCreate() {
}
@Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) {
}
@Override
public String getType(Uri uri) {
return null;
}
@Override
public Uri insert(Uri uri, ContentValues values) {
}
@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
}
@Override
public int update(Uri uri, ContentValues values, String selection,
String[] selectionArgs) {
}
}
清单文件 name : 内容提供者的路径
authorities : 内容提供者的主机名
<provider android:name="com.example.readprivatedata.MyContentProvider"
android:authorities="com.example.myprovider">
</provider>
2.有了增删改查的方法,但是我们去操作哪站表,或者哪张表中的哪些数据呢?者就需要我们配置另外一个参数,用于找到对应的数据。我们使用**UriMatcher**类进行操作。
private static UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);
//参数分别为 主机名 要操作的数据的标识 该标识对应的数字
matcher.addURI("com.example.myprovider", "info", INFO);
当我们进行增删改查操作时,从完整uri,通过match方法获取code码,每个code码对应操作不同的数据
int code = matcher.match(uri);获取当前uri对应的code码
public class MyContentProvider extends ContentProvider {
//用于匹配调用哪个函数
private static final int INFO = 1;
private static UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);
private MyOpenHelper helper;
static{
//可以为matcher添加多个uri,用于操作不同的数据
matcher.addURI("com.example.myprovider", "info", INFO);
}
@Override
public boolean onCreate() {
//创建这个Provider时创建数据库,后面有这个类的源码 其中包含一张名为"info"的表格
helper = new MyOpenHelper(getContext());
return false;
}
@Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) {
//获取code码
int code = matcher.match(uri);
System.out.println("query******code:"+code+"*******");
//当code码和我们为某个数据预定义的code一样时,就进行对应的操作
if(code == INFO)
{
System.out.println("*******query********");
SQLiteDatabase db = helper.getReadableDatabase();
Cursor cursor = db.query("info", projection, selection, selectionArgs, null, null, sortOrder);
return cursor;
}else {
throw new IllegalArgumentException("query路径不匹配 请检查");
}
}
@Override
public String getType(Uri uri) {
return null;
}
@Override
public Uri insert(Uri uri, ContentValues values) {
int code = matcher.match(uri);
System.out.println("insert******code:"+code+"*******");
if(code == INFO)
{
System.out.println("*******insert********");
//匹配成功
SQLiteDatabase db = helper.getReadableDatabase();
//代表插入到了多少行
long insert = db.insert("info", null, values);
return Uri.parse("com.itheima.account.provider"+insert);
}else {
throw new IllegalArgumentException("insert路径不匹配 请检查");
}
}
@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
int code = matcher.match(uri);
System.out.println("delete******code:"+code+"*******");
if(code == INFO)
{
System.out.println("*******delete********");
SQLiteDatabase db = helper.getReadableDatabase();
int delete = db.delete("info", selection, selectionArgs);
//删除了多少行
return delete;
}else {
throw new IllegalArgumentException("3路径不匹配 请检查");
}
}
@Override
public int update(Uri uri, ContentValues values, String selection,
String[] selectionArgs) {
int code = matcher.match(uri);
System.out.println(" update******code:"+code+"*******");
if(code == INFO)
{
System.out.println("*******update********");
SQLiteDatabase db = helper.getReadableDatabase();
int update = db.update("info", values, selection, selectionArgs);
//删除了多少行
return update;
}else {
throw new IllegalArgumentException("4路径不匹配 请检查");
}
}
}
创建数据库的类
public class MyOpenHelper extends SQLiteOpenHelper {
public MyOpenHelper(Context context) {
super(context, "account.db", null, 1);
}
@Override
public void onCreate(SQLiteDatabase db) {
db.execSQL("create table info (_id integer primary key autoincrement,name varchar(20),money varchar(20))");
//初始化2条数据
db.execSQL("insert into info ('name','money') values ('张三','2000')");
db.execSQL("insert into info ('name','money') values ('李四','5000')");
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
System.out.println("onUpgrade");
}
}
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
public void click_insert(View v)
{
ContentValues contentValues = new ContentValues();
contentValues.put("name", "haha");
contentValues.put("money", 10000);
//注意,uri决定了要操作的表,内容提供者在调用insert方法时,根据uri对应的code只找到相应的分支,这个分支对应的操作是操作info指定的数据
getContentResolver().insert(Uri.parse("content://com.example.myprovider/info"), contentValues);
}
public void click_delete(View v)
{
getContentResolver().delete(Uri.parse("content://com.example.myprovider/info"), "name=?", new String[]{"haha"});
}
public void click_update(View v)
{
}
public void click_find(View v)
{
}
}