场景题相关

微信的扫码登录是如何实现的

扫码登录具体的每个步骤:

image

①:用户 A 访问微信网页版,微信服务器为这个会话生成一个全局唯一的 ID,上面的 URL 中 obsbQ-Dzag== 就是这个 ID,此时系统并不知道访问者是谁。

②:用户A打开自己的手机微信并扫描这个二维码,并提示用户是否确认登录。

③:手机上的微信是登录状态,用户点击确认登录后,手机上的微信客户端将微信账号和这个扫描得到的 ID 一起提交到服务器

④:服务器将这个 ID 和用户 A 的微信号绑定在一起,并通知网页版微信,这个 ID 对应的微信号为用户 A,网页版微信加载用户 A 的微信信息,至此,扫码登录全部流程完成

浏览器是如何得知用户已经扫码的呢
答:浏览器轮询服务器,获取扫码状态

// 查看扫码状态
checkLogin: function(e, a) {
    var n = t.defer()
        , a = a || 0;
    return window.code = 0,
    window.checkLoginPromise = $.ajax({
        url: i.API_login + "?loginicon=true&uuid=" + e + "&tip=" + a + "&r=" + ~new Date,
        dataType: "script",
        timeout: 35e3
    }).done(function() {
        new RegExp("/" + location.host + "/");
        if (window.redirect_uri && window.redirect_uri.indexOf("/" + location.host + "/") < 0)
            return void (location.href = window.redirect_uri);
        var e = {
            code: window.code,
            redirect_uri: window.redirect_uri,
            userAvatar: window.userAvatar
        };
        n.resolve(e)
    }).fail(function() {
        n.reject()
    }),
    n.promise
}

然后再根据服务器返回的扫码状态,进行相应的操作

*   408 扫码超时 如果手机没有扫码或没有授权登录,服务器会阻塞约25s,然后返回状态码 408 -> 前端继续轮询

    ![image](//upload-images.jianshu.io/upload_images/14742999-06d0289a4e2b2afc.JPG?imageMogr2/auto-orient/strip%7CimageView2/2/w/1000/format/webp)

    ![image](//upload-images.jianshu.io/upload_images/14742999-2043a0cdc24a7676.JPG?imageMogr2/auto-orient/strip%7CimageView2/2/w/1000/format/webp)

*   400 二维码失效 大约5分钟的时间内不扫码,二维码失效

    ![image](//upload-images.jianshu.io/upload_images/14742999-947e22b0abe02df6.JPG?imageMogr2/auto-orient/strip%7CimageView2/2/w/1000/format/webp)

*   201 已扫码 如果手机已经扫码,服务器立即返回状态码和用户的基本信息 (window.code=201,window.code.userAvator="..."),-> 前端继续轮询

    ![image](//upload-images.jianshu.io/upload_images/14742999-44d09ab883d6e7a6.JPG?imageMogr2/auto-orient/strip%7CimageView2/2/w/1000/format/webp)

*   200 已授权 如果手机点击了确认登录,服务器返回200及token -> 前端停止轮询, 获取到token,重定向到目标页

    ![image](//upload-images.jianshu.io/upload_images/14742999-f64e5128a60fa09e.JPG?imageMogr2/auto-orient/strip%7CimageView2/2/w/1000/format/webp)
// 根据服务器返回的扫码状态,进行相应的操作
function o(c) {
    switch (c.code) {
    case 200:
        t.newLoginPage(c.redirect_uri).then(function(t) {
            var o = t.match(/<ret>(.*)<\/ret>/)
                , r = t.match(/<script>(.*)<\/script>/)
                , c = t.match(/<skey>(.*)<\/skey>/)
                , s = t.match(/<wxsid>(.*)<\/wxsid>/)
                , l = t.match(/<wxuin>(.*)<\/wxuin>/)
                , d = t.match(/<pass_ticket>(.*)<\/pass_ticket>/)
                , f = t.match(/<message>(.*)<\/message>/)
                , u = t.match(/<redirecturl>(.*)<\/redirecturl>/);
            return u ? void (window.location.href = u[1]) : o && "0" != o[1] ? (alert(f && f[1] || "登陆失败"),
            i.report(i.AUTH_FAIL_COUNT, 1),
            void location.reload()) : (e.$emit("newLoginPage", {
                Ret: o && o[1],
                SKey: c && c[1],
                Sid: s && s[1],
                Uin: l && l[1],
                Passticket: d && d[1],
                Code: r
            }),
            void (a.getCookie("webwx_data_ticket") || n.report(n.ReportType.cookieError, {
                text: "webwx_data_ticket 票据丢失",
                cookie: document.cookie
            })))
        });
        break;
    case 201:
        e.isScan = !0,
        n.report(n.ReportType.timing, {
            timing: {
                scan: Date.now()
            }
        }),
        t.checkLogin(e.uuid).then(o, function(t) {
            !t && window.checkLoginPromise && (e.isBrokenNetwork = !0)
        });
        break;
    case 408:
        t.checkLogin(e.uuid).then(o, function(t) {
            !t && window.checkLoginPromise && (e.isBrokenNetwork = !0)
        });
        break;
    case 400:
    case 500:
    case 0:
        var s = a.getCookie("refreshTimes") || 0;
        s < 5 ? (s++,
        a.setCookie("refreshTimes", s, .5),
        document.location.reload()) : e.isNeedRefresh = !0;
        break;
    case 202:
        e.isScan = !1,
        e.isAssociationLogin = !1,
        a.setCookie("login_frequency", 0, 2),
        window.checkLoginPromise && (window.checkLoginPromise.abort(),
        window.checkLoginPromise = null ),
        r()
    }
    e.code = c.code,
    e.userAvatar = c.userAvatar,
    a.log("get code", c.code)
}

总结

  • 轮询采用的是JSONP的形式,排除了跨域问题
  • 轮询采用的后台根据扫码情况阻塞前台请求,优化轮询及减少前端的无效轮询
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容