> Java > java지도 시간 > 본문

Java Zxing을 사용하여 복잡한 QR 코드 이미지를 합성하는 방법

WBOY
풀어 주다: 2023-04-18 15:16:03
앞으로
1413명이 탐색했습니다.

전체 아이디어:

zxing의 성숙한 QR 코드 생성 인터페이스를 도입하여 표준 QR 코드 파일을 생성하고, Java 그래픽 이미지 처리 API를 통해 QR 코드에 관련 텍스트 설명을 추가하고, 필요에 따라 합성 이미지에 관련 배경을 추가합니다. 예시는 아래와 같습니다.

Java Zxing을 사용하여 복잡한 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);
            }
    로그인 후 복사

2.
  •     /**
         * 给二维码下方添加说明文字
         *
         * @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. 파란색 기본 지도와 QR코드 이미지를 결합하면 끝이다.
  • 사진 합성 4단계

  • 1단계: 캔버스를 만듭니다. 캔버스의 너비와 높이를 설정해야 합니다. 단위는 픽셀이어야 합니다.
BufferedImage tag = new BufferedImage(QRCODE_WIDTH, IMAGE_HEIGHT, BufferedImage.TYPE_INT_RGB);
로그인 후 복사

2단계: 그래픽 개체를 만듭니다. 캔버스에 Graphics2D를 사용하여 이미지에 대한 지식을 바탕으로 사용할 수 있습니다. 클릭하면 배경, 전경, 테두리, 너비 및 기타 정보 등의 설정을 이해할 수 있습니다

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 코드와 일치해야 합니다. 그렇지 않으면 QR코드와 배경의 비율이 심각해집니다. QR코드가 잘못 조정되었거나 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 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
원천:yisu.com
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