热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android实现后台服务拍照功能

这篇文章主要为大家详细介绍了Android实现后台服务拍照功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

一、背景介绍

最近在项目中遇到一个需求,实现一个后台拍照的功能。一开始在网上寻找解决方案,也尝试了很多种实现方式,都没有满意的方案。不过确定了难点:即拍照要先预览,然后再调用拍照方法。问题也随之而来,既然是要实现后台拍照,就希望能在Service中或者是异步的线程中进行,这和预览这个步骤有点相矛盾。那有什么方式能够既能正常的实现预览、拍照,又不让使用者察觉呢?想必大家也会想到一个取巧的办法:隐藏预览界面。

说明一下,这只是我在摸索中想到的一种解决方案,能很好的解决业务上的需求。对于像很多手机厂商提供的“找回手机”功能时提供的拍照,我不确定他们的实现方式。如果大家有更好的实现方案,不妨交流一下。

关于这个功能是否侵犯了用户的隐私,影响用户的安全等等问题,不在我们的考虑和讨论范围之内。

二、方案介绍

方案实现步骤大致如下:

1.初始化拍照的预览界面(核心部分);
2.在需要拍照时获取相机Camera,并给Camera设置预览界面;
3.打开预览,完成拍照,释放Camera资源(重要)
4.保存、旋转、上传.......(由业务决定)

先大概介绍下业务需求:从用户登录到注销这段时间内,收到后台拍照的指令后完成拍照、保存、上传。以下会基于这个业务场景来详细介绍各步骤的实现。

1.初始化拍照的预览界面

在测试的过程中发现,拍照的预览界面需要在可显示的情况下生成,才能正常拍照,假如是直接创建SurfaceView实例作为预览界面,然后直接调用拍照时会抛出native层的异常:take_failed。想过看源码寻找问题的原因,发现相机核心的功能代码都在native层上面,所以暂且放下,假定的认为该在拍照时该预览界面一定得在最上面一层显示。

由于应用不管是在前台还是按home回到桌面,都需要满足该条件,那这个预览界面应该是全局的,很容易的联想到使用一个全局窗口来作为预览界面的载体。这个全局窗口要是不可见的,不影响后面的界面正常交互。所以,就想到用全局的context来获取WindowManager对象管理这个全局窗口。接下来直接看代码:

package com.yuexunit.zjjk.service; 
 
import com.yuexunit.zjjk.util.Logger; 
 
import android.content.Context; 
import android.view.SurfaceView; 
import android.view.WindowManager; 
import android.view.WindowManager.LayoutParams; 
 
/** 
 * 隐藏的全局窗口,用于后台拍照 
 * 
 * @author WuRS 
 */ 
public class CameraWindow { 
 
  private static final String TAG = CameraWindow.class.getSimpleName(); 
 
  private static WindowManager windowManager; 
 
  private static Context applicationContext; 
 
  private static SurfaceView dummyCameraView; 
 
  /** 
   * 显示全局窗口 
   * 
   * @param context 
   */ 
  public static void show(Context context) { 
    if (applicatiOnContext== null) { 
      applicatiOnContext= context.getApplicationContext(); 
      windowManager = (WindowManager) applicationContext 
          .getSystemService(Context.WINDOW_SERVICE); 
      dummyCameraView = new SurfaceView(applicationContext); 
      LayoutParams params = new LayoutParams(); 
      params.width = 1; 
      params.height = 1; 
      params.alpha = 0; 
      params.type = LayoutParams.TYPE_SYSTEM_ALERT; 
      // 屏蔽点击事件 
      params.flags = LayoutParams.FLAG_NOT_TOUCH_MODAL 
          | LayoutParams.FLAG_NOT_FOCUSABLE 
          | LayoutParams.FLAG_NOT_TOUCHABLE; 
      windowManager.addView(dummyCameraView, params); 
      Logger.d(TAG, TAG + " showing"); 
    } 
  } 
 
  /** 
   * @return 获取窗口视图 
   */ 
  public static SurfaceView getDummyCameraView() { 
    return dummyCameraView; 
  } 
 
