欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

微信小程序常見的兩種登錄方式詳解

 更新時(shí)間:2024年05月04日 10:20:33   作者:零一行者  
這篇文章主要介紹了微信小程序常見的兩種登錄方式,一種基于手機(jī)號(hào)碼進(jìn)行登錄,另一種是使用用戶在公眾號(hào)下的唯一標(biāo)識(shí)進(jìn)行登錄,需要的朋友可以參考下

小程序登錄

小程序有兩種登錄方式,一種基于手機(jī)號(hào)碼進(jìn)行登錄,另一種是使用用戶在公眾號(hào)下的唯一標(biāo)識(shí)(openid)進(jìn)行登錄(小程序是公眾號(hào)的一種).

接下來先講解下,基于 openid 登錄。

基于openid登錄

先看下圖,描述通過微信小程序提供的 code 換取當(dāng)前用戶在小程序中的唯一標(biāo)識(shí),詳細(xì)流程可以參數(shù)下圖:

接下來通過代碼實(shí)現(xiàn)下大概流程:

獲取 code

uni.login({
  success: async (res) => {
    if (res.errMsg === 'login:ok') {
      const { data } = await login({
        code: res.code,
      });
      // 保存用戶信息
    }
  },
  fail(e) {
    uni.showToast({
      title: e.message,
    });
  },
});

服務(wù)端接收 code 去微信后臺(tái)換取對(duì)應(yīng) openid

// nodejs 部分代碼
const { appid, secret, grant_type } = require('../config/wx');
router.post('/login', (req, res) => {
  const { code } = req.query;
  const { appid, secret, grant_type } = require('../config/wx');
  const { openid } = await request.get('/sns/jscode2session', {
    appid,
    secret,
    js_code: code,
    grant_type,
  });
});

在數(shù)據(jù)庫中查找對(duì)應(yīng) openid 是否存在

const { appid, secret, grant_type } = require('../config/wx');
router.post('/login', (req, res) => {
  // 1. 獲取 code
  const { code } = req.query;
  // 2. 通過 code 獲取 openid 和 session_key
  const { openid } = await request.get('/sns/jscode2session', {
    appid,
    secret,
    js_code: code,
    grant_type,
  });
  // 3. 查找用戶是否已經(jīng)注冊(cè)
  models.user
    .findOne({
      where: {
        openid,
      },
    })
    .then((user) => {
      if (user) {
        // 3.2 如果用戶已經(jīng)注冊(cè),返回用戶信息
        res.json(
          new Result({
            data: user,
            msg: '登錄成功',
          })
        );
      } else {
        // 3.3 如果用戶沒有注冊(cè),創(chuàng)建用戶并返回用戶信息
        const username = randomUserName();
        models.user
          .create({
            nickname: username,
            openid,
            avatar: '/uploads/default-avatar.png',
          })
          .then((user) => {
            res.json(
              new Result({
                data: user,
                msg: '登錄成功',
              })
            );
          });
      }
    });
});

上面就是一個(gè)基于 code 獲取 openid,并通過 openid 創(chuàng)建新的用戶,并將創(chuàng)建好的用戶返回。

為了方便理解,這里簡化描述了登錄邏輯。在實(shí)際業(yè)務(wù)代碼中,通常會(huì)使用 openidsession key 和用戶信息來創(chuàng)建自定義登錄憑證(token),并在登錄時(shí)將用戶信息和 token 一起返回給前端。前端會(huì)將 token 存儲(chǔ)在本地,并在下一次需要登錄的業(yè)務(wù)請(qǐng)求中攜帶 token,從而實(shí)現(xiàn)業(yè)務(wù)鑒權(quán)的功能。這種方式通常使用 JWT(JSON Web Token)等工具來實(shí)現(xiàn)。在后續(xù)的講解中,我們將詳細(xì)介紹這些概念和技術(shù)細(xì)節(jié)。

手機(jī)號(hào)碼快捷登錄

