JAVA生成二维码QRcode

news2024/12/25 10:31:57

JAVA生成二维码QRcode

  • 1 : 配置集成
    • 1.1、配置maven
    • 1.2、配置文件
    • 1.3、logo文件
  • 2 : 代码集成
    • 2.1、加载配置文件
    • 2.2、工具类
    • 2.3、测试类
  • 3 : 测试结果
    • 3.1、生成二维码
    • 3.2、扫描结果
    • 3.3、资源

1 : 配置集成

1.1、配置maven

pom文件中添加一下配置

<!-- QR code -->
<dependency>
    <groupId>com.google.zxing</groupId>
    <artifactId>core</artifactId>
    <version>3.3.0</version>
</dependency>

1.2、配置文件

新建配置文件 qr-image-config.properties
放置路径 /app/qrcode/config

#是否使用默认验证码:true:使用默认验证码;false:不使用默认验证码
useDefaultCode = false
#默认验证码
defaultCode = 123456
#二维码需要嵌入的图片地址,不设置则不嵌入
signImagePath = /app/qrcode/logo/logo.png
#二维码上传至服务器的地址
imageUploadPath = /app/qrcode/generate/
#二维码生成的随机数位数
codeDigit = 6
#二维码生成的随机数类型:1:数字;2:大写字母;3:小写字母;4:数字+大写字母;5:数字+小写字母;6:数字+大小写字母
codeType = 6
#二维码生成的图片类型
fileType = .jpg
#二维码有效期,单位秒
validityTime = 60

1.3、logo文件

可有可无,不需要也可以不用
logo图片 logo.png
放置路径 /app/qrcode/logo
在这里插入图片描述

2 : 代码集成

2.1、加载配置文件

新建 QrImagePathConfig.java

package com.qrcode;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.support.EncodedResource;
import org.springframework.core.io.support.PropertiesLoaderUtils;
import java.io.File;
import java.io.FileReader;
import java.io.IOException;
import java.util.Properties;

/**
 * 获取二维码图片存取路径
 */
public class QrImagePathConfig {
    private Log log = LogFactory.getLog(this.getClass());
	private static final QrImagePathConfig QR_CONFIG = new QrImagePathConfig();
	private static final String CONFIG_FILE_NAME = "/app/qrcode/config/qr-image-config.properties";

    private String useDefaultCode ;//是否使用默认验证码:true:使用默认验证码;false:不使用默认验证码
    private String defaultCode ;//默认验证码
	private String signImagePath ;//二维码需要嵌入的图片地址
	private String imageUploadPath ;//二维码上传至服务器的地址
	private String codeDigit ;//二维码生成的随机数位数
    private String codeType ;//二维码生成的随机数类型:1:数字;2:大写字母;3:小写字母;4:数字+大写字母;5:数字+小写字母;6:数字+大小写字母
	private String fileType ;//二维码生成的图片类型
    private String validityTime ;//二维码有效期,单位秒

    public String getUseDefaultCode() {
        return useDefaultCode;
    }
    public void setUseDefaultCode(String useDefaultCode) {
        this.useDefaultCode = useDefaultCode;
    }
    public String getDefaultCode() {
        return defaultCode;
    }
    public void setDefaultCode(String defaultCode) {
        this.defaultCode = defaultCode;
    }
    public String getSignImagePath() {
        return signImagePath;
    }
    public void setSignImagePath(String signImagePath) {
        this.signImagePath = signImagePath;
    }
    public String getImageUploadPath() {
        return imageUploadPath;
    }
    public void setImageUploadPath(String imageUploadPath) {
        this.imageUploadPath = imageUploadPath;
    }
    public String getCodeDigit() {
        return codeDigit;
    }
    public void setCodeDigit(String codeDigit) {
        this.codeDigit = codeDigit;
    }
    public String getCodeType() {
        return codeType;
    }
    public void setCodeType(String codeType) {
        this.codeType = codeType;
    }
    public String getFileType() {
        return fileType;
    }
    public void setFileType(String fileType) {
        this.fileType = fileType;
    }
    public String getValidityTime() {
        return validityTime;
    }
    public void setValidityTime(String validityTime) {
        this.validityTime = validityTime;
    }

