Java生成二维码解析二维码

package QrCode;

import java.awt.Graphics2D;

import java.awt.image.BufferedImage;

import java.io.File;

import java.io.IOException;

import java.util.HashMap;

import java.util.Map;

import javax.imageio.ImageIO;

import com.google.zxing.BarcodeFormat;

import com.google.zxing.BinaryBitmap;

import com.google.zxing.DecodeHintType;

import com.google.zxing.EncodeHintType;

import com.google.zxing.MultiFormatReader;

import com.google.zxing.MultiFormatWriter;

import com.google.zxing.NotFoundException;

import com.google.zxing.Result;

import com.google.zxing.client.j2se.BufferedImageLuminanceSource;

import com.google.zxing.client.j2se.MatrixToImageConfig;

import com.google.zxing.client.j2se.MatrixToImageWriter;

import com.google.zxing.common.BitMatrix;

import com.google.zxing.common.HybridBinarizer;

import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;

/**

* 026 ZXing工具类

*

* @see

*      --------------------------------------------------------------------------

*      ---------------------------------------------

*

* @see 首页--https://code.google.com/p/zxing

*

* @see 介绍

*      --用于解析多种格式条形码(EAN-13)和二维码(QRCode)的开源Java类库,其提供了多种应用的类库,如javase/jruby/cpp

*      /csharp/android

*

* @see 说明--下载到的ZXing-2.2.zip是它的源码,我们在JavaSE中使用时需用到其core和javase两部分

*

* @see 可直接引入它俩的源码到项目中

*      ,或将它俩编译为jar再引入,这是我编译好的:http://download.csdn.net/detail/jadyer/6245849

*

* @see

*      --------------------------------------------------------------------------

*      ---------------------------------------------

*

* @see 经测试:用微信扫描GBK编码的中文二维码时出现乱码,用UTF-8编码时微信可正常识别

*

* @see 并且MultiFormatWriter.encode()时若传入hints参数来指定UTF-8编码中文时,微信压根就不识别所生成的二维码

*

* @see 所以这里使用的是这种方式new String(content.getBytes("UTF-8"), "ISO-8859-1")

*

* @see

*      --------------------------------------------------------------------------

*      ---------------------------------------------

*

* @see 将logo图片加入二维码中间时,需注意以下几点

*

* @see 1)生成二维码的纠错级别建议采用最高等级H,这样可以增加二维码的正确识别能力(我测试过,不设置级别时,二维码工具无法读取生成的二维码图片)

*

* @see 2)头像大小最好不要超过二维码本身大小的1/5,而且只能放在正中间部位,这是由于二维码本身结构造成的(你就把它理解成图片水印吧)

*

* @see 3)在仿照腾讯微信在二维码四周增加装饰框,那么一定要在装饰框和二维码之间留出白边,这是为了二维码可被识别

*

* @see

*      --------------------------------------------------------------------------

*      ---------------------------------------------

*

* @version v1.0

*

* @history v1.0-->方法新建,目前仅支持二维码的生成和解析,生成二维码时支持添加logo头像

*

* @editor Oct 10, 2014 9:32:23 PM

*

* @create Oct 10, 2013 2:08:16 PM

*

* @author yixian

*/

