SpringBoot中獲取微信用戶信息從未如此簡單!

来源:https://www.cnblogs.com/jerry126/archive/2019/09/17/11531308.html
-Advertisement-
Play Games

前言 不知道你是否參加過拼多多上邀請微信好友砍價功能,這個功能實現首先需要考慮的就是獲取微信用戶的信息。獲取用戶信息就是獲取公眾號下微信用戶的信息,今天我就來講講如何從公眾號下獲取微信用戶信息。 需要聲明一點的是獲取微信公眾號下的用戶信息的許可權是服務號才有,個人訂閱號是沒有該許可權的。 獲取公眾號用戶 ...


前言

不知道你是否參加過拼多多上邀請微信好友砍價功能,這個功能實現首先需要考慮的就是獲取微信用戶的信息。獲取用戶信息就是獲取公眾號下微信用戶的信息,今天我就來講講如何從公眾號下獲取微信用戶信息。

需要聲明一點的是獲取微信公眾號下的用戶信息的許可權是服務號才有,個人訂閱號是沒有該許可權的。

獲取公眾號用戶信息實戰

第一步需要先申請介面測試號併進行網頁授權設置

訪問如下鏈接進行介面測試號申請。

https://developers.weixin.qq.com/doc/offiaccount/Basic_Information/Requesting_an_API_Test_Account.html

選擇介面測試號申請,如下圖所示:在這裡插入圖片描述

在這裡插入圖片描述
點擊登錄進行掃碼登錄,如下圖所示:
在這裡插入圖片描述
登錄後如下圖所示:
在這裡插入圖片描述
在下麵的的網頁賬號一欄添加網頁授權的IP或者功能變數名稱。
在這裡插入圖片描述
在這裡插入圖片描述

為了方便測試我這裡設置成了迴環地址,最好設置成具體的 IP 地址或者功能變數名稱信息。功能變數名稱和 IP 地址不要添加http或者https。這裡 IP 和功能變數名稱可以是內網地址。

到這裡網頁授權設置完畢!

第二步是下載微信web開發者工具,可以在PC 進行測試。
https://developers.weixin.qq.com/doc/offiaccount/OA_Web_Apps/Web_Developer_Tools.html
在這裡插入圖片描述
傻瓜式一步一步安裝即可。

第三步 看微信操作教程並完成代碼實現

接下來就是代碼部分編寫了,在開發前首先需要看一下獲取微信公眾號用戶信息的教程:
通過訪問:https://developers.weixin.qq.com/doc/offiaccount/OA_Web_Apps/Wechat_webpage_authorization.html 。獲取微信用戶信息具體操作有如下4步。
在這裡插入圖片描述
第一步:用戶同意授權,獲取code

代碼的操作就是拼接引導用戶進行微信授權地址,然後重定向到微信服務,微信服務在根據重定向的 URL 地址並攜帶 code 重定向到我們的伺服器。這一步需要配置有公眾號 appid 和 redirect_uri 。

需要註意的是重定向的地址需要 encode 以下,具體操作如下麵代碼所示:

String url = URLEncoder.encode(request.getRequestURL().toString());

具體地址如下圖所示:紅色框位置需要改為我們公眾號 appid 和 redirect_uri 信息,其他的內容不用改動。
在這裡插入圖片描述
程式重定向該地址後會讓用戶進行授權,如下圖所示:
在這裡插入圖片描述
用戶點擊同意後,微信服務會根據重定向地址重定向回我們的服務中並攜帶code。

第二步就是根據 code 獲取網頁授權 access_token 和 openid。

調用如下麵所示微信 API ,紅色框 code 替換成獲取的 code,其他內容不用做任何改動。
在這裡插入圖片描述
第三步:刷新access_token(如果需要)

access_token有效期是7200s,當access_token超時後,可以使用refresh_token進行刷新,refresh_token有效期為30天,當refresh_token失效之後,需要用戶重新授權。這步可做可不做,我們這裡就繞開這步。

第四步:拉取用戶信息(需scope為 snsapi_userinfo)

關於網頁授權的兩種scope的區別說明

1、以snsapi_base為scope發起的網頁授權,是用來獲取進入頁面的用戶的openid的,並且是靜默授權並自動跳轉到回調頁的。用戶感知的就是直接進入了回調頁(往往是業務頁面)

2、以snsapi_userinfo為scope發起的網頁授權,是用來獲取用戶的基本信息的。但這種授權需要用戶手動同意,並且由於用戶同意過,所以無須關註,就可在授權後獲取該用戶的基本信息。

根據第二步獲取的 access_token 和 opendId 獲取微信用戶的信息。調用如下圖所示微信 API,將獲取的 access_token 和 opendId 替換如下圖中紅色框位置內容,其他不用做任何改動。返回的 json 信息就是該公眾好號的用戶信息。
在這裡插入圖片描述

