最近在做android~,恩,就说这么多吧
1,准备工作
支付宝相关文档下载地址:
https://b.alipay.com/order/productDetail.htm?productId=2013080604609688&tabId=4#ps-tabinfo-hash
当然了,假设你已经有了pid(partner)和商户账户(seller),并且开通了手机网页支付功能。
下载的包里面,打开【手机网页即时到账接口】文件夹,看到里面的可以看看里面的pdf文档,【手机网页支付接入与使用规则.pdf】这个文档有教你怎么在支付宝开通相关的东西,这里我提一下,记得把公钥上传,然后私钥生成出来
这个包,对于android,虽然没有需要的lib,但是文档的相关参数说明也是必须的。
看文档,可以知道,网页支付需要两步,第一步是获取授权令牌,第二步根据这个令牌进行交易。
那么在第一步的时候,我们要根据文档提供的url获取令牌,这个时候我们要发起http请求,获得返回的内容,解析出来授权令牌,再去发请求,得到支付宝的支付网页,显示即可
我的做法是,第一步用http请求,第二步是webview加载链接
2,代码相关
假设你已经有了android工程,打开下载的包里面的demo【WS_WAP_PAYWAP-JAVA-UTF-8】在src目录下找到sign这个package然后把RSA.java拿出来,放到你的android工程里,因为我只用了RSA加密所以这里只拿了这一个,如果你需要其他加密方法,那就拿其他的……
还没完,再到util包里面打开AlipaySubmit.java类,找到:
/**
* 解析远程模拟提交后返回的信息,获得token
* @param text 要解析的字符串
* @return 解析结果
* @throws Exception
*/
public static String getRequestToken(String text) throws Exception {
String request_token = "";
//以“&”字符切割字符串
String[] strSplitText = text.split("&");
//把切割后的字符串数组变成变量与数值组合的字典数组
Map<String, String> paraText = new HashMap<String, String>();
for (int i = 0; i < strSplitText.length; i++) {
//获得第一个=字符的位置
int nPos = strSplitText[i].indexOf("=");
//获得字符串长度
int nLen = strSplitText[i].length();
//获得变量名
String strKey = strSplitText[i].substring(0, nPos);
//获得数值
String strValue = strSplitText[i].substring(nPos+1,nLen);
//放入MAP类中
paraText.put(strKey, strValue);
}
if (paraText.get("res_data") != null) {
String res_data = paraText.get("res_data");
//解析加密部分字符串(RSA与MD5区别仅此一句)
if(AlipayConfig.sign_type.equals("0001")) {
res_data = RSA.decrypt(res_data, AlipayConfig.private_key, AlipayConfig.input_charset);
}
//token从res_data中解析出来(也就是说res_data中已经包含token的内容)
Document document = DocumentHelper.parseText(res_data);
request_token = document.selectSingleNode( "//direct_trade_create_res/request_token" ).getText();
}
return request_token;
}
这个方法拿出来,也放到RSA里面去,其中:
res_data = RSA.decrypt(res_data, AlipayConfig.private_key, AlipayConfig.input_charset);
这一句中,private_key是生成的私钥转换成PKCS8的内容(如果你不知道怎么生成,请打开demo里面的openssl工程,里面有一个生成命令文档,按照里面的命令即可),不是直接的RSA私钥,input_charset换成utf-8即可,
最后的:
Document document = DocumentHelper.parseText(res_data);
request_token = document.selectSingleNode( "//direct_trade_create_res/request_token" ).getText();
这两句,请换成成:
Log.d("res_data", res_data);
request_token = res_data.substring(res_data.indexOf("<?"));
int startIndex = request_token.indexOf("<request_token>");
if (startIndex > -1) {
int temp = "<request_token>".length();
String start = request_token.substring(temp+startIndex, temp+startIndex+22);
Log.d("start", start);
int endIndex = request_token.indexOf("</request_token>");
String end = request_token.substring(endIndex - 18, endIndex);
Log.d("end", end);
return start+end;
}
Log.d("request_token", request_token);
return null;
这里说明一下为什么,首先它的这两句无非就是xml的解析,我们没必要这么用,还要加上它的包,另外,我发现解析出来的授权令牌里面有乱码,不管我怎么设置编码都有,本想联系客服问个原因,但是一直都是繁忙中,无奈我只能截取一下,先用着,望有知道的同学告知一下……
看我们的付款layout:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<RelativeLayout
android:id="@+id/top_bar"
android:layout_width="match_parent"
android:layout_height="55dp"
android:layout_alignParentTop="true"
android:background="@color/light_black"
android:padding="4dp">
<TextView
android:id="@+id/back_btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerVertical="true"
android:layout_marginLeft="4dp
android:gravity="center"
android:padding="6dp"
android:text="@string/home_back"
android:textColor="@drawable/top_bar_selector"
android:textSize="20sp" />
<TextView
android:id="@+id/main_app_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:text="支付宝网页版"
android:textColor="@android:color/white"
android:textSize="22sp" />
</RelativeLayout>
<WebView
android:id="@+id/webview"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_below="@+id/top_bar"
android:scrollbars="none" />
</RelativeLayout>
其中的一些color,背景等大家自己调一下,这里面很简单就一个webview
在Activity里面初始化好相关内容,其中关于webview设置如下:
webView.getSettings().setJavaScriptEnabled(true);//允许webkit执行js代码;
webView.setWebChromeClient(new WebChromeClient() {
@Override
public boolean onJsAlert(WebView view, String url, String message,
JsResult result) {
return super.onJsAlert(view, url, message, result);
}
});
webView.setWebChromeClient(new WebChromeClient() {
public void onProgressChanged(WebView view, int newProgress) {
super.onProgressChanged(view, newProgress);
}
});
webView.setWebViewClient(new WebViewClient() {
public boolean shouldOverrideUrlLoading(WebView view, String url) {
return super.shouldOverrideUrlLoading(view, url);
}
public void onPageFinished(WebView view, String url) {
super.onPageFinished(view, url);
if (progressDialog != null) {
progressDialog.dismiss();
}
}
public void onPageStarted(WebView view, String url, Bitmap favicon) {
super.onPageStarted(view, url, favicon);
}
});
其中有一个关闭progressdialog的步骤,这里面如果你没有这个东西,直接注释掉吧。
至此,在我们的oncreate初始话之后呢,加上我们的令牌获取请求,再此之前我们先要初始化一些参数,首先在我们的RSA.java里面添加几个方法:
订单(其中的userid如果没有可以去掉,成功后的页面和通知页面这个你要有自己的处理,当然目前没有的话也可以向我一样写上百度的地址……):
public static String getAliPayWAPOrderBodyInfor(String subject, String price, String refNo, String SUCCESS_URL, String NOTIFY_URL, String userid) {
String orderInfor = "<direct_trade_create_req><su" +
"bject>" + subject +
"</subject><out_trade_no>" + "1282889689801" + "</out_trade_no><total_fee>" + price + "</tot" +
"al_fee><seller_account_name>" + AliPayUtil.SELLER + "</seller_account_name><call_" +
"back_url>" + SUCCESS_URL + "</call_back_" +
"url><notify_url>" + NOTIFY_URL + "</notify_url><out_user>" + userid + "</out_user><pay_expire>3600</pay_expire></direct_trade_create" +
"_req>";
return orderInfor;
}
授权令牌参数:
public static Map<String, String> getAccessTokenParamsMap() {
Map<String, String> sParaTemp = new HashMap<String, String>();
sParaTemp.put("service", "alipay.wap.trade.create.direct");
sParaTemp.put("partner", AliPayUtil.PARTNER);//这里换成你的pid
sParaTemp.put("sec_id", "0001");//这是RSA加密
sParaTemp.put("format", "xml");
sParaTemp.put("v", "2.0");
sParaTemp.put("req_id", UtilDate.getOrderNum());//这个方法可以把util包里面的这个UtilDate.java拿过来,当然了没必要的话直接把相关的方法拿过来
return sParaTemp;
}
交易参数:
public static Map<String, String> getBillParamsMap() {
Map<String, String> sParaTemp = new HashMap<String, String>();
sParaTemp.put("service", "alipay.wap.auth.authAndExecute");
sParaTemp.put("partner", AliPayUtil.PARTNER);//这里换成你的pid
sParaTemp.put("sec_id", "0001");
sParaTemp.put("format", "xml");
sParaTemp.put("v", "2.0");
return sParaTemp;
}
参数拼接方法(在util包的AlipayCore.java中)
/**
* 把数组所有元素排序,并按照“参数=参数值”的模式用“&”字符拼接成字符串
* @param params 需要排序并参与字符拼接的参数组
* @return 拼接后字符串
*/
public static String createLinkString(Map<String, String> params) {
List<String> keys = new ArrayList<String>(params.keySet());
Collections.sort(keys);
String prestr = "";
for (int i = 0; i < keys.size(); i++) {
String key = keys.get(i);
String value = params.get(key);
if (i == keys.size() - 1) {//拼接时,不包括最后一个&字符
prestr = prestr + key + "=" + value;
} else {
prestr = prestr + key + "=" + value + "&";
}
}
return prestr;
}
生成sign签名的方法 :
public static String buildRequestMysign(Map<String, String> sPara) {
String prestr = AlipayCore.createLinkString(sPara); //把数组所有元素,按照“参数=参数值”的模式用“&”字符拼接成字符串
String mysign = "";
mysign = sign(prestr, private_key, "utf-8");
return mysign;
}
然后再你的activity里面这样调用:
<pre name="code" class="html">Map<String, String> sParaTemp = AliPayUtil.getAccessTokenParamsMap();
sParaTemp.put("req_data", RSA.getAliPayWAPOrderBodyInfor(SUBJECT, price, "1282889689801", SUCCESS_URL, NOTIFY_URL, userid));
//生成签名结果
String mysign = RSA.buildRequestMysign(sParaTemp);
//签名结果与签名方式加入请求提交参数组中
sParaTemp.put("sign", mysign);
这样我们的参数map就做好了,当然了 一般我们是用post请求所以好要把它转化一下,像这样:
private List<NameValuePair> generatNameValuePair(Map<String, String> properties) {
List<NameValuePair> nameValuePair = new ArrayList<NameValuePair>(properties.size());
for (Map.Entry<String, String> entry : properties.entrySet()) {
nameValuePair.add(new BasicNameValuePair(entry.getKey(), entry.getValue()));
}
return nameValuePair;
}
这样你就可以以post形式发了,post的url是http://wappaygw.alipay.com/service/rest.htm,参数是上面的list
因为我这边用了一些框架所以对于异步的请求,很好处理,但如果你没有用呢,无非就是handler,message机制,总之,假设你得到了正确的响应结果(不正确的话,看一下code值,和文档中比较一下,j看看是什么错误),然后对结果处理:
String dec = "";
try {
dec = URLDecoder.decode(result, "utf-8");//请将result换成你的请求结果
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
return;
}
String ret;
try {
ret = RSA.getRequestToken(dec);
Log.d("ret", ret);
if (ret!=null) {
wapPay(ret);
} else {
//message
return;
}
} catch (Exception e) {
e.printStackTrace();
return;
}
先做一次urldecode,然后再解析数据:
其中的wapPay方法:
private void wapPay(String token) {
Map<String, String> sParaTemp = RSA.getBillParamsMap();
String req_data = "<auth_and_execute_req><request_token>" + token + "</request_token></auth_and_execute_req>";
sParaTemp.put("req_data", req_data);
String mysign = RSA.buildRequestMysign(sParaTemp);
sParaTemp.put("sign", mysign);
List<NameValuePair> paramspost = generatNameValuePair(sParaTemp);
String url = "http://wappaygw.alipay.com/service/rest.htm?" + URLEncodedUtils.format(paramspost, "UTF-8");
webView.loadUrl(url);
}
好的至此,我们可以看到页面了:
后面的步骤大家就直接看着来把。