public class ZXingUtil {

public ZXingUtil() {

}

/**

*

* 为二维码图片增加logo头像

*

* @see 其原理类似于图片加水印

*

* @param imagePath

*            二维码图片存放路径(含文件名)

*

* @param logoPath

*            logo头像存放路径(含文件名)

*/

private static void overlapImage(String imagePath, String logoPath)

throws IOException {

BufferedImage image = ImageIO.read(new File(imagePath));

int logoWidth = image.getWidth() / 5; // 设置logo图片宽度为二维码图片的五分之一

int logoHeight = image.getHeight() / 5; // 设置logo图片高度为二维码图片的五分之一

Graphics2D graphics = image.createGraphics();

if(logoPath != ""){

int logoX = (image.getWidth() - logoWidth) / 2; // 设置logo图片的位置,这里令其居中

int logoY = (image.getHeight() - logoHeight) / 2; // 设置logo图片的位置,这里令其居中

graphics.drawImage(ImageIO.read(new File(logoPath)), logoX, logoY,

logoWidth, logoHeight, null);

graphics.dispose();

}

ImageIO.write(image,

imagePath.substring(imagePath.lastIndexOf(".") + 1), new File(

imagePath));

}

/**

*

* 生成二维码

*

* @param content

*            二维码内容

*

* @param charset

*            编码二维码内容时采用的字符集(传null时默认采用UTF-8编码)

*

* @param imagePath

*            二维码图片存放路径(含文件名)

*

* @param width

*            生成的二维码图片宽度

*

* @param height

*            生成的二维码图片高度

*

* @param logoPath

*            logo头像存放路径(含文件名,若不加logo则传null即可)

*

* @return 生成二维码结果(true or false)

*/

public  boolean encodeQRCodeImage(String content, String charset,

String imagePath, int width, int height, String logoPath) {

Map<EncodeHintType, Object> hints = new HashMap<EncodeHintType, Object>();

// 指定编码格式

// hints.put(EncodeHintType.CHARACTER_SET, "UTF-8");

// 指定纠错级别(L--7%,M--15%,Q--25%,H--30%)

hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);

// 编码内容,编码类型(这里指定为二维码),生成图片宽度,生成图片高度,设置参数

BitMatrix bitMatrix = null;

try {

bitMatrix = new MultiFormatWriter().encode(

new String(content.getBytes(charset == null ? "UTF-8"

: charset), "ISO-8859-1"), BarcodeFormat.QR_CODE,

width, height, hints);

} catch (Exception e) {

System.out.println("编码待生成二维码图片的文本时发生异常,堆栈轨迹如下");

e.printStackTrace();

return false;

}

// 生成的二维码图片默认背景为白色,前景为黑色,但是在加入logo图像后会导致logo也变为黑白色,至于是什么原因还没有仔细去读它的源码

// 所以这里对其第一个参数黑色将ZXing默认的前景色0xFF000000稍微改了一下0xFF000001,最终效果也是白色背景黑色前景的二维码,且logo颜色保持原有不变

MatrixToImageConfig config = new MatrixToImageConfig(0xFF000001,

0xFFFFFFFF);

// 这里要显式指定MatrixToImageConfig,否则还会按照默认处理将logo图像也变为黑白色(如果打算加logo的话,反之则不须传MatrixToImageConfig参数)

try {

MatrixToImageWriter.writeToFile(bitMatrix,

imagePath.substring(imagePath.lastIndexOf(".") + 1),

new File(imagePath), config);

} catch (IOException e) {

System.out.println("生成二维码图片[" + imagePath + "]时遇到异常,堆栈轨迹如下");

e.printStackTrace();

return false;

}

// 此时二维码图片已经生成了,只不过没有logo头像,所以接下来根据传入的logoPath参数来决定是否加logo头像

if (null == logoPath) {

return true;

} else {

// 如果此时最终生成的二维码不是我们想要的,那么可以扩展MatrixToImageConfig类(反正ZXing提供了源码)

// 扩展时可以重写其writeToFile方法,令其返回toBufferedImage()方法所生成的BufferedImage对象(尽管这种做法未必能解决为题,故需根据实际情景测试)

// 然后替换这里overlapImage()里面的第一行BufferedImage image = ImageIO.read(new

// File(imagePath));

// 即private static void overlapImage(BufferedImage image, String

// imagePath, String logoPath)

try {

// 这里不需要判断logoPath是否指向了一个具体的文件,因为这种情景下overlapImage会抛IO异常

overlapImage(imagePath, logoPath);

return true;

} catch (IOException e) {

System.out.println("为二维码图片[" + imagePath + "]添加logo头像["

+ logoPath + "]时遇到异常,堆栈轨迹如下");

e.printStackTrace();

return false;

}

}

}

/**

*

* 解析二维码

*

* @param imagePath

*            二维码图片存放路径(含文件名)

*

* @param charset

*            解码二维码内容时采用的字符集(传null时默认采用UTF-8编码)

*

* @return 解析成功后返回二维码文本,否则返回空字符串

*/

public  String decodeQRCodeImage(String imagePath, String charset) {

BufferedImage image = null;

try {

image = ImageIO.read(new File(imagePath));

} catch (IOException e) {

e.printStackTrace();

return "";

}

if (null == image) {

System.out.println("Could not decode QRCodeImage");

return "";

}

BinaryBitmap bitmap = new BinaryBitmap(new HybridBinarizer(

new BufferedImageLuminanceSource(image)));

Map<DecodeHintType, String> hints = new HashMap<DecodeHintType, String>();

hints.put(DecodeHintType.CHARACTER_SET, charset == null ? "UTF-8"

: charset);

Result result = null;

try {

result = new MultiFormatReader().decode(bitmap, hints);

return result.getText();

} catch (NotFoundException e) {

System.out.println("二维码图片[" + imagePath + "]解析失败,堆栈轨迹如下");

e.printStackTrace();

return "";

}

}

public static void main(String[] args) {

ZXingUtil zx = new ZXingUtil();

zx.encodeQRCodeImage("my name is zhangsanfeng", null,

"D:/QrCode/myQRCodeImage.png", 300, 300,

"D:/QrCode/Koala.jpg");

System.out.println(zx.decodeQRCodeImage(

"D:/QrCode/myQRCodeImage.png", null));

}

}

