0
点赞
收藏
分享

微信扫一扫

【达内课程】ContentProvider


文章目录

  • ​​介绍​​
  • ​​增加功能的实现​​
  • ​​Unknown URL content://hello_world​​
  • ​​查询数据的功能实现​​
  • ​​拓展:修改音乐播放器3.0​​

介绍

【介绍】
ContentProvider,内容提供者,是 Android 系统的核心组件。用于向其他应用程序提供访问自身数据的机制。

注册 ContentProvider 时,必须配置 ​​android:name​​​、​​android:authorities​​​、​​android:exported=true​​属性。

ContentProvider 向外提供数据访问时,可以提供增删改查这 4 种访问方式中的多种,从实现成本上考虑,通常会结合 SQLite 一起使用。

Android 系统使用 ContentProvider 机制提供了各种系统应用产生的数据的共享方式,例如共享了联系人数据、媒体库数据…

【ContentResolver】
必要的工具类:ContentResolver,访问 ContentResolver 必须使用到 Uri 对象,可以通过 ​​​Uri.parse(String)​​​方法将 String 转为 Uri 类型的数据,在访问 ContentProvider 时,Uri 的字符串格式必须是 ​​content://???​​ 格式的。

Context 类定义了 ​​getContentResolver()​​​方法,用于获取 ContentResolver 对象,并且由 ContextWrapper 重写实现了该方法,所以,Activity 和 Service 均通过继承得到了 ​​getContentResolver()​​方法。

增加功能的实现

在上一节的 ​​联系人项目​​ 基础上做如下修改:

增加 PersonProvider

public class PersonProvider extends ContentProvider {
@Override
public boolean onCreate() {
return false;
}

@Nullable
@Override
public Cursor query(@NonNull Uri uri, @Nullable String[] strings, @Nullable String s, @Nullable String[] strings1, @Nullable String s1) {
return null;
}

@Nullable
@Override
public String getType(@NonNull Uri uri) {
return null;
}

@Nullable
@Override
public Uri insert(@NonNull Uri uri, @Nullable ContentValues contentValues) {
DBOpenHelper dbOpenHelper = new DBOpenHelper(getContext());
SQLiteDatabase db = dbOpenHelper.getWritableDatabase();

String nullColumnHack = "_id";
long id = db.insert("users", nullColumnHack, contentValues);

//释放资源
db.close();
db = null;

return null;
}

@Override
public int delete(@NonNull Uri uri, @Nullable String s, @Nullable String[] strings) {
return 0;
}

@Override
public int update(@NonNull Uri uri, @Nullable ContentValues contentValues, @Nullable String s, @Nullable String[] strings) {
return 0;
}
}

AndroidManifest注册 Provider

<provider
android:name=".provider.PersonProvider"
android:authorities="hello_world"
android:exported="true" />

运行程序,我们先多添加几个数据
【达内课程】ContentProvider_contentprovider
然后新建一个项目 MyApplication2

MainActivity

public class MainActivity extends AppCompatActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//测试增加数据
testInsert();
}

private void testInsert() {
//准备ContentResolver
ContentResolver cr = getContentResolver();
//相当于网址
Uri uri = Uri.parse("content://hello_world");
//向数据库中增加的数据的值
ContentValues values = new ContentValues();
values.put("_name", "Susan");
values.put("_age", 44);
values.put("_phone", "14444444444");
values.put("_email", "susan@qq.com");
//执行增加
cr.insert(uri, values);
}
}

Unknown URL content://hello_world

这是在 Android 11 下才会出现的问题,简单来说,就是出于安全考虑,Android 11 要求应用事先说明需要访问的其他软件包。

解决办法:
在 My Application2 中的 AndroidManifest.xml 中增加

<queries>
<package android:name="com.example.testapplication" />
</queries>

改好程序好,先运行第一个项目,然后运行 MyApplication2 ,然后再打开第一个项目看下数据,发现已经增加成功了
【达内课程】ContentProvider_数据_02

查询数据的功能实现

PersonProvider

@Nullable
@Override
public Cursor query(@NonNull Uri uri, @Nullable String[] strings, @Nullable String s, @Nullable String[] strings1, @Nullable String s1) {
//获取SQLiteDatabase对象
DBOpenHelper dbOpenHelper = new DBOpenHelper(getContext());
SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
//执行业务:查询数据
Cursor c = db.query("users", strings, s, strings1, null, null, s1);
return c;
}

第二个项目中的 MainActivity

public class MainActivity extends AppCompatActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//测试增加数据
//testInsert();
//测试查询数据
testQuery();
}
......
private void testQuery() {
ContentResolver cr = getContentResolver();
Uri uri = Uri.parse("content://hello_world");
//执行查询
Cursor c = cr.query(uri, null, null, null, null);
for (c.moveToFirst(); !c.isAfterLast(); c.moveToNext()) {
int id = c.getInt(c.getColumnIndex("_id"));
String name = c.getString(c.getColumnIndex("_name"));
int age = c.getInt(c.getColumnIndex("_age"));
String phone = c.getString(c.getColumnIndex("_phone"));
String email = c.getString(c.getColumnIndex("_email"));
Log.d("CONTENT_PROVIDER", "id:" + id + ";name:" + name + ";age:" + age + ":phone:" + phone + ";email:" + email);
}
}
}