  /** 
   * 隐藏窗口 
   */ 
  public static void dismiss() { 
    try { 
      if (windowManager != null && dummyCameraView != null) { 
        windowManager.removeView(dummyCameraView); 
        Logger.d(TAG, TAG + " dismissed"); 
      } 
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
  } 
} 

代码很简单,主要功能就是显示这个窗口、获取用于预览的SurfaceView以及关闭窗口。

在这个业务中,show方法可以直接在自定义的Application类中调用。这样,在应用启动后,窗口就在了,只有在应用销毁(注意,结束所有Activity不会关闭,因为它初始化在Application中,它的生命周期就为应用级的,除非主动调用dismiss方法主动关闭)。

完成了预览界面的初始化,整个实现其实已经非常简单了。可能许多人遇到的问题就是卡在没有预览界面该如何拍照这里,希望这样一种取巧的方式可以帮助大家在以后的项目中遇到无法直接解决问题时,可以考虑从另外的角度切入去解决问题。

2.完成Service拍照功能

这里将对上面的后续步骤进行合并。先上代码:

package com.yuexunit.zjjk.service; 
 
import java.io.File; 
import java.io.FileNotFoundException; 
import java.io.FileOutputStream; 
import java.io.IOException; 
 
import android.app.Service; 
import android.content.Intent; 
import android.graphics.Bitmap; 
import android.graphics.BitmapFactory; 
import android.graphics.BitmapFactory.Options; 
import android.hardware.Camera; 
import android.hardware.Camera.CameraInfo; 
import android.hardware.Camera.PictureCallback; 
import android.os.IBinder; 
import android.os.Message; 
import android.text.TextUtils; 
import android.view.SurfaceView; 
 
import com.yuexunit.sortnetwork.android4task.UiHandler; 
import com.yuexunit.sortnetwork.task.TaskStatus; 
import com.yuexunit.zjjk.network.RequestHttp; 
import com.yuexunit.zjjk.util.FilePathUtil; 
import com.yuexunit.zjjk.util.ImageCompressUtil; 
import com.yuexunit.zjjk.util.Logger; 
import com.yuexunit.zjjk.util.WakeLockManager; 
 
/** 
 * 后台拍照服务,配合全局窗口使用 
 * 
 * @author WuRS 
 */ 
public class CameraService extends Service implements PictureCallback { 
 
  private static final String TAG = CameraService.class.getSimpleName(); 
 
  private Camera mCamera; 
 
  private boolean isRunning; // 是否已在监控拍照 
 
  private String commandId; // 指令ID 
 
  @Override 
  public void onCreate() { 
    Logger.d(TAG, "onCreate..."); 
    super.onCreate(); 
  } 
 
  @Override 
  public int onStartCommand(Intent intent, int flags, int startId) { 
    WakeLockManager.acquire(this); 
    Logger.d(TAG, "onStartCommand..."); 
    startTakePic(intent); 
    return START_NOT_STICKY; 
  } 
 
  private void startTakePic(Intent intent) { 
    if (!isRunning) { 
      commandId = intent.getStringExtra("commandId"); 
      SurfaceView preview = CameraWindow.getDummyCameraView(); 
      if (!TextUtils.isEmpty(commandId) && preview != null) { 
        autoTakePic(preview); 
      } else { 
        stopSelf(); 
      } 
    } 
  } 
 
  private void autoTakePic(SurfaceView preview) { 
    Logger.d(TAG, "autoTakePic..."); 
    isRunning = true; 
    mCamera = getFacingFrontCamera(); 
    if (mCamera == null) { 
      Logger.w(TAG, "getFacingFrontCamera return null"); 
      stopSelf(); 
      return; 
    } 
    try { 
      mCamera.setPreviewDisplay(preview.getHolder()); 
      mCamera.startPreview();// 开始预览 
      // 防止某些手机拍摄的照片亮度不够 
      Thread.sleep(200); 
      takePicture(); 
    } catch (Exception e) { 
      e.printStackTrace(); 
      releaseCamera(); 
      stopSelf(); 
    } 
  } 
 