講完操作流程,接下來就是代碼實現。具體獲取微信用戶信息 Controller 內容如下:

@RestController
@RequestMapping("/weixin")
public class WeiXinDemoController {
    @Autowired
    private WeiXinService weiXinService;

    @RequestMapping("/getWeiXinUserInfo")
    public String getWeiXinUserInfo(String code,HttpServletRequest request,HttpServletResponse response,HttpSession session) throws IOException{
        //第一步:用戶同意授權,獲取code
        if (code == null) {
            String url = URLEncoder.encode(request.getRequestURL().toString());
            String authorizeUrl = weiXinService.buildAuthorizeURL(url);
            response.sendRedirect(authorizeUrl);
            return null;
        }
        //第二步:通過code換取網頁授權access_token和openid
        String htmlInfo = "";
        Map<String, Object> openIdInfo = weiXinService.getOpenIdInfo(code);
        String errcode = (String)openIdInfo.get("errcode");
        if(StringUtils.isEmpty(errcode)){
            //第四步:拉取用戶信息(需scope為 snsapi_userinfo)根據access_token和OpenId
            Map<String, Object> weiXinUserInfo = weiXinService.getWeiXinUserInfo(openIdInfo);
            String userInfohtml = createUserInfoHtml(weiXinUserInfo);
            return userInfohtml;
        }
        return htmlInfo;
    }
@Component
@ConfigurationProperties(prefix="wx")
public class WeiXinConfig {
    
    private String appID;
    private String mchID;
    private String appsecret;
    private String key;
    
    //省略getter and setter
}

application.properties 配置內容如下:
在這裡插入圖片描述
微信核心處理都在 WeiXinService中,微信介面調用時通過 RestTemplate來實現的。

拼接引導用戶進行微信授權地址代碼如下:

    /**
     * 拼接用戶授權重定向的URL
     * @param url
     * @return
     */
    public String buildAuthorizeURL(String url){
        
        return concatAuthorizeURL(url);
    }
        private String concatAuthorizeURL(String url) {
        StringBuilder authorizeUrl = new StringBuilder(AUTHORIZEURL);
        authorizeUrl.append("?appid=").append(weiXinConfig.getAppID());
        authorizeUrl.append("&redirect_uri=").append(url);
        authorizeUrl.append("&response_type=code");
         //snsapi_base (不彈出授權頁面,直接跳轉,只能獲取用戶openid),
         //snsapi_userinfo (彈出授權頁面,可通過openid拿到昵稱、性別、所在地。並且, 即使在未關註的情況下,只要用戶授權,也能獲取其信息 )
        authorizeUrl.append("&scope=snsapi_userinfo");
        authorizeUrl.append("&state=").append("STATE");
        authorizeUrl.append("#wechat_redirect");
        return authorizeUrl.toString();
    }

根據 code 獲取網頁授權 access_token 和 openid 代碼如下:

    /**
     * 獲取 access_token 和 openid
     * @param code
     * @return
     */
    public Map<String,Object> getOpenIdInfo(String code){
        
        String getAccessTokenUrl = concatGetOpenIdInfoURL(code);
        String json = postRequestForWechat(getAccessTokenUrl);
        Map<String,Object> map = jsonToMap(json);
        
        return map;
    }
    private String concatGetOpenIdInfoURL(String code) {
        StringBuilder getAccessTokenUrl = new StringBuilder(GE_TACCESSTOKEN_URL);
        getAccessTokenUrl.append("?appid=").append(weiXinConfig.getAppID());
        getAccessTokenUrl.append("&secret=").append(weiXinConfig.getAppsecret());
        getAccessTokenUrl.append("&code=").append(code);
        getAccessTokenUrl.append("&grant_type=authorization_code");
        return getAccessTokenUrl.toString();
    }
    private String postRequestForWechat(String getAccessTokenUrl) {
        ResponseEntity<String> postForEntity = restTemplate.postForEntity(getAccessTokenUrl, null, String.class);
        String json = postForEntity.getBody();
        return json;
    }
        private Map jsonToMap(String json) {
        Gson gons = new Gson();
        Map map = gons.fromJson(json, new TypeToken<Map>(){}.getType());
        return map;
    }

通過 access_token 和 openid 獲取微信用戶信息 代碼如下:

