Android上使用ZXing识别条形码与二维码的方法

 更新时间:2014年08月09日 16:18:18   投稿:shichen2014  
这篇文章主要介绍了Android上使用ZXing识别条形码与二维码的方法,需要的朋友可以参考下

目前有越来越多的手机具备自动对焦的拍摄功能,这也意味着这些手机可以具备条码扫描的功能。手机具备条码扫描的功能,可以优化购物流程,快速存储电子名片(二维码)等。

本文所述实例就使用了ZXing 1.6实现条码/二维码识别。ZXing是个很经典的条码/二维码识别的开源类库,早在很久以前,就有开发者在J2ME上使用ZXing了,只不过需要支持JSR-234规范(自动对焦)的手机才能发挥其威力,而目前已经有不少Android手机具备自动对焦的功能。

本文代码运行的结果如下,使用91手机助手截图时,无法截取SurfaceView的实时图像:

本文使用了ZXing1.6的core,即把/zxing-1.6/core/下的src复制覆盖工程的src;另外还要使用到/zxing-1.6/android/下的PlanarYUVLuminanceSource.java。

此处需要注意:/zxing-1.6/android/ 是BarcodeScanner的源码,本文程序相当于BarcodeScanner的精简版,只保留最基本的识别功能。

本文完整源码点击此处本地下载

源码目录结果如下图,ChecksumException.java下面还有很多源文件,截图尚未列出:

本文例子必须要开摄像头和自动对焦的权限,不然启动时会报异常,所用的权限如下:

<uses-permission android:name="android.permission.CAMERA"></uses-permission>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>
<uses-feature android:name="android.hardware.camera" />
<uses-feature android:name="android.hardware.camera.autofocus" />

main.xml源码如下,main.xml必须要用到FrameLayout才能重叠控件实现“范围框”的效果:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout android:id="@+id/FrameLayout01"
 android:layout_width="fill_parent" android:layout_height="fill_parent"
 xmlns:android="http://schemas.android.com/apk/res/android">
 <SurfaceView android:layout_height="fill_parent"
 android:id="@+id/sfvCamera" android:layout_width="fill_parent"></SurfaceView>
 <RelativeLayout android:id="@+id/RelativeLayout01"
 android:layout_height="fill_parent" android:layout_width="fill_parent">
 <ImageView android:id="@+id/ImageView01"
 android:layout_height="100dip" android:layout_width="160dip"></ImageView>
 <View android:layout_centerVertical="true"
 android:layout_centerHorizontal="true" android:layout_width="300dip"
 android:background="#55FF6666" android:id="@+id/centerView"
 android:layout_height="180dip"></View>
 <TextView android:layout_centerHorizontal="true"
 android:layout_width="wrap_content" android:layout_below="@+id/centerView"
 android:layout_height="wrap_content" android:text="Scanning..."
 android:id="@+id/txtScanResult" android:textColor="#FF000000"></TextView>
 </RelativeLayout>
</FrameLayout>

testCamera.java是主类,负责控制Camera和对图像做解码,源码如下:

