13

我正在开发一个 Android 应用程序,我必须在其中检测 Android SD 卡中音频文件的更改,其中包含文件名、文件路径和对其执行的操作。例如,如果我在我的 SD 卡中添加一个文件,那么我想知道

  1. 添加的文件名
  2. 文件路径
  3. 操作——添加

以前我尝试过文件观察器但为此我必须将它应用于每个目录。所以我搜索了一些其他解决方案并获得了有关Audio.Media.EXTERNAL_CONTENT_URI的信息。然后我创建了一个像这样的内容观察者

UriObserver.java -- 内容观察者

class UriObserver extends ContentObserver {

    public UriObserver(Handler handler) {
        super(handler);
        // TODO Auto-generated constructor stub
    }

    @Override
    public void onChange(boolean selfChange) {
        // TODO Auto-generated method stub
        super.onChange(selfChange);

        Log.d("INSTANT", "GETTING CHANGES");
    }

}

这是它的注册代码

UriObserver observer = new UriObserver(new Handler());

Log.d("INSTANT", "registered content observer");

this.getApplicationContext()
    .getContentResolver()
    .registerContentObserver(
    MediaStore.Images.Media.EXTERNAL_CONTENT_URI, false,
    observer);

Log.d("INSTANT", "registered content observer");

它让我知道与音频文件相关的 sdcard 发生了一些变化。但它没有提供有关添加、编辑或删除哪个文件的任何信息。

然后我搜索了解决方案并得到了这篇文章

Android:通过 MTP 连接时如何检测 MediaStore 的变化

在这篇文章中, Bhiefer给出了一些代码作为答案,我认为它可以工作,所以我试图实现它,但我无法做到。

我能为此做些什么?

更新

我可以查询 Audio.Media.EXTERNAL_CONTENT_URI 的最新更改吗?这段代码:

mCursor = context.getContentResolver().query(
                    Audio.Media.EXTERNAL_CONTENT_URI, null, null, null, "_id");

mCursor.moveToLast();

没有给出最新的变化。还有其他方法可以获取最新更改吗?

4

2 回答 2

9

让我试着放松一下

内容观察者

它不会为您提供有关已更改内容的信息

这是每个设计。文档中没有任何内容表明它将为您提供此信息。

文件观察者

它不是递归的

是的。这是知道的问题。遍历所有目录并设置观察者有什么问题?根据我的理解,默认情况下不应该有很多(比如说十几个)。

Android:通过 MTP 连接时如何检测 MediaStore 的变化

您找到的代码只是包裹在 UriObserver 中的 ContentObserver。

它做了几件事

  • 他获得了一个内容提供的光标(在他的情况下,我相信它是来自 MediaStore 的图像)

  • 他为此注册了一名观察员

  • 一旦发生一些变化,它会将这些变化转发给外部监听器

但是,此解决方案有两个限制:

  • 它继承了 ContentObserver 的问题,即它不报告数据发生了什么。

  • 我相信它只会报告在此 MediaStore 内容提供程序中注册的文件的更改。我相信系统只会扫描 SD 卡上的特殊目录以检查图像等。因此,如果文件将被放置在另一个目录中,则此解决方案将看不到它。

那么,您对他的代码有什么疑问?

概括

在这种情况下,如果您想知道 scdard 上所有文件的确切更改类型,我认为您找不到比 FileObserver 更好的东西

更新 1

结合更多的想法,这可能不适合你。如果您可以根设备,那么您可以选择为文件系统编写过滤器驱动程序,因此每次发生更改时都会调用您的驱动程序。

你可以看看这个链接: http ://www.gossamer-threads.com/lists/linux/kernel/355190

或者您可以重用一些现有的 linux 更改通知系统。例如,看看这个: http ://stefan.buettcher.org/cs/fschange/ 。但是,可能 FileObserver 正是基于它。

无论如何,这两种方法都是低级的,需要更多的时间来弄清楚。

于 2013-04-12T16:48:18.977 回答
1

您可以通过查询 DATE_ADDED 和 DATE_MODIFIED 列来获取最新的添加/修改,但您不会得到 DELETIONS

这是我的做法:

SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(context);
long lastDatabaseUpdateTime = preferences.getLong("lastDatabaseUpdateTime", 0);
long newDatabaseUpdateTime = (new Date()).getTime();

String[] cols = new String[] {MediaStore.Audio.Media._ID /* and other columns */};

String where = "("+MediaStore.MediaColumns.DATE_ADDED + ">" + (lastDatabaseUpdateTime/1000);
where += " or " + MediaStore.MediaColumns.DATE_MODIFIED + ">" + (lastDatabaseUpdateTime/1000);
where += ") and "+MediaStore.Audio.AudioColumns.IS_MUSIC+"==1 ";

Cursor cursor = MusicUtils.query(context, MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, cols, where, null, null);

/* Do my work on the cursor and close the cursor */

//If no exceptions, then save the new timestamp
SharedPreferences.Editor editor = preferences.edit();
editor.putLong("lastDatabaseUpdateTime", newDatabaseUpdateTime);
editor.commit();
于 2015-10-13T20:33:49.927 回答