我在Android Marshmallow中无法使用 FIleObserver 和 ContentObserver 。我正在使用这个东西来检测文件夹内发生的变化。我为marshmallow设置了运行时权限。但在那之后它也没有显示任何事件。它在其他版本中完美运行。请帮我解决这个问题。
首先,我在服务中尝试使用内容解析器来检测后台文件夹的更改。
public class TestService extends Service {
@Override
public void onCreate() {
super.onCreate();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
initial();
return START_STICKY;
}
public void initial(){
getContentResolver().registerContentObserver(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
true,
new ContentObserver(new Handler()) {
@Override
public boolean deliverSelfNotifications() {
Log.d("hai", "deliverSelfNotifications");
return super.deliverSelfNotifications();
}
@Override
public void onChange(boolean selfChange) {
super.onChange(selfChange);
}
@Override
public void onChange(boolean selfChange, Uri uri) {
if (uri.toString().matches(MediaStore.Images.Media.EXTERNAL_CONTENT_URI.toString() + "/[0-9]+")) {
Cursor cursor = null;
try {
cursor = getContentResolver().query(uri, new String[] {
MediaStore.Images.Media.DISPLAY_NAME,
MediaStore.Images.Media.DATA
}, null, null, null);
if (cursor != null && cursor.moveToFirst()) {
final String fileName = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DISPLAY_NAME));
final String path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
// TODO: apply filter on the file name to ensure it's screen shot event
Log.d("file", "FILE CHANGE OCCURED " + fileName + " " + path);
}
} finally {
if (cursor != null) {
cursor.close();
}
}
}
super.onChange(selfChange, uri);
}
}
);
}
}
并将运行时权限视为:
private void getPermission(){
boolean hasPermission = (ContextCompat.checkSelfPermission(this,
Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED);
if (!hasPermission) {
ActivityCompat.requestPermissions(this,
new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},
REQUEST_READ_STORAGE);
}
}
并且获得了onRequestPermissionsResult
的权限结果。
这种方法对我不起作用。所以我尝试在该服务中使用FileObserver
。那个时候它也适用于所有其他平台,但不适用于Marshmallow。
答案 0 :(得分:3)
这似乎是Marshmallow的一个错误,see here。
你只能通过轮询你需要的任何信息来尝试解决它。
这对你有用的程度取决于你的用例。我发现它可用于跟踪下载进度:下载开始时以一秒的间隔开始轮询,并在下载完成时停止。
如果您预计会发生非常罕见的更改,您可以尝试增加时间间隔 - 代价是更改发生之间可能会有更长的延迟,并且您的应用会将其提取起来。