二維碼的應用很廣泛,這里不多介紹。我們的業務場景是給家政人員電子簡歷增加二維碼,便于客戶快速掃描識別。問題在于漢字的亂碼。
關于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下最終效果如下:
調試過程比較費時間,中間logo的大小,位置,下面從id變成漢字。
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
一定要重啟服務:
通過驗證程序,確定二維碼圖片上的小方塊正確顯示為中文。
新聞熱點
疑難解答