BarcodeScanner源码分析

项目地址:BarcodeScanner,本文分析版本: 78278e7

1.简介

项目开发中我们经常遇到需要扫描二维码功能的需求,二维码识别算法我们都知道有zxing,但是它提供的demo里并不支持竖屏扫描,而且定制性也不好,不能拿来即用。所以github上又产生一些对zxing封装的项目,能很简易的集成而且又提供良好定制性。今天我们就来介绍其中一个项目BarcodeScanner,也是我认为相当不错的一个项目。

BarcodeScanner同时提供了zxing扫描方案和zbar扫描方案。zbar的扫描算法是C实现的,扫描速度要比zxing快,但是错误率好像要高于zxing。我并没有在项目中实际使用过zbar,具体是参照这里。本文我们并不分析zxing的实现方式(我也不会。。),我们只分析BarcodeScanner具体的使用和实现方式。

2.使用方法

BarcodeScanner的集成和使用方法相当简单,我们放上一个集成好的Activity代码:


public class SimpleScannerActivity extends BaseScannerActivity implements ZXingScannerView.ResultHandler {
    private ZXingScannerView mScannerView;
    private static final String TAG = "SimpleScannerActivity";

    @Override
    public void onCreate(Bundle state) {
        super.onCreate(state);
        setContentView(R.layout.activity_simple_scanner);
        setupToolbar();

        ViewGroup contentFrame = (ViewGroup) findViewById(R.id.content_frame);
        mScannerView = new ZXingScannerView(this);
        contentFrame.addView(mScannerView);
        Log.d(TAG, "onCreate id : " + Thread.currentThread().getName());
    }

    @Override
    public void onResume() {
        super.onResume();
        mScannerView.setResultHandler(this);
        mScannerView.startCamera();
    }

    @Override
    public void onPause() {
        super.onPause();
        mScannerView.stopCamera();
    }

    @Override
    public void handleResult(Result rawResult) {
        Toast.makeText(this, "Contents = " + rawResult.getText() +
                ", Format = " + rawResult.getBarcodeFormat().toString(), Toast.LENGTH_SHORT).show();
    }
}
  • 首先实现ZXingScannerView.ResultHandler接口,该接口的handleResult(Result rawResult);回调方法就是用来处理扫描出的结果rawResult的.
  • 如果你不需要自定制扫描的界面,BarcodeScanner提供给了我们一个默认实现的样式,所以使用默认样式直接mScannerView = new ZXingScannerView(this);如果需要自定义界面则需要重写ZXingScannerView的父类BarcodeScannerView中的createViewFinderView(Context context)方法,代码如下:

        mScannerView = new ZXingScannerView(this) {
            @Override
            protected IViewFinder createViewFinderView(Context context) {
                //返回我们自定义的IViewFinder
                return new CustomViewFinderView(context);
            }
        };
  • 分别在onResume()onPause()里执行响应的函数。

此外BarcodeScanner还支持:

  • 开关闪光灯
  • 是否开启自动对焦
  • 自定义扫描条形码或者二维码的格式
  • 前后置摄像头切换

代码这里就不贴了,大家可以参照demo中具体的这个类FullScannerActivity

3.类关系图

BarcodeScanner.png

从类图上看大部分类都跟BarcodeScannerView相关,BarcodeScannerView是继承自FrameLayout的,这个布局是用来放置我们用于相机预览的SurfaceView以及我们自定义的扫描界面IViewFinder以及实现控制闪光灯的功能。先大致了解这么多,下面我们详细的来看。

4.源码分析

BarcodeScanner的实现并不复杂,我们也就按照惯例,从它的调用流程开始看:

1.BarcodeScannerView的实现

可以在使用方法中首先看到,我们在ActivityonCreate()方法里实例化了一个ZXingScannerView对象,并添加到我们的布局里。进入ZXingScannerView类里看到它是继承自BarcodeScannerView类的,所以我们再跟进到BarcodeScannerView中去看看具体的实现,发现BarcodeScannerView继承自FrameLayout并且实现了Camera.PreviewCallback接口,下面是BarcodeScannerView的具体实现,省略了部分非主要代码:


