#Java实现二维码登录_java开发生成登陆二维码-程序员宅基地

技术标签: spring  java  后端  redis  

二维码登录的流程:

1)用户选择二维码登录;

前端发送获取二维码请求,服务器收到请求后生成一个uuid(用于绑定二维码),然后根据指定网址和uuid生成对应的二维码,将uuid作为key,一个对应的code(生成二维码成功)作为value存入Redis,再将生成二维码流返回给前端展示。前端成功获取二维码后开始两秒一次轮询后端接口,直到登录成功或二维码超时(当然也包括关闭页面)

这里使用的是Redis集群模板:
@Resource
private RedisClusterTemplate jedis;

	public AjaxResultModel generateQrCode() {
    
        AjaxResultModel resultModel = new AjaxResultModel();
        // 生成唯一ID
        String uuid = UUID.randomUUID().toString();
        String uuidStr = PREFILED + uuid;
        jedis.setEx(uuidStr, GENERATEQRCODE, TIMEOUTVALUE);
        resultModel.setCode(ResultCodeEnum.SUCCESS_STATUS.getCode());
        resultModel.setMessage(ResultCodeEnum.SUCCESS_STATUS.getMessage());
        resultModel.setData(createQrCode("http://baidu.com" + "?uuid=" + uuid));

        log.info("二维码登录生成uuid:{}", uuid);

        return resultModel;
    }

2)用户扫描;

用户在移动端登录后扫描网页二维码,二维码解析出来的是一个网址,经websocket直接进行跳转,跳转到确认登录页面,该页面在初始化时调用后端接口(该请求中携带uuid),改变Redis中的code(扫码成功未登录)。

	/**
	 * 扫码成功进入确认登录页面记录状态
	 * @param uuid
	 * @return
	 */
	@RequestMapping(value = "/scanCode.ajax", method = RequestMethod.POST)
    @ResponseBody
    public AjaxResultModel scanCode(@RequestBody String uuid) {
    
        AjaxResultModel resultModel = new AjaxResultModel();
        SessionUserCommon user = SessionUserCommonUtil.getSessionUser();
        if (user.getId() == null) {
    
            resultModel.setCode(ResultCodeEnum.SESSION_OUT.getCode());
            resultModel.setMessage(ResultCodeEnum.SESSION_OUT.getMessage());
            return resultModel;
        }
        String uuidStr = PREFILED + uuid;
        jedis.setEx(uuidStr, CONFIRMLOGINSCODE, TIMEOUTVALUE);
        resultModel.setCode(SCANCODE);
        resultModel.setMessage(SCANMSG);
        return resultModel;
    }

3)用户确认登录;

用户点击确认登录按钮,移动端发送一个确认登录请求(该请求中携带uuid),后端收到请求后改变Redis中的code(确认登录成功),并获取请求中的token等用户相关的登录信息,再以uuid+后缀 为key,登录信息为value存入Redis。

	/**
	 * 扫码确认登录
	 * @param uuid
	 * @return
	 */
	@RequestMapping(value = "/confirmLogin.ajax", method = RequestMethod.GET)
    @ResponseBody
    public AjaxResultModel confirmLogin(@RequestParam("uuid") String uuid, HttpServletRequest request) {
    
        AjaxResultModel resultModel = new AjaxResultModel();
        resultModel.setStatus("200");
        SessionUserCommon user = SessionUserCommonUtil.getSessionUser();
        if (user.getCode() == null) {
    
            resultModel.setCode(ResultCodeEnum.SESSION_OUT.getCode());
            resultModel.setMessage(ResultCodeEnum.SESSION_OUT.getMessage());
            resultModel.setSuccess(false);
            return resultModel;
        }
        String uuidStr = PREFILED + uuid;
        // 判断确认登录时二维码是否过期
        if (!jedis.exists(uuidStr)) {
    
            resultModel.setCode(QRCODEEXPIREDCODE);
            resultModel.setMessage(QRCODEEXPIREDMSG);
            resultModel.setSuccess(false);
            return resultModel;
        }
        String uuidValue = PREFILED + uuid + ENDFILED;
        String authToken = request.getHeader("AUTH_TOKEN");
        Map<String, Object> infoMap = new HashMap<>();
        // 将token和userCode放入json对象
        infoMap.put("authToken", authToken);
        infoMap.put("userCode", user.getCode());
        JSONObject jsonObject = new JSONObject(infoMap);
        jedis.setEx(uuidStr, CONFIRMLOGINSCODE, TIMEOUTVALUE);
        jedis.setEx(uuidValue, jsonObject.toJSONString(), TIMEOUTVALUE);
        resultModel.setCode(ResultCodeEnum.SUCCESS_STATUS.getCode());
        resultModel.setMessage(ResultCodeEnum.SUCCESS_STATUS.getMessage());
        resultModel.setData(uuid);
        log.info("二维码登录>>>>>> 移动端扫码确认登录成功,确认人:{}", user.getName());
        return resultModel;
    }

