1. 程式人生 > >實現手機掃描二維碼進行登入

實現手機掃描二維碼進行登入

專案結構:

實現流程:

pc端:

1:開啟二維碼登入網頁index.html

2:index.html呼叫GetQrCodeServlet

3:GetQrCodeServlet幹2件事

  a:生成隨機的uuid,是一個唯一標識,該標識貫穿整個流程

  b:生成二維碼圖片,二維碼資訊:http://xx.xx.xx.xx:8080/QrCodeLoginPro/Login.html?uuid=" + uuid

4:index頁面展示二維碼

5:index頁面呼叫LongConnectionCheckServlet進行長連線輪詢操作,引數為uuid

6:LongConnectionCheckServlet只幹1件事

  a:拿到uuid後迴圈檢查loginUserMap中uuid是否不為null。

7:如果為null則代表沒有登入,index.html將繼續進行輪詢

  ps: LongConnectionCheckServlet 一個長連線請求檢測登入狀態

    loginUserMap 是一個靜態的map結構的登入池,uuid為key , 登入資訊為value

手機端:

1:掃描pc端的二維碼

2:開啟二維碼中的網頁 http://xx.xx.xx.xx:8380/QrCodeLoginPro/Login.html?uuid=" + uuid

3:登入,將uname upwd uuid 傳遞給登入程式PhoneLoginServlet

4:PhoneLoginServlet幹2件事

  a:檢測登入

  b:登入成功後將登入資訊插入到loginUserMap中去,uuid為key

pc端:

  1:繼續輪詢檢測uuid中是否為null

  2:登入後的uuid中就不為null了,此時LongConnectionCheckServlet停止迴圈,返回登入狀態。

程式碼:

cn.kuwo下的3個servlet

複製程式碼
package cn.kuwo;

import java.io.IOException;
import java.io.PrintWriter;
import java.util.Date;
import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import cn.kuwo.util.TwoDimensionCode; /** * 生成二維碼圖片以及uuid * @author zijuntang * */ public class GetQrCodeServlet extends HttpServlet { private static final long serialVersionUID = 1L; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { PrintWriter out = response.getWriter(); //生成唯一ID int uuid = (int) (Math.random() * 100000); //二維碼內容 String content = "http://xx.xx.xx.xx:8380/QrCodeLoginPro/Login.html?uuid=" + uuid; //生成二維碼 String imgName = uuid + "_" + (int) (new Date().getTime() / 1000) + ".png"; String imgPath = "/home/web/apache/htdocs/QrCodeLogin/" + imgName; TwoDimensionCode handler = new TwoDimensionCode(); handler.encoderQRCode(content, imgPath, "png"); //生成的圖片訪問地址 String qrCodeImg = "http://xx.xx.xx.xx/QrCodeLogin/" + imgName; String jsonStr = "{\"uuid\":" + uuid + ",\"qrCodeImg\":\"" + qrCodeImg + "\"}"; out.print(jsonStr); out.flush(); out.close(); } }
複製程式碼 複製程式碼
package cn.kuwo;

import java.io.IOException;
import java.io.PrintWriter;
import java.util.Date;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import cn.kuwo.vo.LoginUserVo;
import cn.kuwo.vo.UserVo;

/**
 * 用長連線,檢查登入狀態
 * @author zijuntang
 *
 */
public class LongConnectionCheckServlet extends HttpServlet {
    private static final long serialVersionUID = 1L;
    public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }
    public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        String uuid = request.getParameter("uuid");
        String jsonStr = "";
        System.out.println("in");
        System.out.println("uuid:" + uuid);
        long inTime = new Date().getTime();
        Boolean bool = true;
        while (bool) {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            //檢測登入
            UserVo userVo = LoginUserVo.getLoginUserMap().get(uuid);
            System.out.println("userVo:" + userVo);
            if(userVo != null){
                bool = false;
                jsonStr = "{\"uname\":\""+userVo.getUname()+"\"}";
                LoginUserVo.getLoginUserMap().remove(uuid);
            }else{
                if(new Date().getTime() - inTime > 5000){
                    bool = false;
                }
            }
        }
        System.out.println("login ok : " + jsonStr);
        PrintWriter out = response.getWriter();
        out.print(jsonStr);
        out.flush();
        out.close();
    }
}
複製程式碼 複製程式碼
package cn.kuwo;

