二维码登录原理及生成与解析

一、前言

  这几天在研究二维码的扫码登录。初来乍到,还有好多东西不懂。在网上看到有人写了一些通过QRCode或者Zxing实现二维码的生成和解码。一时兴起,决定自己亲手试一试。本人是通过QRCode实现的,下面具体的说一下。

二、二维码原理

  基础知识参考:http://news.cnblogs.com/n/191671/

  很重要的一部分知识:二维码一共有 40 个尺寸。官方叫版本 Version。Version 1 是 21 x 21 的矩阵,Version 2 是 25 x 25 的矩阵,Version 3 是 29 的尺寸,每增加一个 version,就会增加 4 的尺寸,公式是:(V-1)*4 + 21(V是版本号) 最高 Version 40,(40-1)*4+21 = 177,所以最高是 177 x 177 的正方形。

  

三、二维码生成和解码工具

1.效果如下图所示。

  

生成二维码(不含有logo)                                                   生成二维码(带有logo)

  

对应的解码

  工具很简单,但是很实用。界面还可以美化,功能还可以加强,初心只是为了练习一下二维码的生成和解析。

2.二维码生成和解析的核心类

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.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;import javax.imageio.ImageIO;import com.swetake.util.Qrcode;import jp.sourceforge.qrcode.QRCodeDecoder;
import jp.sourceforge.qrcode.exception.DecodingFailedException;  public class TwoDimensionCode {  //二维码 SIZEprivate static final int CODE_IMG_SIZE = 235;// LOGO SIZE (为了插入图片的完整性,我们选择在最中间插入,而且长宽建议为整个二维码的1/7至1/4)private static final int INSERT_IMG_SIZE = CODE_IMG_SIZE/5;/** * 生成二维码(QRCode)图片 * @param content 存储内容 * @param imgPath 二维码图片存储路径 * @param imgType 图片类型 * @param insertImgPath logo图片路径*/  public void encoderQRCode(String content, String imgPath, String imgType, String insertImgPath) {  try {  BufferedImage bufImg = this.qRCodeCommon(content, imgType, insertImgPath);  File imgFile = new File(imgPath);if (!imgFile.exists()){imgFile.mkdirs();}// 生成二维码QRCode图片  
            ImageIO.write(bufImg, imgType, imgFile);  } catch (Exception e) {  e.printStackTrace();  }  }   /** * 生成二维码(QRCode)图片 * @param content 存储内容 * @param output 输出流 * @param imgType 图片类型 */  public void encoderQRCode(String content, OutputStream output, String imgType) {  try {  BufferedImage bufImg = this.qRCodeCommon(content, imgType, null);  // 生成二维码QRCode图片  
            ImageIO.write(bufImg, imgType, output);  } catch (Exception e) {  e.printStackTrace();  }  }  /*** @param content* @param imgType* @param size* @param imgPath    嵌入图片的名称* @return*/private BufferedImage qRCodeCommon(String content, String imgType, String imgPath){BufferedImage bufImg = null;  try {  Qrcode qrcodeHandler = new Qrcode();  // 设置二维码排错率,可选L(7%)、M(15%)、Q(25%)、H(30%),排错率越高可存储的信息越少,但对二维码清晰度的要求越小  qrcodeHandler.setQrcodeErrorCorrect('M');  qrcodeHandler.setQrcodeEncodeMode('B');  // 设置设置二维码尺寸,取值范围1-40,值越大尺寸越大,可存储的信息越大  qrcodeHandler.setQrcodeVersion(15);  // 获得内容的字节数组,设置编码格式  byte[] contentBytes = content.getBytes("utf-8");  // 图片尺寸  int imgSize = CODE_IMG_SIZE;  bufImg = new BufferedImage(imgSize, imgSize, BufferedImage.TYPE_INT_RGB);  Graphics2D gs = bufImg.createGraphics();  // 设置背景颜色  
            gs.setBackground(Color.WHITE);  gs.clearRect(0, 0, imgSize, imgSize);  // 设定图像颜色> BLACK  
            gs.setColor(Color.BLACK);  // 设置偏移量,不设置可能导致解析出错  final int pixoff = 2;final int sz = 3;// 输出内容> 二维码  if (contentBytes.length > 0 && contentBytes.length < 800) {  boolean[][] codeOut = qrcodeHandler.calQrcode(contentBytes);  for (int i = 0; i < codeOut.length; i++) {  for (int j = 0; j < codeOut.length; j++) {  if (codeOut[j][i]) {  gs.fillRect(j * sz + pixoff, i * sz + pixoff, sz, sz);  }  }  }  } else {  throw new Exception("QRCode content bytes length = " + contentBytes.length + " not in [0, 800].");  }  //嵌入logoif(imgPath != null)this.insertImage(bufImg, imgPath, true);gs.dispose();  bufImg.flush();  } catch (Exception e) {  e.printStackTrace();  }  return bufImg;  }private 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 > INSERT_IMG_SIZE) {  width = INSERT_IMG_SIZE;  }  if (height > INSERT_IMG_SIZE) {  height = INSERT_IMG_SIZE;  }  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 = (CODE_IMG_SIZE - width) / 2;  int y = (CODE_IMG_SIZE - height) / 2;  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();  }  /** * 解析二维码(QRCode) * @param imgPath 图片路径 * @return */  public String decoderQRCode(String imgPath) {  // QRCode 二维码图片的文件  File imageFile = new File(imgPath);  BufferedImage bufImg = null;  String content = null;  try {  bufImg = ImageIO.read(imageFile);  QRCodeDecoder decoder = new QRCodeDecoder();  content = new String(decoder.decode(new TwoDimensionCodeImage(bufImg)), "utf-8");   } catch (IOException e) {  System.out.println("Error: " + e.getMessage());  e.printStackTrace();  } catch (DecodingFailedException dfe) {  System.out.println("Error: " + dfe.getMessage());  dfe.printStackTrace();  }  return content;  }  /** * 解析二维码(QRCode) * @param input 输入流 * @return */  public String decoderQRCode(InputStream input) {  BufferedImage bufImg = null;  String content = null;  try {  bufImg = ImageIO.read(input);  QRCodeDecoder decoder = new QRCodeDecoder();  content = new String(decoder.decode(new TwoDimensionCodeImage(bufImg)), "utf-8");   } catch (IOException e) {  System.out.println("Error: " + e.getMessage());  e.printStackTrace();  } catch (DecodingFailedException dfe) {  System.out.println("Error: " + dfe.getMessage());  dfe.printStackTrace();  }  return content;  }  
}  
View Code

3.具体注意的地方

//二维码 SIZE
private static final int CODE_IMG_SIZE = 235;
// LOGO SIZE (为了插入图片的完整性,我们选择在最中间插入,而且长宽建议为整个二维码的1/7至1/4)
private static final int INSERT_IMG_SIZE = CODE_IMG_SIZE/5;

  对于二维码图片大小还是不会计算,如果有人看到这里,方便的话可以告诉小弟一声。我这里的这个值(235)是通过设定好QrcodeVersion(版本15),以及绘制图像时偏移量pixoff=2和black区域的size=3,最终生成图片后,将图片通过ps打开,然后确定图片的尺寸信息。

  还有就是中间的logo不要过大,否则会导致QRCode解析出错,但是手机扫码不一定会出错。感觉手机扫码解析比QRCode解析能力强。

Qrcode qrcodeHandler = new Qrcode();  
// 设置二维码排错率,可选L(7%)、M(15%)、Q(25%)、H(30%),排错率越高可存储的信息越少,但对二维码清晰度的要求越小  
qrcodeHandler.setQrcodeErrorCorrect('M');  
qrcodeHandler.setQrcodeEncodeMode('B');  
// 设置设置二维码尺寸,取值范围1-40,值越大尺寸越大,可存储的信息越大  
qrcodeHandler.setQrcodeVersion(15);  

  一般设置version就好了,网上好多都是7或者8,我尝试下更大的值,15的话二维码看起来很密集。

// 设置偏移量,不设置可能导致解析出错  
final int pixoff = 2;
final int sz = 3;
// 输出内容> 二维码  
if (contentBytes.length > 0 && contentBytes.length < 800) {  boolean[][] codeOut = qrcodeHandler.calQrcode(contentBytes);  for (int i = 0; i < codeOut.length; i++) {  for (int j = 0; j < codeOut.length; j++) {  if (codeOut[j][i]) {  gs.fillRect(j * sz + pixoff, i * sz + pixoff, sz, sz);  }  }  }  
}

  绘制black区域的时候要设置偏移量,要不然可能导致二维码识别出错。 black区域的大小根据实际情况来就好。

四、二维码登录原理

1.原理图

   按照自己的理解画的,结合上图,看一下代码吧。

2.GetQrCodeController.java

/*** @author hjzgg* 获取二维码图片*/
@Controller
public class GetQrCodeController {@RequestMapping(value="/getTwoDemensionCode")@ResponseBodypublic String getTwoDemensionCode(HttpServletRequest request){String uuid = UUID.randomUUID().toString().substring(0, 8);String ip = "localhost";try {ip = InetAddress.getLocalHost().getHostAddress();} catch (UnknownHostException e) {e.printStackTrace();}//二维码内容String content = "http://" + ip + ":8080/yycc-portal/loginPage?uuid=" + uuid;//生成二维码String imgName =  uuid + "_" + (int) (new Date().getTime() / 1000) + ".png";String imgPath = request.getServletContext().getRealPath("/") + imgName;//String insertImgPath = request.getServletContext().getRealPath("/")+"img/hjz.jpg";TwoDimensionCode handler = new TwoDimensionCode();handler.encoderQRCode(content, imgPath, "png", null);//生成的图片访问地址String qrCodeImg = "http://" + ip + ":8080/yycc-portal/" + imgName;JSONObject json = new JSONObject();json.put("uuid", uuid);json.put("qrCodeImg", qrCodeImg);return json.toString();}
}

  用户请求扫码方式登录,后台生成二维码,将uuid和二维码访问地址传给用户。

3.LongConnectionCheckController.java

@Controller
public class LongConnectionCheckController {private static final int LONG_TIME_WAIT = 30000;//30s
    @Autowiredprivate RedisTemplate<String, Object> redisTemplate;@RequestMapping(value="/longUserCheck")public String longUserCheck(String uuid){long inTime = new Date().getTime();Boolean bool = true;while (bool) {try {Thread.sleep(500);} catch (InterruptedException e) {e.printStackTrace();}//检测登录UserVo userVo = (UserVo) redisTemplate.opsForValue().get(uuid);System.out.println("LongConnectionCheckAction:" + userVo);if(userVo != null){redisTemplate.delete(uuid);return "forward:/loginTest?username=" + userVo.getUsername() + "&password=" + userVo.getPassword();}else{if(new Date().getTime() - inTime > LONG_TIME_WAIT){bool = false;redisTemplate.delete(uuid);}}}return "forward:/longConnectionFail";}@RequestMapping(value="/longConnectionFail")@ResponseBodypublic String longConnectionFail(){JSONObject json = new JSONObject();json.put("success", false);json.put("message", "长连接已断开!");return json.toString();}
}

  用户获得uuid和二维码之后,请求后台的长连接(携带uuid),不断检测uuid是否有对应的用户信息,如果有则转到登录模块(携带登录信息)。

4.PhoneLoginController.java

/*** @author hjzgg*    手机登录验证*/
@Controller
public class PhoneLoginController {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;@RequestMapping(value="/phoneLogin")public void phoneLogin(String uuid, String username, String password){UserVo user = (UserVo) redisTemplate.opsForValue().get(uuid);if(user == null) {user = new UserVo(username, password);}System.out.println(user);redisTemplate.opsForValue().set(uuid, user);}@RequestMapping(value="/loginPage")public String loginPage(HttpServletRequest request, String uuid){request.setAttribute("uuid", uuid);return "phone_login";}
}

  用户通过手机扫码之后,在手机端输入用户信息,然后进行验证(携带uuid),后台更新uuid对应的用户信息,以便长连接可以检测到用户登录信息。

五、源码下载

  二维码登录例子以及二维码生成解析工具源码下载:https://github.com/hjzgg/QRCodeLoginDemo

转载于:https://www.cnblogs.com/hujunzheng/p/5661443.html

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

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

相关文章

knockout+echarts实现图表展示

一、需要学习的知识 knockout, require, director, echarts, jquery。简单的入一下门&#xff0c;网上的资料很多&#xff0c;最直接就是进官网校习。 二、效果展示 三、require的配置 require.config.js中可以配置我们的自定义模块的加载。 require.config({baseUrl: ".&…

React中使用Ant Table组件

一、Ant Design of React http://ant.design/docs/react/introduce 二、建立webpack工程 webpackreact demo下载 项目的启动&#xff0c;参考 三、简单配置 1.工程下载下来之后&#xff0c;在src目录下新建目录“table”&#xff0c;新建app.js&#xff0c;内容如下。 import R…

解决“Dynamic Web Module 3.0 requires Java 1.6 or newer.”错误

一、问题描述 1.错误截图如下。 2.设计的问题 在Eclipse中新建了一个Maven工程, 然后更改JDK版本为1.6, 结果每次使用Maven > Update project的时候JDK版本都恢复成1.5。 二、原因分析 Maven官方文档有如下描述&#xff1a; 编译器插件用来编译项目的源文件.从3.0版本开始, …

解决cookie跨域访问

一、前言 随着项目模块越来越多&#xff0c;很多模块现在都是独立部署。模块之间的交流有时可能会通过cookie来完成。比如说门户和应用&#xff0c;分别部署在不同的机器或者web容器中&#xff0c;假如用户登陆之后会在浏览器客户端写入cookie&#xff08;记录着用户上下文信息…

React使用antd Table生成层级多选组件

一、需求 用户对不同的应用需要有不同的权限&#xff0c;用户一般和角色关联在一起&#xff0c;新建角色的时候会选择该角色对应的应用&#xff0c;然后对应用分配权限。于是写了一种实现的方式。首先应用是一个二级树&#xff0c;一级表示的是应用分组&#xff0c;二级表示的是…

junit4进行单元测试

一、前言 提供服务的时候&#xff0c;为了保证服务的正确性&#xff0c;有时候需要编写测试类验证其正确性和可用性。以前的做法都是自己简单写一个控制层&#xff0c;然后在控制层里调用服务并测试&#xff0c;这样做虽然能够达到测试的目的&#xff0c;但是太不专业了。还是老…

快速搭建springmvc+spring data jpa工程

一、前言 这里简单讲述一下如何快速使用springmvc和spring data jpa搭建后台开发工程&#xff0c;并提供了一个简单的demo作为参考。 二、创建maven工程 http://www.cnblogs.com/hujunzheng/p/5450255.html 三、配置文件说明 1.application.properties jdbc.drivercom.mysql.jd…

git亲测命令

一、Git新建本地分支与远程分支关联问题 git checkout -b branch_name origin/branch_name 或者 git branch --set-upstream branch_name origin/branch_name 或者 git branch branch_name git branch --set-upstream-toorigin/branch_name branch_name 二、查看本地分支所关…

mysql 7下载安装及问题解决

mysql 7安装及问题解决 一、mysql下载 下载地址&#xff1a;https://www.mysql.com/downloads/Community (GPL) DownloadsMySQL Community Server (GPL)Windows (x86, 64-bit), ZIP ArchiveNo thanks, just start my download.二、mysql安装 解压到指定目录在mysql bin目录下打…

tomcat开发远程调试端口以及利用eclipse进行远程调试

一、tomcat开发远程调试端口 方法1 WIN系统 在catalina.bat里&#xff1a;   SET CATALINA_OPTS-server -Xdebug -Xnoagent -Djava.compilerNONE -Xrunjdwp:transportdt_socket,servery,suspendn,address8899   Linux系统 在catalina.sh里&#xff1a;   CATALINA_OPTS&q…

webpack+react+redux+es6开发模式

一、预备知识 node, npm, react, redux, es6, webpack 二、学习资源 ECMAScript 6入门 React和Redux的连接react-redux Redux 入门教程 redux middleware 详解 Redux研究 React 入门实例教程 webpack学习demo NPM 使用介绍 三、工程搭建 之前有写过 webpackreactes6开发模式…

fiddler发送post请求

1.指定为 post 请求&#xff0c;输入 url Content-Type: application/x-www-form-urlencoded;charsetutf-8 request body中的参数格式&#xff1a;userNameadminicxp&userPassword123qwe!# 这种方式可以用 request.getParameter的方式来获得。 2.指定为 post 请求&#xff…

Kettle之数据抽取、转换、装载

Kettle 官网 ETL利器Kettle实战应用解析系列 利用kettle组件导入excel文件到数据库 kettle中实现动态SQL查询 java中调用kettle转换文件 kettle 7.x版本下载&#xff1a;https://pan.baidu.com/s/1nvnzzCH  密码&#xff1a;6f5c mac 下运行spoon.sh,  windows下为spoon.bat…

webpack+react+redux+es6开发模式---续

一、前言 之前介绍了webpackreactreduxes6开发模式 &#xff0c;这个项目对于一个独立的功能节点来说是没有问题的。假如伴随着源源不断的需求&#xff0c;前段项目会涌现出更多的功能节点&#xff0c;需要独立部署运行。为了更好地管理这些独立的功能节点&#xff0c;我们需要…

RabbitMQ安装和使用(和Spring集成)

一、安装Rabbit MQ   Rabbit MQ 是建立在强大的Erlang OTP平台上&#xff0c;因此安装Rabbit MQ的前提是安装Erlang。通过下面两个连接下载安装3.2.3 版本&#xff1a; 下载并安装 Eralng OTP For Windows (vR16B03)运行安装 Rabbit MQ Server Windows Installer (v3.2.3)具体…

单点登录实现(spring session+redis完成session共享)

一、前言 项目中用到的SSO&#xff0c;使用开源框架cas做的。简单的了解了一下cas&#xff0c;并学习了一下 单点登录的原理&#xff0c;有兴趣的同学也可以学习一下&#xff0c;写个demo玩一玩。 二、工程结构 我模拟了 sso的客户端和sso的服务端&#xff0c; sso-core中主要是…

加密策略

一、前言 这两天研究了一下项目中的密码加密&#xff0c;可以说得上是学到了很多。下面来大致说一下。 二、常用加密 1.单向加密算法 单向加密算法主要用来验证数据传输的过程中&#xff0c;是否被篡改过。 BASE64 严格地说&#xff0c;属于编码格式&#xff0c;而非加密算法 …

Spring Data JPA: 实现自定义Repository

一、前言 由于项目中的 实体&#xff08;entity&#xff09;默认都是继承一个父类&#xff08;包含一些公共的属性&#xff0c;比如创建时间&#xff0c;修改时间&#xff0c;是否删除&#xff0c;主键id&#xff09;。为了实现逻辑删除&#xff0c;一般会自己实现RepositoryFa…

js冲刺一下

js中__proto__和prototype的区别和关系 1.对象有属性__proto__,指向该对象的构造函数的原型对象。  2.方法除了有属性__proto__,还有属性prototype&#xff0c;prototype指向该方法的原型对象。 深入浅出妙用 Javascript 中 apply、call、bind ***两道面试题*** 关于js中伪数…

Jackson序列化实例

参考文章 Jackson使用ContextualSerializer在序列化时获取字段注解的属性 使用BeanSerializerModifier定制jackson的自定义序列化(null值的处理) 关于使用ContextualSerializer的补充 BeanSerializerFactory中有如下代码&#xff0c; 关于设置SerializerModifier&#xff0c;如…