public abstract class BarcodeScannerView extends FrameLayout implements Camera.PreviewCallback  {
    //Camera对象
    private Camera mCamera;
    //Camera预览对象,继承自SurfaceView,负责刷新相机预览界面
    private CameraPreview mPreview;
    //自定义的扫描遮罩,控制扫描框的大小
    private IViewFinder mViewFinderView;
    //扫描区域的大小
    private Rect mFramingRectInPreview;
    //相机HandlerThread,负责在子线程中开启相机
    private CameraHandlerThread mCameraHandlerThread;
    //闪光灯状态
    private Boolean mFlashState;
    private boolean mAutofocusState = true;

    public BarcodeScannerView(Context context) {
        super(context);
    }

    public BarcodeScannerView(Context context, AttributeSet attributeSet) {
        super(context, attributeSet);
    }

    public final void setupLayout(Camera camera) {
        removeAllViews();
        //实例化相机预览对象,并添加到一个RelativeLayout里
        mPreview = new CameraPreview(getContext(), camera, this);
        RelativeLayout relativeLayout = new RelativeLayout(getContext());
        relativeLayout.setGravity(Gravity.CENTER);
        relativeLayout.setBackgroundColor(Color.BLACK);
        relativeLayout.addView(mPreview);
        addView(relativeLayout);
        //实例化mViewFinderView对象,并添加到布局中
        mViewFinderView = createViewFinderView(getContext());
        if (mViewFinderView instanceof View) {
            addView((View) mViewFinderView);
        } else {
            throw new IllegalArgumentException("IViewFinder object returned by " +
                    "'createViewFinderView()' should be instance of android.view.View");
        }
    }

    //创建二维码扫描样式,重写此方法即可自定义样式,
    protected IViewFinder createViewFinderView(Context context) {
        return new ViewFinderView(context);
    }

    //通过HandlerThread开启相机
    public void startCamera(int cameraId) {
        if(mCameraHandlerThread == null) {
            mCameraHandlerThread = new CameraHandlerThread(this);
        }
        mCameraHandlerThread.startCamera(cameraId);
    }

    //开启相机预览
    public void setupCameraPreview(Camera camera) {
        mCamera = camera;
        if(mCamera != null) {
            setupLayout(mCamera);
            mViewFinderView.setupViewFinder();
            if(mFlashState != null) {
                setFlash(mFlashState);
            }
            setAutoFocus(mAutofocusState);
        }
    }

    public void startCamera() {
        startCamera(-1);
    }

    //关闭相机
    public void stopCamera() {
        if(mCamera != null) {
            mPreview.stopCameraPreview();
            mPreview.setCamera(null, null);
            mCamera.release();
            mCamera = null;
        }
        if(mCameraHandlerThread != null) {
            mCameraHandlerThread.quit();
            mCameraHandlerThread = null;
        }
    }

    //停止预览
    public void stopCameraPreview() {
        if(mPreview != null) {
            mPreview.stopCameraPreview();
        }
    }

    //恢复预览
    protected void resumeCameraPreview() {
        if(mPreview != null) {
            mPreview.showCameraPreview();
        }
    }

    //得到预览区域中需要扫描的区域的Rect
    public synchronized Rect getFramingRectInPreview(int previewWidth, int previewHeight) {
        if (mFramingRectInPreview == null) {
            Rect framingRect = mViewFinderView.getFramingRect();
            int viewFinderViewWidth = mViewFinderView.getWidth();
            int viewFinderViewHeight = mViewFinderView.getHeight();
            if (framingRect == null || viewFinderViewWidth == 0 || viewFinderViewHeight == 0) {
                return null;
            }

            Rect rect = new Rect(framingRect);
            rect.left = rect.left * previewWidth / viewFinderViewWidth;
            rect.right = rect.right * previewWidth / viewFinderViewWidth;
            rect.top = rect.top * previewHeight / viewFinderViewHeight;
            rect.bottom = rect.bottom * previewHeight / viewFinderViewHeight;

            mFramingRectInPreview = rect;
        }
        return mFramingRectInPreview;
    }

