• 识别图书ISBN号并输出查询结果的示例 | Marshal's Blog


    识别图书ISBN号并输出查询结果的示例 | Marshal's Blog

    识别图书ISBN号并输出查询结果的示例

    写了个技术原型,通过手机识别图书的ISBN条形码,然后通过豆瓣网的API,输出ISBN号对应的图书信息。

    image

    示例很简单,上来是个按钮,点击后,进入zxing的activity,用于扫描图书ISBN号。

    image

    使用zxing的扫描功能,这次还出现了点儿问题。我的nexus one摄像头比较脏了,造成对焦后无法识别,或者识别的很慢。吓了我一跳,因为以前没有问题的。

     

    更换了几个版本zxing,问题依旧,擦拭了镜头,问题立刻得到了解决。

    nexus one的像素和自动对焦能力,也可能包括cpu的处理因素,稍稍对中条码,立刻得到结果。之前的G1就要慢一些。

    这也带来了捕获上面截图的困难,因为一瞬间扫描就结束了。怎么办呢?我用手在镜头上揉搓了几下,就可以了,哈哈。

    image

    提示找到图书后,通过豆瓣API,获取到图书的XML,这里要说,豆瓣的图书信息还是比较全的,我用国外图书也找到了信息。

    image

    因为使用联通WCDMA卡,整个过程很快。

    下面说说代码中的一些要点。

    如何扫描条形码

    现在做条形码扫描,java一般用zxing。zxing有多种复用方式,最简单的方式是,安装zxing软件,通过google market。在自己的应用中,通过Intent调用zxing的扫描条码的Activity,然后将扫描到的结果返回自己的Activity。

    比较复杂的办法是,将zxing库包含在自己的程序中,这比较适合正式的应用。

    本文使用的是前者。

    示例代码

    示例源代码见:

    http://easymorse.googlecode.com/svn/tags/sou.book-0.2

    还有一个更为简单的示例,可用于理解调用zxing和豆瓣API的基本过程,见:

    http://easymorse.googlecode.com/svn/tags/sou.book-0.1

    实现activity跳转

    在0.2版本中,涉及到3个activity,首先是搜索图书的activity,见最上面的图。点击按钮后,进入zxing的activity,从视觉上看,zxing获取到ISBN号后,进入了第三个activity,显示图书的详细信息。

    其实,从zxing得到ISBN号以后,还是返回到搜索图书的activity,但是该activity马上发起新的intent,进入新的activity。另外,这里也演示了如果通过intent共享activity之间的参数。

    以下代码创建新的activity,并设置参数:

    Intent intent = new Intent();
    intent.setClass(this, SearchBookActivity.class);
    intent.putExtra("ISBN", data.getExtras().getString("SCAN_RESULT"));
    this.startActivity(intent);

    以下是从intent中获取到参数:

    this.getIntent().getExtras().getString("ISBN");
    有关zxing的使用,可以参考我另外一个示例,android扫描商品条形码

    webview使用本地页面的考虑

    在显示图书查询结果的activity中,使用webview,这样显示可以做的比较灵活和简单。

    显示页面有多种做法,比如可以是远程服务器上的页面,这要求有服务器端的支持。看起来这样比较省心,尤其是比如有iphone,有android等等异构终端的情况下。手机开发人员可以将这部分都空出来,交给服务器端去解决。而且是统一的解决方案。

    但是,实践中,有两个问题:

    • 服务器端开发人员要为不同手机客户端实现不同的样式,而且服务器端开发者往往并不配备手机设备,这也确实是难以做到的,难道要为一个做适配的服务器端开发人员配备iphone、android和s60手机么?
    • 手机终端开发者无法灵活的控制webview中的细节,webview(这里指android环境,iphone环境类似的)支持通过javascript访问手机的服务或者对象,但这是平台相关的,android有自己的代码,iphone也有自己的代码,很难在服务器端统一解决。

    我在这个原型中试图解决,即,HTML页面在手机端本地,手机端和服务器交互,只是取得数据,而不包含显示样式(也就是HTML页面的内容),手机端通过javascript,将取得的数据注入到本地页面中显示。

    这种方式的好处是:

    • 手机端和服务器端交互是获取数据,这是各种手机系统都通用的;
    • 这种方式类似与AJAX,对数据流量的要求也会下降;
    • 手机端开发人员可以灵活的决定页面的样式和交互动作。

    在本例中,将html和css放置在项目的assets目录下,另外,该目录下,还有一张图片,用于未加载到图书图片时占位。想像一下,如果不使用webview,这个需求需要编程处理,是比较繁琐的,可参见android异步加载ListView中的图片中的处理。

    实现webview的代码很简单,这是一个更简单的示例,见android编写简单的Webview。本例相关代码:

    this.resultWeb = (WebView) this.findViewById(R.id.resultWeb);

    this.resultWeb.getSettings().setSupportZoom(false);

    this.resultWeb.getSettings().setJavaScriptCanOpenWindowsAutomatically(

            true);

    this.resultWeb.getSettings().setJavaScriptEnabled(true);

    this.resultWeb.loadUrl(file:///android_asset/results.html);

    这里要注意,setJavaScriptEnabled启用javascript,否则javascript无效。

    android java与html页面中javascript互操作

    javascript和java之间的通信问题,本例只通过javascript调用java,还是比较简单的。如果是java调用javascript,应该需要启动单独的线程来实现。

    需要为webview对象创建一个javascript调用的接口对象:

    this.resultWeb.addJavascriptInterface(new Object() {

        public String getBookName() {

            return bookInfo.getName();

        }

        public String getBookSummary() {

            return bookInfo.getSummary();

        }

        public String getBookImageUrl() {

            return bookInfo.getImageUrl();

        }

        public String getBookAuthor() {

            return bookInfo.getAuthor();

        }

    }, "searchResult");

    其中第一个参数是javascript调用的对象,第二个参数,是调用对象的名称。

    看看javascript中怎样调用这个对象:

    document.getElementById("bookName").innerHTML=window.searchResult.getBookName();

    可以看到window对象多了个searchResult对象属性,这个searchResult就是上面方法中第二个参数。通过该名字,就调用到了上面第一个参数创建的内部类对象。

    使用豆瓣API

    豆瓣API官方网址:

    http://www.douban.com/service/apidoc/

    这里仅用到根据ISBN得到图书信息xml的功能。在这里:

    http://www.douban.com/service/apidoc/reference/subject#获取书籍信息

    如果匿名使用这个API,有一定的限制:

    API调用被限制为每分钟请求不超过10次。

    这里的调用次数是每ip地址的。对于手机应用来说,大于6秒钟扫描一次图书,应该是够用了。

    解析豆瓣XML查询结果

    对豆瓣的xml查询结果,使用了android自带的xmlpull api进行解析。

    这里和android通过xmlpull解析xml稍有差别在于,后者是获取xml的属性值,这里是获取节点中的文本内容。

    需要:

    if (i == XmlPullParser.START_TAG

            && parser.getName().equals("attribute")

            && parser.getAttributeValue(0).equals("title")) {

        bookInfo.setName(parser.nextText());

        Log.v("soubook", "title>>" + bookInfo.getName());

        continue;

    }

    通过parser.nextText()方法获取。另外,要注意在获取完毕后,continue结束本次循环。

    HTML与CSS

    本例中显示查询结果的webview使用了比较难看的灰色背景,主要是为了演示外部css可以很好的和html结合使用。

    同理,也应该可以在html页面中引用外部的javascript,打算在下一个版本中加入jquery。

  • 相关阅读:
    Balanced Number [ZOJ 3416]
    动态树
    Jason的特殊爱好 [FZU 2113]
    Raney引理
    Tri Tiling [POJ 2663]
    糖尿病的虾青素+胰岛素疗法 (转)
    JAVASCRIPT 开发工具:aptana ,WebStorm
    众志和达,英文SOUL 存储与数据安全提供商
    健身音乐及其它
    nodejs + CompoundJS 资源
  • 原文地址:https://www.cnblogs.com/lexus/p/2878389.html
Copyright © 2020-2023  润新知