文章目录
- 介绍
- 增加功能的实现
- 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 {
    
    public boolean onCreate() {
        return false;
    }
    
    
    public Cursor query( Uri uri,  String[] strings,  String s,  String[] strings1,  String s1) {
        return null;
    }
    
    
    public String getType( Uri uri) {
        return null;
    }
    
    
    public Uri insert( Uri uri,  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;
    }
    
    public int delete( Uri uri,  String s,  String[] strings) {
        return 0;
    }
    
    public int update( Uri uri,  ContentValues contentValues,  String s,  String[] strings) {
        return 0;
    }
}AndroidManifest注册 Provider
<provider
  android:name=".provider.PersonProvider"
  android:authorities="hello_world"
  android:exported="true" />运行程序,我们先多添加几个数据
 然后新建一个项目 MyApplication2
MainActivity
public class MainActivity extends AppCompatActivity {
    
    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 ,然后再打开第一个项目看下数据,发现已经增加成功了
查询数据的功能实现
PersonProvider
    
    public Cursor query( Uri uri,  String[] strings,  String s,  String[] strings1,  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 {
    
    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);
        }
    }
}先运行第一个项目,然后运行第二个项目,日志输出如下
拓展:修改音乐播放器3.0
我们打开 Device File Exploer,我们在 data/data 下会发现有很多包含 providers 的文件夹,这些是安卓内置的使用 ContentProvider 共享数据的程序
 打开com.android.providers.media中 databases,导出 internal.db
 歌曲信息既然存储在表中,我们可以改进音乐播放器3
新增 MediaStoreMusicDao
public class MediaStoreMusicDao implements IDao<Music> {
    private Context content;
    public MediaStoreMusicDao(Context content) {
        this.content = content;
    }
    
    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
 
<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);
    }
    
    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;
    }
}运行程序:
 同时之前通过 MediaPlayer 获得时长的代码player.getDuration() 可以全部替换成musics.get(currentMusicIndex).getDuration(),可以通过数据库读取了。