    //省略了部分闪光灯相关代码
}

可以看到BarcodeScannerView是一个抽象类,它主要是用来组织我们用来预览图像的SurfaceView以及决定我们二维码扫描样式的IViewFinder对象,以及提供出一些控制相机和闪光灯的方法。以及实现了Camera.PreviewCallback接口,但是在这里并未实现,所以继承自BarcodeScannerView的子类需要实现这个接口,并处理相机的回调数据。

所以在BarcodeScannerView里有还有三个对象比较重要依次是:CameraPreview,IViewFinderCameraHandlerThread,这些都是具体用来做什么的呢?我相信大家通过名字和其中的一些方法应该已经大致知道,下面我们就一起来看看这些类到底有些什么作用.

2.CameraPreview的实现

CameraPreview是继承自SurfaceView的,做过相机开发的都知道,一般我们需要在SurfaceView中更新相机的预览画面,所以CameraPreview就是一个标准的SurfaceView的实现,这里我们就不再在分析SurfaceView是如何使用的.我只要知道CameraPreview类里是真正控制
相机预览以及显示相机预览的就可以了,具体的代码希望大家自行去研究。
值得一提的是CameraPreview中设置预览画面大小的方法值得我们学习。因为我们知道不同手机使用的摄像头是不同的,而且摄像头所支持的分辨率也不尽相同,而且我们更有可能给我们的扫码界面设置不一样的大小,所以一个好的扫码库一定要做到自适应大小才是最好的,那让我们看看它是怎么做的:


    public void setupCameraParameters() {
        //首先根据当前View的宽高,再根据所有相机支持的分辨率
        //来找出一个最适合的相机Size
        Camera.Size optimalSize = getOptimalPreviewSize();
        //获得相机的参数并且设定
        Camera.Parameters parameters = mCamera.getParameters();
        parameters.setPreviewSize(optimalSize.width, optimalSize.height);
        mCamera.setParameters(parameters);
        //再调整View的宽高
        adjustViewSize(optimalSize);
    }

实现的流程已经在上面了,详细的方法我们这里就不展开说了,有兴趣的可以自己查看.

3.IViewFinder的实现

IViewFinder是一个为了让我们能自定义扫码框而抽象的一个接口,代码如下:


public interface IViewFinder {

    //当相机预览开始的时候调用,一般推荐用来更新扫描框的大小和刷新View
    void setupViewFinder();

    //返回扫码识别的区域
    Rect getFramingRect();

    //返回View的宽度,因为View中已经实现,所以你不需要重写这个方法.
    int getWidth();

    //返回View的高度,因为View中已经实现,所以你不需要重写这个方法.
    int getHeight();
}

IViewFinder是为了帮助我们能自定义扫码框以及扫码的区域抽象出来的一个接口,在前面我们知道.我们在实现任何扫码框的样式的时候都需要继承自View否则在添加时将会抛出异常,实现起来就比较简单了,当然BarcodeScanner中为我们默认实现了一个ViewFinderView。代码我们就补贴了,大家可以自行阅读。

4.CameraHandlerThread的实现


public class CameraHandlerThread extends HandlerThread {
    private static final String LOG_TAG = "CameraHandlerThread";

    private BarcodeScannerView mScannerView;

    public CameraHandlerThread(BarcodeScannerView scannerView) {
        super("CameraHandlerThread");
        mScannerView = scannerView;
        start();
    }

    public void startCamera(final int cameraId) {
        Handler localHandler = new Handler(getLooper());
        localHandler.post(new Runnable() {
            @Override
            public void run() {
                //在子线程中获得camera对象
                final Camera camera = CameraUtils.getCameraInstance(cameraId);
                Handler mainHandler = new Handler(Looper.getMainLooper());
                mainHandler.post(new Runnable() {
                    @Override
                    public void run() {
                        mScannerView.setupCameraPreview(camera);
                    }
                });
            }
        });
    }
}

