二维码带汉字Linux环境乱码解决

时间:2022-06-04 11:00:48

一背景

  二维码的应用很广泛,这里不多介绍。我们的业务场景是给家政人员电子简历增加二维码,便于客户快速扫描识别。问题在于汉字的乱码。

二问题

  关于java二维码生成,主要就是两种方式,qrcode跟zxing.网上例子很多,不多介绍。我是从网上找了直接用的。

代码如下:

/**
*
*/
package com.bj58.daojia.crm.crmCustom.utils;


import com.google.zxing.BarcodeFormat;
import com.google.zxing.EncodeHintType;
import com.google.zxing.MultiFormatWriter;
import com.google.zxing.WriterException;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;


import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.geom.RoundRectangle2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;

/**
* @author zhangliang
*
* 2016年12月3日下午6:36:21
*/
public class QrcodeUtil {

private static final int QRCOLOR = 0xFF000000; //默认是黑色
private static final int BGWHITE = 0xFFFFFFFF; //背景颜色
/** * 生成带logo的二维码图片 * * @param qrPic * @param logoPic */
public static byte[] getLogoQRCode(String qrUrl,String productName)
{
// String filePath = request.getSession().getServletContext().getRealPath("/") + "resources/images/logoImages/llhlogo.png";
//filePath是二维码logo的路径,但是实际中我们是放在项目的某个路径下面的,所以路径用上面的,把下面的注释就好
String filePath = "/opt/web/dianshangwuxian_crm_daojia/webapps/image/logo.png"; //TODO
String content = qrUrl;
try
{

BufferedImage bim = getQR_CODEBufferedImage(content, BarcodeFormat.QR_CODE, 400, 400, getDecodeHintType());
byte[] file= addLogo_QRCode(bim, new File(filePath), new LogoConfig(), productName);
return file;
}
catch (Exception e)
{
e.printStackTrace();
}
return null;
}

/** * 给二维码图片添加Logo * * @param qrPic * @param logoPic */
public static byte[] addLogo_QRCode(BufferedImage bim, File logoPic, LogoConfig logoConfig, String productName)
{
try
{
/** * 读取二维码图片,并构建绘图对象 */
BufferedImage image = bim;
Graphics2D g = image.createGraphics();
int matrixWidth = image.getWidth();
int matrixHeigh = image.getHeight();

/** * 读取Logo图片 */
BufferedImage logo = ImageIO.read(logoPic);
//开始绘制图片
g.drawImage(logo,matrixWidth/5*2,matrixHeigh/5*2, matrixWidth/4, matrixHeigh/4, null);//绘制
BasicStroke stroke = new BasicStroke(5,BasicStroke.CAP_ROUND,BasicStroke.JOIN_ROUND);
g.setStroke(stroke);// 设置笔画对象
//指定弧度的圆角矩形
RoundRectangle2D.Float round = new RoundRectangle2D.Float(matrixWidth/5*2, matrixHeigh/5*2, matrixWidth/4, matrixHeigh/4,20,20);
g.setColor(Color.white);
g.draw(round);// 绘制圆弧矩形

//设置logo 有一道灰色边框
BasicStroke stroke2 = new BasicStroke(1,BasicStroke.CAP_ROUND,BasicStroke.JOIN_ROUND);
g.setStroke(stroke2);// 设置笔画对象
RoundRectangle2D.Float round2 = new RoundRectangle2D.Float(matrixWidth/5*2, matrixHeigh/5*2, matrixWidth/4, matrixHeigh/4,20,20);
g.setColor(new Color(128,128,128));
g.draw(round2);// 绘制圆弧矩形

g.dispose();
// g.drawRoundRect(x, y, widthLogo, heightLogo, 15, 15);
// g.setStroke(new BasicStroke(logoConfig.getBorder()));
// g.setColor(logoConfig.getBorderColor());
// g.drawRect(x, y, widthLogo, heightLogo);
g.dispose();

//把商品名称添加上去,商品名称不要太长哦,这里最多支持两行。太长就会自动截取啦
if (productName != null && !productName.equals("")) {
//新的图片,把带logo的二维码下面加上文字
BufferedImage outImage = new BufferedImage(400, 445, BufferedImage.TYPE_4BYTE_ABGR);
Graphics2D outg = outImage.createGraphics();
//画二维码到新的面板
outg.drawImage(image, 0, 0, image.getWidth(), image.getHeight(), null);
//画文字到新的面板
outg.setColor(Color.BLACK);
outg.setFont(new Font("宋体",Font.BOLD,30)); //字体、字型、字号
int strWidth = outg.getFontMetrics().stringWidth(productName);
if (strWidth > 399) {
// //长度过长就截取前面部分
// outg.drawString(productName, 0, image.getHeight() + (outImage.getHeight() - image.getHeight())/2 + 5 ); //画文字
//长度过长就换行
String productName1 = productName.substring(0, productName.length()/2);
String productName2 = productName.substring(productName.length()/2, productName.length());
int strWidth1 = outg.getFontMetrics().stringWidth(productName1);
int strWidth2 = outg.getFontMetrics().stringWidth(productName2);
outg.drawString(productName1, 200 - strWidth1/2, image.getHeight() + (outImage.getHeight() - image.getHeight())/2 + 12 );
BufferedImage outImage2 = new BufferedImage(400, 485, BufferedImage.TYPE_4BYTE_ABGR);
Graphics2D outg2 = outImage2.createGraphics();
outg2.drawImage(outImage, 0, 0, outImage.getWidth(), outImage.getHeight(), null);
outg2.setColor(Color.BLACK);
outg2.setFont(new Font("宋体",Font.BOLD,30)); //字体、字型、字号
outg2.drawString(productName2, 200 - strWidth2/2, outImage.getHeight() + (outImage2.getHeight() - outImage.getHeight())/2 + 5 );
outg2.dispose();
outImage2.flush();
outImage = outImage2;
}else {
outg.drawString(productName, 200 - strWidth/2 , image.getHeight() + (outImage.getHeight() - image.getHeight())/2 + 12 ); //画文字
}
outg.dispose();
outImage.flush();
image = outImage;
}
logo.flush();
image.flush();
ByteArrayOutputStream baos = new ByteArrayOutputStream();
baos.flush();
ImageIO.write(image, "png", baos);

//二维码生成的路径,但是实际项目中,我们是把这生成的二维码显示到界面上的,因此下面的折行代码可以注释掉
//可以看到这个方法最终返回的是这个二维码的imageBase64字符串
//前端用 <img src="data:image/png;base64,${imageBase64QRCode}"/> 其中${imageBase64QRCode}对应二维码的imageBase64字符串
//ImageIO.write(image, "png", new File("D:/qrcodeImages/" + new Date().getTime() + ".png")); //TODO
return baos.toByteArray();

}
catch (Exception e)
{
e.printStackTrace();
}
return null;
}


/** * 构建初始化二维码 * * @param bm * @return */
public BufferedImage fileToBufferedImage(BitMatrix bm)
{
BufferedImage image = null;
try
{
int w = bm.getWidth(), h = bm.getHeight();
image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);

for (int x = 0; x < w; x++)
{
for (int y = 0; y < h; y++)
{
image.setRGB(x, y, bm.get(x, y) ? 0xFF000000 : 0xFFCCDDEE);
}
}

}
catch (Exception e)
{
e.printStackTrace();
}
return image;
}

