關(guān)于如何申請 “微信公眾平臺開發(fā)賬號” 和 “微信商戶平臺賬號” 這些步驟,在這里就不再做介紹了,本文假設(shè)我們已經(jīng)做好了所有的申請并已經(jīng)被微信后臺審核通過了。
第一步,打開 “微信公眾平臺” 后臺,添加你的APP。在管理中心當中點擊 “創(chuàng)建移動應用”, 填寫一大堆資料后,通過審核,就可以獲得一個AppID,例如:wx*********************。這里大家需要注意一點是,在你創(chuàng)建移動應用的過程當中,有一個叫做“應用簽名”的參數(shù),如何生成這個參數(shù)呢?這里有些要點,打開微信公眾平臺后臺頁面,點擊 資源中心 > 資源下載 > 簽名生成工具(https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&verify=1&id=open1419319167&token=&lang=zh_CN),下載這個工具,這個東西下載后是一個.zip文件,然后把這個文件的后綴名改成.apk,發(fā)送到你的手機當中(記住這個手機要安裝了你的APP才可以),然后安裝這個apk,打開這個apk,輸入你自己app的包名,什么是app包名,打開你的android studio打開model的build.gradle文件,找到applicationId,這個就是你要填寫的包名。在填入包名后,就可以生成簽名了,沒錯,就是這么簡陋,手動把這個簽名填到你的app信息當中,千萬不要填錯了。
第二步,當申請的APP通過審核后,進入后臺在管理中心點擊“查看”你自己申請的APP,就可以看到當前APP的詳細信息,其中有一項“微信支付”,我們就是要申請這個功能,點擊申請,然后又是填寫一大堆資料,然后等審核通過,通過后,前期的工作就算準備好了。
第三步,正式接入SDK,首先當然去下載SDK,本人是使用AS,所以直接build.gradel直接加入這個compile 'com.tencent.mm.opensdk:wechat-sdk-android-with-mta:+'就可以了。用Eclipse的兄弟,可以自行參考官方文檔下載SDK導入即可,但為了方便還是盡快使用gradle管理項目。
第四步,在你的項目根目錄新建一個包,名字固定為“wxapi”,全部小寫,例如:
在“wxapi”包下新建一個Activity,名字規(guī)定為“WXPayEntryActivity”,這個activity幾乎為固定的內(nèi)容,此activity的作用是當你成功調(diào)起微信支付并成功支付后,微信就會調(diào)用這個activity,說清楚,是成功調(diào)起微信支付并得到支付結(jié)果(成功或失敗)后,微信SDK才會調(diào)起這個activity,此Activity無關(guān)重要但又不可缺少,如果你喜歡,甚至可以忽略為此activity做任何布局文件。此Activity看起來大概就是這個樣子,只要存在于“wxapi”包之下,就可以了。
public class WXPayEntryActivity extends Activity implements IWXAPIEventHandler {
//這個就是你自己APP盛申請成功后在后臺看到的appId。
private final String APP_ID = "wx******************";
private IWXAPI api;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
api = WXAPIFactory.createWXAPI(this, APP_ID);
api.handleIntent(getIntent(), this);
}
@Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
setIntent(intent);
api.handleIntent(intent, this);
}
@Override
public void onReq(BaseReq req) {
//
}
//這個方法是支付結(jié)果的回調(diào),大家可以隨意開始你們的表演,也可以什么都不表演,像我一樣直接finish。
@Override
public void onResp(BaseResp resp) {
if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX) {
if (resp.errCode == 0) {
System.out.println("支付成功");
} else {
System.out.println("支付失敗, errorCode = " + resp.errCode);
}
finish();
}
}
}
最后在app清單當中聲明此activity
<activity
android:name=".wxapi.WXPayEntryActivity"
android:exported="true"
android:launchMode="singleTop">
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<data android:scheme="wx**********************" />這個是你申請到的appId。
</intent-filter>
</activity>
以上步驟基本為固定步驟,一下的內(nèi)容才是描述如何調(diào)起微信支付SDK。
第五步,在你需要調(diào)起微信支付的Activity去實現(xiàn)一個接口implements IWXAPIEventHandler,隨意一個都可以。
public class XXXActivity extends Activity implements IWXAPIEventHandler {
private final String APP_ID = "wx**********************";
private IWXAPI mWeChatAPI;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_pay);
}
@Override
public void onReq(BaseReq baseReq) {
}
@Override
public void onResp(BaseResp baseResp) {
}
private void payByWeChat(String item, String userId) {
Call<PrePayResponse> call = HttpManager.INSTANCE.getPrePay(item, userId);
if (call != null) {
call.enqueue(new Callback<PrePayResponse>() {
@Override
public void onResponse(@NonNull Call<PrePayResponse> call, @NonNull Response<PrePayResponse> response) {
final PrePayResponse prePayResponse = response.body();
if (prePayResponse != null) {
final String appId = prePayResponse.getAppId();
final String nonceStr = prePayResponse.getNonceStr();
final String partnerId = prePayResponse.getPartnerId();
final String timeStamp = prePayResponse.getTimeStamp();
final String prepayId = prePayResponse.getPrepayId();
final String packageValue = prePayResponse.getPackageValue();
String prePaySign = "appid=" + appId +
"&noncestr=" + nonceStr +
"&package=Sign=WXPay" +
"&partnerid=" + partnerId +
"&prepayid=" + prepayId +
"×tamp=" + timeStamp +
"&key=*******************************************";
final String paySign = CommonUtil.getMessageDigest(prePaySign.getBytes()).toUpperCase();
System.out.println("appId = " + appId);
System.out.println("nonceStr = " + nonceStr);
System.out.println("partnerId = " + partnerId);
System.out.println("sign = " + paySign);
System.out.println("timeStamp = " + timeStamp);
System.out.println("prepayId = " + prepayId);
System.out.println("packageValue = " + packageValue);
call.cancel();
new Thread(new Runnable() {
@Override
public void run() {
mWeChatAPI = WXAPIFactory.createWXAPI(PayEntryActivity.this, null);
mWeChatAPI.registerApp(APP_ID);
PayReq payReq = new PayReq();
payReq.appId = appId;
payReq.nonceStr = nonceStr;
payReq.partnerId = partnerId;
payReq.sign = paySign;
payReq.timeStamp = timeStamp;
payReq.prepayId = prepayId;
payReq.packageValue = packageValue;
mWeChatAPI.sendReq(payReq);
}
}).start();
finish();
} else {
System.out.println("prePayResponse body is null");
}
}
@Override
public void onFailure(@NonNull Call<PrePayResponse> call, @NonNull Throwable t) {
}
});
}
}
}
我們關(guān)注的是payByWeChat這個方法,不管你使用什么網(wǎng)絡(luò)框架和你自己的服務器后臺通信,你需要得到的都是一串Json,這個工作流程就是你自己的服務器后臺首先需要去調(diào)用微信的統(tǒng)一支付接口,獲得預付訂單的返回信息,這里有2種情況,你公司的后臺開發(fā)是一個極度懶惰的人(例如我公司那個),他就會把從微信得到的返回信息甚至原封不動的轉(zhuǎn)發(fā)給你,最后你自己還要在app端對json做各種截取字符串的操作,自己進行二次簽名。第二種情況是你公司的后臺開發(fā)是一個負責人的人,他已經(jīng)把各種你要的參數(shù)都處理好了,你從json里面得到的數(shù)據(jù)就是直接用于調(diào)起微信支付所需要的數(shù)據(jù)。現(xiàn)在我們從一個懶惰后臺開發(fā)者的角度說一說如何處理這個json。
從你的服務器后臺獲取原始的預付訂單信息后,我們從Json當中可以獲得什么,舉個例子:
{"appId":"wx******************",
"nonceStr":"****************************************",
"package":"prepay_id=wx******************************************",
"partnerId":"99999999999",
"paySign":"2767257DC4034263D73F3C22B718FCEA",
"signType":"MD5",
"timeStamp":"1507441751"}
這就是你從那個懶惰后臺那里得到的,原始的微信預付訂單信息,參考微信的文檔,我們一共需要傳遞7個參數(shù)才能成功調(diào)起微信SDK的支付接口,他們分別是:
PayReq payReq = new PayReq();
payReq.appId = appId;
payReq.nonceStr = nonceStr;
payReq.partnerId = partnerId;
payReq.sign = paySign;
payReq.timeStamp = timeStamp;
payReq.prepayId = prepayId;
payReq.packageValue = packageValue;
新建一個PayReq對象,并為這7個參數(shù)賦值,其中一些,我們可以從json當中直接獲取,例如appid,nonceStr ,partnerId等,但其中有3個參數(shù)我們需要額外注意,就是payReq.prepayId,payReq.packageValue和payReq.sign,payReq.packageValue是一個固定的值,值為“Sign=WXPay”,payReq.prepayId需要你從那個懶惰后臺得到的json當中,從packge那個字段里面取出來,從“wx”這里開始取,前面的“prepay_id=”要去掉。下面重點說一下payReq.sign如何處理。
payReq.sign這個就是我們說的二次簽名,需要什么來組成這個簽名呢?答案是6個參數(shù)+1個key,6個參數(shù)就是上面PayReq 對象7個值當中除去payReq.sign這個值,key就是你在“微信商戶平臺”里面設(shè)置的那個key,看清楚,是微信商戶平臺后臺里面設(shè)置的那個key,然后把這7個參數(shù)按照一定循序組合成一個字符串:
String prePaySign = "appid=" + appId +
"&noncestr=" + nonceStr +
"&package=Sign=WXPay" +
"&partnerid=" + partnerId +
"&prepayid=" + prepayId +
"×tamp=" + timeStamp +
"&key=你自己的key";
大家一定要注意,這個字符串拼接的順序不能錯,直接復制我的或者去https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=20_1這個頁面,在校驗方式里面選自定義,然后把這7個參數(shù)填入去,讓工具為你自動生成這個順序,這7個參數(shù)的名字也不能錯,參考https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_12&index=2這個頁面當中的7個參數(shù),直接復制參數(shù)的名字。在拼接完這個字符串后,我們就要用MD5為這個字符串生成校驗碼,不要用安卓自帶的MD5,這里使用微信SDK內(nèi)使用的那個方法:
static String getMessageDigest(byte[] buffer) {
char hexDigits[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};
try {
MessageDigest mdTemp = MessageDigest.getInstance("MD5");
mdTemp.update(buffer);
byte[] md = mdTemp.digest();
int j = md.length;
char str[] = new char[j * 2];
int k = 0;
for (byte byte0 : md) {
str[k++] = hexDigits[byte0 >>> 4 & 0xf];
str[k++] = hexDigits[byte0 & 0xf];
}
return new String(str);
} catch (Exception e) {
return "";
}
}
到此為止,你的二次簽名算是完成了,最后湊齊7個龍珠,可以召喚神龍了,記得在另一個線程召喚。
new Thread(new Runnable() {
@Override
public void run() {
mWeChatAPI = WXAPIFactory.createWXAPI(PayEntryActivity.this, null);
mWeChatAPI.registerApp(APP_ID);
PayReq payReq = new PayReq();
payReq.appId = appId;
payReq.nonceStr = nonceStr;
payReq.partnerId = partnerId;
payReq.sign = paySign;
payReq.timeStamp = timeStamp;
payReq.prepayId = prepayId;
payReq.packageValue = packageValue;
mWeChatAPI.sendReq(payReq);
}
}).start();
如無意外,大家應該可以正確的召喚出微信支付的界面了。