android實(shí)現(xiàn)微信聯(lián)合登錄開(kāi)發(fā)

隨著現(xiàn)在社交app的越來(lái)越普及,微信,QQ,微博幾乎成了我們每個(gè)人的手機(jī)必備,而且第三方應(yīng)用也越來(lái)越多,不是每個(gè)用戶都想為你的應(yīng)用注冊(cè)一個(gè)獨(dú)立帳號(hào)的,所以聯(lián)合登錄也就顯得尤為重要,成了我們不得不掌握的開(kāi)發(fā)要點(diǎn),但你要用微信等帳號(hào)實(shí)現(xiàn)聯(lián)合登錄,人家也有要求,就是你必須遵守OAuth2.0協(xié)議標(biāo)準(zhǔn)(https://oauth.net/2/)就行。

接下來(lái)我們就以android studio為平臺(tái),介紹下接入微信的整個(gè)流程:

這里有一個(gè)官方參考資料:https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&verify=1&id=open1419317851&token=6bfe3acd2969037c7217667f24f8eeaf714e5113&lang

那么在接入之前,我們還有一些準(zhǔn)備工作要做:

1):登錄微信開(kāi)放平臺(tái),完成賬號(hào)申請(qǐng):https://open.weixin.qq.com

2):提交app完成審核,并開(kāi)通相應(yīng)權(quán)限:

image.png

在提交app完成審核的過(guò)程中,我們需要提供app的包名與簽名:

包名就是我們創(chuàng)建app時(shí)候的名稱(chēng),可以通過(guò)manifest.xml和build.gradle中查看,如在manifest.xml中:

<?xml version="1.0" encoding="utf-8"?>
<manifest 
xmlns:android="http://schemas.android.com/apk/res/android"
package="com.yeejay.reader"
>

...

在build.gradle中:

defaultConfig {
    applicationId "com.yeejay.reader"
    ...
}

如果manifest.xml與build.gradle中不一致,將以build.gradle中的聲明為準(zhǔn):

應(yīng)用簽名的獲取會(huì)有一些麻煩,因?yàn)槲覀冃枰ㄟ^(guò)簽名工具對(duì)我們的應(yīng)用進(jìn)行簽名,如果不進(jìn)行簽名,每次安裝apk時(shí)android平臺(tái)會(huì)為我們生成默認(rèn)簽名,由于平臺(tái)的不一致性,導(dǎo)致我們的apk裝到不同的手機(jī)上簽名不一致,最后導(dǎo)致無(wú)法正常使用微信登錄與分享功能,所以我們要提前做好統(tǒng)一簽名,整個(gè)簽名流程如下:

(1)點(diǎn)擊 build ---> Generate Signed Apk... ---> 選擇app module ---> 填寫(xiě)生成keystore.jks的目錄,密鑰,有效期等信息;

(2)將生成的keystore.jks文件拷貝到project目錄下的app目錄下;

(3)在build.gradle文件中進(jìn)行配置,代碼如下:

signingConfigs {

    release {
        storeFile file('keystore.jks')
        storePassword 'xxx'
        keyAlias 'xxx'
        keyPassword 'xxx'
    }
}

buildTypes {
    release {
        minifyEnabled false
        proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        signingConfig signingConfigs.release
    }

    debug {
        signingConfig signingConfigs.release
    }
}

將配置中的xxx改為生成keystore.jks時(shí)的密碼信息;

(4)Sync同步下項(xiàng)目代碼,完成后在右邊的Gradle projects/app/build目錄下雙擊assembleDebug或assembleRelease即可生成對(duì)應(yīng)版本帶有簽名的apk。

(5)下載apk簽名檢測(cè)工具:
https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&verify=1&id=open1419319167&token=&lang=zh_CN
在最下面有一個(gè)簽名生成工具,單擊下載進(jìn)行下載;

(6)將下載的apk簽名工具安裝到手機(jī)上,將之前生成的帶有簽名認(rèn)證的apk也裝到手機(jī)上,打開(kāi)簽名工具apk,如下圖:

image.png

輸入包名,點(diǎn)擊按鈕生成對(duì)應(yīng)的數(shù)字簽名;

(7) 將包名與數(shù)字簽名注冊(cè)到我們的app審核過(guò)程中;

(8)漫長(zhǎng)的等待...,也許是兩三天,也許一兩個(gè)禮拜。當(dāng)我們的應(yīng)用申請(qǐng)通過(guò)時(shí),微信開(kāi)放平臺(tái)會(huì)返回給我們APP_ID與APP_SECRET信息。

當(dāng)應(yīng)用審核通過(guò)后,就到我們的代碼實(shí)施階段了,不過(guò)在寫(xiě)代碼之前,我們先來(lái)了解一下OAuth2.0它的一個(gè)認(rèn)證流程:

image.png