package com.testCamera;
import java.util.Timer;
import java.util.TimerTask;
import com.google.zxing.BinaryBitmap;
import com.google.zxing.MultiFormatReader;
import com.google.zxing.Result;
import com.google.zxing.Android.PlanarYUVLuminanceSource;
import com.google.zxing.common.HybridBinarizer;
import android.app.Activity;
import android.graphics.Bitmap;
import android.hardware.Camera;
import android.os.Bundle;
import android.view.SurfaceView;
import android.view.View;
import android.widget.ImageView;
import android.widget.TextView;
public class testCamera extends Activity {
 /** Called when the activity is first created. */
 private SurfaceView sfvCamera;
 private SFHCamera sfhCamera;
 private ImageView imgView;
 private View centerView;
 private TextView txtScanResult;
 private Timer mTimer;
 private MyTimerTask mTimerTask;
 // 按照标准HVGA
 final static int width = 480;
 final static int height = 320;
 int dstLeft, dstTop, dstWidth, dstHeight;
 @Override
 public void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.main);
 this.setTitle("Android条码/二维码识别Demo-----hellogv");
 imgView = (ImageView) this.findViewById(R.id.ImageView01);
 centerView = (View) this.findViewById(R.id.centerView);
 sfvCamera = (SurfaceView) this.findViewById(R.id.sfvCamera);
 sfhCamera = new SFHCamera(sfvCamera.getHolder(), width, height,
 previewCallback);
 txtScanResult=(TextView)this.findViewById(R.id.txtScanResult);
 // 初始化定时器
 mTimer = new Timer();
 mTimerTask = new MyTimerTask();
 mTimer.schedule(mTimerTask, 0, 80);
 }
 class MyTimerTask extends TimerTask {
 @Override
 public void run() {
 if (dstLeft == 0) {//只赋值一次
 dstLeft = centerView.getLeft() * width
  / getWindowManager().getDefaultDisplay().getWidth();
 dstTop = centerView.getTop() * height
  / getWindowManager().getDefaultDisplay().getHeight();
 dstWidth = (centerView.getRight() - centerView.getLeft())* width
  / getWindowManager().getDefaultDisplay().getWidth();
 dstHeight = (centerView.getBottom() - centerView.getTop())* height
  / getWindowManager().getDefaultDisplay().getHeight();
 }
 sfhCamera.AutoFocusAndPreviewCallback();
 }
 }
 /**
 * 自动对焦后输出图片
 */
 private Camera.PreviewCallback previewCallback = new Camera.PreviewCallback() {
 @Override
 public void onPreviewFrame(byte[] data, Camera arg1) {
 //取得指定范围的帧的数据
 PlanarYUVLuminanceSource source = new PlanarYUVLuminanceSource(
  data, width, height, dstLeft, dstTop, dstWidth, dstHeight);
 //取得灰度图
 Bitmap mBitmap = source.renderCroppedGreyscaleBitmap();
 //显示灰度图
 imgView.setImageBitmap(mBitmap);
 BinaryBitmap bitmap = new BinaryBitmap(new HybridBinarizer(source));
 MultiFormatReader reader = new MultiFormatReader();
 try {
 Result result = reader.decode(bitmap);
 String strResult = "BarcodeFormat:"
  + result.getBarcodeFormat().toString() + " text:"
  + result.getText();
 txtScanResult.setText(strResult);
 } catch (Exception e) {
 txtScanResult.setText("Scanning");
 }
 }
 };
}

SFHCamera.java是Camera控制类,源码如下:

package com.testCamera;
import java.io.IOException;
import android.graphics.PixelFormat;
import android.hardware.Camera;
import android.util.Log;
import android.view.SurfaceHolder;
public class SFHCamera implements SurfaceHolder.Callback{
 private SurfaceHolder holder = null;
 private Camera mCamera;
 private int width,height;
 private Camera.PreviewCallback previewCallback;
 public SFHCamera(SurfaceHolder holder,int w,int h,Camera.PreviewCallback previewCallback) {
 this.holder = holder; 
 this.holder.addCallback(this); 
 this.holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
  width=w;
  height=h;
  this.previewCallback=previewCallback;
 }
 @Override
 public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2, int arg3) {
 Camera.Parameters parameters = mCamera.getParameters(); 
  parameters.setPreviewSize(width, height);//设置尺寸 
  parameters.setPictureFormat(PixelFormat.JPEG);
  mCamera.setParameters(parameters); 
  mCamera.startPreview();//开始预览
  Log.e("Camera","surfaceChanged");
 }
 @Override
 public void surfaceCreated(SurfaceHolder arg0) {
 mCamera = Camera.open();//启动服务 
  try { 
   mCamera.setPreviewDisplay(holder);//设置预览 
   Log.e("Camera","surfaceCreated");
  } catch (IOException e) { 
   mCamera.release();//释放 
   mCamera = null; 
  }
 }
 @Override
 public void surfaceDestroyed(SurfaceHolder arg0) {
 mCamera.setPreviewCallback(null);
 mCamera.stopPreview();//停止预览 
  mCamera = null;
  Log.e("Camera","surfaceDestroyed");
 }
 /**
 * 自动对焦并回调Camera.PreviewCallback
 */
 public void AutoFocusAndPreviewCallback()
 {
 if(mCamera!=null)
 mCamera.autoFocus(mAutoFocusCallBack);
 }
 /**
  * 自动对焦
  */
 private Camera.AutoFocusCallback mAutoFocusCallBack = new Camera.AutoFocusCallback() { 
  @Override 
  public void onAutoFocus(boolean success, Camera camera) {  
   if (success) { //对焦成功,回调Camera.PreviewCallback
    mCamera.setOneShotPreviewCallback(previewCallback); 
   } 
  } 
 };
}