獲取手機(jī)號(hào)碼的前提:

  • 非個(gè)人小程序
  • 認(rèn)證的小程序
  • 非海外的企業(yè)認(rèn)證

下面是大概業(yè)務(wù)流程圖:

獲取對(duì)應(yīng)code

<template>
  <button
    class="login-btn"
    open-type="getPhoneNumber"
    @getphonenumber="getPhoneNumber"
  >
    手機(jī)號(hào)碼登錄
  </button>
</template>
<script>
export default {
  setup() {
    // 目前該接口針對(duì)非個(gè)人開發(fā)者,且完成了認(rèn)證的小程序開放(不包含海外主體)
    const getPhoneNumber = (e) => {
      const { code, errMsg } = e.detail;
      if (errMsg === 'getPhoneNumber:ok') {
        const { data } = await loginByPhone({
          code,
        });
      } else {
        uni.showToast({
          title: errMsg,
        });
      }
    };
    return {
      getPhoneNumber,
    };
  },
};
</script>

后端處理邏輯

// 基于手機(jī)號(hào)登錄
router.post('/loginByPhone', async function (req, res) {
  try {
    // 1. 獲取 code 和 loginCode
    const { code } = req.body;
    // 2. 獲取接口調(diào)用憑據(jù),理論上這里需要緩存 access_token,避免頻繁調(diào)用接口
    const { access_token } = await request.get('/cgi-bin/token', {
      grant_type: 'client_credential',
      appid,
      secret,
    });
    // 3. 獲取手機(jī)號(hào)
    const { phone_info } = await request.post(
      `/wxa/business/getuserphonenumber?access_token=${access_token}`,
      {
        code,
      }
    );
    // 4. 查找用戶是否已經(jīng)注冊(cè)
    // 4.1 根據(jù) phone 查找用戶
    const { purePhoneNumber } = phone_info;
    models.user
      .findOne({
        where: {
          purePhoneNumber,
        },
      })
      .then((user) => {
        if (user) {
          // 4.2 如果用戶已經(jīng)注冊(cè),返回用戶信息
          res.json(
            new Result({
              data: user,
              msg: '登錄成功',
            })
          );
        } else {
          // 4.3 如果用戶沒有注冊(cè),創(chuàng)建用戶并返回用戶信息
          const username = randomUserName();
          models.user
            .create({
              nickname: username,
              avatar: '/uploads/default-avatar.png',
              phone: phone_info.purePhoneNumber,
            })
            .then((user) => {
              res.json(
                new Result({
                  data: user,
                  msg: '登錄成功',
                })
              );
            });
        }
      });
  } catch (error) {
    res.json(
      new Result({
        code: 'BIZ_ERROR',
        msg: error.errmsg || error.message,
      })
    );
  }
});

上面代碼,實(shí)現(xiàn)獲取手機(jī)號(hào)碼并使用手機(jī)號(hào)碼作為唯一標(biāo)識(shí),進(jìn)行用戶創(chuàng)建和查找的操作。

從登錄的角度來看,使用手機(jī)號(hào)碼作為唯一標(biāo)識(shí)符是沒有問題的。然而,如果用戶嘗試使用非手機(jī)號(hào)碼(例如 OpenID)進(jìn)行登錄,并在數(shù)據(jù)庫中找不到匹配的記錄時(shí),系統(tǒng)會(huì)創(chuàng)建一個(gè)新的賬號(hào)。這可能導(dǎo)致同一個(gè)用戶在系統(tǒng)中存在多個(gè)賬號(hào)的情況。

為了優(yōu)化這種情況,可以考慮以下幾種方法:

  • 當(dāng)用戶使用 openid 登錄后,檢測未綁定手機(jī)號(hào)碼時(shí),進(jìn)行號(hào)碼綁定
  • 當(dāng)用戶使用手機(jī)號(hào)碼登錄時(shí),提前調(diào)用 wx.login 獲取對(duì)應(yīng) code,換取 openid 把他與手機(jī)號(hào)碼進(jìn)行關(guān)聯(lián)