    public static QrImagePathConfig getQrConfig() {
		return QR_CONFIG;
	}
	static{
        QR_CONFIG.loadProperties();
	}
	private void loadProperties(){
		 try {
			 Properties properties = null;
			 if(CONFIG_FILE_NAME.startsWith("/")){
                 properties = new Properties();
                 properties.load(new FileReader(new File(CONFIG_FILE_NAME)));
			 } else {
                 properties = PropertiesLoaderUtils.loadProperties(new EncodedResource(new ClassPathResource(CONFIG_FILE_NAME),"UTF-8"));
			 }
			 log.info("加载文件" + CONFIG_FILE_NAME + "成功!");
             String useDefaultCode = properties.getProperty("useDefaultCode");
             String defaultCode = properties.getProperty("defaultCode");
			 String signImagePath = properties.getProperty("signImagePath");
			 String imageUploadPath = properties.getProperty("imageUploadPath");
			 String codeDigit = properties.getProperty("codeDigit");
             String codeType = properties.getProperty("codeType");
			 String fileType = properties.getProperty("fileType");
			 String validityTime = properties.getProperty("validityTime");
			 log.info("useDefaultCode:"+useDefaultCode+"defaultCode:"+defaultCode+"signImagePath:"+signImagePath+",imageUploadPath:"+imageUploadPath+",codeDigit:"+codeDigit+",codeType:"+codeType+",fileType:"+fileType+",validityTime:"+validityTime);
             QR_CONFIG.useDefaultCode = useDefaultCode;
             QR_CONFIG.defaultCode = defaultCode;
			 QR_CONFIG.signImagePath = signImagePath;
             QR_CONFIG.imageUploadPath = imageUploadPath;
             QR_CONFIG.codeDigit = codeDigit;
             QR_CONFIG.codeType = codeType;
             QR_CONFIG.fileType = fileType;
             QR_CONFIG.validityTime = validityTime;
		} catch (IOException e) {
			log.warn("加载配置文件:" + CONFIG_FILE_NAME + " 失败",e);
		}
	}
}

2.2、工具类

新建BufferedImageLuminanceSource.java

package com.qrcode;

import java.awt.Graphics2D;
import java.awt.geom.AffineTransform;
import java.awt.image.BufferedImage;
import com.google.zxing.LuminanceSource;

public class BufferedImageLuminanceSource extends LuminanceSource {
    private final BufferedImage image;
    private final int left;
    private final int top;

    public BufferedImageLuminanceSource(BufferedImage image) {
        this(image, 0, 0, image.getWidth(), image.getHeight());
    }

    public BufferedImageLuminanceSource(BufferedImage image, int left, int top, int width, int height) {
        super(width, height);

        int sourceWidth = image.getWidth();
        int sourceHeight = image.getHeight();
        if (left + width > sourceWidth || top + height > sourceHeight) {
            throw new IllegalArgumentException("Crop rectangle does not fit within image data.");
        }

        for (int y = top; y < top + height; y++) {
            for (int x = left; x < left + width; x++) {
                if ((image.getRGB(x, y) & 0xFF000000) == 0) {
                    image.setRGB(x, y, 0xFFFFFFFF); // = white
                }
            }
        }

        this.image = new BufferedImage(sourceWidth, sourceHeight, BufferedImage.TYPE_BYTE_GRAY);
        this.image.getGraphics().drawImage(image, 0, 0, null);
        this.left = left;
        this.top = top;
    }

    public byte[] getRow(int y, byte[] row) {
        if (y < 0 || y >= getHeight()) {
            throw new IllegalArgumentException("Requested row is outside the image: " + y);
        }
        int width = getWidth();
        if (row == null || row.length < width) {
            row = new byte[width];
        }
        image.getRaster().getDataElements(left, top + y, width, 1, row);
        return row;
    }

