Android后台服务拍照的解决方式

摘要:
实现一个后台拍照的功能。一開始在网上寻找解决方式。说明一下,这仅仅是我在摸索中想到的一种解决方式。能非常好的解决业务上的需求。然后直接调用拍照时会抛出native层的异常:take_failed。接下来直接看代码:packagecom.yuexunit.zjjk.service;importcom.yuexunit.zjjk.util.Logger;importandroid.content.Context;importandroid.view.SurfaceView;importandroid.view.WindowManager;importandroid.view.WindowManager.LayoutParams;/***隐藏的全局窗体。主要功能就是显示这个窗体、获取用于预览的SurfaceView以及关闭窗体。
一、背景介绍

近期在项目中遇到一个需求。实现一个后台拍照的功能。

一開始在网上寻找解决方式。也尝试了非常多种实现方式,都没有惬意的方案。只是确定了难点:即拍照要先预览,然后再调用拍照方法。问题也随之而来。既然是要实现后台拍照,就希望能在Service中或者是异步的线程中进行,这和预览这个步骤有点相矛盾。

那有什么方式可以既能正常的实现预览、拍照,又不让使用者察觉呢?想必大家也会想到一个取巧的办法:隐藏预览界面。

说明一下,这仅仅是我在摸索中想到的一种解决方式。能非常好的解决业务上的需求。

对于像非常多手机厂商提供的“找回手机”功能时提供的拍照。我不确定他们的实现方式。假设大家有更好的实现方案。最好还是交流一下。

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

二、方案介绍

方案实现步骤大致例如以下:

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

先大概介绍下业务需求:从用户登录到注销这段时间内,收到后台拍照的指令后完毕拍照、保存、上传。

下面会基于这个业务场景来具体介绍各步骤的实现。

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

在測试的过程中发现,拍照的预览界面须要在可显示的情况下生成,才干正常拍照,假如是直接创建SurfaceView实例作为预览界面。然后直接调用拍照时会抛出native层的异常:take_failed。想过看源代码寻找问题的解决办法。发现相机核心的功能代码都在native层上面,所以暂且放下。假定的觉得该在拍照时该预览界面一定得在最上面一层显示。
因为应用无论是在前台还是按home回到桌面,都须要满足该条件,那这个预览界面应该是全局的,非常easy的联想到使用一个全局窗体来作为预览界面的载体。

这个全局窗体要是不可见的。不影响后面的界面正常交互。所以。就想到用全局的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();
		}
	}
}

代码非常easy。主要功能就是显示这个窗体、获取用于预览的SurfaceView以及关闭窗体。
在这个业务中,show方法能够直接在自己定义的Application类中调用。这样。在应用启动后,窗体就在了,仅仅有在应用销毁(注意,结束全部Activity不会关闭,由于它初始化在Application中,它的生命周期就为应用级的,除非主动调用dismiss方法主动关闭)。
完毕了预览界面的初始化。整个实现事实上已经很easy了。

可能很多人遇到的问题就是卡在没有预览界面该怎样拍照这里,希望这样一种取巧的方式能够帮助大家在以后的项目中遇到无法直接解决这个问题时。能够考虑从另外的角度切入去解决这个问题。

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 < numberOfCameras; i++) {
			Camera.getCameraInfo(i, cameraInfo);
			if (cameraInfo.facing == CameraInfo.CAMERA_FACING_FRONT) {
				try {
					return Camera.open(i);
				} catch (Exception e) {
					e.printStackTrace();
				}
			}
		}
		return null;
	}

	@Override
	public void onPictureTaken(byte[] data, Camera camera) {
		Logger.d(TAG, "onPictureTaken...");
		releaseCamera();
		try {
			// 大于500K,压缩预防内存溢出
			Options opts = null;
			if (data.length > 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标签。其他经常使用权限这里就不赘述。
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.CAMERA" />

免责声明:文章转载自《Android后台服务拍照的解决方式》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇ubuntu 设置静态ipGolang: 接收GET和POST参数下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

PyQt5-按钮关闭窗体-4

import sys from PyQt5.QtWidgets import QApplication, QWidget,QToolTip,QPushButton from PyQt5.QtGui import QIcon,QFont from PyQt5.QtCore import QCoreApplication #demo_4:通过一个按钮关闭窗体...

VB.NET Winform的一些功能实现

近段时间,开发的需要,需要写一个winform的程序。用VB.NET来写。 开发开始,需要实现一个窗体设为多文档界面 (MDI) 子窗体的容器。实现这个功能,开始找资料,得知设置一个属性:Form.IsMdiContainer,它默认值为False,没为True即可。 或者是form Load时添加一句程序: 接下来,又需要为窗体容器的背景设置颜色: M...

实现WinForm窗体的美化(借助第三方控件)

在winform项目中,其实皮肤就是一个第三方的控件,名字是IrisSkin4.dll只要添加到你的工具箱里就可以和其它控件一样使用了 一.添加控件IrisSkin4.dll。方法: 先把IrisSkin4.dll文件添加到当前项目引用(解决方案资源管理器->当前项目->引用->右键->添加引用,找到IrisSkin4.dll文件....

WPF 窗体程序入口简介

1,直接指定StartupUri为某一个window的子类Window1.xaml(属性指定法) <Application x:Class="brush.App"xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.micros...

解决关闭窗口,C#报错"在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke"

情况:在C#开发的过程中多线程委托是经常用的,今天在测试以前写的软件的时候发现有个问题,报 在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke。 这样的错误。 解决方法:加上 if (this.IsHandleCreated) 1、首先分析问题,句柄:是对象的引用名,存于栈区(可以理解为对象的指针),对象是存于堆区,通过操控栈区...

winform程序中界面的跳转问题

 首先是我们进行窗口间的跳转,尤其注意的是winform程序里面的空间都是中线程安全的。但是注意的是如果你在一个线程中操纵另外的控件,这时候会提示你一个错误,这个错误的解决方法准备单独的在另一篇文章中来讲。    好了,这时候回到主题上吧,怎么来实现界面间的跳转吧!假设我们想要从Form1跳转到Form2,那么可以有以下的集中方法:   1. 使用for...