4)登录完成;

在整个过程中网页端一直在轮询后端接口,当uuid对应的code为“确认登录成功”时将用户相关的登录信息返回,此时轮询结束。

	/**
     * PC端轮询
     * 放开鉴权
     * @param uuid
     * @return
     */
    @RequestMapping(value = "/polling.ajax", method = RequestMethod.GET)
    @ResponseBody
    public AjaxResultModel polling (@RequestParam("uuid") String uuid) {
    
        AjaxResultModel resultModel = new AjaxResultModel();
        String uuidValue = PREFILED + uuid;
        String uuidInfoValue = uuidValue + ENDFILED;
        String uuidStr = PREFILED + uuid;
        String status = jedis.get(uuidValue);
        // 判断过期
        if (!jedis.exists(uuidStr)) {
    
            status = QRCODEEXPIREDCODE;
        }
        switch (status) {
    
            case GENERATEQRCODE:
                resultModel.setCode(GENERATEQRCODE);
                resultModel.setMessage(GENERATEQRMSG);
                break;
            case SCANCODE:
                resultModel.setCode(SCANCODE);
                resultModel.setMessage(SCANMSG);
                break;
            case CONFIRMLOGINSCODE:
                String info = jedis.get(uuidInfoValue);
                jedis.del(uuidValue);
                jedis.del(uuidInfoValue);
                resultModel.setCode(CONFIRMLOGINSCODE);
                resultModel.setMessage(CONFIRMLOGINSMSG);
                resultModel.setData(info);
                break;
            case QRCODEEXPIREDCODE:
                jedis.del(uuidValue);
                jedis.del(uuidInfoValue);
                resultModel.setCode(QRCODEEXPIREDCODE);
                resultModel.setMessage(QRCODEEXPIREDMSG);
                resultModel.setSuccess(false);
                break;
             default:
                 resultModel.setCode(UUIDNOTEXITCODE);
                 resultModel.setMessage(UUIDNOTEXITMSG);
                 break;
        }
        return resultModel;
    }

注:生成二维码代码:

package com.example.demo;

import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Image;
import java.awt.Shape;
import java.awt.geom.RoundRectangle2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;

import javax.imageio.ImageIO;

import com.google.zxing.*;
import com.google.zxing.client.j2se.BufferedImageLuminanceSource;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.common.HybridBinarizer;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;

/**
 * 生成二维码
 * @author chuangchuang.yang
 * @date 2020年4月19日
 * @version v6
 */
public class QrCodeUtil {
    
    private static final int BLACK = Color.black.getRGB();
    private static final int BLUE = Color.blue.getRGB();
    private static final int WHITE = Color.WHITE.getRGB();
    private static final int DEFAULT_QR_SIZE = 183;
    // 图片类型:位图
    private static final String DEFAULT_QR_FORMAT = "png";
    private static final byte[] EMPTY_BYTES = new byte[0];

    public static byte[] createQrCode(String content, int size, String extension) {
    
        Image image = null;
        try {
    
            image = ImageIO.read(new File("D:\\download\\mmexport1586329085259.jpg"));
        } catch (IOException e) {
    
            e.printStackTrace();
        }
        return createQrCode(content, size, extension, image);
    }