    public byte[] getMatrix() {
        int width = getWidth();
        int height = getHeight();
        int area = width * height;
        byte[] matrix = new byte[area];
        image.getRaster().getDataElements(left, top, width, height, matrix);
        return matrix;
    }

    public boolean isCropSupported() {
        return true;
    }

    public LuminanceSource crop(int left, int top, int width, int height) {
        return new BufferedImageLuminanceSource(image, this.left + left, this.top + top, width, height);
    }

    public boolean isRotateSupported() {
        return true;
    }

    public LuminanceSource rotateCounterClockwise() {
        int sourceWidth = image.getWidth();
        int sourceHeight = image.getHeight();
        AffineTransform transform = new AffineTransform(0.0, -1.0, 1.0, 0.0, 0.0, sourceWidth);
        BufferedImage rotatedImage = new BufferedImage(sourceHeight, sourceWidth, BufferedImage.TYPE_BYTE_GRAY);
        Graphics2D g = rotatedImage.createGraphics();
        g.drawImage(image, transform, null);
        g.dispose();
        int width = getWidth();
        return new BufferedImageLuminanceSource(rotatedImage, top, sourceWidth - (left + width), getHeight(), width);
    }
}

新建 QRCodeUtil.java

package com.qrcode;

import java.awt.*;
import java.awt.geom.RoundRectangle2D;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.OutputStream;
import java.util.Hashtable;
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.Result;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.common.HybridBinarizer;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;

public class QRCodeUtil {
    private static final String CHARSET = "utf-8";
    private static final String FORMAT_NAME = "JPG";
    // 二维码尺寸
    private static final int QR_CODE_SIZE = 300;
    // LOGO宽度
    private static final int WIDTH = 60;
    // LOGO高度
    private static final int HEIGHT = 60;

