Android Camera生成bmp格式的图片
来源:互联网 发布:json转化为字符串java 编辑:程序博客网 时间:2024/05/21 23:34
Android Camera拍照默认会生成jpg格式的图片,这是一种有损压缩后的图片格式。前段时间项目需要生成一张无压缩的bmp格式的图片,这就不能通过拍照来实现,而是需要通过预览时的某一帧数据来生成这样的图片。这个过程暂时可以简单的概括为 yuv—-》rgb—-》bmp。
首先,需要进行相机的开发工作,在Android自定义相机实践记录可以完整的看到开发一个相机的过程。
然后,在预览模式下获取数据:
@Override public void surfaceCreated(SurfaceHolder holder) { Log.e(TAG,"surfaceCreated"); mCamera = Camera.open(mCameraIndex); } @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { Log.e(TAG,"surfaceChanged"); //会在surfaceCreated之后至少调用一次 //设置相机的各种参数 if (mCamera != null){ if (mPreviewRunning ) { mCamera.stopPreview(); } Camera.Parameters parameters = mCamera.getParameters(); //获取当前手机支持的相机预览尺寸 List<Camera.Size> sizes = parameters.getSupportedPreviewSizes(); // 设置自动对焦 parameters.setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_VIDEO); parameters.setPreviewSize(sizes.get(0).width, sizes.get(0).height); //设置预览时的数据格式,这个地方可以设置为RGB_565 parameters.setPreviewFormat(ImageFormat.YV12); mCamera.setParameters(parameters); try { mCamera.setPreviewDisplay(holder); //开始预览 mCamera.startPreview(); mPreviewRunning = true; //预览回调监听接口 mCamera.setPreviewCallback(new Camera.PreviewCallback() { @Override public void onPreviewFrame(byte[] bytes, Camera camera) { //这个方法在预览模式下会一直被回调 //在这里获取预览模式下的数据,这里是数据格式会根据setPreviewFormat来决定 saveBMPpicture(mCameraIndex,data,MainActivity.this); } }); } catch (IOException e) { mCamera.release(); mCamera = null; e.printStackTrace(); } } } @Override public void surfaceDestroyed(SurfaceHolder holder) { if (mCamera != null){ mCamera.setPreviewCallback(null); mCamera.stopPreview(); mCamera.release(); mCamera = null; } }public static String saveBMPpicture(int which ,Bitmap bm, Context context) { if(!Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())) { return ACCESS_ERROR; } File file =null; if (which == CameraManager.INDEX_BACK_CAMERA){ file = new File(PHOTO_CAMERA,PHOTO_BACK_CAMERA); }else if(which == CameraManager.INDEX_FRONT_CAMERA){ file = new File(PHOTO_CAMERA,PHOTO_FRONT_CAMERA); } System.out.println(file.getPath()); if(!file.getParentFile().exists()) { file.getParentFile().mkdirs(); } FileOutputStream out = null; try { out = new FileOutputStream(file); } catch (FileNotFoundException e) { e.printStackTrace(); return FILE_ERROR; } int w = bm.getWidth(); int h = bm.getHeight(); int[] pixels = new int[w * h]; bm.getPixels(pixels, 0, w, 0, 0, w, h); byte[] rgb = addBMP_RGB_888(pixels, w, h); byte[] header = addBMPImageHeader(rgb.length); byte[] infos = addBMPImageInfosHeader(w, h); byte[] buffer = new byte[54 + rgb.length]; System.arraycopy(header, 0, buffer, 0, header.length); System.arraycopy(infos, 0, buffer, 14, infos.length); System.arraycopy(rgb, 0, buffer, 54, rgb.length); try { out.write(buffer); out.flush(); out.close(); } catch (IOException e) { e.printStackTrace(); return SAVE_ERROR; } return file.getPath(); } private static byte[] addBMP_RGB_888(int[] b, int w, int h) { int len = b.length; System.out.println(b.length); byte[] buffer = new byte[w * h * 4]; int offset = 0; for (int i = len - 1; i >= w; i -= w) { int end = i, start = i - w + 1; for(int j = start; j <= end; j++) { buffer[offset] = (byte)(b[j] >> 0); buffer[offset + 1] = (byte)(b[j] >> 8); buffer[offset + 2] = (byte)(b[j] >> 16); buffer[offset + 3] = (byte)(b[j] >> 24); offset += 4; } } return buffer; } //BMP文件信息头 private static byte[] addBMPImageInfosHeader(int w, int h) { byte[] buffer = new byte[40]; //这个是固定的 BMP 信息头要40个字节 buffer[0] = 0x28; buffer[1] = 0x00; buffer[2] = 0x00; buffer[3] = 0x00; //宽度 地位放在序号前的位置 高位放在序号后的位置 buffer[4] = (byte) (w >> 0); buffer[5] = (byte) (w >> 8); buffer[6] = (byte) (w >> 16); buffer[7] = (byte) (w >> 24); //长度 同上 buffer[8] = (byte) (h >> 0); buffer[9] = (byte) (h >> 8); buffer[10] = (byte) (h >> 16); buffer[11] = (byte) (h >> 24); //总是被设置为1 buffer[12] = 0x01; buffer[13] = 0x00; //比特数 像素 32位保存一个比特 这个不同的方式(ARGB 32位 RGB24位不同的!!!!) buffer[14] = 0x20; buffer[15] = 0x00; //0-不压缩 1-8bit位图 //2-4bit位图 3-16/32位图 //4 jpeg 5 png buffer[16] = 0x00; buffer[17] = 0x00; buffer[18] = 0x00; buffer[19] = 0x00; //说明图像大小 buffer[20] = 0x00; buffer[21] = 0x00; buffer[22] = 0x00; buffer[23] = 0x00; //水平分辨率 buffer[24] = 0x00; buffer[25] = 0x00; buffer[26] = 0x00; buffer[27] = 0x00; //垂直分辨率 buffer[28] = 0x00; buffer[29] = 0x00; buffer[30] = 0x00; buffer[31] = 0x00; //0 使用所有的调色板项 buffer[32] = 0x00; buffer[33] = 0x00; buffer[34] = 0x00; buffer[35] = 0x00; //不开颜色索引 buffer[36] = 0x00; buffer[37] = 0x00; buffer[38] = 0x00; buffer[39] = 0x00; return buffer; } //BMP文件头 private static byte[] addBMPImageHeader(int size) { byte[] buffer = new byte[14]; //magic number 'BM' buffer[0] = 0x42; buffer[1] = 0x4D; //记录大小 buffer[2] = (byte) (size >> 0); buffer[3] = (byte) (size >> 8); buffer[4] = (byte) (size >> 16); buffer[5] = (byte) (size >> 24); buffer[6] = 0x00; buffer[7] = 0x00; buffer[8] = 0x00; buffer[9] = 0x00; buffer[10] = 0x36; buffer[11] = 0x00; buffer[12] = 0x00; buffer[13] = 0x00; return buffer; }
生成的bmp格式的图片一般都比较大,因为这是原始的无压缩的图片。我们也可以在网上找到一些使用JNI进行转化的方法。使用JNI也是可以的。
0 0
- Android Camera生成bmp格式的图片
- Android camera生成bmp格式的图片
- BMP图片的格式
- BMP图片的格式
- 处理bmp格式的图片
- BMP格式图片的数据格式
- Android解析获取网络上的图片(支持bmp格式)
- delphi根据不同图片生成不规则窗口的实现(仅限于BMP格式)
- PHP读取BMP格式图片的函数
- BMP格式验证码图片的破解。
- Bmp 格式的 图片文件读写
- 关于图片存储格式的整理(BMP)
- 将bmp格式的图片反色
- Flex 读取 bmp 格式的图片
- C语言读取BMP格式的图片
- php处理bmp格式的图片
- PHP读取BMP格式图片的函数
- Qt之生成png/jpg/bmp格式图片
- Linux修改环境变量
- PAT BASIC LEVEL 1011. A+B和C (15)
- java 图片的下载并将其转化为Base64码
- Spring定时任务计划中注入service、数据源的问题
- 2016总结
- Android Camera生成bmp格式的图片
- python中如何把string 转换成int
- 国际资讯丨最近土耳其断电也是由网络攻击造成的
- 如何理解 python UnicodeEncodeError :python 的 string 和 unicode
- 文件分割器的实现
- 工作中python总结
- [教程]手把手教你体验小程序,附送小程序体验清单
- jbpm.hibernate.cfg.xml
- Spring 拦截器