文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

android中对截图事件进行监听的原理是什么

2023-05-30 23:33

关注

这篇文章给大家介绍android中对截图事件进行监听的原理是什么,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。

监听截屏图片所在目录变化(FileObserver)

监听媒体库的变化(ContentObserver) 

上面两种方法均不是万能的,需要结合使用才能达到良好的效果,首先看看如何监控目录

在android中,我们可以通过FileObserver来监听目录变化,先来看看如何使用

private static final File DIRECTORY_PICTURES = new File(Environment.getExternalStorageDirectory(), Environment.DIRECTORY_PICTURES); private static final File DIRECTORY_DCIM = new File(Environment.getExternalStorageDirectory(), Environment.DIRECTORY_DCIM);  if (manufacturer.equalsIgnoreCase("xiaomi")) {   DIRECTORY_SCREENSHOT = new File(DIRECTORY_DCIM, "Screenshots"); } else {   DIRECTORY_SCREENSHOT = new File(DIRECTORY_PICTURES, "Screenshots"); }  FILE_OBSERVER = new FileObserver(DIRECTORY_SCREENSHOT.getPath(), FileObserver.ALL_EVENTS) {   @Override   public void onEvent(int event, String path) {     if (event == FileObserver.CREATE) {       String newPath = new File(DIRECTORY_SCREENSHOT, path).getAbsolutePath();       Log.d(TAG, "path: " + newPath);     }   } };

我们对指定目录的指定事件监听即可,当事件被触发时onEvent会回调。这里我们只关心目录中有没有新的文件生成。

坑1:在实践中发现,并不是所有手机都允许如此监听或者说都能收到回调。有的手机上面无法收到CREATE事件,但是可以收到其他事件。

我还发现,有的时候收到的事件并没有在FileObserver中定义,比如32768!下面是Linux中相应event对应的含义,32768=IN_IGNORED,但是为什么会ignore,并不清楚。

http://rswiki.csie.org/lxr/http/source/include/linux/inotify.h?a=m68k#L45

还遇到过1073741856(1073741856 = 0x40000000 | 0x20,即IN_OPEN | IN_ISDIR)和1073741840(1073741840 = 0x40000000 | 0x10,即IN_CLOSE_NOWRITE | IN_ISDIR)。

坑2:不同手机,监听的目录并不一致。小米需要监听Environment.DIRECTORY_DCIM,其他监听Environment.DIRECTORY_PICTURES即可。

关于FileObserver这里再多说两句,FileObserver无法进行递归监听,也就是说,我们监听的文件夹中如果有子文件夹,并且我们想知道其中变化,这种方式是不可行的。需要手动对子文件进行操作。

另外,当我们监听的目录/文件被删除后又重新建立了一个同名的目录/文件,之前的FileObserver不会继续工作,需要重新设置监听才行。

还要注意,FileObserver回调并不在主线程中,而是在FileObserver线程中。

鉴于上述原因,我们还要使用方法2,监听媒体库变化。这个方法使用ContentObserver即可。

private static final ContentObserver CONTENT_OBSERVER = new ContentObserver(HANDLER) {   @Override   public void onChange(boolean selfChange, Uri uri) {     //记得先检查读文件的权限     ContentResolver resolver = GeneralInfoHelper.getContext().getContentResolver();     if (uri.toString().matches(MediaStore.Images.Media.EXTERNAL_CONTENT_URI + "(/\\d+)?")) {       Cursor cursor = resolver.query(uri, PROJECTION, null, null, MediaStore.MediaColumns.DATE_ADDED + " DESC");       if (cursor != null && cursor.moveToFirst()) {         //完整路径         String newPath = cursor.getString(cursor.getColumnIndex(MediaStore.MediaColumns.DATA));         File file = new File(newPath);         //file.exists() 判断文件是否存在       }       if (cursor != null) {         cursor.close();       }     }   } };  getContentResolver().registerContentObserver(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, true, CONTENT_OBSERVER);

坑3:实践中发现,并不是所有手机都是监听相同的Uri,有的带数字,有的不带。

坑4:查询数据库时记得按MediaStore.MediaColumns.DATE_ADDED字段排序,注意,这个时间单位是秒,不是毫秒

坑5:即使排了序,你拿到的仍然有可能不是正确的,在魅族E2上面出现了这个问题。但是当我删除了魅族E2截图文件夹之后,一切又恢复正常了……这里我做了一个简单的判断,如何DATE_ADDED和当前时间相差两秒以内,那么从数据库查出的这条数据我视为有效

坑6:当用户删除了截图文件夹的时候,媒体库此时会更新,所以此时onChange会收到大量回调,所以这里需要判断判断文件是否存在。

可能有人会问,为什么不直接用第二种方法?

原因有2,首先从坑5可以看出第二种方法也并非100%有效,其次,这种方法速度很慢,通常会有2-3秒的延迟。而第一种方法如果有效,通常都会比后者快很多。

好了,障碍基本扫清,下面开始融合两种方法

首先使用成员变量记录截图文件路径

private static String sScreenshotPath;

当方法1或者方法2收到结果时,用收到的结果与sScreenshotPath对比,如果是同一个文件,那么就无需再次通知了,否则则进行通知。

逻辑太简单,代码就不写了。但是实际情况是不会这么乐观的。

坑7:在实践中发现,有的系统不直接保存截图,而是先生成一个隐藏文件,比如叫.截图.jpg,然后再修改文件名(去掉“.”)。这种情况下,我们可能就会收到两次用户截图事件的回调(方法1和方法2都可能收到回调),但实际用户只截了一次。

这里我做了一个特殊处理,在判断是否是同一个文件时,只判断文件名,而不去管文件的完整路径也不管文件是否隐藏(也就是不比较文件名前面的“.”)

//仅靠文件名而不是全路径判断是否为同一个截图文件,因为有些系统对截图有move操作 private static boolean isSameFile(String newPath) {   if (TextUtils.isEmpty(sScreenshotPath)) {     return false;   }    return TextUtils.equals(removePrefixDot(new File(sScreenshotPath).getName()), removePrefixDot(new File(newPath).getName())); }  private static String removePrefixDot(@NonNull String filename) {   if (filename.startsWith(".")) {     return filename.substring(1);   }   return filename; }

关于android中对截图事件进行监听的原理是什么就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