整個(gè)過(guò)程的交互一共七個(gè)流程,當(dāng)用戶使用第三方應(yīng)用請(qǐng)求微信登錄時(shí),此時(shí)會(huì)調(diào)起微信登錄確認(rèn)頁(yè)面,并請(qǐng)求微信OAuth2.0進(jìn)行授權(quán),當(dāng)用戶確認(rèn)登錄時(shí),這時(shí)微信平臺(tái)會(huì)拉起我們的第三方應(yīng)用,并將帶有臨時(shí)票據(jù)的信息返回給我們的第三方應(yīng)用,當(dāng)我們的應(yīng)用拿到微信的臨時(shí)票據(jù)時(shí),我們就可以根據(jù)它的票據(jù)的code信息,再加上我們應(yīng)用的appId與appSecret信息換取access_token。當(dāng)然,為了用戶帳號(hào)信息的保存,這個(gè)過(guò)程一般都是由我們的第三方應(yīng)用的服務(wù)器完成的,我們只需要將我們的code傳給我們的服務(wù)器,由服務(wù)器最終完成與微信平臺(tái)的token換取。

接下來(lái)就是開(kāi)始寫(xiě)代碼了,首先我們需要下載微信的sdk包,在build.gradle中,添加下面一行代碼:

dependencies {

...

compile 'com.tencent.mm.opensdk:wechat-sdk-android-with-mta:+'

...

}

sync,將會(huì)下載我們最新的sdk版本。如果下載失敗我們也可以使用離線包,添加到app/lib目錄中,并添加引用即可(由于前段時(shí)間jCenter下架了微信的sdk,導(dǎo)致開(kāi)發(fā)的過(guò)程中發(fā)現(xiàn)不能直接使用,這時(shí)候可以考慮在lib中添加即可);

根據(jù)微信開(kāi)發(fā)平臺(tái)的官方要求,調(diào)起微信登錄的處理邏輯必須放在我們應(yīng)用的包名.wxapi目錄下,否則會(huì)報(bào)錯(cuò)。這點(diǎn)可以參考微信提供的官方sdkDemo,https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&verify=1&id=open1419319167&token=&lang=zh_CN 在范例代碼進(jìn)行下載。不過(guò)這里下載的代碼可能會(huì)因?yàn)榫幋a的原因,導(dǎo)致漢語(yǔ)注釋變成了混亂字符,后來(lái)將UTF-8編碼改為GBK就OK了。

我們看一下代碼的實(shí)現(xiàn)邏輯:

public class WXEntryActivity extends Activity implements IWXAPIEventHandler {

public static final String APP_ID = "xxxxxxxxxx";
public static final String APP_SECRET = "xxxxxxxxxx";
private static final String WEIXIN_SCOPE = "snsapi_userinfo";
private static final String WEIXIN_STATE = "login_wx";

private IWXAPI api;
private SendAuth.Req req;

private TextView loginWx;
private User user;
private String wxCode;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.login_wx);
    loginWx = (TextView) findViewById(R.id.login_wx);
    api = WXAPIFactory.createWXAPI(this, APP_ID, false);
    api.registerApp(APP_ID);

    try {
        api.handleIntent(getIntent(), this);
    } catch (NullPointerException e) {
        e.printStackTrace();
    }

    initListener();
}

private void initListener() {
    loginWx.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            sendAuth();
        }
    });
}

private void sendAuth() {
    req = new SendAuth.Req();
    req.scope = WEIXIN_SCOPE;
    req.state = WEIXIN_STATE;
    api.sendReq(req);
}

@Override
public void onReq(BaseReq baseReq) {
}

@Override
public void onResp(BaseResp baseResp) {
    int type = baseResp.getType();
    if (type == 1) {
        // 登錄
    SendAuth.Resp sendAuthResp = (SendAuth.Resp) baseResp;
    wxCode = sendAuthResp.code;
    String state = ((SendAuth.Resp) baseResp).state;
    int errCode = baseResp.errCode;
    Intent intent = new Intent(WXEntryActivity.this, HomeActivity.class);
    intent.putExtra("login", "login");
    intent.putExtra("wxCode", wxCode);
    startActivity(intent);
    YeeUtils.animOpenActivity(this);
    finish();
    } 
}
    
@Override
public void finish() {
    super.finish();
    YeeUtils.animCloseActivity(this);
}

@Override
protected void onDestroy() {
    super.onDestroy();
    api.unregisterApp();
}
}

要實(shí)現(xiàn)微信登錄的功能,我們的代碼就必須實(shí)現(xiàn)IWXAPIEventHandler接口,實(shí)現(xiàn)onReq()與onResp()接口,首先我們要在activity創(chuàng)建的時(shí)候需要進(jìn)行一些初始化的操作,這個(gè)照寫(xiě)就是:

api = WXAPIFactory.createWXAPI(this, APP_ID, false);
api.registerApp(APP_ID);

try {
    api.handleIntent(getIntent(), this);
} catch (NullPointerException e) {
    e.printStackTrace();
}