import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import cn.kuwo.vo.LoginUserVo;
import cn.kuwo.vo.UserVo;

/**
 * 二維碼手機端登入
 * @author zijuntang
 *
 */
public class PhoneLoginServlet extends HttpServlet {
    private static final long serialVersionUID = 1L;
    public PhoneLoginServlet() {
        super();
        // TODO Auto-generated constructor stub
    }
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        String uuid = request.getParameter("uuid");
        String uname = request.getParameter("uname");
        String upwd = request.getParameter("upwd");
        System.out.println(uuid);
        System.out.println(uname);
        System.out.println(upwd);
        //TODO 驗證登入
        boolean bool = true;
        if(bool){
            //將登陸資訊存入map
            UserVo userVo = LoginUserVo.getLoginUserMap().get(uuid);
            if(userVo == null){
                userVo = new UserVo();
                userVo.setUname(uname);
                userVo.setUpwd(upwd);
                LoginUserVo.getLoginUserMap().put(uuid, userVo);
            }
        }
        PrintWriter out = response.getWriter();
        out.print(bool);
        out.flush();
        out.close();
    }
}
複製程式碼

cn.kuwo.util包下的生成二維碼的封裝類

複製程式碼
package cn.kuwo.util;  
  
import java.awt.Color;  
import java.awt.Graphics2D;  
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 jp.sourceforge.qrcode.QRCodeDecoder;  
import jp.sourceforge.qrcode.exception.DecodingFailedException;  
import com.swetake.util.Qrcode;  
  
public class TwoDimensionCode {  
      
    /** 
     * 生成二維碼(QRCode)圖片 
     * @param content 儲存內容 
     * @param imgPath 圖片路徑 
     */  
    public void encoderQRCode(String content, String imgPath) {  
        this.encoderQRCode(content, imgPath, "png", 7);  
    }  
      
    /** 
     * 生成二維碼(QRCode)圖片 
     * @param content 儲存內容 
     * @param output 輸出流 
     */  
    public void encoderQRCode(String content, OutputStream output) {  
        this.encoderQRCode(content, output, "png", 7);  
    }  
      
    /** 
     * 生成二維碼(QRCode)圖片 
     * @param content 儲存內容 
     * @param imgPath 圖片路徑 
     * @param imgType 圖片型別 
     */  
    public void encoderQRCode(String content, String imgPath, String imgType) {  
        this.encoderQRCode(content, imgPath, imgType, 7);  
    }  
      
    /** 
     * 生成二維碼(QRCode)圖片 
     * @param content 儲存內容 
     * @param output 輸出流 
     * @param imgType 圖片型別 
     */  
    public void encoderQRCode(String content, OutputStream output, String imgType) {  
        this.encoderQRCode(content, output, imgType, 7);  
    }  
  
