Android之ContentProvide(内容提供者)

ContentProvider使用方法。
ContentProvide的使用涉及ContentProvider、 ContentObserver、 ContentReslover 、ContentValues、Cursor、Uri、UriMatcher以及数据库等。
content Provider。它最大的特点是可以在应用之间共享数据
1.定义自己的ContentProvide类,
2.在AndroidManifest.xml上注册,并指定authorities属性,给外界提供固定Uri对外提供服务
提供的方法
操作方法:insert、getType、delete、query、update、oncreate用于创建数据库或链接数据库,以及提供为contentResolver增删改查的方法。

ContentValues是用于存储ContentResolver处理的值。

ContentResolver用于访问content模型,其实就是访问和操作contentprovider。
实现方式:Context提供了getContentResolver方法,这表明Activity、Service组件都可通过该方法获取该类的实例
ContentResolver提供的操作方法:insert、getType、delete、query、update等与ProvideContent相对应,通过Uri链接ContentProvide和ContentResolver
Cursor类,游标。应用时一般通过查询query得到。
Uri:协议部分、域名、路径、ID、字段
其中协议部分:content://
域名对应authoties属性

路径:资源部分或数据部分

ContentProvider示例:

package org.crazyit.content;

import android.content.ContentProvider;
import android.content.ContentValues;
import android.database.Cursor;
import android.net.Uri;

public class FirstProvider extends ContentProvider
{
	// 第一次创建该ContentProvider时调用该方法
	@Override
	public boolean onCreate()
	{
		System.out.println("===onCreate方法被调用===");
		return true;
	}

	// 该方法的返回值代表了该ContentProvider所提供数据的MIME类型
	@Override
	public String getType(Uri uri)
	{
		System.out.println("~~getType方法被调用~~");
		return null;
	}

	// 实现查询方法,该方法应该返回查询得到的Cursor
	@Override
	public Cursor query(Uri uri, String[] projection, String where,
		String[] whereArgs, String sortOrder)
	{
		System.out.println(uri + "===query方法被调用===");
		System.out.println("where参数为:" + where);
		return null;
	}

	// 实现插入的方法,该方法应该新插入的记录的Uri
	@Override
	public Uri insert(Uri uri, ContentValues values)
	{
		System.out.println(uri + "===insert方法被调用===");
		System.out.println("values参数为:" + values);
		return null;
	}

	// 实现删除方法,该方法应该返回被删除的记录条数
	@Override
	public int delete(Uri uri, String where, String[] whereArgs)
	{
		System.out.println(uri + "===delete方法被调用===");
		System.out.println("where参数为:" + where);
		return 0;
	}

	// 实现删除方法,该方法应该返回被更新的记录条数
	@Override
	public int update(Uri uri, ContentValues values, String where,
		String[] whereArgs)
	{
		System.out.println(uri + "===update方法被调用===");
		System.out.println("where参数为:"
			+ where + ",values参数为:" + values);
		return 0;
	}
}
<pre name="code" class="java">ContentProvider:提供了增、删、改、查的方法,同时也可以联系本地SQLite,对外提供数据。

 配置ContentProvide 

<provider
		    android:exported="true"
		    android:name=".FirstProvider"
		    android:authorities="org.crazyit.providers.firstprovider">
</provider>
name:指定的Android类
authoruties:指定ContentProvide的uri
exported:


使用ContentResolve调用方法

package org.crazyit.resolver;

import org.crazyit.resolver.R;

import android.app.Activity;
import android.content.ContentResolver;
import android.content.ContentValues;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.view.View;
import android.widget.Toast;

public class FirstResolver extends Activity
{
	ContentResolver contentResolver;
	Uri uri = Uri.parse("content://org.crazyit.providers.firstprovider/");

	@Override
	public void onCreate(Bundle savedInstanceState)
	{
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		// 获取系统的ContentResolver对象
		contentResolver = getContentResolver();
	}

	public void query(View source)
	{
		// 调用ContentResolver的query()方法。
		// 实际返回的是该Uri对应的ContentProvider的query()的返回值
		Cursor c = contentResolver.query(uri, null
			, "query_where", null, null);
		Toast.makeText(this, "远程ContentProvide返回的Cursor为:" + c,
			Toast.LENGTH_LONG).show();
	}

	public void insert(View source)
	{
		ContentValues values = new ContentValues();
		values.put("name", "fkjava");
		// 调用ContentResolver的insert()方法。
		// 实际返回的是该Uri对应的ContentProvider的insert()的返回值
		Uri newUri = contentResolver.insert(uri, values);
		Toast.makeText(this, "远程ContentProvide新插入记录的Uri为:"
			+ newUri, Toast.LENGTH_LONG).show();
	}

	public void update(View source)
	{
		ContentValues values = new ContentValues();
		values.put("name", "fkjava");
		// 调用ContentResolver的update()方法。
		// 实际返回的是该Uri对应的ContentProvider的update()的返回值
		int count = contentResolver.update(uri, values
			, "update_where", null);
		Toast.makeText(this, "远程ContentProvide更新记录数为:"
			+ count, Toast.LENGTH_LONG).show();
	}

	public void delete(View source)
	{
		// 调用ContentResolver的delete()方法。
		// 实际返回的是该Uri对应的ContentProvider的delete()的返回值
		int count = contentResolver.delete(uri
			, "delete_where", null);
		Toast.makeText(this, "远程ContentProvide删除记录数为:"
			+ count, Toast.LENGTH_LONG).show();
	}
}
通过this. getContentResolver()方法获取ContentResolve。ContentResolve与ContentProvide通过Uri联系在一起,通过contentResolve调用增删改查,直接调用ContentProvide的增删改查方法。


Android系统为ContentProvide提供工具类:UriMatcher和ContentUris