  private void takePicture() throws Exception { 
    Logger.d(TAG, "takePicture..."); 
    try { 
      mCamera.takePicture(null, null, this); 
    } catch (Exception e) { 
      Logger.d(TAG, "takePicture failed!"); 
      e.printStackTrace(); 
      throw e; 
    } 
  } 
 
  private Camera getFacingFrontCamera() { 
    CameraInfo cameraInfo = new CameraInfo(); 
    int numberOfCameras = Camera.getNumberOfCameras(); 
    for (int i = 0; i  500 * 1024) { 
        opts = new Options(); 
        opts.inSampleSize = 2; 
      } 
      Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length, 
          opts); 
      // 旋转270度 
      Bitmap newBitmap = ImageCompressUtil.rotateBitmap(bitmap, 270); 
      // 保存 
      String fullFileName = FilePathUtil.getMonitorPicPath() 
          + System.currentTimeMillis() + ".jpeg"; 
      File saveFile = ImageCompressUtil.convertBmpToFile(newBitmap, 
          fullFileName); 
      ImageCompressUtil.recyleBitmap(newBitmap); 
      if (saveFile != null) { 
        // 上传 
        RequestHttp.uploadMonitorPic(callbackHandler, commandId, 
            saveFile); 
      } else { 
        // 保存失败,关闭 
        stopSelf(); 
      } 
    } catch (Exception e) { 
      e.printStackTrace(); 
      stopSelf(); 
    } 
  } 
 
  private UiHandler callbackHandler = new UiHandler() { 
 
    @Override 
    public void receiverMessage(Message msg) { 
      switch (msg.arg1) { 
      case TaskStatus.LISTENNERTIMEOUT: 
      case TaskStatus.ERROR: 
      case TaskStatus.FINISHED: 
        // 请求结束,关闭服务 
        stopSelf(); 
        break; 
      } 
    } 
  }; 
 
  // 保存照片 
  private boolean savePic(byte[] data, File savefile) { 
    FileOutputStream fos = null; 
    try { 
      fos = new FileOutputStream(savefile); 
      fos.write(data); 
      fos.flush(); 
      fos.close(); 
      return true; 
    } catch (FileNotFoundException e) { 
      e.printStackTrace(); 
    } catch (IOException e) { 
      e.printStackTrace(); 
    } finally { 
      if (fos != null) { 
        try { 
          fos.close(); 
        } catch (IOException e) { 
          e.printStackTrace(); 
        } 
      } 
    } 
    return false; 
  } 
 
  private void releaseCamera() { 
    if (mCamera != null) { 
      Logger.d(TAG, "releaseCamera..."); 
      mCamera.stopPreview(); 
      mCamera.release(); 
      mCamera = null; 
    } 
  } 
 
  @Override 
  public void onDestroy() { 
    super.onDestroy(); 
    Logger.d(TAG, "onDestroy..."); 
    commandId = null; 
    isRunning = false; 
    FilePathUtil.deleteMonitorUploadFiles(); 
    releaseCamera(); 
    WakeLockManager.release(); 
  } 
 
  @Override 
  public IBinder onBind(Intent intent) { 
    return null; 
  } 
} 

代码也不多,不过有几个点需要特别注意下,

1.相机在通话时是用不了的,或者别的应用持有该相机时也是获取不到相机的,所以需要捕获camera.Open()的异常,防止获取不到相机时应用出错;

2.在用华为相机测试时,开始预览立马拍照,发现获取的照片亮度很低,原因只是猜测,具体需要去查资料。所以暂且的解决方案是让线程休眠200ms,然后再调用拍照。

3.在不使用Camera资源或者发生任何异常时,请记得释放Camera资源,否则为导致相机被一直持有,别的应用包括系统的相机也用不了,只能重启手机解决。代码大家可以优化下, 把非正常业务逻辑统一处理掉。或者是,使用自定义的UncaughtExceptionHandler去处理未捕获的异常。

4.关于代码中WakeLocaManager类,是我自己封装的唤醒锁管理类,这也是大家在处理后台关键业务时需要特别关注的一点,保证业务逻辑在处理时,系统不会进入休眠。等业务逻辑处理完,释放唤醒锁,让系统进入休眠。