    /**
     * 獲取微信用戶信息通過  access_token 和 openid
     * @param map
     * @return
     */
    public Map getWeiXinUserInfo(Map<String, Object> map) {
        
        String getUserInfoUrl = concatGetWeiXinUserInfoURL(map);
        String json = getRequestForWechat(getUserInfoUrl);
        Map userInfoMap = jsonToMap(json);
        
        return userInfoMap;
    }
        private String concatGetWeiXinUserInfoURL(Map<String, Object> map) {
        String openId = (String) map.get("openid");
        String access_token = (String) map.get("access_token");
        // 繞過檢驗授權憑證(access_token)是否有效
        StringBuilder getUserInfoUrl = new StringBuilder(GE_USERINFO_URL);
        getUserInfoUrl.append("?access_token=").append(access_token);
        getUserInfoUrl.append("&openId=").append(openId);
        getUserInfoUrl.append("&lang=zh_CN");
        
        return getUserInfoUrl.toString();
    }
    private String getRequestForWechat(String getUserInfoUrl) {
        ResponseEntity<String> postForEntity = restTemplate.getForEntity(getUserInfoUrl.toString(), String.class);
        String json = postForEntity.getBody();
        return json;
    }

測試

下載微信web開發者工具完成後,根據傻瓜式安裝即可。安裝完成後打開web開發者工具,如下圖所示選擇公眾號網頁。
在這裡插入圖片描述
輸入 http:127.0.1:8090/sbe2/weixin/getWeiXinUserInfo,就會看到該測試公眾號的用戶信息。
在這裡插入圖片描述
一般情況下我們通過介面測試號配置成測試環境功能變數名稱或 IP,測試通過後就可以在服務號上配置網頁授權功能變數名稱,順便在把我們項目配置的 appid 換成服務號的 appid 即可。具體操作如下:
在這裡插入圖片描述
在這裡插入圖片描述

小結

獲取微信公眾號用戶信息步驟就是:第一步拼接引導用戶授權的地址然後根據該地址重定向到微信服務,第二步獲取授權code,根據 code 獲取 access_token 和 OpenId,第三步根據 access_token 和 OpenId 獲取微信用戶信息。

我這裡介紹最簡三步,正常來講還需要驗證access_token 是否有效,這一步也可以通過緩存access_token到Reid中並設置過期時間,當其失效後刷新access_token。

在這裡再三強調一下,微信提供操作文檔一定要多看幾篇,因為很多細節都在文檔中進行了說明。當你把微信提供操作文檔看透,你就會覺得其實就是個API 調用而已。

代碼示例

具體代碼示例請查看我的GitHub 倉庫 springbootexamples 中的 spring-boot-2.x-weixin 查看。

GitHub:https://github.com/zhuoqianmingyue/springbootexamples


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 場景 Docker-Compose簡介與Ubuntu Server 上安裝Compose: https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/100902301 在上面實現Compose成功安裝的基礎上,使用Compose啟動項目。 ...
  • Compose簡介 Compose是Docker官方的開源項目,負責對Docker容器集群的快速編排。 Compose是定義和運行多個Docker容器的應用。 舉例來說: 一個項目除了Tomcat容器外,還需要mysql服務容器,Compose允許用戶通過一個單獨的 docker-compose.y ...
  • 什麼是常量 用final修飾的成員變數表示常量,值一旦給定就無法改變! final修飾的變數有三種:靜態變數、實例變數和局部變數,分別表示三種類型的常量。 Class文件中的常量池 在Class文件結構中,最頭的4個位元組用於存儲魔數Magic Number,用於確定一個文件是否能被JVM接受,再接著 ...
  • 引言: 原型模式是什麼?它是在什麼場景下被提出的呢?本章節,我們將詳細瞭解下原型模式。 在軟體系統中,當創建一個類的實例過程過於昂貴或複雜,並且我們需要創建多個這樣類的實例時,如果我們通過new來創建類實例,這就會增加創建類的複雜度和創建過程與客戶代碼複雜的耦合度。如果採用工廠模式來創建這樣的實例對 ...
  • 概述 在開發RESTFull API 和普通的表單提交都需要對用戶提交的數據進行校驗,例如:用戶姓名不能為空,年齡必須大於0 等等。這裡我們主要說的是後臺的校驗,在 SpringBoot 中我們可以通過使用 Hibernate Validator 來進行後臺的數據校驗的。 閑話少說!接下來就開始介紹 ...
  • Devtools 介紹 SpringBoot 提供了熱部署的功能,那啥是熱部署累?SpringBoot官方是這樣說的:只要類路徑上的文件發生更改,就會自動重新啟動應用程式。在IDE中工作時,這可能是一個有用的功能,因為它為代碼更改提供了非常快速的反饋迴圈。預設情況下,將監視類路徑上指向文件夾的任何條 ...
  • 前言 Thymeleaf是一個適用於Web和獨立環境的現代伺服器端Java模板引擎。 Thymeleaf的主要目標是為您的開發工作流程帶來優雅的自然模板 可以在瀏覽器中正確顯示的HTML,也可以用作靜態原型,從而在開發團隊中實現更強大的協作。 通過Spring Framework模塊,與您喜歡的工具 ...
  • 前言 操作前建議先參考我的另一篇博客: "玩轉 SpringBoot 2 快速整合 | Thymeleaf 篇" 查看如何在SpringBoot 中使用 Thymeleaf。還有一點需要註意的是:模版頁面中的 html 上需要聲明 Thymeleaf 的命名空間,具體代碼如下: 接下來就可以開始 T ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...