    /**
     * 生成带图片的二维码
     * @param content  二维码中要包含的信息
     * @param size  大小
     * @param extension  文件格式扩展
     * @param insertImg  中间的logo图片
     * @return
     */
    public static byte[] createQrCode(String content, int size, String extension, Image insertImg) {
    
        if (size <= 0) {
    
            throw new IllegalArgumentException("size (" + size + ")  cannot be <= 0");
        }
        ByteArrayOutputStream baos = null;
        try {
    
            Map<EncodeHintType, Object> points = new HashMap<EncodeHintType, Object>();
            points.put(EncodeHintType.CHARACTER_SET, "utf-8");
            points.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.M);

            //使用信息生成指定大小的点阵
            BitMatrix matrix = new MultiFormatWriter().encode(content, BarcodeFormat.QR_CODE, size, size, points);

            //去掉白边
            matrix = updateBit(matrix, 0);

            int width = matrix.getWidth();
            int height = matrix.getHeight();

            //将BitMatrix中的信息设置到BufferdImage中,形成黑白图片
            BufferedImage bufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
            for (int i = 0; i < width; i++) {
    
                for (int j = 0; j < height; j++) {
    
                    bufferedImage.setRGB(i, j, matrix.get(i, j) ? BLUE : WHITE);
                }
            }
            if (insertImg != null) {
    
                // 插入中间的logo图片
                insertImage(bufferedImage, insertImg, matrix.getWidth());
            }
            //将因为去白边而变小的图片再放大
            bufferedImage = zoomInImage(bufferedImage, size, size);
            baos = new ByteArrayOutputStream();
            ImageIO.write(bufferedImage, extension, baos);

            return baos.toByteArray();
        } catch (Exception e) {
    
        } finally {
    
            if(baos != null)
                try {
    
                    baos.close();
                } catch (IOException e) {
    
                    e.printStackTrace();
                }
        }
        return EMPTY_BYTES;
    }

    /**
     * 自定义二维码白边宽度
     * @param matrix
     * @param margin
     * @return
     */
    private static BitMatrix updateBit(BitMatrix matrix, int margin) {
    
        int tempM = margin * 2;
        int[] rec = matrix.getEnclosingRectangle(); // 获取二维码图案的属性
        int resWidth = rec[2] + tempM;
        int resHeight = rec[3] + tempM;
        BitMatrix resMatrix = new BitMatrix(resWidth, resHeight); // 按照自定义边框生成新的BitMatrix
        resMatrix.clear();
        for (int i = margin; i < resWidth - margin; i++) {
     // 循环,将二维码图案绘制到新的bitMatrix中
            for (int j = margin; j < resHeight - margin; j++) {
    
                if (matrix.get(i - margin + rec[0], j - margin + rec[1])) {
    
                    resMatrix.set(i, j);
                }
            }
        }
        return resMatrix;
    }

    // 图片放大缩小
    public static BufferedImage zoomInImage(BufferedImage originalImage, int width, int height) {
    
        BufferedImage newImage = new BufferedImage(width, height, originalImage.getType());
        Graphics g = newImage.getGraphics();
        g.drawImage(originalImage, 0, 0, width, height, null);
        g.dispose();
        return newImage;
    }

    private static void insertImage(BufferedImage source, Image insertImg, int size) {
    
        try {
    
            int width = insertImg.getWidth(null);
            int height = insertImg.getHeight(null);
            width = width > size / 6 ? size / 6 : width;  // logo设为二维码的六分之一大小
            height = height > size / 6 ? size / 6 : height;
            Graphics2D graph = source.createGraphics();
            int x = (size - width) / 2;
            int y = (size - height) / 2;
            graph.drawImage(insertImg, 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();
        } catch (Exception e) {
    
            e.printStackTrace();
        }
    }

    public static byte[] createQrCode(String content) {
    
        return createQrCode(content, DEFAULT_QR_SIZE, DEFAULT_QR_FORMAT);
    }

    /** 
          * 二维码解析 
          * @param analyzePath    二维码路径 
          * @return 
          * @throws IOException 
          */  
    @SuppressWarnings({
     "rawtypes", "unchecked" })
    public static String zxingCodeAnalyze(String analyzePath) throws Exception{
    
        MultiFormatReader formatReader = new MultiFormatReader();
        String resultStr = null;
        try {
    
            File file = new File(analyzePath);
            if (!file.exists())
            {
    
                return "二维码不存在";
            }
            BufferedImage image = ImageIO.read(file);
            LuminanceSource source = new BufferedImageLuminanceSource(image);
            Binarizer binarizer = new HybridBinarizer(source);
            BinaryBitmap binaryBitmap = new BinaryBitmap(binarizer);
            Map hints = new HashMap();
            hints.put(EncodeHintType.CHARACTER_SET, "UTF-8");
            Result result = formatReader.decode(binaryBitmap, hints);
            resultStr = result.getText();
        } catch (NotFoundException e) {
    
            e.printStackTrace();
        }
        return resultStr;
    }

    public static void main(String[] args){
    
        try {
    
            // 生成唯一ID
            int uuid = (int) (Math.random() * 100000);
            // 生成二维码时绑定uuid (移动端扫码后跳转到确认登录的页面,此时已经携带该uuid,在确认登录时携带该uuid访问后端接口)
//            byte[] qrCode = createQrCode("" + "?uuid=" + uuid);

            FileOutputStream fos = new FileOutputStream("D:\\usuallyUse\\erweima.png");
            fos.write(createQrCode("http://baidu.com" + "?uuid=" + uuid));
            fos.close();

//            System.out.println(zxingCodeAnalyze("D:\\usuallyUse\\erweima.png"));
        } catch (Exception e) {
    
            e.printStackTrace();
        }

    }

}

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_43962314/article/details/106438376

