【Android】多功能二维码实现思路,自动连接WI-FI

来源:互联网 发布:金坛网络维护维修电话 编辑:程序博客网 时间:2024/04/30 03:31

现在项目的需求是:
1. 带AP功能的机顶盒端能生成二维码,供手机客户端扫描
1.1 如果用非特定应用(手机助手)扫描,则跳转下载手机助手界面
1.2 如果用手机助手扫描,自动连接到该机顶盒的WI-FI
2. 不带AP功能的机顶盒也能生成二维码
2.1 同1.1
2.2 如果用手机助手扫描,自动连接到该机顶盒所连接到的WI-FI

首先,必须了解什么是二维码?

简单来说,二维码就是把一段纯文本用图形样式转换出来了,以便于快速扫描读出。

现代应用中,二维码最常存的文本就是URL,所以也可以想象成二维码其实就是一个URL地址。所以扫描二维码可以跳转到某个界面。

比如我现在做的项目的二维码URL是“http://appproxy.topway.cn:8080/index.htm”,用手机端打开就能够跳转到下载页面,用电脑端打开却显示不出内容。这是因为服务器端会对访问端进行判断,看是否是移动设备,然后进行相应的操作(跳转下载界面也要区分iOS和Android端)。

为什么用微信扫一扫能直接加关注某个人而不是跳转URL呢?

这是因为微信是一个”特别”的应用,扫的是”特别”的二维码。

学过web开发的都知道,网络请求有一种GET方式,是直接把参数放在URL后面的,比如下面扩展的URL:

http://appproxy.topway.cn:8080/index.htm?ssid=xxx&pwd=xxx

这个字符串就带了WI-FI名称和密码。像一些速食店现在都有一个二维码贴在桌子上的,目的就是让用户扫一扫然后自动连接上WI-FI了。但是前提是要用他们公司的应用扫才有用。这是因为URL后面带的参数大家定义的都不同,需要由协商好的软件去处理大家不同的需求,所以才会出现需要用专门的软件去扫一扫。
(同理,网上有的网站说你输入你家店的WI-FI和密码自动免费帮你生成二维码,让店家瞬间高大上,但是客人要连上你家WI-FI必须下它家的产品才可以,原理已经说过了,就是这样来扩展市场的)

好了,扫盲讲到这应该差不多了。

现在针对我的项目需求讲下思路。

1. 带AP功能的机顶盒端能生成二维码,供手机客户端扫描

有必要再科普一下,AP就是Access Point,接入点的意思,就是说这个机顶盒能够自己发射WI-FI供别的设备接入。

那第一个问题最直接的就是:怎么生成二维码?

用到的是google的一个开源二维码项目——zxing,目前基本上和二维码打交道的东西,都会用到它。

只提思路,具体怎么实现另搜百度就好。

(PS:这里发现把二维码改成其他颜色扫描无效,只有黑色可以被应用扫描到,背景改为透明没有关系)

然后我这边建立了个Service,读取机顶盒的AP信息,包括SSID和密码,与访问应用地址形成最终的URL,再通过zxing生成二维码。

到这一步需求1.1已经完成了,因为其他应用扫描二维码会忽略到后面的参数,只识别前面的地址,就会跳转到下载界面。

为了实现1.2,我们在自己的应用扫描时做特别判断,也就是获取后面的参数值,都获取到WI-FI和密码了,就可以通过代码进行自动连接了!~

2. 不带AP功能的机顶盒也能生成二维码

通过前面的分析,2.1不用改代码就可以实现,关键是2.2,如何能获得本机已经连接过的WI-FI的密码?

有两种方法,第一种通过系统API,在11年以后已经不能获得明文密码了,有密码全部用*代替值返回

WifiManager wifiManager = (WifiManager) getSystemService(WIFI_SERVICE);  List<WifiConfiguration> conList = wifiManager.getConfiguredNetworks();for (WifiConfiguration wifiConfiguration : conList) {    Log.d("wifi", "SSID = " + wifiConfiguration.SSID);    Log.d("wifi", "psk = " + wifiConfiguration.preSharedKey);} 