    private static BufferedImage createImage(String content, String imgPath, boolean needCompress) throws Exception {
        Hashtable hints = new Hashtable();
        hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);
        //设置编码字符集
        hints.put(EncodeHintType.CHARACTER_SET, CHARSET);
        hints.put(EncodeHintType.MARGIN, 1);
        //1、生成二维码
        BitMatrix bitMatrix = new MultiFormatWriter().encode(content, BarcodeFormat.QR_CODE, QR_CODE_SIZE, QR_CODE_SIZE, hints);
        //2、获取二维码宽高
        int width = bitMatrix.getWidth();
        int height = bitMatrix.getHeight();
        //3、将二维码放入缓冲流
        BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
        for (int x = 0; x < width; x++) {
            for (int y = 0; y < height; y++) {
                //4、循环将二维码内容定入图片
                image.setRGB(x, y, bitMatrix.get(x, y) ? 0xFF000000 : 0xFFFFFFFF);
            }
        }
        if (imgPath == null || "".equals(imgPath)) {
            return image;
        }
        // 插入图片
        QRCodeUtil.insertImage(image, imgPath, needCompress);
        return image;
    }

    private static void insertImage(BufferedImage source, String imgPath, boolean needCompress) throws Exception {
        File file = new File(imgPath);
        if (!file.exists()) {
            System.err.println("" + imgPath + "   该文件不存在!");
            return;
        }
        Image src = ImageIO.read(new File(imgPath));
        int width = src.getWidth(null);
        int height = src.getHeight(null);
        if (needCompress) { // 压缩LOGO
            if (width > WIDTH) {
                width = WIDTH;
            }
            if (height > HEIGHT) {
                height = HEIGHT;
            }
            Image image = src.getScaledInstance(width, height, Image.SCALE_SMOOTH);
            BufferedImage tag = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
            Graphics g = tag.getGraphics();
            g.drawImage(image, 0, 0, null); // 绘制缩小后的图
            g.dispose();
            src = image;
        }
        // 插入LOGO
        Graphics2D graph = source.createGraphics();
        int x = (QR_CODE_SIZE - width) / 2;
        int y = (QR_CODE_SIZE - height) / 2;
//        graph.setColor(Color.BLUE);//设定嵌入的图像边框颜色
        graph.drawImage(src, x, y, width, height, null);
        Shape shape = new RoundRectangle2D.Float(x, y, width, width, 6, 6);
        graph.setStroke(new BasicStroke(3f));
        graph.draw(shape);
        graph.dispose();
    }

    public static void encode(String content, String imgPath, String destPath, boolean needCompress) throws Exception {
        BufferedImage image = QRCodeUtil.createImage(content, imgPath, needCompress);
        mkdirs(destPath);
        // String file = new Random().nextInt(99999999)+".jpg";
        // ImageIO.write(image, FORMAT_NAME, new File(destPath+"/"+file));
        ImageIO.write(image, FORMAT_NAME, new File(destPath));
    }

    public static BufferedImage encode(String content, String imgPath, boolean needCompress) throws Exception {
        BufferedImage image = QRCodeUtil.createImage(content, imgPath, needCompress);
        return image;
    }

    public static void mkdirs(String destPath) {
        File file = new File(destPath);
        // 当文件夹不存在时,mkdirs会自动创建多层目录,区别于mkdir.(mkdir如果父目录不存在则会抛出异常)
        if (!file.exists() && !file.isDirectory()) {
            file.mkdirs();
        }
    }

    public static void encode(String content, String imgPath, String destPath) throws Exception {
        QRCodeUtil.encode(content, imgPath, destPath, false);
    }
    // 被注释的方法
	/*
	 * public static void encode(String content, String destPath, boolean
	 * needCompress) throws Exception { QRCodeUtil.encode(content, null, destPath,
	 * needCompress); }
	 */

    public static void encode(String content, String destPath) throws Exception {
        QRCodeUtil.encode(content, null, destPath, false);
    }

    public static void encode(String content, String imgPath, OutputStream output, boolean needCompress)
            throws Exception {
        BufferedImage image = QRCodeUtil.createImage(content, imgPath, needCompress);
        ImageIO.write(image, FORMAT_NAME, output);
    }

    public static void encode(String content, OutputStream output) throws Exception {
        QRCodeUtil.encode(content, null, output, false);
    }

    public static String decode(File file) throws Exception {
        BufferedImage image;
        image = ImageIO.read(file);
        if (image == null) {
            return null;
        }
        BufferedImageLuminanceSource source = new BufferedImageLuminanceSource(image);
        BinaryBitmap bitmap = new BinaryBitmap(new HybridBinarizer(source));
        Result result;
        Hashtable hints = new Hashtable();
        hints.put(DecodeHintType.CHARACTER_SET, CHARSET);
        result = new MultiFormatReader().decode(bitmap, hints);
        String resultStr = result.getText();
        return resultStr;
    }

    public static String decode(String path) throws Exception {
        return QRCodeUtil.decode(new File(path));
    }
}

2.3、测试类

新建 QrCodeTest.java

package com.qrcode;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Random;
import java.util.UUID;

public class QrCodeTest {