智能推荐

linux下c编程实例_linuxc有趣的编程例子-程序员宅基地

文章浏览阅读911次。linux下c编程使用gcc编译器,下载gcc:sudo yum install -b gcc current编写程序一、最简单的hellowordvi hello.c#include int main() { printf("Hello World\n"); return 0;}编译:gcc hello.c -o hel_linuxc有趣的编程例子

nginx下WrodPress永久链接-程序员宅基地

文章浏览阅读102次。nginx下实现了WrodPress的永久链接rewrite。简单的方法,修改nginx.conf文件:location / {if (-f $request_filename/index.html){rewrite (.*) $1/index.html break;}if (-f $request_filename/index.php){rewrite (.*) $1..._nginx 永久连接

r语言如何输出html格式,R语言设置write.table()输出的文件格式-程序员宅基地

文章浏览阅读3.2k次。write.table(),是保存数据为文件的函数。> xiaohuqingdan > xiaohuqingdan[1] 3900088702 3900072499 3900021029> write.table(xiaohuqingdan,"1234.txt")得到的1234.txt,打开是这个样子:"x""1" 3900088702"2" 3900072499"3" 3900..._r语言怎么保存成html

Nginx/Apache恶意User-Agent过滤列表 屏蔽蜘蛛爬虫-程序员宅基地

文章浏览阅读1.8k次。从网上收集的一些恶意或无用User-Agent关键词,通过User-Agent屏蔽访问,包含漏洞扫描,无用蜘蛛,采集爬虫,cc攻击,sq注入等等。(Abonti|aggregator|AhrefsBot|asterias|BDCbot|BLEXBot|BuiltBotTough|Bullseye|BunnySlippers|ca-crawler|CCBot|Cegbfeieh|CheeseBot|CherryPicker|CopyRightCheck|cosmos|Crescent|discobot|D_恶意user-agent

Java多线程之间如何进行通信?-程序员宅基地

文章浏览阅读1.1k次。1、Object.wait()与Object.notify()/notifyAll()2、Thread.join() 面试题:新建 T1、T2、T3 三个线程,如何保证它们按顺序执行?代码如下:package com.autocoding.juc;import java.util.concurrent.TimeUnit;import lombok.extern.slf4j.Slf4j;/** * * 面试题:新建 T1、T2、T3 三个线程,如何保证它们按顺..._多线程之间如何进行通信

c++ 循环程序的作业,2017年10月10日作业题。_c++阅读程序、循环题-程序员宅基地

文章浏览阅读457次。作业1:需求:输出一个由 * 符号所组成的矩形,要求每行有50个 * ,一共需要有60行。使用双重for循环完成。作业2:需求:输出一个由 * 符号所组成的三角形,要求第一行一个 * ,第二行 两个 * 第三行 三个 * 依次类推,最后一行10个 *。使用双重for循环完成。作业3:需求:输出一个由 * 符号所组成的三角形,要求第一行十个 * ,第二行 九个 * 第三行 八个 * 依次类_c++阅读程序、循环题

随便推点

Gradle依赖替换_dependencysubstitution-程序员宅基地

文章浏览阅读6.6k次,点赞6次,收藏11次。前言在项目开发过程中,不可避免的需要引入一些第三方库,而不同的第三方库之间,可能存在一些依赖关系。例如:你依赖了库A与B,而同时B也依赖于A。这样就可能存在这种情况:你依赖的A的版本与B中依赖的A的版本不同。1. gradle的默认解决方案通常情况下,gradle通过选择最高版本解决来解决版本冲突,下面以RxJava为例: implementation("io.reactivex.r..._dependencysubstitution

从oracle中读取blob图片资源_java oracle clob 图片-程序员宅基地

文章浏览阅读1.3k次。1.对应图片资源这张表创建对应的实体类Ibatisblob.javaprivate byte[] picresult;public byte[] getPicresult() { return picresult; }public void setPicresult(byte[] picresult) { this.picresult = picresul_java oracle clob 图片

进制转换_1500转换为二进制-程序员宅基地

文章浏览阅读1.5k次。进制转换常用的进制有二进制(计算机的语言),八进制,十进制(生活中所用的就是十进制),十六进制(内存地址一般用十六进制显示)。那么这些进制是如何相互转换的呢首先说二进制 :给你一个十进制,比如6,如果将它转换成二进制数呢?10进制数转换成二进制数,这是一个连续除以2的过程:把要转换的数,除以2,得到商和余数,将商继续除以2,直到商为0。最后将所有余数倒序排列,得到数就是转换结果。二..._1500转换为二进制

阿里巴巴 开源软件列表-程序员宅基地

文章浏览阅读1k次。服务框架 DubboDubbo 是阿里巴巴公司开源的一个高性能优秀的服务框架,使得应用可通过高性能的 RPC 实现服务的输出和输入功能,可以和 Spring框架无缝集成。JDBC连接池、监控组件 DruidDruid是一个JDBC组件,它包括三部分:代理Driver,能够提供基于Filter-Chain模式的插件体系。DruidDataSo..._阿里simple el

玩转Redis-Lua脚本入门到实战-树形结构存储及查询_redis 部门tree-程序员宅基地

文章浏览阅读2.6k次。《玩转Redis》系列文章 by zxiaofan 系列第【16】篇,Lua脚本入门到实战、调试Lua脚本、树形结构的存储方案、“邻接表”、“路径枚举”、查找部门的所有上级部门。_redis 部门tree

Linux面试题及答案,适合毕业生和小白,适合查缺补漏-程序员宅基地

文章浏览阅读1.7k次,点赞3次,收藏25次。linux问题回答好了有两大作用,第一能证明你除了做过编码以外,还能通过看linux日志排查线上问题,第二能证明你在linux上做过部署和发布相关工作。 相反,如果你连linux基本命令都不会,那么面试官轻则认为你只有在windows上写代码的能力,重则就会质疑你的项目是学习项目。 在本文里,就将结合“排查线上问题”和“在linux上部署项目”这两大主题,全面分析下面试中经常会问到的linux相关面试题。1、怎么以绝对路径和相对路径的方式进入目录?怎么看当前处在哪个目录下?..._linux面试题