現(xiàn)在基于上面的代碼,采用第二種方案,只需要微調(diào)下代碼就能解決這個(gè)問題。

  • 登錄時(shí)把 wx.login 獲取 code 傳遞給后端
<template>
  <button
    class="login-btn"
    open-type="getPhoneNumber"
    @getphonenumber="getPhoneNumber"
  >
    手機(jī)號(hào)碼登錄
  </button>
</template>
<script>
export default {
  setup() {
    // 目前該接口針對(duì)非個(gè)人開發(fā)者,且完成了認(rèn)證的小程序開放(不包含海外主體)
    const getPhoneNumber = (e) => {
      const { code, errMsg } = e.detail;
      if (errMsg === 'getPhoneNumber:ok') {
        uni.login({
          success: async (res) => {
            if (res.errMsg === 'login:ok') {
              const { data } = await loginByPhone({
                code,
                loginCode: res.code,
              });
              userStore.setUserInfo(data);
              uni.navigateBack();
            }
          },
          fail(e) {
            uni.showToast({
              title: e.message,
            });
          },
        });
      } else {
        uni.showToast({
          title: errMsg,
        });
      }
    };
    return {
      getPhoneNumber,
    };
  },
};
</script>

服務(wù)端基于 loginCode 換取 openid

// 基于手機(jī)號(hào)登錄
router.post('/loginByPhone', async function (req, res) {
  try {
    // 1. 獲取 code 和 loginCode
    const { code, loginCode } = req.body;
    // 2. 獲取接口調(diào)用憑據(jù),理論上這里需要緩存 access_token,避免頻繁調(diào)用接口
    const { access_token } = await request.get('/cgi-bin/token', {
      grant_type: 'client_credential',
      appid,
      secret,
    });
    // 3. 獲取 openid
    const { openid } = await request.get('/sns/jscode2session', {
      appid,
      secret,
      js_code: loginCode,
      grant_type,
    });
    // 4. 獲取手機(jī)號(hào)
    const { phone_info } = await request.post(
      `/wxa/business/getuserphonenumber?access_token=${access_token}`,
      {
        code,
        openid,
      }
    );
    // 5. 查找用戶是否已經(jīng)注冊(cè)
    // 5.1 根據(jù) openid 查找用戶
    models.user
      .findOne({
        where: {
          openid,
        },
      })
      .then((user) => {
        if (user) {
          // 5.2 如果用戶已經(jīng)注冊(cè),返回用戶信息
          res.json(
            new Result({
              data: user,
              msg: '登錄成功',
            })
          );
        } else {
          // 5.3 如果用戶沒有注冊(cè),創(chuàng)建用戶并返回用戶信息
          const username = randomUserName();
          models.user
            .create({
              nickname: username,
              openid,
              avatar: '/uploads/default-avatar.png',
              phone: phone_info.purePhoneNumber,
            })
            .then((user) => {
              res.json(
                new Result({
                  data: user,
                  msg: '登錄成功',
                })
              );
            });
        }
      });
  } catch (error) {
    res.json(
      new Result({
        code: 'BIZ_ERROR',
        msg: error.errmsg || error.message,
      })
    );
  }
});

這種方案被視為最佳的解決方案,能夠有效解決多賬號(hào)和綁定手機(jī)號(hào)碼等問題。 實(shí)際上,采用哪種方式取決于具體的業(yè)務(wù)場景,因?yàn)樵谀承┣闆r下,用戶可能會(huì)擔(dān)心手機(jī)號(hào)碼泄露而不愿采用這種方式。

注意

  • 獲取手機(jī)號(hào)碼是需要收費(fèi),每次調(diào)用需要 0.03 元。
  • wx.logingetPhoneNumber 中獲取的 code 不是同一個(gè)