先运行第一个项目,然后运行第二个项目,日志输出如下
【达内课程】ContentProvider_android_03

拓展:修改音乐播放器3.0

我们打开 Device File Exploer,我们在 data/data 下会发现有很多包含 providers 的文件夹,这些是安卓内置的使用 ContentProvider 共享数据的程序
【达内课程】ContentProvider_android_04
打开​​​com.android.providers.media​​​中 databases,导出 internal.db
【达内课程】ContentProvider_ide_05
歌曲信息既然存储在表中,我们可以改进音乐播放器3

新增 MediaStoreMusicDao

public class MediaStoreMusicDao implements IDao<Music> {
private Context content;

public MediaStoreMusicDao(Context content) {
this.content = content;
}

@Override
public List<Music> getData() {
//声明返回值
List<Music> musics = new ArrayList<>();
//准备ContentResolver
ContentResolver cr = content.getContentResolver();
//准备Uri
Uri uri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
//读数据,获取Cursor对象
String[] projection = {
"_id",//0
"_data",//1->path
"title",//2
"duration",//3
"album",//4
"artist",//5
"album_artist"//
};//读取的字段
String selection = null;
String[] selectionArgs = null;
String sortOrder = null;
Cursor c = cr.query(uri, projection, selection, selectionArgs, sortOrder);
//遍历Cursor,向返回值中添加数据
for (c.moveToFirst(); !c.isAfterLast(); c.moveToNext()) {
Music music = new Music();
music.setId(c.getLong(0));
music.setPath(c.getString(1));
music.setTitle(c.getString(2));
music.setDuration(c.getInt(3));
music.setAlbum(c.getString(4));
music.setArtist(c.getString(5));
music.setAlbumArtist(c.getString(6));
musics.add(music);
}
//释放资源
c.close();
c = null;
return musics;
}
}

修改 MusicDaoFactory

public class MusicDaoFactory {
//如果是单例,就用getInstance();
public static IDao<Music> newInstance(Context context) {
return new MediaStoreMusicDao(context);
}
}

修改 Music 类,我们要存储更多的数据

public class Music {
private long id;
private String title;
private String path;
private int duration;
private String album;
private String artist;
private String albumArtist;

public long getId() {
return id;
}

public void setId(long id) {
this.id = id;
}

public String getTitle() {
return title;
}

public void setTitle(String title) {
this.title = title;
}

public String getPath() {
return path;
}

public void setPath(String path) {
this.path = path;
}

public int getDuration() {
return duration;
}

public void setDuration(int duration) {
this.duration = duration;
}

public String getAlbum() {
return album;
}

public void setAlbum(String album) {
this.album = album;
}

public String getArtist() {
return artist;
}

public void setArtist(String artist) {
this.artist = artist;
}

public String getAlbumArtist() {
return albumArtist;
}

public void setAlbumArtist(String albumArtist) {
this.albumArtist = albumArtist;
}
}

我们之前的音乐播放器只显示了标题和路径,我们可以调整界面,显示更多的内容了

item_music.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:padding="10dp">

<TextView
android:id="@+id/tv_music_title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:maxLines="1"
android:text="title"
android:textColor="#222222"
android:textSize="15dp" />

<TextView
android:id="@+id/tv_music_duration"
android:layout_width="50dp"
android:layout_height="wrap_content"
android:layout_alignParentRight="true"
android:layout_marginLeft="10dp"
android:maxLines="1"
android:text="00:00"
android:textColor="#222222"
android:textSize="15dp" />

<TextView
android:id="@+id/tv_music_album"
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_below="@+id/tv_music_title"
android:maxLines="1"
android:text="path......"
android:textColor="#999999"
android:textSize="15dp" />

<TextView
android:id="@+id/tv_music_artist"
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_below="@+id/tv_music_title"
android:layout_marginLeft="10dp"
android:layout_toRightOf="@+id/tv_music_album"
android:maxLines="1"
android:text="path......"
android:textColor="#999999"
android:textSize="15dp" />
</RelativeLayout>

修改 MusicAdapter

public class MusicAdapter extends BaseAdapter<Music> {
public MusicAdapter(Context context, List data) {
super(context, data);
}

@Override
public View getView(int i, View view, ViewGroup viewGroup) {
Music music = getData().get(i);

ViewHolder holder;

if (view == null) {
view = getInflater().inflate(R.layout.item_music, null);
holder = new ViewHolder();
holder.title = view.findViewById(R.id.tv_music_title);
holder.duration = view.findViewById(R.id.tv_music_duration);
holder.album = view.findViewById(R.id.tv_music_album);
holder.artist = view.findViewById(R.id.tv_music_artist);
view.setTag(holder);
} else {
holder = (ViewHolder) view.getTag();
}

holder.title.setText(music.getTitle());
holder.duration.setText(CommonUtils.getFormattedTime(music.getDuration()));
holder.album.setText(music.getAlbum());
holder.artist.setText(music.getArtist());

return view;
}

class ViewHolder {
TextView title;
TextView duration;
TextView album;
TextView artist;
}
}

运行程序:
【达内课程】ContentProvider_contentprovider_06
同时之前通过 MediaPlayer 获得时长的代码​​​player.getDuration()​​​ 可以全部替换成​​musics.get(currentMusicIndex).getDuration()​​,可以通过数据库读取了。

举报

相关推荐

0 条评论