第一个Android应用 扫描宝 欲挑战传统扫描枪

前言

      很久没有上博客园,更是很久没更新过随笔了。这个小小的Android应用还是去年年底开发的,过完年后一直都很忙,也比较懒,今天在整理资料的时候,觉得这个小小的应用算是学习Android开发的一个毕业作业吧,也跟大家分享一下。

      本来一直在做.net平台的开发,但是项目的需要要做一个移动终端的版本,但是这个移动终端版本并不是用原生开发的,而是基于PhoneGap框架, 开发人员能够使用熟悉的HTML,CSS 和JavaScript 构建跨平台的移动本地应用,也可以开发原生插件配合使用。关于PhoneGap的具体介绍,大家可以google一下,这里只简短说一下优点和不足。

优点:

1、开发效率高,可以使用传统的web技术(html+css+javascript),开发人员不必学习原生开发语言;

2、成本低,一次开发维护一份代码可以打包发布多个平台,主要是Android和IOS平台;

3、可以开发原生插件,配合一些特殊功能实现,毕竟web可以做的事情还是有限的;

不足:

1、性能上和原生应用还是差一点,在IOS上明显比Android流畅,特别是运行动画和手势操作的时候,但是一般的应用型APP还是足够了;

 

     学习Android开发,纯粹是个人兴趣,在去年年底利用业余时间,自学了一下,并且现在移动应用开发这么热门,多学习一点,了解一下还是好的。对于想入门Android开发的童鞋,我也讲一下我学习的过程,供大家参考一下(仅供参考)。我没有买相关的书籍,因为有一点JAVA的基础,跳过了JAVA语言的学习,直接到http://developer.android.com/samples/index.html 上,按照里面的课程把初级和进阶的学习了,高级部分只挑选了一部分内容看了下,然后就去 一些Android论坛找一下相关的DEMO例子,最多的还是有什么疑问不懂的就直接Google,初学的话基本上你遇到的问题,别人都遇到过,也会有很多的解决方法。

      最后就是为什么要开发这样的一个应用,其实就是结合了生活的实际,有个朋友开了个淘宝小店,小卖家一天就10来单,但是每天点发货的时候,手工输入快递单号很麻烦,还要重复核对有没输错,所以就产生了开发一个 扫描宝 应用的想法,既可以满足不想购买市面上的扫描枪,不想手工输入条形码的盆友,又可以把之前学习的知识串联起来,重新巩固一下。

 

功能描述

  首先通过一张顺序图来描述一下该应用的功能:

  这就是该应用的主要功能,其实就是传统扫描枪的功能,将条码扫描到电脑上,只是这里利用现在智能手机的便利性和高配置,使用无线局域网,把这该功能简单化了,还省了买扫描枪的钱。现在市面上一般的扫描枪都要几十RMB,我之前在朋友那里用过一款,从淘宝上买的80大洋,有时扫描条码要几秒十几秒,还有的就是扫描不出来,郁闷到只能手工输入。假如各位童鞋身边有这样需求的朋友,也可以推荐使用以下这个APP,不用花钱。

  其他的辅助小功能,在光线不好的情况下可以开启闪光灯,在夜深人静的时候还在埋头扫描,可以关闭提示音,当然后面如果我还继续给这个应用维护或新增功能的话,就可以检查更新,继续使用新功能了。

 

应用截图