UriMatcher:主要作用是为了确定该ContentProvide能处理的Uri,以及确定每个方法中Uri参数多操作的数据

ContentUris:主要是处理增加数据时,对返回值Uri的处理,insert时返回一个携带id的uri





  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,以下是一个简单的Android Studio ContentProvider实验程序,可以满足您的要求。请注意,这个程序使用了系统提供的ContactsContract内容提供器来访问联系人信息。 首先,在AndroidManifest.xml文件中添加以下权限: ```xml <uses-permission android:name="android.permission.READ_CONTACTS" /> <uses-permission android:name="android.permission.WRITE_CONTACTS" /> ``` 然后,在MainActivity.java文件中定义以下代码: ```java import android.Manifest; import android.content.ContentProviderOperation; import android.content.ContentResolver; import android.content.ContentUris; import android.content.ContentValues; import android.content.Context; import android.content.OperationApplicationException; import android.content.pm.PackageManager; import android.database.Cursor; import android.net.Uri; import android.os.Bundle; import android.os.RemoteException; import android.provider.ContactsContract; import android.provider.ContactsContract.CommonDataKinds.Phone; import android.provider.ContactsContract.Contacts; import android.provider.ContactsContract.RawContacts; import android.util.Log; import android.widget.Toast; import androidx.annotation.NonNull; import androidx.appcompat.app.AppCompatActivity; import androidx.core.app.ActivityCompat; import androidx.core.content.ContextCompat; import java.util.ArrayList; public class MainActivity extends AppCompatActivity { private static final int READ_CONTACTS_PERMISSIONS_REQUEST = 1; private static final int WRITE_CONTACTS_PERMISSIONS_REQUEST = 2; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 请求读取联系人权限 if (ContextCompat.checkSelfPermission(this, Manifest.permission.READ_CONTACTS) != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.READ_CONTACTS}, READ_CONTACTS_PERMISSIONS_REQUEST); } else { readContacts(); } // 请求写入联系人权限 if (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_CONTACTS) != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_CONTACTS}, WRITE_CONTACTS_PERMISSIONS_REQUEST); } else { addContact(); } } // 读取联系人信息 private void readContacts() { ContentResolver contentResolver = getContentResolver(); Cursor cursor = contentResolver.query(Contacts.CONTENT_URI, null, null, null, null); if (cursor != null && cursor.getCount() > 0) { while (cursor.moveToNext()) { String name = cursor.getString(cursor.getColumnIndex(Contacts.DISPLAY_NAME)); String id = cursor.getString(cursor.getColumnIndex(Contacts._ID)); // 查询联系人的电话号码 Cursor phoneCursor = contentResolver.query(Phone.CONTENT_URI, null, Phone.CONTACT_ID + " = ?", new String[]{id}, null); if (phoneCursor != null && phoneCursor.getCount() > 0) { while (phoneCursor.moveToNext()) { String phoneNumber = phoneCursor.getString(phoneCursor.getColumnIndex(Phone.NUMBER)); Log.d("Contact", name + " " + phoneNumber); } } if (phoneCursor != null) { phoneCursor.close(); } } } if (cursor != null) { cursor.close(); } } // 查询指定电话的联系人信息 private void queryContactByPhoneNumber(String phoneNumber) { ContentResolver contentResolver = getContentResolver(); Uri uri = Uri.withAppendedPath(Phone.CONTENT_FILTER_URI, Uri.encode(phoneNumber)); Cursor cursor = contentResolver.query(uri, new String[]{Phone.DISPLAY_NAME}, null, null, null); if (cursor != null && cursor.getCount() > 0) { cursor.moveToNext(); String name = cursor.getString(cursor.getColumnIndex(Phone.DISPLAY_NAME)); Log.d("Contact", "Phone number " + phoneNumber + " belongs to " + name); } else { Log.d("Contact", "No contact found for phone number " + phoneNumber); } if (cursor != null) { cursor.close(); } } // 添加一个新的联系人 private void addContact() { ArrayList<ContentProviderOperation> ops = new ArrayList<>(); // 添加一个新的RawContact ops.add(ContentProviderOperation.newInsert(RawContacts.CONTENT_URI) .withValue(RawContacts.ACCOUNT_TYPE, null) .withValue(RawContacts.ACCOUNT_NAME, null) .build()); // 添加联系人姓名 ops.add(ContentProviderOperation.newInsert(Contacts.CONTENT_URI) .withValue(Contacts.DISPLAY_NAME, "Test Contact") .build()); // 添加联系人电话号码 ops.add(ContentProviderOperation.newInsert(Phone.CONTENT_URI) .withValueBackReference(Phone.RAW_CONTACT_ID, 0) .withValue(Phone.NUMBER, "+8613800000000") .withValue(Phone.TYPE, Phone.TYPE_MOBILE) .build()); try { getContentResolver().applyBatch(ContactsContract.AUTHORITY, ops); Toast.makeText(MainActivity.this, "Contact added successfully", Toast.LENGTH_SHORT).show(); } catch (RemoteException | OperationApplicationException e) { e.printStackTrace(); } } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); switch (requestCode) { case READ_CONTACTS_PERMISSIONS_REQUEST: if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { readContacts(); } break; case WRITE_CONTACTS_PERMISSIONS_REQUEST: if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { addContact(); } break; } } } ``` 这个程序中,readContacts()方法用于读取所有联系人信息,并且查询每个联系人的电话号码。queryContactByPhoneNumber()方法用于查询指定电话号码的联系人信息。addContact()方法用于添加一个新的联系人,其中包括联系人姓名和电话号码。 请注意,这个程序只是一个简单的示例,实际的应用程序可能需要更多的错误处理和用户提示。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值