總結(jié)

  • 基于 openid 或 手機(jī)號(hào)碼快捷登錄
  • 獲取手機(jī)號(hào)碼前置條件
  • 如何解決多賬號(hào)的問題
  • 講解前端、后端、微信登錄過程中完整交互流程,方便更好去理解小程序登錄

以上就是微信小程序常見的兩種登錄方式詳解的詳細(xì)內(nèi)容,更多關(guān)于小程序登錄方式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • uniapp微信小程序授權(quán)登錄并獲取手機(jī)號(hào)的方法

    uniapp微信小程序授權(quán)登錄并獲取手機(jī)號(hào)的方法

    這篇文章主要給大家介紹了關(guān)于uniapp微信小程序授權(quán)登錄并獲取手機(jī)號(hào)的相關(guān)資料,我們?cè)趗niapp開發(fā)微信小程序的過程中,經(jīng)常需要在微信端登錄,需要的朋友可以參考下
    2023-06-06
  • 詳解微信小程序文件下載--視頻和圖片

    詳解微信小程序文件下載--視頻和圖片

    這篇文章主要介紹了微信小程序文件下載視頻和圖片,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • JavaScript 高效運(yùn)行代碼分析

    JavaScript 高效運(yùn)行代碼分析

    傳統(tǒng)上,網(wǎng)頁中不會(huì)有大量的腳本,至少腳本很少會(huì)影響網(wǎng)頁的性能。
    2010-03-03
  • javascript實(shí)現(xiàn)列表滾動(dòng)的方法

    javascript實(shí)現(xiàn)列表滾動(dòng)的方法

    這篇文章主要介紹了javascript實(shí)現(xiàn)列表滾動(dòng)的方法,較為詳細(xì)的分析了javascript實(shí)現(xiàn)列表滾動(dòng)的頁面布局及javascript滾動(dòng)效果的實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-07-07
  • 常用JS代碼實(shí)例小結(jié)

    常用JS代碼實(shí)例小結(jié)

    跟網(wǎng)上的一些常用的不太一樣,個(gè)人都有個(gè)人常用的代碼,大家看到好東西,不一定什么都會(huì),起碼要知道有這么個(gè)東西。方便以后用也方便找。
    2009-04-04
  • js replace() 文本替換你所不知的

    js replace() 文本替換你所不知的

    今天看了一個(gè)函數(shù),功能是把形如word-word的字符串轉(zhuǎn)化為wordWord
    2010-03-03
  • Ant Design Pro 下實(shí)現(xiàn)文件下載的實(shí)現(xiàn)代碼

    Ant Design Pro 下實(shí)現(xiàn)文件下載的實(shí)現(xiàn)代碼

    這篇文章主要介紹了Ant Design Pro 下實(shí)現(xiàn)文件下載的實(shí)現(xiàn)代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-12-12
  • js截取小數(shù)點(diǎn)后幾位的寫法

    js截取小數(shù)點(diǎn)后幾位的寫法

    截取小數(shù)點(diǎn)后幾位的方法有很多,下面為大家介紹下使用js是如何實(shí)現(xiàn)的
    2013-11-11
  • GoJs中導(dǎo)出圖片或者SVG實(shí)現(xiàn)示例詳解

    GoJs中導(dǎo)出圖片或者SVG實(shí)現(xiàn)示例詳解

    這篇文章主要為大家介紹了GoJs中導(dǎo)出圖片或者SVG實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05
  • 定時(shí)器(setTimeout/setInterval)調(diào)用帶參函數(shù)失效解決方法

    定時(shí)器(setTimeout/setInterval)調(diào)用帶參函數(shù)失效解決方法

    setInterval()方法可按照指定的周期(以毫秒計(jì))來調(diào)用函數(shù)或計(jì)算表達(dá)式,setTimeout()方法用于在指定的毫秒數(shù)后調(diào)用函數(shù)或計(jì)算表達(dá)式,詳細(xì)使用方法可以參考下本文
    2013-03-03

最新評(píng)論