    public static void main(String[] args) throws Exception {
        //text存放在二维码中的内容
        String num = "";
        if("true".equals(QrImagePathConfig.getQrConfig().getUseDefaultCode())){
            num = QrImagePathConfig.getQrConfig().getDefaultCode();
        } else {
            num = getRandomCodeByDigitAndType(Integer.valueOf(QrImagePathConfig.getQrConfig().getCodeDigit()),Integer.valueOf(QrImagePathConfig.getQrConfig().getCodeType()));
        }
        String text = "二维码测试,验证码为:\n" + "*********************\n" + "****** "+ num +" ******\n" + "*********************";
        //imgPath嵌入二维码的图片路径
//        String imgPath = "E:\\app\\qrcode\\config\\logo.png";
//        String imgPath = "/app/qrcode/config/logo.png";
        String imgPath = QrImagePathConfig.getQrConfig().getSignImagePath();
        //destPath生成的二维码的路径及名称
        String fileName = UUID.randomUUID().toString().replaceAll("-", "").substring(0,10);
        String destPath = QrImagePathConfig.getQrConfig().getImageUploadPath() + fileName + QrImagePathConfig.getQrConfig().getFileType();
        //生成二维码
        //text:编码到二维码中的内容;imgPath:要嵌入二维码的图片路径,如果不写或者为null则生成一个没有嵌入图片的纯净的二维码
        //destPath:生成的二维码的存放路径;true:表示将嵌入二维码的图片进行压缩,如果为“false”则表示不压缩.
        QRCodeUtil.encode(text, imgPath, destPath, true);
        //解析二维码
        //destPath:将要解析的二维码的存放路径;该方法返回值为String类型,即返回解析出的文字或者数字等.
        String str = QRCodeUtil.decode(destPath);
        //打印出解析出的内容
        System.out.println(str);
        //二维码存的信息越多,二维码图片也就越复杂,容错率也就越低,识别率也越低,并且二维码能存的内容大小也是有限的(大概500个汉字左右)
    }

    /**
     * 根据位数和类型获取验证码
     * @param codeDigit  随机数位数
     * @param codeType  随机数类型:1:数字;2:大写字母;3:小写字母;4:数字+大写字母;5:数字+小写字母;6:数字+大小写字母
     * @return
     */
    public static String getRandomCodeByDigitAndType(Integer codeDigit,Integer codeType) {
        String string = "";
        String numStr = "0123456789";
        String lowerCaseLetter = "abcdefghijklmnopqrstuvwxyz";
        String upperCaseLetter = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
        switch (codeType){
            case 1:
                string = numStr;
                break;
            case 2:
                string = upperCaseLetter;
                break;
            case 3:
                string = lowerCaseLetter;
                break;
            case 4:
                string = numStr + upperCaseLetter;
                break;
            case 5:
                string = numStr + lowerCaseLetter;
                break;
            case 6:
                string = numStr + lowerCaseLetter + upperCaseLetter;
                break;
        }
        string = getStringForDisorderSorting(string);//将字符串的顺序打乱
        StringBuilder stringBuilder = new StringBuilder();//声明一个StringBuffer对象保存验证码
        for (int i = 0; i < codeDigit; i++) {
            Random random = new Random();//创建随机数生成器
            int index = random.nextInt(string.length());//返回[0,string.length)范围的int值,保存下标
            char charAt = string.charAt(index);//charAt(),返回指定索引处的char值
            stringBuilder.append(charAt);//将charAt字符串追加到此序列
        }
        return stringBuilder.toString();//返回字符串
    }

    /**
     * 随机打乱一个字符串的排序方式
     * @param string
     * @return
     */
    private static String getStringForDisorderSorting(String string) {
        char[] chars = string.toCharArray();
        List<Character> characterList = new ArrayList<Character>();
        for(char ch:chars){
            characterList.add(ch);
        }
        Collections.shuffle(characterList);
        StringBuilder stringBuilder = new StringBuilder();
        for(Character character:characterList){
            stringBuilder.append(character);
        }
        return stringBuilder.toString();
    }

    /**
     * 根据位数获取数字验证码
     * @param codeDigit
     * @return
     */
    private static String getRandomNum(Integer codeDigit){
        StringBuilder stringBuilder=new StringBuilder();
        Random random = new Random();
        for (int i = 0; i < codeDigit; i++) {
            stringBuilder.append(random.nextInt(10));
        }
        return stringBuilder.toString();
    }
}

3 : 测试结果

3.1、生成二维码

在路径 /app/qrcode/generate
在这里插入图片描述

3.2、扫描结果

在这里插入图片描述

3.3、资源