记得在Manifest文件中添加许可

<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"></uses-permission>

第二种用代码写命令去访问(《Android之查看Wifi密码》)

我已经找到data/misc/wifi/wpa_supplicant.conf里确实有明文密码。

public StringBuffer read() throws Exception {        Process process = null;        DataOutputStream dataOutputStream = null;        DataInputStream dataInputStream = null;        StringBuffer wifiConf = new StringBuffer();        try {            process = Runtime.getRuntime().exec("su");            dataOutputStream = new DataOutputStream(process.getOutputStream());            dataInputStream = new DataInputStream(process.getInputStream());            dataOutputStream.writeBytes("cat /data/misc/wifi/*.conf\n");            dataOutputStream.writeBytes("exit\n");            dataOutputStream.flush();            InputStreamReader inputStreamReader = new InputStreamReader(                    dataInputStream, "UTF-8");            BufferedReader bufferedReader = new BufferedReader(                    inputStreamReader);            String line = null;            while ((line = bufferedReader.readLine()) != null) {                wifiConf.append(line);            }            bufferedReader.close();            inputStreamReader.close();            process.waitFor();        } catch (Exception e) {            throw e;        } finally {            try {                if (dataOutputStream != null) {                    dataOutputStream.close();                }                if (dataInputStream != null) {                    dataInputStream.close();                }                process.destroy();            } catch (Exception e) {                throw e;            }        }        StringBuffer sb = new StringBuffer();        Pattern network = Pattern.compile("network=\\{([^\\}]+)\\}",                Pattern.DOTALL);        Matcher networkMatcher = network.matcher(wifiConf.toString());        while (networkMatcher.find()) {            String networkBlock = networkMatcher.group();            Pattern ssid = Pattern.compile("ssid=\"([^\"]+)\"");            Matcher ssidMatcher = ssid.matcher(networkBlock);            if (ssidMatcher.find()) {                sb.append(ssidMatcher.group(1));                Pattern psk = Pattern.compile("psk=\"([^\"]+)\"");                Matcher pskMatcher = psk.matcher(networkBlock);                if (pskMatcher.find()) {                    sb.append(pskMatcher.group(1));                } else {                    sb.append("无密码" + "/n");                }            }        }        return sb;}

上面的方法我实现行不通,报错java.io.IOException: write failed: EPIPE (Broken pipe),应该是权限不够。

这种要求有root权限,且应用还有访问权限,我在Manifest里加上android:sharedUserId="android.uid.system",然后到源码里去编译。

额外的想法:

又想过能不能直接用输入输出流访问data/misc/wifi/wpa_supplicant.conf

不过总觉得有隐患,另一个想法时,在连接WI-FI时,我们就额外保存一份密码到别处,然后供我们其他的应用访问,这样安全性好像又不好。

这个问题我还没有解决,若有人已经解决或者有解决思路,望留言告诉一下,提前谢过!~


2015.9.25 更新:已解决获取data/misc/wifi/wpa_supplicant.conf获取密码问题

前提:1.有源码环境 。2.作为系统应用。

经过四天苦战,终于搞定没有权限的问题了。我们的项目是在机顶盒端,而我的应用是作为“系统应用”存在的,而盒子又不会给应用开放 root 权限,但是我的应用能在源码编译成系统应用,是能够获取系统权限的,可是用上述代码还是访问不了data/misc/wifi/wpa_supplicant.conf文件内的内容。

以下讲解决办法:

1.在Manifest中加入

<manifest ...    package="com.azz.wifipsk"    ...    coreApp="true"    android:sharedUserId="android.uid.system">

这样做以后,就能够获取系统权限,但必须在源码环境中编译了(无法在eclipse中编译)。