三、总结

该方案问题也比较多,只是提供一种思路。全局窗口才是这个方案的核心。相机的操作需要谨慎,获取的时候需要捕获异常(native异常,连接相机错误,相信大家也遇到过),不使用或异常时及时释放(可以把相机对象写成static,然后在全局的异常捕获中对相机做释放,防止在持有相机这段时间内应用异常时导致相机被异常持有),不然别的相机应用使用不了。
代码大家稍作修改就可以使用,记得添加相关的权限。以下是系统窗口、唤醒锁、相机的权限。如果用到自动对焦再拍照,记得声明以下uses-feature标签。其它常用权限这里就不赘述。



以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 在计算机技术的学习道路上,51CTO学院以其专业性和专注度给我留下了深刻印象。从2012年接触计算机到2014年开始系统学习网络技术和安全领域,51CTO学院始终是我信赖的学习平台。 ... [详细]
  • 深入理解 Oracle 存储函数:计算员工年收入
    本文介绍如何使用 Oracle 存储函数查询特定员工的年收入。我们将详细解释存储函数的创建过程,并提供完整的代码示例。 ... [详细]
  • 本文总结了2018年的关键成就,包括职业变动、购车、考取驾照等重要事件,并分享了读书、工作、家庭和朋友方面的感悟。同时,展望2019年,制定了健康、软实力提升和技术学习的具体目标。 ... [详细]
  • CSS 布局:液态三栏混合宽度布局
    本文介绍了如何使用 CSS 实现液态的三栏布局,其中各栏具有不同的宽度设置。通过调整容器和内容区域的属性,可以实现灵活且响应式的网页设计。 ... [详细]
  • Linux 系统启动故障排除指南:MBR 和 GRUB 问题
    本文详细介绍了 Linux 系统启动过程中常见的 MBR 扇区和 GRUB 引导程序故障及其解决方案,涵盖从备份、模拟故障到恢复的具体步骤。 ... [详细]
  • 本文探讨了Hive中内部表和外部表的区别及其在HDFS上的路径映射,详细解释了两者的创建、加载及删除操作,并提供了查看表详细信息的方法。通过对比这两种表类型,帮助读者理解如何更好地管理和保护数据。 ... [详细]
  • 本文介绍了如何使用jQuery根据元素的类型(如复选框)和标签名(如段落)来获取DOM对象。这有助于更高效地操作网页中的特定元素。 ... [详细]
  • 本文将详细介绍如何使用剪映应用中的镜像功能,帮助用户轻松实现视频的镜像效果。通过简单的步骤,您可以快速掌握这一实用技巧。 ... [详细]
  • 深入理解Cookie与Session会话管理
    本文详细介绍了如何通过HTTP响应和请求处理浏览器的Cookie信息,以及如何创建、设置和管理Cookie。同时探讨了会话跟踪技术中的Session机制,解释其原理及应用场景。 ... [详细]
  • 本文介绍如何在 Xcode 中使用快捷键和菜单命令对多行代码进行缩进,包括右缩进和左缩进的具体操作方法。 ... [详细]
  • 本文介绍了一款用于自动化部署 Linux 服务的 Bash 脚本。该脚本不仅涵盖了基本的文件复制和目录创建,还处理了系统服务的配置和启动,确保在多种 Linux 发行版上都能顺利运行。 ... [详细]
  • 在Linux系统中配置并启动ActiveMQ
    本文详细介绍了如何在Linux环境中安装和配置ActiveMQ,包括端口开放及防火墙设置。通过本文,您可以掌握完整的ActiveMQ部署流程,确保其在网络环境中正常运行。 ... [详细]
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
  • 如何在WPS Office for Mac中调整Word文档的文字排列方向
    本文将详细介绍如何使用最新版WPS Office for Mac调整Word文档中的文字排列方向。通过这些步骤,用户可以轻松更改文本的水平或垂直排列方式,以满足不同的排版需求。 ... [详细]
author-avatar
非船_725
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有