CSDN下载链接:https://download.csdn.net/download/qq_38254635/87351537
百度云下载链接:https://pan.baidu.com/s/1zBaTmv2WgG8Ihx-HIEdoow?pwd=v7mc
提取码: v7mc

有什么不对的还望指正,书写不易,觉得有帮助就点个赞吧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/124227.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

立根铸魂 崛起数智时代 欧拉部署超300万套

操作系统产业峰会2022于12月28日在线上举办。本次峰会由开放原子开源基金会、中国软件行业协会、CCF&#xff08;中国计算机学会&#xff09;开源专委会、绿色计算产业联盟、中关村科学城管委会共同主办&#xff0c;以“立根铸魂 崛起数智时代”为主题&#xff0c;汇聚全产业链…

Day842.Future -Java 并发编程实战

Future Hi&#xff0c;我是阿昌&#xff0c;今天学习记录的是关于Future的内容。 ThreadPoolExecutor 的 void execute(Runnable command) 方法&#xff0c;利用这个方法虽然可以提交任务&#xff0c;但是却没有办法获取任务的执行结果&#xff08;execute() 方法没有返回值&…

Java Web高级面试题(二)

✅作者简介&#xff1a;热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏&#xff1a;Java面试题…

总体分布、样本分布、抽样分布的区别

总体分布、样本分布、抽样分布的区别 参考自&#xff1a;THREE DISTRIBUTIONS 1.总体分布 The population is the whole set of values, or individuals, you are interested in. The population which consists of a set of scores (5, 6, 7, 8) which distribute around a …

比Hive还快10倍的大数据计算引擎

最近几年&#xff0c;Presto这个大数据组件越来越多地出现在程序员的岗位需求中&#xff0c;很多应届同学一番自我检查后发现&#xff0c;在学校都没怎么接触过&#xff0c;更不用说了解了。 某游戏公司岗位需求 Presto到底是个啥&#xff1f; 有什么用&#xff1f; 适合哪些…

月薪9K!前台测试男生偷偷努力,工资翻倍转行5G网络优化工程师,“卷死”所有人!

你是否有过这样的生活&#xff1f; 每天重复着外场测试&#xff0c;抱着电脑&#xff0c;手机在户外验证基站信号;长期的出差&#xff0c;频繁的更换城市;每个月领着3500块钱的工资&#xff0c;可能再工作2-3年会涨到4000元&#xff0c;技术也不可能有任何的突破。生活重复单调…

Ubuntu 上安装conan + Cmake

目录 1. 在用户目录下创建共享目录&#xff0c;然后将共享目录挂载到上去 2. 安装新版本的cmake 3. 安装conan 4. 配置CONAN环境 5. 配置conan的访问服务器URL 6. Linux 平台需要切换gcc编译器版本为c11以及version 7. 安装opengl开发库 1. 在用户目录下创建共享目录&…

springboot+mybatis如何快速插入大量数据

在公司业务开发过程中&#xff0c;我们经常会遇到往数据库表中插入大量数据的场景&#xff0c;比如excel批量导入数据。那么该如何快速地插入数据呢&#xff1f;我们可以考虑使用批量插入来实现&#xff0c;该方案实测每秒能达到35000条&#xff0c;后附具体实现代码。接下来我…

经典蓝牙连接过程

ACL是连接基础&#xff0c;sco需要再ACL连接成功后建立连接。基本链接流程如下: 一、创建连接 从用户点击扫描到的蓝牙设备开始&#xff0c;发起连接的设备会发送create_connection给controller。controller会进行page过程。 而后会底层链接完成&#xff0c;这个完成仅仅是确认…

CSS初级教程【第四天】

CSS初级教程【第四天】【1】CSS 外边距【2】CSS 外边距合并【3】所有 CSS 外边距属性【4】CSS 内边距【5】内边距和元素宽度【6】所有 CSS 内边距属性【7】CSS 高度和宽度【8】设置 CSS 尺寸属性CSS上回学习链接 CSS初级教程【第一天】 CSS初级教程【第二天】 CSS初级教程【第三…