    /** 
     * 生成二維碼(QRCode)圖片 
     * @param content 儲存內容 
     * @param imgPath 圖片路徑 
     * @param imgType 圖片型別 
     * @param size 二維碼尺寸 
     */  
    public void encoderQRCode(String content, String imgPath, String imgType, int size) {  
        try {  
            BufferedImage bufImg = this.qRCodeCommon(content, imgType, size);  
              
            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 圖片型別 
     * @param size 二維碼尺寸 
     */  
    public void encoderQRCode(String content, OutputStream output, String imgType, int size) {  
        try {  
            BufferedImage bufImg = this.qRCodeCommon(content, imgType, size);  
            // 生成二維碼QRCode圖片  
            ImageIO.write(bufImg, imgType, output);  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
      
    /** 
     * 生成二維碼(QRCode)圖片的公共方法 
     * @param content 儲存內容 
     * @param imgType 圖片型別 
     * @param size 二維碼尺寸 
     * @return 
     */  
    private BufferedImage qRCodeCommon(String content, String imgType, int size) {  
        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(size);  
            // 獲得內容的位元組陣列,設定編碼格式  
            byte[] contentBytes = content.getBytes("utf-8");  
            // 圖片尺寸  
            int imgSize = 67 + 12 * (size - 1);  
            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);  
            // 設定偏移量,不設定可能導致解析出錯  
            int pixoff = 2;  
            // 輸出內容> 二維碼  
            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 * 3 + pixoff, i * 3 + pixoff, 3, 3);  
                        }  
                    }  
                }  
            } else {  
                throw new Exception("QRCode content bytes length = " + contentBytes.length + " not in [0, 800].");  
            }  
            gs.dispose();  
            bufImg.flush();  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
        return bufImg;  
    }  
      
    /** 
     * 解析二維碼(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;  
    }  
  
    public static void main(String[] args) {  
        String imgPath = "D:/aaa/Michael_QRCode.png";  
        String encoderContent = "http://xx.xx.xx.xx:8380/QrCodeLoginPro/Login.html";  
        TwoDimensionCode handler = new TwoDimensionCode();  
        handler.encoderQRCode(encoderContent, imgPath, "png");
        
        
        /*
        System.out.println("========encoder success");  
        String decoderContent = handler.decoderQRCode(imgPath);  
        System.out.println("解析結果如下:");  
        System.out.println(decoderContent);  
        System.out.println("========decoder success!!!");  
        */
    }  
}  
複製程式碼 複製程式碼
package cn.kuwo.util;
  
import java.awt.image.BufferedImage;  
import jp.sourceforge.qrcode.data.QRCodeImage;  
  
public class TwoDimensionCodeImage implements QRCodeImage {  
  
    BufferedImage bufImg;  
      
    public TwoDimensionCodeImage(BufferedImage bufImg) {  
        this.bufImg = bufImg;  
    }  
      
    @Override  
    public int getHeight() {  
        return bufImg.getHeight();  
    }  
  
    @Override  
    public int getPixel(int x, int y) {  
        return bufImg.getRGB(x, y);  
    }  
  
    @Override  
    public int getWidth() {  
        return bufImg.getWidth();  
    }  
  
}  
複製程式碼

cn.kuwo.vo下的2個數據層

複製程式碼
package cn.kuwo.vo;

import java.util.HashMap;

public class LoginUserVo {
    private static HashMap<String, UserVo> loginUserMap = new HashMap<String, UserVo>();
    private static LoginUserVo loginUserVo;
    public static LoginUserVo getVo(){
        if(loginUserVo == null){
            loginUserVo = new LoginUserVo();
        }
        return loginUserVo;
    }
    public static HashMap<String, UserVo> getLoginUserMap() {
        return loginUserMap;
    }
}
複製程式碼 複製程式碼
package cn.kuwo.vo;

public class UserVo {
    private String uname;
    private String upwd;
    public String getUname() {
        return uname;
    }
    public void setUname(String uname) {
        this.uname = uname;
    }
    public String getUpwd() {
        return upwd;
    }
    public void setUpwd(String upwd) {
        this.upwd = upwd;
    }
}
複製程式碼

2個網頁

複製程式碼
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<script type="text/javascript" src="js/jquery-1.11.0.min.js"></script>
<body>
    <div id="divCon">
        <img src="" id="QrCodeImg" />
    </div>
</body>
<script type="text/javascript">
    $(document).ready(function() {
        var uuid;
        $.get("/QrCodeLoginPro/GetQrCodeServlet", function(data, status) {
            var obj = eval("(" + data + ")");
            //儲存UUID
            uuid = obj.uuid;
            //顯示二維碼
            $("#QrCodeImg").attr("src", obj.qrCodeImg);
            //開始驗證登入
            validateLogin();
        });

        function validateLogin(){
            $.get("/QrCodeLoginPro/LongConnectionCheckServlet?uuid=" + uuid , function(data, status) {
                if(data == ""){
                    validateLogin();
                }else{
                    var obj = eval("(" + data + ")");
                    alert("登入成功了:" + obj.uname);
                }
            });
        }
    });
</script>
</html>
複製程式碼