当前位置:首页 > 嵌入式 > 嵌入式软件
[导读] 如何获取Android设备上的详细的摄像头信息呢? 目前Samsung的Galaxy Tab和Nexus S均有前置摄像头,获取Android摄像头的详细信息,在Android 2.3SDK中得到了增强:在android

 如何获取Android设备上的详细的摄像头信息呢? 目前Samsung的Galaxy Tab和Nexus S均有前置摄像头,获取Android摄像头的详细信息,在Android 2.3SDK中得到了增强:

在android.hardware.Camera类中,API Level 9的SDK中加入了两个比较重要的方法,使用getNumberOfCameras这个static类型方法可以获取当前Android设备上的摄像头数量,比如Nexus S有两个,方法原型如下

public static int getNumberOfCameras ()

而对于具体的每个摄像头的信息,可以通过Camera类的getCameraInfo()这个静态方法获取,该方法有两个参数,参数一的ID,我们通过getNumberOfCameras获取的值减1即可,类似数组索引从0开始一样,用循环遍历每个摄像头信息,参数二是android.hardware.Camera.CameraInfo类,有关getCameraInfo方法的原型如下:

public static void getCameraInfo (int cameraId, Camera.CameraInfo cameraInfo)

对于Camera.CameraInfo类而言,比较简单,包含两个字段

public int facing 代表摄像头的方位,目前有定义值两个分别为CAMERA_FACING_FRONT前置和CAMERA_FACING_BACK后置

public int orientation 下面是拍照的旋转方向,一般自然些有0度、90度、180度和270度,这样可以获取我们正确的手握设备是横着还是竖着,有关拍照时的方向设置,可以参考下面的代码设置

public static void setCameraDisplayOrientation(Activity activity,

int cameraId, android.hardware.Camera camera) {

android.hardware.Camera.CameraInfo info =

new android.hardware.Camera.CameraInfo();

android.hardware.Camera.getCameraInfo(cameraId, info);

int rotation = activity.getWindowManager().getDefaultDisplay()

.getRotation();

int degrees = 0;

switch (rotation) {

case Surface.ROTATION_0: degrees = 0; break;

case Surface.ROTATION_90: degrees = 90; break;

case Surface.ROTATION_180: degrees = 180; break;

case Surface.ROTATION_270: degrees = 270; break;

}

int result;

if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {

result = (info.orientation + degrees) % 360;

result = (360 - result) % 360; // compensate the mirror

} else { // back-facing

result = (info.orientation - degrees + 360) % 360;

}

camera.setDisplayOrientation(result);

}

使用摄像头拍照

1. 使用 SurfaceView 控件来显示摄像头捕捉到的画面

2. 具体细节

/* 获取 SurfaceView 控件 */

SurfaceView surfaceView = (SurfaceView)this.findViewById(R.id.surfaceView);

/* 设置分辨率 */

surfaceView.getHolder().setFixedSize(176, 144);

/*下面设置Surface不维护自己的缓冲区,而是等待屏幕的渲染引擎将内容推送到用户面前*/

surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);

/* 打开摄像头,注意这里是 android.hardware.Camera */

Camera camera = Camera.open();

/* 为 Camera 设置摄像参数 */

Camera.Parameters parameters = camera.getParameters();

/* 设置预览照片的大小,此处设置为全屏 */

WindowManager wm = (WindowManager) getSystemService(Context.WINDOW_SERVICE); // 获取当前屏幕管理器对象

Display display = wm.getDefaultDisplay(); // 获取屏幕信息的描述类

parameters.setPreviewSize(display.getWidth(), display.getHeight()); // 设置

/* 每秒从摄像头捕获5帧画面, */

parameters.setPreviewFrameRate(5);

/* 设置照片的输出格式:jpg */

parameters.setPictureFormat(PixelFormat.JPEG);

/* 照片质量 */

parameters.set("jpeg-quality", 85);

/* 设置照片的大小:此处照片大小等于屏幕大小 */

parameters.setPictureSize(display.getWidth(), display.getHeight());

/* 将参数对象赋予到 camera 对象上 */

camera.setParameters(parameters);

/* 设置用 SurfaceView 作为承载镜头取景画面的显示 */

camera.setPreviewDisplay(surfaceView.getHolder());

/* 开始预览 */

camera.startPreview();

/* 自动对焦 */

camera.autoFocus(null);

/* 拍照片 */

camera.takePicture(null, null, null, jpegCallback);

/* 停止预览 */

camera.stopPreview();

/* 释放摄像头 */

camera.release();

3. 添加使用摄像头的权限

4. 测试: 目前模拟器不支持拍照环境的模拟,必须使用真实手机测试。

5. 代码清单

** string values : strings.xml

view plaincopy to clipboardprint?

手机拍照程序

 

手机拍照程序

 

** AndroidManifest.xml

view plaincopy to clipboardprint?

[!--empirenews.page--]

package="wjh.android.takepicture"

android:versionCode="1"

android:versionName="1.0">

 

 

 

 

package="wjh.android.takepicture"

android:versionCode="1"

android:versionName="1.0">

 

 

 

 

** main.xml

view plaincopy to clipboardprint?

android:orientation="vertical"

android:layout_width="fill_parent"

android:layout_height="fill_parent"

>