2.把上述read代码Runtime.getRuntime().exec("su");改一下,已经有系统权限了,就不要执行su了(不然会报uid 1000 not allowed to su的错误)

    class MyThread implements Runnable {        public void run() {            Process process;            StringBuilder content = new StringBuilder();            String cmd = "cat /data/misc/wifi/wpa_supplicant.conf"; //(不能用*代替,要写具体文件名)            try {                process = Runtime.getRuntime().exec(cmd);   //有系统权限后直接执行命令                  DataOutputStream dataOutputStream = new DataOutputStream(process.getOutputStream());                DataInputStream dataIntputStream = new DataInputStream(process.getInputStream());                DataInputStream dataErrorStream = new DataInputStream(process.getErrorStream());                dataOutputStream.writeBytes(cmd + "\n");                            dataOutputStream.flush();                Thread.sleep(2000);                Log.d("wifi", "input = " + dataIntputStream.readLine());                Log.d("wifi", "error = " + dataErrorStream.readLine());                String line = "";                if (dataIntputStream.available() > 0)                {                    String error = "";                    int total = dataIntputStream.available();                    Log.e("TotalCount", Integer.toString(total));                    int i = 0;                    while(i < total)                    {                           line = dataIntputStream.readLine();                        if(line.trim().startsWith("ssid=") || line.trim().startsWith("psk="))                        {                            content.append(line + "\n");                        }                        i += line.length() + 1;                    }                    dataOutputStream.close();                    dataErrorStream.close();                    dataErrorStream.close();                }            } catch (IOException e) {                // TODO Auto-generated catch block                e.printStackTrace();                Log.e("Exception1", e.toString());            } catch (InterruptedException e) {                // TODO Auto-generated catch block                e.printStackTrace();                Log.e("Exception2", e.toString());            }            Message msg = new Message();            Bundle b = new Bundle();// 存放数据            b.putString("info", content.toString());            Log.e("info", content.toString());            msg.setData(b);            MainActivity.this.myHandler.sendMessage(msg); // 向Handler发送消息,更新UI        }    }

3.最重要的一点,做底层的发现我的应用已经有了system权限,但是还是读取不到wifi文件夹里的内容,于是他通过串口,发现/data/misc/wifi目录的权限是wifi.wifi,即wifi组创建,且属于wifi组,那么只有wifi组成员才能够访问,通过命令root@XXXX:/data/misc # chown system.wifi wifi/将wifi组的权限改为system.wifi,然后“系统应用”就可以访问了。

这里写图片描述

4.在源码环境中编译,并把编成的apk装在/system/app/目录下(好像我用adb装在/data/目录下也可以)

这里写图片描述

5.在上面的前提下,发现直接用File读取文件(不能用*代替,要写具体文件名),也可以读到,代码如下:

    String path = "/data/misc/wifi/wpa_supplicant.conf";    File file = new File(path);    InputStream in = null;    String line, content = "";    try {        in = new FileInputStream(file);        InputStreamReader inReader = new InputStreamReader(in);        BufferedReader bufferedReader = new BufferedReader(inReader);        while((line = bufferedReader.readLine()) != null) {            content += line + "\n";        }        Log.d("file", "path = " + path + ", content = " + content);    } catch (Exception e) {        Log.e("file", e.getMessage());    } finally {        if(in != null) {            in.close();        }    }

这里写图片描述

2016.1.7更新:修改二维码的容错率和边框

看zxing源码,Writer其实还有个方法是

public ByteMatrix encode(System.String contents, BarcodeFormat format, int width, int height, System.Collections.Hashtable hints){}

所以只要加上hints配置参数,就能够修改一些配置了

public static Bitmap Create2DCode(String str, int picWidth, int picHeight) throws WriterException {    Hash table hints = new Hashtable();    hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrentionLevel.L); //容错率,L为7%,M为15%,Q为25%,H为30%,容错率越高,二维码点数越多    hints.put(EncodeHintType.MARGIN, 0); //边框,默认为4    //将hints设置进去    BitMatrix matrix = new MultiFormatWriter().encode(str, BarcodeFormat.QR_CODE, w, h, hints);    ...}

Reference:
1.知乎 -《为何用二维码扫描App扫描微信名片都能直接跳转到微信?这是如何实现的?》
2.《Android之查看Wifi密码》
3.《Android-WIFI密码破解工具编写初探》
4.《Android平台利用ZXING生成二维码图片》
5.《提高zxing生成二维码的容错率及zxing生成二维码的边框设置》

1 0
原创粉丝点击