windows本地安装openjdk环境

1讲一下为什么要选openjdk 先讲一下为什么小编在这里选择openjdk&#xff0c;而不是像绝大多数人一样选择jdk。那是因为&#xff0c;jdk在1.8的某个版本之后&#xff0c;就不是免费的版本了&#xff0c;不能免费用于商用。有过工作经验的就知道&#xff0c;公司一般对电脑上安装…

掌握这17张图,没人比你更懂RecyclerView的预加载

回顾上一篇文章&#xff0c;我们为了减少描述问题的维度&#xff0c;于演示之前附加了许多限制条件&#xff0c;比如禁用了RecyclerView的预拉取机制。 实际上&#xff0c;预拉取(prefetch)机制作为RecyclerView的重要特性之一&#xff0c;常常与缓存复用机制一起配合使用、共…

Go语言设计与实现 -- Mutex源码剖析

上图来自面向信仰编程 上图中&#xff0c;第一列为常见的同步原语&#xff0c;第二列为容器&#xff0c;第三列为互斥锁。 接下来我们来逐一介绍一下&#xff1a; Mutex 我们先来看一下sync.Mutex的结构体&#xff1a; type Mutex struct {// 当前互斥锁的状态state int32…

代码随想录算法训练营第一天 java : 704.二分查找法、27.移除算法

文章目录Leecode 704.二分查找题目连接&#xff1a;[Leecode 704.二分查找](https://leetcode.cn/problems/remove-element/)遇到的问题题目二分法的第一种写法 &#xff08;左闭右闭)第二种解法&#xff08;左闭右开 代码呈现&#xff09;Leecode 27.移除元素题目链接&#xf…

MyBatis【创建与使用】

MyBatis【创建与使用】&#x1f34e;一. MyBatis&#x1f352;1.1. MyBatis 是什么&#xff1f;&#x1f352;1.2 没有使用MyBatis时的操作流程&#x1f352;1.3 MyBatis的操作与数据库之间的流程&#x1f34e;二.创建MyBatis项目&#x1f352;2.1 idea创建&#x1f352;2.2 配…

【Git】一文带你入门Git分布式版本控制系统(撤销修改、删除文件)

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;也会涉及到服务端 &#x1f4c3;个人状态&#xff1a; 在校大学生一枚&#xff0c;已拿多个前端 offer&#xff08;秋招&#xff09; &#x1f680;未…

Debezium故障演练

1、搭建演练环境 postgresql及wal2json插件安装:https://blog.csdn.net/li281037846/article/details/128411222 kafka及kafka-connect安装&#xff0c;略 //添加debezium connector curl -i -X POST -H "Content-Type:application/json" -H "Accepted:applic…

Qt样式(qss)使用小结(软件换肤,比如暗黑模式)

1.背景&#xff1a; Qt style sheet&#xff08;qss&#xff09;跟前端技术一样&#xff0c;就是为了美化界面。关键是&#xff0c;太好用了。之前还为此写过一篇博客。 Qt样式&#xff08;qss&#xff09;手册小结_大橘的博客-CSDN博客 其中主要是记录如何获取手册细节。 …

6、GPIO输入按键检测(轮询检测)

目录 0x01、简介 0x02、硬件设计 0x03、编写函数 0x001、按键初始化 0x002、按键检测 0x003、按键led翻转 0x04、源程序下载地址 0x01、简介 本次实验主要实现按键控制LED灯。 由于机械按键在按下和抬起的时候会产生按键抖动&#xff0c;所以在设计的时候需要考虑如何消除抖…

Pytorch可视化特征图(代码 亲测可用)

2013年Zeiler和Fergus发表的《Visualizing and Understanding Convolutional Networks》 早期LeCun 1998年的文章《Gradient-Based Learning Applied to Document Recognition》中的一张图也非常精彩&#xff0c;个人觉得比Zeiler 2013年的文章更能给人以启发。从下图的F6特征&…