时间: 2024-08-09 22:03:05

Java生成二维码解析二维码的相关文章

Spring 源码解析之HandlerAdapter源码解析(二)

Spring 源码解析之HandlerAdapter源码解析(二) 前言 看这篇之前需要有Spring 源码解析之HandlerMapping源码解析(一)这篇的基础,这篇主要是把请求流程中的调用controller流程单独拿出来了 解决上篇文章遗留的问题 getHandler(processedRequest) 这个方法是如何查找到对应处理的HandlerExecutionChain和HandlerMapping的,比如说静态资源的处理和请求的处理肯定是不同的HandlerMapping ge

GlusterFS源码解析 —— GlusterFS 源码安装

安装环境: CentOS6.2 glusterfs-3.4.3 GlusterFS 挂载需要 fuse 支持,如果你的内核版本低于 2.6.16 则需要下载fuse的源码包自行编译安装,也可下载 fuse 的rpm包.安装fuse的方法我就不说了,不会源码安装的直接去rpmfind.net上下载rpm即可.高于此版本的内核中已经有了fuse.ko的模块,需要的时候可以执行以下命令进行加载: modprobe -b fuse 1.下载GlusterFS的源码包,目前已经有更新版本 : wget h

chenglei1986/DatePicker源码解析(二)

接上一篇文章chenglei1986/DatePicker源码解析(一),我们继续将剩余的部分讲完,其实剩余的内容,就是利用Numberpicker来组成一个datePicker,代码非常的简单 为了实现自定义布局的效果,我们给Datepciker定制了一个layout,大家可以定制自己的layout <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="h

erlang下lists模块sort(排序)方法源码解析(二)

上接erlang下lists模块sort(排序)方法源码解析(一),到目前为止,list列表已经被分割成N个列表,而且每个列表的元素是有序的(从大到小) 下面我们重点来看看mergel和rmergel模块,因为我们先前主要分析的split_1_*对应的是rmergel,我们先从rmergel查看,如下 ....................................................... split_1(X, Y, [], R, Rs) -> rmergel([[Y, X

AFNetworking2.0源码解析&lt;二&gt;

本篇我们继续来看看AFNetworking的下一个模块 — AFURLRequestSerialization. AFURLRequestSerialization用于帮助构建NSURLRequest,主要做了两个事情: 1.构建普通请求:格式化请求参数,生成HTTP Header. 2.构建multipart请求. 分别看看它在这两点具体做了什么,怎么做的. 1.构建普通请求 A.格式化请求参数 一般我们请求都会按key=value的方式带上各种参数,GET方法参数直接加在URL上,POST方

volley源码解析(二)--Request&lt;T&gt;类的介绍

在上一篇文章中,我们已经提到volley的使用方式和设计的整体思路,从这篇文章开始,我就要结合具体的源码来给大家说明volley功能的具体实现. 我们第一个要介绍的类是Request<T>这个一个抽象类,我将Request称为一个请求,通过继承Request<T>来自定义request,为volley提供了更加灵活的接口. Request<T>中的泛型T,是指解析response以后的结果.在上一篇文章中我们知道,ResponseDelivery会把response分派

java生成word文档【二】

/** * <p>expWordByItext方法主要用于-采用itext导出文档到word.</p> * <p>优缺点-可将随意控制生成word的格式,但代码控制上比采用模板的方式稍微复杂,可设置页眉页脚等其他文档属性.</p> * <p>第三方jar-itext-2.1.7.jar.itext-rtf-2.1.7.jar.<br> * import com.lowagie.text.Document;<br> * im

第37篇 Asp.Net源码解析(二)--详解HttpApplication

这篇文章花了点时间,差点成烂到电脑里面,写的过程中有好几次修改,最终的这个版本也不是很满意,东西说的不够细,还需要认真的去看下源码才能有所体会,先这样吧,后面有时间把细节慢慢的再修改.顺便对于开发的学习,个人是觉得源码的阅读是最快的提高方式,当然阅读不是走马观花,应该多次阅读. 上次说到获得HttpApplication对象的创建,创建完成后调用InitInternal方法,这个方法任务比较多,也比较长,这里就不贴全码了,一个一个过程的去说: 初始化HttpModule 对于HttpModule

dmytrodanylyk/circular-progress-button源码解析(二)

源码下载http://download.csdn.net/detail/kangaroo835127729/8755815 在上篇文章http://blog.csdn.net/crazy__chen/article/details/46278423中,我主要讲述了circular-progress-button状态切换的动画过程,接下来我们看一个最特殊的状态,就是加载状态,这个状态会显示一个圆环来表示当前加载的进度,但是其实circular-progress-button提供给了我们两个选择,一