其中testCamera.java的Camera.PreviewCallback previewCallback 是整个程序的逻辑核心,作为回调函数给SFHCamera.java的内部Camera类调用。

PS:本站还提供了一个功能非常强大的二维码生成工具,感兴趣的朋友可以参考一下:

http://tools.jb51.net/transcoding/jb51qrcode

希望本文实例对大家学习Android程序设计能有所帮助。

相关文章

  • 强制去除Unity自动添加的Android隐私权限

    强制去除Unity自动添加的Android隐私权限

    大家好,本篇文章主要讲的是强制去除Unity自动添加的Android隐私权限,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2021-12-12
  • 使用Flutter 构建Web应用逻辑解析

    使用Flutter 构建Web应用逻辑解析

    这篇文章主要为大家介绍了使用Flutter 构建Web应用逻辑解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-12-12
  • Android自定义View之渐变色折线图的实现

    Android自定义View之渐变色折线图的实现

    折线图的实现方法在github上有很多开源的程序,但是对于初学者来讲,简单一点的教程可能更容易入门,下面这篇文章主要给大家介绍了关于Android自定义View之渐变色折线图的相关资料,需要的朋友可以参考下
    2022-04-04
  • Android RxJava与Retrofit结合使用详解

    Android RxJava与Retrofit结合使用详解

    RxJava和Retrofit的结合使用估计已经相当普遍了,自己工作中也是一直都在使用。在使用的过程中我们都会对其进行封装使用,GitHub上也有很多封装好的项目可以直接拿来使用,其实对于开源框架的二次封装有时候针对不同的业务逻辑封装的过程中也多多少少有些不同
    2023-03-03
  • Android Fragment实现列表和内容联动

    Android Fragment实现列表和内容联动

    这篇文章主要为大家详细介绍了Android Fragment实现列表和内容联动效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-01-01
  • Android向node.js编写的服务器发送数据并接收请求

    Android向node.js编写的服务器发送数据并接收请求

    这篇文章主要为大家详细介绍了Android向node.js编写的服务器发送数据,并接收请求,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10
  • Android 游戏开发中绘制游戏触摸轨迹的曲线图

    Android 游戏开发中绘制游戏触摸轨迹的曲线图

    本文主要介绍 Android 绘制游戏触摸轨迹的曲线图的简单示例,这里详细说明使用方法,并附示例代码,有需要的小伙伴可以参考下
    2016-08-08
  • Android集成微信登录的步骤详解

    Android集成微信登录的步骤详解

    这篇文章给大家整理Android集成微信登录的详细步骤,通过文章整理的内容,大家只需要几行代码就可以实现微信登录的功能了,刚兴趣的朋友们下面来一起看看吧。
    2016-09-09
  • Android TabWidget底部显示效果

    Android TabWidget底部显示效果

    这篇文章主要为大家详细介绍了Android TabWidget底部显示效果的三种方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-03-03
  • Android实现文字上下滚动效果

    Android实现文字上下滚动效果

    这篇文章主要为大家详细介绍了Android实现文字上下滚动效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12

最新评论