android:layout_width="fill_parent"

android:layout_height="fill_parent"

/>

 

android:orientation="vertical"

android:layout_width="fill_parent"

android:layout_height="fill_parent"

>

android:layout_width="fill_parent"

android:layout_height="fill_parent"

/>

 

** MainActicity

view plaincopy to clipboardprint?

public class MainActicity extends Activity {

private Camera camera;

private boolean preview = false ;

@Override

public void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

/*

*设置窗口属性:一定要在 setContentView(R.layout.main) 之前

*/

// 窗口标题

requestWindowFeature(Window.FEATURE_NO_TITLE);

// 全屏

getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);

setContentView(R.layout.main);

SurfaceView surfaceView = (SurfaceView) findViewById(R.id.surfaceView);

surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);

surfaceView.getHolder().setFixedSize(176, 164);

surfaceView.getHolder().addCallback(new SurfaceViewCallback());

}

private final class SurfaceViewCallback implements Callback {

/**

* surfaceView 被创建成功后调用此方法

*/

@Override

public void surfaceCreated(SurfaceHolder holder) {

/*

* 在SurfaceView创建好之后 打开摄像头

* 注意是 android.hardware.Camera

*/

camera = Camera.open();

Camera.Parameters parameters = camera.getParameters();[!--empirenews.page--]

/* 设置预览照片的大小,此处设置为全屏 */

WindowManager wm = (WindowManager) getSystemService(Context.WINDOW_SERVICE); // 获取当前屏幕管理器对象

Display display = wm.getDefaultDisplay(); // 获取屏幕信息的描述类

parameters.setPreviewSize(display.getWidth(), display.getHeight()); // 设置

/* 每秒从摄像头捕获5帧画面, */

parameters.setPreviewFrameRate(5);

/* 设置照片的输出格式:jpg */

parameters.setPictureFormat(PixelFormat.JPEG);

/* 照片质量 */

parameters.set("jpeg-quality", 85);

/* 设置照片的大小:此处照片大小等于屏幕大小 */

parameters.setPictureSize(display.getWidth(), display.getHeight());

/* 将参数对象赋予到 camera 对象上 */

camera.setParameters(parameters);

preview = true;

}

@Override

public void surfaceChanged(SurfaceHolder holder, int format, int width,

int height) {

}

/**

* SurfaceView 被销毁时释放掉 摄像头

*/

@Override

public void surfaceDestroyed(SurfaceHolder holder) {

if(camera != null) {

/* 若摄像头正在工作,先停止它 */

if(preview) {

camera.stopPreview();

preview = false;

}

camera.release();

}

}

}

/**

* 手机键盘按键事件

* 返回 true, 将阻止事件继续传递,例如搜索键,他默认会触发和打开系统的搜索引擎。返回true后,将不会触发。

*/

@Override

public boolean onKeyDown(int keyCode, KeyEvent event) {

/*

* event.getRepeatCount() 为重复按键的次数,例如,快速地对某个键连续按了两次,则此值为一,表示重复了一次。往上可以累推。

* 按键只被按了一次,则此值为 0。

* 这有点类似于鼠标的 "单击" 和 "双击"。

*/

if(camera != null && event.getRepeatCount() == 0 ) {

switch (keyCode) {

case KeyEvent.KEYCODE_SEARCH: // 搜索键

/* 按下搜索键自动对焦 , 如果要关注它的事件,

* 可以实现 AutoFocusCallback 接口,并实例化其对象传入 */

camera.autoFocus(null);

break;

case KeyEvent.KEYCODE_CAMERA: // 拍照键

case KeyEvent.KEYCODE_DPAD_CENTER: // 中间确认键

/*

* @param shutter : 照片被捕获之后的回调对象

* @param raw : 此回调对象可以生产为压缩的图片数据

* @param jpeg : 此回调对象可以产生压缩后的图片数据,其onPictureTaken将被调用

*/

camera.takePicture(null, null, new TakePictureCallback());

/* 拍完照后回到预览状态,继续取景 -- 错误的方式 */

// camera.startPreview();必须写在 onPictureTaken 方法内部,因为 takePicture 内部是另开了一条线程异步的完成保存照片等操作。

// 虽然 takePicture 方法完成了,但是并不代表其内部的工作全部完成,也不代表摄像头以及从上一次“拍照”任务中工作完毕

break;

default:

break;

}

return true;

}

return super.onKeyDown(keyCode, event);

}

/**

* 处理照片被拍摄之后的事件

*/

private final class TakePictureCallback implements PictureCallback {

@Override

public void onPictureTaken(byte[] data, Camera camera) {

Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);

/* 照片将被保存到 SD 卡跟目录下,文件名为系统时间,后缀名为".jpg" */

File file = new File(Environment.getExternalStorageState(), System.currentTimeMillis() + ".jpg");

try {

FileOutputStream fos = new FileOutputStream(file);

/* 位图格式为JPEG

* 参数二位 0-100 的数值,100为最大值,表示无损压缩

* 参数三传入一个输出流对象,将图片数据输出到流中

*/

bitmap.compress(CompressFormat.JPEG, 100, fos);

fos.close();

/* 拍完照后回到预览状态,继续取景 */

camera.startPreview();

} catch (IOException e) {

e.printStackTrace();

}

}

}

}

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