Android———ContentProvider理解与使用
ContentProvider
ContentProvider是Android四大组件之一,与Activity,service和BroadcastReceiver处于同一级别,都是Android中重要的组件,但ContentProvider使用的却很少。
ContentProvider为存储和读取数据提供了统一的接口,实现了程序间的数据共享。ContentProvider为应用间的数据交互提供了一个安全的环境。它准许你把自己的应用数据根据需求开放给其他应用进行增、删、改、查,而不用担心直接开放数据库权限而带来的安全问题。
Uri
Uri 通用资源标志符(Universal Resource Identifier)Uri代表要操作的数据,Android中可用的每种资源 - 图像、视频片段等都可以用Uri来表示。
Uri的标准写法
content://com.example.app.provider/table
content://是固定的写法,表明这是一个Uri
com.example.app.provider代表包名
table表示数据表
UriMatch
UriMatcher 类主要用于匹配Uri.这里的匹配是发生在ContentProvider中的,假如我们向ContentProvider中插入一条数据,不可能为所欲为的想怎么干就怎么干,在ContentProvider肯定要做一个判断,只有在符合条件下才会去执行你想要执行的操作,这里的判断就是用UriMatch进行匹配,假如是系统的ContentProvider如联系人、图库、视频库等,这些系统都提供了Uri我们可以根据系统提供的Uri来操作相应的数据
创建自定义ContentProvider步骤
- 使用SQLite技术,创建好数据库和数据表;
- 新建类继承ContentProvider;
- 创建UriMatcher定义Uri规则;
- 重写6个抽象方法(onCreate,insert,delete,getType,update,query);
- 在AndroidManifest中注册provide;
- ContentResolver对ContentProvider共享数据进行增删改查。
使用ContentProvider进行进程间的数据共享
代码实例
首先在AndroidManifest中注册provide
<provider android:authorities="com.example.app.provider" android:name=".MyProvider" android:exported="true" android:enabled="true">provider>
创建一个类继承ContentProvider
public class MyProvider extends ContentProvider { //定义数据库 private StudentSQLite studentSQLite; private String TAG="MyProvider"; //定义一个静态布局代码 //TABLE1_DIR代表table1表中所有数据 //TABLE1_ITEM表示table1表中单条数据 public static final int TABLE1_DIR= 0; public static final int TABLE1_ITEM= 1; public static final int TABLE2_DIR= 2; public static final int TABLE2_ITEM= 3; private static UriMatcher uriMatcher; static { //创建UriMatch实例,调用addURI()方法将期望匹配的内容Uri格式传递进去 uriMatcher = new UriMatcher(UriMatcher.NO_MATCH); uriMatcher.addURI("com.example.app.provider","student",TABLE1_DIR); } private ContentResolver contentResolver; @Override public boolean onCreate() { Log.e(TAG, "onCreate*******************************" ); //获取数据表的数据 Context context = getContext(); contentResolver = context.getContentResolver(); studentSQLite = new StudentSQLite(context,"student",null,1); return true; } @Nullable @Override public Cursor query(@NonNull Uri uri, @Nullable String[] strings, @Nullable String s, @Nullable String[] strings1, @Nullable String s1) { Log.e(TAG, "query*******************************" ); Cursor cursor = null; //判断uri调取的是哪张表,查询数据 if(uriMatcher.match(uri) == TABLE1_DIR){ SQLiteDatabase database = studentSQLite.getReadableDatabase(); cursor = database.query("student", strings, s, strings1, null, null, s1); cursor.setNotificationUri(contentResolver,uri); } return cursor; } @Nullable @Override public String getType(@NonNull Uri uri) { Log.e(TAG, "getType*******************************" ); return null; } @Nullable @Override public Uri insert(@NonNull Uri uri, @Nullable ContentValues contentValues) { Uri u = null; if(uriMatcher.match(uri) == TABLE1_DIR){ SQLiteDatabase database = studentSQLite.getWritableDatabase(); // 插入数据 database.insert("student", "name", contentValues); u = ContentUris.withAppendedId(uri,d); contentResolver.notifyChange(u,null); } Log.e(TAG, "insert*******************************" ); return null; } @Override public int delete(@NonNull Uri uri, @Nullable String s, @Nullable String[] strings) { //删除数据 SQLiteDatabase database=studentSQLite.getWritableDatabase(); database.delete("student",s,strings); Log.e(TAG, "delete------------------------------" ); return 0; } @Override public int update(@NonNull Uri uri, @Nullable ContentValues contentValues, @Nullable String s, @Nullable String[] strings) { //更改数据 SQLiteDatabase database=studentSQLite.getWritableDatabase(); database.update("student",contentValues,s,strings); Log.e(TAG, "update*******************************" ); return 0; }}
在增删改查中都要判断是在哪一张表中,可以用switch()方法来判断,这里我就不一一写出了。
要跨的进程中代码,在此代码中实现增删改查操作
public class ContresolverActivity extends AppCompatActivity implements View.OnClickListener{ //定义增删改查的按钮控件 private Button testBtn; private Button addBtn; private Button queryBtn; private Button deleteBtn; private Button updateBtn; private EditText shuruEt; private EditText deleteEt; private String TAG="ContresolverActivity"; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_contresolver); bindID(); } //绑定id private void bindID() { testBtn=findViewById(R.id.test_tv); addBtn=findViewById(R.id.add_tv); queryBtn=findViewById(R.id.query_tv); deleteBtn=findViewById(R.id.delete_tv); updateBtn=findViewById(R.id.update_tv); shuruEt=findViewById(R.id.shuru_edit); deleteEt=findViewById(R.id.delete_edit); testBtn.setOnClickListener(this); addBtn.setOnClickListener(this); queryBtn.setOnClickListener(this); updateBtn.setOnClickListener(this); } //设置监听事件 @Override public void onClick(View view) { switch (view.getId()){ //删除 case R.id.delete_tv: //Uri.parse方法给出路径 Uri uri=Uri.parse("content://com.example.app.provider/student"); String name=deleteEt.getText().toString(); ContentValues values=new ContentValues(); values.put("name",name); ContentResolver resolver=getContentResolver(); resolver.delete(uri,"name=?",new String[]{name}); Log.e(TAG, "*********************************************"); break; //添加 case R.id.add_tv: //content://com.example.app.provider/student/#是第一张表的单条数据 Uri uri1=Uri.parse("content://com.example.app.provider/student/#"); String name1=shuruEt.getText().toString(); ContentValues values1=new ContentValues(); ContentResolver resolver1=getContentResolver(); values1.put("name",name1); resolver1.insert(uri1,values1); Log.e(TAG, values1+"*********************************************"); break; //查询 case R.id.query_tv: Uri uri2=Uri.parse("content://com.example.app.provider/teacher"); ContentResolver resolver2=getContentResolver(); resolver2.query(uri2,null,null,null,null,null); break; //更改 case R.id.update_tv: Uri uri3=Uri.parse("content://com.example.app.provider/student"); String name=deleteEt.getText().toString(); ContentValues values=new ContentValues(); values.put("name",name); ContentResolver resolver3=getContentResolver(); resolver3.update(uri3,values,"name=?",new String[]{name}); break; } }}
更多相关文章
- 一句话锁定MySQL数据占用元凶
- 我的android 第24天 - ContentUris和ContentProvider
- Android系统架构和四大组件
- Android下SD卡文件操作与数据读写
- 告别 USB,用 wifi 进行 Android(安卓)真机调试
- 关于Android(安卓)SQLite3多线程并发问题
- android之DEX文件格式
- Android应用数据存储几种方式(2)
- 页面js判断浏览器类型