當(dāng)我們進(jìn)行微信登錄時(shí),整個(gè)認(rèn)證流程也就從sendAuth()開(kāi)始了,當(dāng)調(diào)起微信登錄并確認(rèn)后,這個(gè)時(shí)候微信sdk會(huì)調(diào)起我們的當(dāng)前activity并觸發(fā)onResp()回調(diào),并傳回code信息,這時(shí)候我們就可以將接下來(lái)的處理交給我們的服務(wù)器,由它完成對(duì)微信token的換取。

當(dāng)然如果你想自己實(shí)現(xiàn)對(duì)微信token的獲取以及接下來(lái)的操作,也是可以的,不過(guò)不建議這樣使用,下面就是獲取token的代碼:

private void getResult(String code) {
    String url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid="
            + APP_ID
            + "&secret="
            + APP_SECRET
            + "&code="
            + code
            + "&grant_type=authorization_code";
    JsonObjectRequest request = new JsonObjectRequest(Request.Method.POST, url, null, new SuccessResponse(), new ErrorListener());
    StudyApplication.getInstance().newRequestQueue(this).add(request);
}

class SuccessResponse implements Response.Listener<JSONObject> {

    @Override
    public void onResponse(JSONObject response) {
 
        if (null != response) {
            String openid = null;
            try {
                openid = response.getString("openid")
                        .toString().trim();
            String access_token = response
                    .getString("access_token").toString().trim();          
               SharePreferenceHelper.putValue(getApplicationContext(), "access_token", access_token);
                getUID(openid, access_token);
            }
            catch (JSONException e) {
                e.printStackTrace();
            }
        }
    }
}

通過(guò)這一步那么我們就拿到了用戶的token信息,使用token我們就可以獲取到用戶信息,包括昵稱(chēng),性別,頭像,地址等信息,我們接下來(lái)看getUid()中的實(shí)現(xiàn):

private void getUID(final String openId, final String accessToken) {
    String url = "https://api.weixin.qq.com/sns/userinfo?access_token=" + accessToken + "&openid=" + openId;
    JsonObjectRequest request = new JsonObjectRequest(Request.Method.POST, url, null, new UIDSuccessResponse(), new ErrorListener());
    StudyApplication.getInstance().newRequestQueue(this).add(request);
    }

class UIDSuccessResponse implements Response.Listener<JSONObject> {

        @Override
        public void onResponse(JSONObject response) {
            if (response == null) {
                return;
            }
            String nickname = null;
            nickname = response.optString("nickname");
            String unionid = response.optString("unionid");
            String headUrl = response.optString("headimgurl");
            SharePreferenceHelper.putValue(getApplicationContext(), "nickname", nickname);
            SharePreferenceHelper.putValue(getApplicationContext(), "photourl", headUrl);
            SharePreferenceHelper.putValue(getApplicationContext(), "uin", unionid);
            if (user == null) {
                user = new User();
            }
            user.setUin(unionid);
            user.setNickName(nickname);
            user.setHeadImgUrl(headUrl);

            ...
        }
    };

當(dāng)拿到用戶的私人信息后,我們就完成了整個(gè)流程,但這些信息不應(yīng)該由我們自己處理,所以當(dāng)需要獲取用戶信息時(shí),我們應(yīng)該請(qǐng)求我們自己的服務(wù)器,然后由服務(wù)器再去請(qǐng)求微信服務(wù)器,這樣的流程才是安全可靠的。

除了微信登錄,微信sdk還集成了微信好友分享,朋友圈分享,小程序分享(比較早的版本是沒(méi)有的...)等功能,分享內(nèi)容包括圖片,文本,網(wǎng)頁(yè)等,這就看我們的需求了,分享這塊我就不寫(xiě)了,大家可以參考下官方的sdkDemo關(guān)于分享的代碼。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,702評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,143評(píng)論 3 415
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 175,553評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 62,620評(píng)論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,416評(píng)論 6 405
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 54,940評(píng)論 1 321
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,024評(píng)論 3 440
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,170評(píng)論 0 287
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,709評(píng)論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,597評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,784評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,291評(píng)論 5 357
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,029評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,407評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,663評(píng)論 1 280
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,403評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,746評(píng)論 2 370

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,597評(píng)論 25 707
  • 轉(zhuǎn)載需要著名出處: 之前寫(xiě)過(guò)微信登錄分享支付第一版: 前言 大部分的app都有接入第三方sdk的需求。例如第三方登...
    Android開(kāi)發(fā)666閱讀 6,465評(píng)論 3 38
  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,776評(píng)論 18 139
  • 第2天·21天OH卡魔法美顏瘦身課#玩卡不卡·每日一抽# 每一位都可以通過(guò)這張卡片覺(jué)察自己: 1、直覺(jué)他叫什么名字...
    聽(tīng)味十足的故事閱讀 204評(píng)論 0 0
  • 如果陽(yáng)光不那么高遠(yuǎn)我愿意摘取一籃每天把它種在地里施肥、除草、澆水直到收獲 如果大地不這么廣袤我愿擇取一隅種些糧食和...
    郭安安閱讀 327評(píng)論 11 15