二维码之所以能够封装图片、文件等主要是把图片、文件等URL编码成二维码,用户一扫手机自动访问。
上一次在《【jQuery】使用jquery-qrcode插件把网址转化成二维码,手机扫一扫即可访问》(点击打开链接)使用jquery-qrcode插件配合jquery能够把网址编码成字符串,在网址上显示编码后的二维码,移动设备能够直接扫一扫。
这次将在Java中实现对二维码的编码与解码输出到磁盘上,无需在网页中进行,而且还可以解码,但是步骤比较复杂而已。
Java自身当然没有二维码的编码与解码的功能,需要下载两个插件,也就是额外的jar包,一个用来编码的,把字符串、网址、URL等转化成二维码的com.swetake.util.Qrcode,可以从它的官网下载(点击打开链接),我也为大家上传了一份(点击打开链接),另一个用来解码,把二维码转化成原来的字符串、网址、URL等的jp.sourceforge.qrcode.data.QRCodeImage,这个也可以从它的官网中下载(点击打开链接),我也为大家上传了一份(点击打开链接)
编码的com.swetake.util.Qrcode下载之后是一个qrcode_java0.50beta10.tar.gz,最新版的Winrar可以解压,解压之后把里面lib目录下的qrcode.jar取出来,改个名字叫qrcodeencoding.jar,解码的jp.sourceforge.qrcode.data.QRCodeImage下载之后的是一个redir.zip,解压之后同样把里面lib目录下的qrcode.jar取出来,改个名字叫qrcodedecoding.jar。本来这些第三方jar包,无需修改可以扔到Java工程中使用。但是,这两家日本公司,实在是太有缘了,开发的二维码包撞衫了。因此,要改好名字才能放到一起。改成其他名字也行,反正就不能同时叫qrcode,我的编码叫qrcode.jar,译码叫qrcodedecoding.jar,主要是一开始不知道两个jar的名字相同,又引入了编码的com.swetake.util.Qrcode,才没改编码包的名字。
在Eclipse中新建一个java工程,叫什么随意,我的叫qrcode,之后最后手动打开eclipse的工程目录,找到这个文件夹,在里面新建一个lib,把qrcodeencoding.jar与qrcodedecoding.jar放进去,然后在Eclipse中刷新一下这个工程,右键点最下面的一项属性,在Java Build Path中的Libraries选择Add Jars..如果没有拷到工程目录下则选择Add External Jars.添加这两个包:
之后先在src下的package下新建一个QRcode.java,也就是一个class。于是,整个工程目录结构如下所示,此时应该没有test.png,test.png是后来生成的二维码文件:
准备工作写好之后就能够正式写代码。
直接贴上全代码,则一行一行给大家剖析,是怎么做的。
package qrcode; import java.awt.*; import java.awt.image.*; import java.io.*; import javax.imageio.*; import jp.sourceforge.qrcode.QRCodeDecoder; import jp.sourceforge.qrcode.data.QRCodeImage; import com.swetake.util.Qrcode; public class QRcode { public static void makeQRcode(String content, String imgPath) { Qrcode qrcode = new Qrcode(); qrcode.setQrcodeErrorCorrect('M'); qrcode.setQrcodeEncodeMode('B'); qrcode.setQrcodeVersion(7); BufferedImage bufImg = new BufferedImage(325, 325, BufferedImage.TYPE_INT_RGB); Graphics2D gs = bufImg.createGraphics(); gs.setBackground(Color.WHITE); gs.clearRect(0, 0, 325, 325); gs.setColor(Color.BLACK); try { byte[] contentBytes = content.getBytes("utf-8"); if (0 < contentBytes.length && contentBytes.length < 120) { boolean[][] codeOut = qrcode.calQrcode(contentBytes); for (int i = 0; i < codeOut.length; i++) { for (int j = 0; j < codeOut.length; j++) { if (codeOut[j][i]) { gs.fillRect(j * 7 + 5, i * 7 + 5, 7, 7); } } } } gs.dispose(); bufImg.flush(); File imgFile = new File(imgPath); //生成为png ImageIO.write(bufImg, "png", imgFile); System.out.println("生成成功!"); } catch (Exception e) { System.out.println("出错了!"); } } public static String decodeQRCode(String imgPath) throws IOException { File imageFile = new File(imgPath); BufferedImage bufImg = null; String decodedData = null; bufImg = ImageIO.read(imageFile); QRCodeDecoder decoder = new QRCodeDecoder(); decodedData = new String(decoder.decode(new J2SEImage(bufImg))); System.out.println("解析成功!"); return decodedData; } public static void main(String args[]) throws IOException { String content = "http://www.a.com"; File directory = new File(""); String imgPath = directory.getCanonicalPath() + "\\test.png"; makeQRcode(content, imgPath); System.out.println(decodeQRCode(imgPath)); } } class J2SEImage implements QRCodeImage { BufferedImage bufImg; public J2SEImage(BufferedImage bufImg) { this.bufImg = bufImg; } public int getWidth() { return bufImg.getWidth(); } public int getHeight() { return bufImg.getHeight(); } public int getPixel(int x, int y) { return bufImg.getRGB(x, y); } }
import java.awt.*; import java.awt.image.*; import java.io.*; import com.swetake.util.Qrcode;
//由于这个类中同时有主函数,所以要加上static属性 /* * @content 要编码的内容 * @imgPath 二维码生成的路径 */ //据说上面这种说明类参数注释方法,符合名企标准,╮(╯▽╰)╭你TMD看懂就行了 public static void makeQRcode(String content, String imgPath) { //以下很多是指定动作,不为什么,这个类就要这样写, //可以改的部分我会注释 Qrcode qrcode = new Qrcode(); //这里是容错率,有L,M,Q三种,L为最小,一般如果你还要在二维码上面加图片,就用Q //注意这里必须使用单引号,因为这里是char或者int类型,而不是String类型 qrcode.setQrcodeErrorCorrect('M'); qrcode.setQrcodeEncodeMode('B'); qrcode.setQrcodeVersion(7); //新建一个325x325px的图像,这个大小不是说改就改的,有讲究的,与下面gs.fillRect语句相关联 BufferedImage bufImg = new BufferedImage(325, 325, BufferedImage.TYPE_INT_RGB); Graphics2D gs = bufImg.createGraphics(); //图像的背景色是白色 gs.setBackground(Color.WHITE); //这里的325x325必须与上面BufferedImage bufImg = new BufferedImage(325, 325, BufferedImage.TYPE_INT_RGB);对应 //上面是325x325你这里必须是多少 gs.clearRect(0, 0, 325, 325); //二维码的颜色是黑色 gs.setColor(Color.BLACK); //它丫不抛出,不让编译通过╮(╯▽╰)╭ try { //编码方式为utf-8,现在还写gb2312与gbk那就是作死,都什么年代了,编码也全球化了! byte[] contentBytes = content.getBytes("utf-8"); //可以编码的字符串长度0-120字节,二维码的固定长度来的,别改 if (0 < contentBytes.length && contentBytes.length < 120) { boolean[][] codeOut = qrcode.calQrcode(contentBytes); for (int i = 0; i < codeOut.length; i++) { for (int j = 0; j < codeOut.length; j++) { if (codeOut[j][i]) { //这里是每一个点为7x7 //这个7统一改,如果你要每一个点为6x6,那么这句必须写成gs.fillRect(j * 6 + 5, i * 6 + 5, 6, 6); //那个5是边缘的留白,就是这个二维码外面还有一些白色的边框,如果这个数是0,紧贴图片边缘据说会出错 //设置为0我没试过,只是为了好看留5px的空白 //每个点为7x7,还有5px的留白,得出图片大小为7*45+5*2=325 //如果你每个点是6*6,留白3px,那么图片大小就是6*45+3*2=276。 //也就是上面的BufferedImage bufImg = new BufferedImage(325, 325, BufferedImage.TYPE_INT_RGB);与gs.clearRect(0, 0, 325, 325);自己改好 //这公式中的45与2是固定的,二维码就是这样,不为什么。 gs.fillRect(j * 7 + 5, i * 7 + 5, 7, 7); } } } } gs.dispose(); bufImg.flush(); File imgFile = new File(imgPath); //生成为png ImageIO.write(bufImg, "png", imgFile); System.out.println("生成成功!"); } catch (Exception e) { System.out.println("出错了!"); } }
3、然后下面的主函数写入要编码的字符串,比如http://www.a.com这个网址,与保存的图片目录,比如当前工程目录,就能完成二维码的编码工作:
public static void main(String args[]) throws IOException { //要编码的字符串 String content = "http://www.a.com"; //取出当前的工程目录,并指明保存文件名为test.png File directory = new File(""); String imgPath = directory.getCanonicalPath() + "\\test.png"; //执行makeQRcode编码的方法 makeQRcode(content, imgPath); }
主函数需要抛出异常是因为用到了File中的取工程目录函数,它JAVA害怕取不到工程目录会出错
import javax.imageio.*; import jp.sourceforge.qrcode.QRCodeDecoder; import jp.sourceforge.qrcode.data.QRCodeImage;
class J2SEImage implements QRCodeImage { BufferedImage bufImg; public J2SEImage(BufferedImage bufImg) { this.bufImg = bufImg; } public int getWidth() { return bufImg.getWidth(); } public int getHeight() { return bufImg.getHeight(); } public int getPixel(int x, int y) { return bufImg.getRGB(x, y); } }这接口封装起来很简单嘛!就是要取走图片、长宽,各个像素的颜色。
6、把QRCodeImage这个接口的封装成J2SEImage之后,写一个decodeQRCode这个类,代码如下:
<span style="white-space:pre"> </span>public static String decodeQRCode(String imgPath) throws IOException { /* * @imageFile 被解码的文件 * @BufferedImage 此乃所谓的图片缓冲流 * @decodedData 存放解码的内容 */ File imageFile = new File(imgPath); BufferedImage bufImg = null; String decodedData = null; bufImg = ImageIO.read(imageFile); QRCodeDecoder decoder = new QRCodeDecoder(); //封装后直接用decode方法就能完成二维码的解码 decodedData = new String(decoder.decode(new J2SEImage(bufImg))); System.out.println("解析成功!"); //最后返回解码的内容就OK return decodedData; }