CameraHandlerThread的实现看起来相当简单,但是它的作用却是相当重要的,根据代码我们可以看到,其实是在子线程中获取到了camera对象,再通过Looper.getMainLooper()初始化一个运行在主线程中的Handler通知mScannerView.setupCameraPreview(camera);。这样处理之后我们整个的相机就运行在子线程中了,包括相机预览的刷新以及Camera.PreviewCallbackonPreviewFrame()方法的回调都在子线程中了。这样做的好处是什么呢?我大致列举了两个

  • 在子线程中初始化相机,能增加扫码Activity的进入速度。
  • 由于相机更新是在子线程中的,那么我们如果再在主线程中实现各种动画就会相当流畅。不会有卡顿现象。

所以CameraHandlerThread还是相当有必要的。

5.ZXingScannerView的实现

所以在ZXingScannerView中只要做初始化Zxing的相关类以及在onPreviewFrame(byte[] data, Camera camera)回调方法中处理我们相机回调的数据扫描即可,为了验证onPreviewFrame(byte[] data, Camera camera)是运行在子线程中我们可以看到方法的最后:


    @Override
    public void onPreviewFrame(byte[] data, Camera camera) {
        if(mResultHandler == null) {
            return;
        }
        
        .....省略部分扫码相关代码

        final Result finalRawResult = rawResult;

        if (finalRawResult != null) {
            Handler handler = new Handler(Looper.getMainLooper());
            handler.post(new Runnable() {
                @Override
                public void run() {
                    ResultHandler tmpResultHandler = mResultHandler;
                    mResultHandler = null;

                    stopCameraPreview();
                    if (tmpResultHandler != null) {
                        tmpResultHandler.handleResult(finalRawResult);
                    }
                }
            });
        } else {
            camera.setOneShotPreviewCallback(this);
        }
    }

通过一个Handler在主线程中回调handleResult()方法。所以到此我们就大致知道了BarcodeScannerZxing的扫码是如何实现的了,聪明的同学应该也已经知道这个库中Zbar扫码的模块也应该是如何实现的了,无非也是继承自BarcodeScannerView只不过是使用Zbar的扫描算法来完成扫码识别,那么到底是不是这样?还欢迎大家把BarcodeScanner这个库clone来自己验证了!好了源码实现我们就写到这。

5.个人评价

BarcodeScanner是我在项目开发中使用过的二维码扫描库中个人觉得结构和可拓展性都比较好的库,在当初使用这个库的时候,它并没有实现CameraHandlerThread,当时自己也实现了在HandlerThread中开启相机的功能,实现的方法这里就不写了,因为BarcodeScanner的这种实现方式要比我自己实现的要好。所以总体来说值得推荐!
多说一句,因为最近眼睛做了手术,所以我只写了一遍博客,并没有review难免有错字或者写错的地方,还希望大家帮忙指出,最后谢谢大家的支持。_

我每周会写一篇源代码分析的文章,以后也可能会有其他主题.
如果你喜欢我写的文章的话,欢迎关注我的新浪微博@达达达达sky
地址: http://weibo.com/u/2030683111
每周我会第一时间在微博分享我写的文章,也会积极转发更多有用的知识给大家.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容

  • 二维码扫描最近两年简直是风靡移动互联网时代,尤其在国内发展神速。围绕条码扫码功能,首先说说通过本文你可以知道啥。一...
    55book阅读 4,134评论 0 1
  • 1. 二维码扫码库介绍 二维码又称QR Code,QR全称Quick Response,是一个近几年来移动设备上超...
    Jinwong阅读 12,208评论 1 40
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,389评论 25 707
  • 这一周,没有按时完成每天的文字任务。总觉得自己太忙,没有抽出整块的时间来写作。 之前,再给别人分享自己如何用碎片化...
    长安笔客阅读 528评论 4 8
  • 拥有爱情是自己的荣耀和幸福,但这荣耀不是逢人就炫耀的资本。 当自己对心仪的对象发起猛追的时候,对方可能也对你有好感...
    骆驼白杨阅读 812评论 0 0