zxing의 성숙한 QR 코드 생성 인터페이스를 도입하여 표준 QR 코드 파일을 생성하고, Java 그래픽 이미지 처리 API를 통해 QR 코드에 관련 텍스트 설명을 추가하고, 필요에 따라 합성 이미지에 관련 배경을 추가합니다. 예시는 아래와 같습니다.
1. 먼저 비트맵을 가져오겠습니다. QR 코드 이미지를 생성하는 핵심 코드는 다음과 같습니다/** * 定义二维码的参数 */ HashMap<EncodeHintType, Object> hints = new HashMap(); //指定字符编码为“utf-8” hints.put(EncodeHintType.CHARACTER_SET, "utf-8"); //指定二维码的纠错等级为中级 hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H); //设置图片的边距 hints.put(EncodeHintType.MARGIN, 1); /** * 生成二维码 */ try { BitMatrix bitMatrix = new MultiFormatWriter().encode(content, BarcodeFormat.QR_CODE, QRCODE_WIDTH, QRCODE_HEIGHT, hints); Path file = new File(filePath).toPath(); MatrixToImageWriter.writeToPath(bitMatrix, format, file); } catch (Exception e) { log.error("二维码生成出错:/permitDownload: error", e); }
/** * 给二维码下方添加说明文字 * * @param image 原二维码 * @param topText 顶部说明文字 * @param downText 底部说明文字 * @return 带说明文字的二维码 */ private static BufferedImage addNote(BufferedImage image, String topText, String downText) { Image src = image.getScaledInstance(QRCODE_WIDTH, QRCODE_HEIGHT, Image.SCALE_DEFAULT); BufferedImage tag = new BufferedImage(QRCODE_WIDTH, IMAGE_HEIGHT, BufferedImage.TYPE_INT_RGB); Graphics2D g2 = tag.createGraphics();//设置文字 g2.setColor(Color.BLACK); g2.setBackground(Color.WHITE); g2.clearRect(0,0,QRCODE_WIDTH, IMAGE_HEIGHT); //设置顶部文本并计算坐标 // 保证操作系统包含“宋体”字体,如果没有上传字体至JAVA_HOME/jre/lib/fonts下 FontMetrics fm = getFontByWidth(new Font("宋体", Font.PLAIN, DEFAULT_FONT_SIZE), topText, g2); //文字的宽度 int fontWidth = fm.stringWidth(topText); //文字高度 int fontHeight = fm.getHeight(); /** * 顶部添加文字并居中 */ g2.drawString(topText, (QRCODE_WIDTH - fontWidth) / 2, (TEXT_DEFAULT_HEIGHT - fontHeight) / 2 + fm.getFont().getSize()); /** * 绘制二维码 */ g2.drawImage(src, 0, TEXT_DEFAULT_HEIGHT, null); // 设置底部文字字体并计算坐标 // 保证操作系统包含“宋体”字体,如果没有上传字体至JAVA_HOME/jre/lib/fonts下 fm = getFontByWidth(new Font("宋体", Font.PLAIN, DEFAULT_FONT_SIZE), downText, g2); //文字的宽度 fontWidth = fm.stringWidth(downText); //文字高度 fontHeight = fm.getHeight(); /** * 添加底部文字 */ g2.drawString(downText, (QRCODE_WIDTH - fontWidth) / 2, QRCODE_HEIGHT + TEXT_DEFAULT_HEIGHT+(TEXT_DEFAULT_HEIGHT - fontHeight) / 2 + fm.getFont().getSize()); g2.dispose(); image = tag; image.flush(); return image; }
알아두어야 할 점: 현재 디자인에서는 텍스트 한 줄만 넣기 때문에 텍스트 크기는 단어 수에 따라 동적으로 변경됩니다. 합리적인 범위(인식하기에 너무 작지 않음). 이 점은 대부분의 학생들에게 유용합니다. 도움말
3. 글꼴과 크기를 동적으로 수정하세요.
/** * 根据文字长度改变文字大小 * * @param font 默认字体 * @param note 文字内容 * @param g2 图像画布 * @return 处理后的字体封装 */ private static FontMetrics getFontByWidth(Font font, String note, Graphics2D g2) { FontMetrics fm = g2.getFontMetrics(font); int textWidth = fm.stringWidth(note);//文字的宽度 if (textWidth > QRCODE_WIDTH) { int fontSize = (int) ((TEMP_PARAM / textWidth) * font.getSize()); font = new Font(font.getName(), font.getStyle(), fontSize); } g2.setFont(font); return g2.getFontMetrics(font); }
사진 합성 4단계
BufferedImage tag = new BufferedImage(QRCODE_WIDTH, IMAGE_HEIGHT, BufferedImage.TYPE_INT_RGB);
Graphics2D g2 = tag.createGraphics();//设置文字
3단계: 사진을 합성하는 과정에서 사진 추가 순서, 사진 크기, 좌표 위치가 결정됩니다. 최종 프리젠테이션 효과에 영향을 미칩니다. 최종 효과가 디자인 요구 사항을 충족하지 못하는 경우 이 세 가지 매개 변수를 조정하면 확실히 도움이 됩니다
Image src = image.getScaledInstance(QRCODE_WIDTH, QRCODE_HEIGHT, Image.SCALE_DEFAULT); ... /** * 绘制二维码 */ g2.drawImage(src, 0, TEXT_DEFAULT_HEIGHT, null);
4단계: 최종 새 사진 생성
/** * 给二维码图片添加背景图片 * * @param qrPic 二维码 * @param backImage 背景图片 */ private static void createNewPng(File qrPic, BufferedImage backImage) { try { if (!qrPic.isFile()) { log.error("二维码临时路径不存在!"); } /** * 读取背景图片,并构建绘图对象--画布 */ Graphics2D g = backImage.createGraphics(); /** * 读取二维码图片 */ BufferedImage qrcodeImage = ImageIO.read(qrPic); //开始绘制图片 g.drawImage(qrcodeImage, 48, 120, qrcodeImage.getWidth(), qrcodeImage.getHeight(), null); g.dispose(); ImageIO.write(backImage, "png", qrPic); } catch (Exception e) { log.error("绘制二维码出错!"); } }
두 번째 사진은 첫 번째 사진과 동일합니다. 텍스트 [2. QR 코드 추가 텍스트의 순서를 위쪽, 가운데, 아래쪽에서 왼쪽 및 오른쪽으로 변경하세요.
함정
QR코드에 깨진 텍스트 개발 환경(윈도우), 테스트 환경(센토스 서버 버전)이 추가되었습니다. 로컬 개발 및 테스트에는 문제가 없습니다. 패키징하여 서버에 배포한 후 모든 한자가 깨져 보이는데(다양한 트랜스코딩, 글꼴 설정, 디버깅이 오랫동안 지속됨) 문제는 여전히 변경되지 않았습니다. 그러다가 문득 시스템 폰트 문제인가 의심이 들어서 테스트 환경(centos)의 폰트 정보를 확인해 봤는데 정말 "송티"는 시스템 자체 폰트나 기본 폰트로 설정되어 있지 않더군요. 문제는 여전히 지속됩니다. 마지막으로 개발시스템(c:windowsfontssimsun.ttc)에서 테스트시스템(JAVA_HOME/jre/libs/fonts)으로 폰트를 복사하고 애플리케이션을 다시 시작하면 문제가 완벽하게 해결됩니다. 시스템 설치 및 배포를 위해 글꼴을 준비하는 것이 약간 번거롭습니다. 더 나은 방법이 있는지 모르겠습니다. 글꼴에는 논리적 글꼴과 물리적 글꼴이 포함됩니다.
배경 이미지 로딩 문제 프로젝트는 springboot 프로젝트입니다. 배경 이미지가 리소스 폴더에 저장되어 있습니다. 로컬 개발 및 테스트에서 예외가 발견되지 않았습니다. 서버에 패키징하고 배포한 후 배경 이미지를 찾을 수 없습니다. 원본 코드는 다음과 같습니다
String backFilePath = "template/down.png"; ClassPathResource resource = new ClassPathResource(backFilePath); File file = resource.getFile();
온라인에서 찾아보세요. 이미지 로딩 방법을 다 써봤지만 효과가 없습니다. 마지막으로 입력 스트림으로 변경했더니 이미지 합성이 정상이네요. 다음과 같습니다
/** * 必须通过流方式,否则不同操作系统无法拿到背景图片信息 */ String backFilePath = "template/down.png"; ClassPathResource resource = new ClassPathResource(backFilePath); BufferedImage bi = ImageIO.read(resource.getInputStream());
위 내용은 Java Zxing을 사용하여 복잡한 QR 코드 이미지를 합성하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!