/** * 生成二维码bufferedImage图片 * * @param content * 编码内容 * @param barcodeFormat * 编码类型 * @param width * 图片宽度 * @param height * 图片高度 * @param hints * 设置参数 * @return */
public static BufferedImage getQR_CODEBufferedImage(String content, BarcodeFormat barcodeFormat, int width, int height, Map<EncodeHintType, ?> hints)
{
MultiFormatWriter multiFormatWriter = null;
BitMatrix bm = null;
BufferedImage image = null;
try
{
multiFormatWriter = new MultiFormatWriter();
// 参数顺序分别为:编码内容,编码类型,生成图片宽度,生成图片高度,设置参数
bm = multiFormatWriter.encode(content, barcodeFormat, width, height, hints);
int w = bm.getWidth();
int h = bm.getHeight();
image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);

// 开始利用二维码数据创建Bitmap图片,分别设为黑(0xFFFFFFFF)白(0xFF000000)两色
for (int x = 0; x < w; x++)
{
for (int y = 0; y < h; y++)
{
image.setRGB(x, y, bm.get(x, y) ? QRCOLOR : BGWHITE);
}
}
}
catch (WriterException e)
{
e.printStackTrace();
}
return image;
}

/** * 设置二维码的格式参数 * * @return */
public static Map<EncodeHintType, Object> getDecodeHintType()
{
// 用于设置QR二维码参数
Map<EncodeHintType, Object> hints = new HashMap<EncodeHintType, Object>();
// 设置QR二维码的纠错级别(H为*别)具体级别信息
hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);
// 设置编码方式
hints.put(EncodeHintType.CHARACTER_SET, "utf-8");
hints.put(EncodeHintType.MARGIN, 0);
// hints.put(EncodeHintType.MAX_SIZE, 350);
// hints.put(EncodeHintType.MIN_SIZE, 100);

return hints;
}


public static void main(String[] args){
String url ="http://sso.daojia-inc.com/views/login.jsp";
String note ="洪加荣";
String test = QrcodeUtil.getLogoQRCode(url,note).toString();

}
}

里面有注解,说明下。jdk版本不同,应用的jar版本也不同。高版本的需要jdk1.8支持。

 <dependency>
<groupId>com.google.zxing</groupId>
<artifactId>core</artifactId>
<version>3.2.0</version>
</dependency>
<dependency>
<groupId>com.google.zxing</groupId>
<artifactId>javase</artifactId>
<version>3.2.0</version>
</dependency>

我们目前使用的是jdk1.6,所以版本是2.2的。Windows下最终效果如下:

二维码带汉字Linux环境乱码解决

调试过程比较费时间,中间logo的大小,位置,下面从id变成汉字。

二维码带汉字Linux环境乱码解决

二维码带汉字Linux环境乱码解决

linux环境下测试不显示汉字,

二维码带汉字Linux环境乱码解决二维码带汉字Linux环境乱码解决

3解决

首先排除编码问题

先后测试gbk,utf8等不同方式,确定编码格式无误,log打印不乱吗,生成图片就是空格。

怀疑是字体问题:

outg.setFont(new Font("宋体",Font.BOLD,30));

没有root权限,但是可以修改jdk的字体。

搜一下find / -name 'fonts' -type -d

........

/opt/soft/java/jre/lib/fonts
/opt/soft/jdk1.7.0_79/jre/lib/fonts
/opt/soft/jdk1.8.0_92/jre/lib/fonts

找到很多,目前我们版本是1.6,所以就针对性修改即可。

本地的字体是“C:\WINDOWS\Fonts\simsun.ttc,win7名字不同。以ttf结尾。可以试试。

上传到fonts下面:

rz waiting to receive.
Starting zmodem transfer.  Press Ctrl+C to cancel.
Transferring simsun.ttf...
  100%   10261 KB     100 KB/sec    00:01:42       0 Errors 

一定要重启服务:

通过验证程序,确定二维码图片上的小方块正确显示为中文。