开发思路和关键代码

  Client:

  1、摄像头扫描二维码和条形码

  这部分功能使用Google提供的ZXing开源项目,它提供二维码和条形码的扫描,就不需要自己再去详细研究具体实现了,只要知道怎么用就可以了。假如童鞋需要研究透切,可以下载源码:https://github.com/zxing/zxing ,相信对你的开发水平提高不少。这个开源项目内容很多,但是我这里要用的只是扫描二维码和条形码的功能,所以需要做一些精剪的操作,我也是主要参看以下这两篇博文,http://blog.csdn.net/ryantang03/article/details/7831826另一篇http://www.cnblogs.com/dolphin0520/p/3355728.html

  闪光灯的开启和关闭是用全局变量isLightVisiable标识,通过菜单选项控制,ZXing的CameraManager类已经公开了setTorch方法。

 1     @Override
 2     public boolean onOptionsItemSelected(MenuItem item) {
 3 
 4         switch (item.getItemId()) {
 5         //闪光灯
 6         case ITEM_LIGHT_OFF:
 7             if (this.isLightVisiable) {
 8                 cameraManager.setTorch(false);
 9                 Toast.makeText(this, "闪光  已关", Toast.LENGTH_LONG).show();
10                 item.setTitle("闪光  开");
11             } else {
12                 cameraManager.setTorch(true);
13                 Toast.makeText(this, "闪光  已开", Toast.LENGTH_LONG).show();
14                 item.setTitle("闪光  关");
15             }
16             this.isLightVisiable = !this.isLightVisiable;
17             break;
18         //提示音
19         case ITEM_VOLUE_ON:
20             if (this.isBeepVolue) {
21                 Toast.makeText(this, "提示音   已关", Toast.LENGTH_LONG).show();
22                 item.setTitle("提示音 开");
23             } else {
24                 Toast.makeText(this, "提示音   已开", Toast.LENGTH_LONG).show();
25                 item.setTitle("提示音 关");
26             }
27             this.isBeepVolue = !this.isBeepVolue;
28             break;
29         //帮助
30         case ITEM_HELP:
31             Intent intent = new Intent(this, AboutActivity.class);
32             startActivity(intent);
33             break;
34         //检查更新
35         case ITEM_CHECK_UPDATE:
36             UpdateManager updateManager = new UpdateManager(CaptureActivity.this);
37             updateManager.checkUpdate();
38             break;
39         //退出
40         case ITEM_EXIT:
41             dialog_Exit(this);
42             break;
43         }
44 
45         return false;
46     }

    提示音的开启和关闭时用全局变量isBeepVolue标识,通过菜单选项控制,在ZXing的CaptrueActivity类的方法handleDecode做判断。

 1 public void handleDecode(Result rawResult, Bitmap barcode, float scaleFactor) {
 2         inactivityTimer.onActivity();
 3         lastResult = rawResult;
 4         ResultHandler resultHandler = ResultHandlerFactory.makeResultHandler(
 5                 this, rawResult);
 6 
 7         if (this.isBeepVolue) {
 8             beepManager.playBeepSoundAndVibrate();
 9         }
10         handleDecodeInternally(rawResult, resultHandler, barcode);
11     }

 

  2、扫描二维码与服务端建立连接

  这部分内容主要是socket网络编程的知识,具体的连接方式这里就不介绍了,Google里面有很多,也比较简单。这里介绍一下具体流程。首先,在电脑上启动服务端,生成二维码(后面再细讲),用手机扫描二维码并Decode,在handleDecodeInternally方法判断扫描结果是不是正确的服务器地址,再与服务器做TCP连接。

 1 // Put up our own UI for how to handle the decoded contents.
 2     private void handleDecodeInternally(Result rawResult,
 3             ResultHandler resultHandler, Bitmap barcode) {
 4         // statusView1.setText();
 5         String resultStr = resultHandler.getResult().getDisplayResult();
 6         if (!this.isContentServer) {
 7             if (resultStr != null && resultStr.length() > 0) {
 8                 if (this.socketThread != null) {
 9                     this.socketThread.interrupt();
10                     this.socketThread = null;
11                 }
12                 if (this.socketThread == null) {
13                     String ip = "";
14                     int port = 8099;
15 
16                     String[] results = resultStr.split(":");
17                     if (results.length == 3) {
18                         if (results[0].equals("barcodeServer")) {
19                             if (this.isIPAdress(results[1])) {
20                                 ip = results[1];
21                                 try {
22                                     port = Integer.parseInt(results[2]);
23                                     this.socketThread = new SocketThread(
24                                             this.mHandler, new ProgressHandle(
25                                                     this), ip, port);
26                                     this.socketThread.start();
27                                 } catch (Exception en) {
28                                     showErrorDialog("请扫描正确的服务器二维码");
29                                 }
30 
31                             } else {
32                                 showErrorDialog("请扫描正确的服务器二维码");
33                             }
34                         } else {
35                             showErrorDialog("请扫描正确的服务器二维码.");
36                         }
37                     } else {
38                         showErrorDialog("请扫描正确的服务器二维码");
39                     }
40                 }
41             } else {
42                 showErrorDialog("连接服务器失败");
43             }
44         } else {
45             if (this.socketThread != null) {
            //发送条形码到服务端
46 this.socketThread.SendMessage(resultStr); 47 ClipboardInterface.setText(resultStr, this); 48 Toast.makeText(this, "已经添加到剪切板", Toast.LENGTH_LONG).show(); 49 } else { 50 showErrorDialog("已断开服务器"); 51 ((TextView) findViewById(R.id.status_server)).setText("已断开服务器"); 52 this.isContentServer = false; 53 } 54 } 55 56 new Thread() { 57 public void run() { 58 try { 59 Thread.sleep(3000); 60 Message msg = new Message(); 61 msg.obj = 1; 62 wiatHandler.sendMessage(msg); 63 } catch (InterruptedException e) { 64 e.printStackTrace(); 65 } 66 67 } 68 }.start(); 69

   建立连接后,扫描条形码发送到服务端也是在此方法中进行,其实就是客户端与服务端的相互发送和接收信息,和局域网实时聊天实例原理是一样的。

  3、检查更新

  在Android和IOS两个平台中有比较大的区别,Android比较乱有很多的应用市场,但是IOS基本上就只有AppStore,所以Android应用的话可以在应用市场上做版本更新控制,也可以在应用内部提供检查更新功能,把版本更新文件放置在自己的服务器上。本应用是使用第二种方式,下面介绍一下实现的过程。

  首先将应用安装包APK和版本控制文件XML(主要包括versionCode,appName,appUrl属性),放置到web服务器上,我这里只是简单地放在了博客园的文件管理里。在检查版本更新时先从服务器上把版本控制文件XML获取下来(Android4.2之后不允许在主线程进行HttpConnection的操作,现在网上找的相关资料大部分都比较旧的直接在主线程上下载,4.2以上的需要开启新的线程异步下载,因为这个的提示错误不是很明显,所以说明一下,稍微注意一下就可以了),拿到versionCode与本地安装的应用versionCode比较,服务器的版本号比本地的新时提示下载更新。

 1     public void checkUpdate() {
 2         new DownloadWebpageText(this).execute("http://files.cnblogs.com/lijie198871/barcodeClientUpdate.xml");
 3     }
 4 
 5 @SuppressWarnings("rawtypes")
 6     private class DownloadWebpageText extends AsyncTask {
 7         UpdateManager  updateManager = null;
 8         
 9         public DownloadWebpageText(UpdateManager um){
10             this.updateManager = um;
11         }
12         
13         @Override
14         protected Object doInBackground(Object... params) {
15             // TODO Auto-generated method stub
16             try {
17                 return downloadUrl(params[0].toString());
18             } catch (Exception en) {
19                 return en.getMessage();
20             }
21         }
22 
23         @SuppressWarnings("unchecked")
24         @Override
25         protected void onPostExecute(Object result) {
26             // TODO Auto-generated method stub
27             super.onPostExecute(result);
28 
29             this.updateManager.getVersionCode(Integer.parseInt(result.toString()));
30 //            ((TextView) findViewById(R.id.networkResult)).setText(result
31 //                    .toString());
32         }
33 
34         private String downloadUrl(String myurl) throws IOException {
35             InputStream is = null;
36             HttpURLConnection httpConn = null;
37             
38             try {
39                 URL url = new URL(myurl);
40                 httpConn = (HttpURLConnection) url.openConnection();
41                 httpConn.setReadTimeout(10000);
42                 httpConn.setConnectTimeout(15000);
43                 httpConn.setRequestMethod("GET");
44                 httpConn.setDoInput(true);
45 
46                 httpConn.connect();
47                 int resultCode = httpConn.getResponseCode();
48                 if (resultCode == 200) {
49                     is = httpConn.getInputStream();
50 
51                     PraseXmlService pxmlService = new PraseXmlService();
52                     HashMap<String,String> map = pxmlService.praseXml(is);
53                     return map.get("versionCode");
54                 } else {
55                     return "错误代码:" + resultCode;
56                 }
57 
58             } finally {
59                 if (is != null) {
60                     is.close();
61                 }
62                 if (httpConn != null) {
63                     httpConn.disconnect();
64                 }
65             }
66         }
67         
68         private String readIt(InputStream is, int len) throws IOException {
69             Reader reader = new InputStreamReader(is, "UTF-8");
70             char[] charBuffer = new char[len];
71             reader.read(charBuffer);
72 
73             return new String(charBuffer);
74         }
75     }
从服务器上获取版本控制文件,并读取版本号
 1 public class PraseXmlService {
 2     public HashMap<String,String> praseXml(InputStream inStream) throws Exception
 3     {
 4         HashMap<String,String> updateMap = new HashMap<String,String>();
 5         
 6         DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
 7         DocumentBuilder builder = factory.newDocumentBuilder();
 8         Document document = builder.parse(inStream);
 9         Element element = document.getDocumentElement();
10         
11         NodeList nodeList = element.getChildNodes();
12         int nodeCount = nodeList.getLength();
13         if(nodeList!=null && nodeCount>0){
14             for(int i=0;i<nodeCount;i++){
15                 Node node = nodeList.item(i);
16                 if(node.getNodeType() == Node.ELEMENT_NODE){
17                     Element childElement = (Element)node;
18                     if("versionCode".equals(childElement.getNodeName())){
19                         updateMap.put("versionCode", childElement.getFirstChild().getNodeValue());
20                     }else if("versionName".equals(childElement.getNodeName())){
21                         updateMap.put("versionName", childElement.getFirstChild().getNodeValue());
22                     }else if("url".equals(childElement.getNodeName())){
23                         updateMap.put("url", childElement.getFirstChild().getNodeValue());
24                     }
25                 }
26             }
27         }
28         return updateMap;
29     }
30 }
解析XML文件
 1 private int getAppVersionCode() {
 2         int appVersionCoode = 0;
 3         PackageManager pm = this.context.getPackageManager();
 4         try {
 5             appVersionCoode = pm.getPackageInfo("com.lijie.client.android", 0).versionCode;
 6         } catch (Exception e) {
 7             e.printStackTrace();
 8         }
 9 
10         return appVersionCoode;
11     }
12 
13     private boolean isUpdate(int newVersionCode) {
14         int appVersionCode = this.getAppVersionCode();
15         
16         if (newVersionCode > appVersionCode) {
17             return true;
18         } else {
19             return false;
20         }
21         
22     }
23 
24     private void getVersionCode(int newVersionCode){
25         if (this.isUpdate(newVersionCode)) {
26             this.showNoticeDialog();
27         } else {
28             Toast.makeText(this.context, "当前已经是最新版本", Toast.LENGTH_LONG).show();
29         }
30     }
版本号比较

 

  Server:使用.net framework2.0

  1、生成服务器地址二维码

  首先获取服务器电脑IP,并随机生成端口号,拼接成服务器地址,如:192.168.1.123:8023 。同样使用ZXing的.net framew2.0版本将地址生成二维码。

  1 private void CreateBarcode()
  2         {
  3             serverIp = GetLocalIP();
  4 
  5             Random random = new Random();
  6             while (true)
  7             {
  8                 int port = random.Next(8088, 20480);
  9                 if (!this.isPortUsed(port))
 10                 {
 11                     serverPort = port.ToString();
 12                     break;
 13                 }
 14             }
 15 
 16             EncodingOptions options = null;
 17             BarcodeWriter writer = null;
 18             options = new QrCodeEncodingOptions
 19             {
 20                 DisableECI = true,
 21                 CharacterSet = "UTF-8",
 22                 Width = this.picBarCode.Width,
 23                 Height = this.picBarCode.Height
 24             };
 25             writer = new BarcodeWriter();
 26             writer.Format = BarcodeFormat.QR_CODE;
 27             writer.Options = options;
 28             Bitmap bitmap = writer.Write("barcodeServer:" + this.serverIp + ":" + this.serverPort);
 29             this.picBarCode.Image = bitmap;
 30         }
 31 
 32 
 33 /// <summary>
 34         /// 得到本机IP
 35         /// </summary>
 36         private string GetLocalIP()
 37         {
 38             NetworkInterface[] interfaces = NetworkInterface.GetAllNetworkInterfaces();
 39             foreach (NetworkInterface ni in interfaces)
 40             {
 41                 foreach (UnicastIPAddressInformation ip in
 42                     ni.GetIPProperties().UnicastAddresses)
 43                 {
 44                     if (ip.Address.AddressFamily == System.Net.Sockets.AddressFamily.InterNetwork)
 45                     {
 46                         return ip.Address.ToString();
 47                     }
 48                 }
 49             }
 50 
 51             //本机IP地址
 52             string strLocalIP = "";
 53             //得到计算机名
 54             string strPcName = Dns.GetHostName();
 55             //得到本机IP地址数组
 56             IPAddress[] ipAddress = Dns.GetHostAddresses(strPcName);
 57             //遍历数组
 58             foreach (var IPadd in ipAddress)
 59             {
 60                 //判断当前字符串是否为正确IP地址
 61                 if (IsRightIP(IPadd.ToString()))
 62                 {
 63                     //得到本地IP地址
 64                     strLocalIP = IPadd.ToString();
 65                     //结束循环
 66                     break;
 67                 }
 68             }
 69 
 70             //返回本地IP地址
 71             return strLocalIP;
 72         }
 73 
 74         /// <summary>
 75         /// 判断是否为正确的IP地址
 76         /// </summary>
 77         /// <param name="strIPadd">需要判断的字符串</param>
 78         /// <returns>true = 是 false = 否</returns>
 79         public static bool IsRightIP(string strIPadd)
 80         {
 81             //利用正则表达式判断字符串是否符合IPv4格式
 82             if (Regex.IsMatch(strIPadd, "[0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3}"))
 83             {
 84                 //根据小数点分拆字符串
 85                 string[] ips = strIPadd.Split('.');
 86                 if (ips.Length == 4 || ips.Length == 6)
 87                 {
 88                     //如果符合IPv4规则
 89                     if (System.Int32.Parse(ips[0]) < 256 && System.Int32.Parse(ips[1]) < 256 & System.Int32.Parse(ips[2]) < 256 & System.Int32.Parse(ips[3]) < 256)
 90                         //正确
 91                         return true;
 92                     //如果不符合
 93                     else
 94                         //错误
 95                         return false;
 96                 }
 97                 else
 98                     //错误
 99                     return false;
100             }
101             else
102                 //错误
103                 return false;
104         }
105 
106         private bool isPortUsed(int port)
107         {
108             bool isUsed = false;
109             IPGlobalProperties ipProperties = IPGlobalProperties.GetIPGlobalProperties();
110             IPEndPoint[] ipEndPoints = ipProperties.GetActiveTcpListeners();
111 
112             foreach (IPEndPoint endPoint in ipEndPoints)
113             {
114                 if (endPoint.Port == port)
115                 {
116                     isUsed = true;
117                     break;
118                 }
119             }
120             return isUsed;
121         }
View Code

   2、与客户端建立连接

  这部分其实也是socket的编程,具体也不介绍了,然后输出到光标位置,直接使用 SendKeys.Send("内容") 就可以了。

 

使用方法目前版本需要手机和服务端电脑连接在同一个无线局域网 WIFI下)

  1、下载服务器端,http://files.cnblogs.com/lijie198871/BarcodeScannerServer.rar

  2、下载Android手机客户端到本地电脑再安装到手机,http://files.cnblogs.com/lijie198871/BarcodeClient.apk

    也可以直接扫描二维码安装,或者直接在 豌豆荚 上搜索 扫描宝,就可以直接下载安装了。

  使用UC浏览器扫描或其他扫描工具。

  3、解压BarcodeScannerServer.rar直接运行 扫描宝Server.exe,出现二维码;

  4、使用手机运行 扫描宝 应用,扫描服务端的二维码,与服务器成功连接;

  5、将电脑光标定位到任意输入框,手机随意找一个条形码或二维码 进行扫描,就可以将内容输入到电脑上了;

 

 后续版本畅想

  1、当前版本只局限在WIFI同一个无线局域网下,应该再加上USB的连接方式、蓝牙等等;

  2、如果要按一个产品的定位去发展的话,应该要开发IOS版本,可惜苹果的开发设备太贵,买不起啊,有人资助就好了;

  3、应该加入互联网的发展方式,支持扫描云同步,现在只是简单的扫描条码到电脑上,应该还有很多适用场景,比如超市的库存清点、商品信息查看和更新云同步;快递员送货信息及时更新,一个扫描枪成本不高,但是每个快递员都带一个扫描枪,成本就非常高了, 并且手机基本上每个人都有,而且是越来越新款的智能机,装上一个扫描软件成本几乎为零,只是服务端的成本;当然现在市面上已经有的扫描比价,扫描翻译等等的功能也可以集成进去;有朝一日把传统的扫描枪淘汰掉(貌似想法有点狂妄,但是按目前互联网的发展态势,很多传统行业都在逐步被淘汰,如 拍照开片机、GPS导航等)

  4、盈利模式,还没想到,如果只是靠广告的话利润太低了,而且这样的应用别人很容易仿照复制;

  只是这样想想,后续应该不会做太大的更新了,把源码共享出来,靠各位童鞋建设了,或者放到开源社区,大家一起添砖加瓦;

 

源码下载

  服务端源码:下载

  客户端源码:下载

 

  亲,记得点赞“推荐”哦!

 

  本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接。如有问题,可以邮件:373008218(at)qq(dot)com  联系我,非常感谢。

 

posted @ 2014-03-29 23:20 小李飛菜刀 阅读(...) 评论(...) 编辑 收藏