手机的在线支付,被认为是2012年最看好的功能,我个人认为这也是移动互联网较传统互联网将会大放光彩的一个功能。
人人有手机,人人携带手机,花钱买东西,不再需要取钱付现,不再需要回家上网银,想买什么,扫描一下,或者搜索一下,然后下单,不找零,直接送到你家,这将是手机支付给我们带来的全新交易体验。
谷歌刚推出了谷歌钱包,这必是我们后面要使用的主要手段,但是鉴于当前国情,我觉得有必要介绍一下android手机集成支付宝功能。

1.下载官方架包和说明文档
其实官方已经提供了安装指南,下载地址:
https://mobiless.alipay.com/product/product_down_load.htm?code=SECURITY_PAY
里面有有个pdf,详细说明了说用指南,写的比较详细,可以重点参考。


下载下来,我们主要是用到Android(20120104)目录下的alipay_plugin.jar和AppDemo/assets下的alipay_plugin223_0309.apk,这两个文件是我们不能修改的支付宝api和安装包。

2. 商户签约
现在的安全机制,都是这样,客户端需要先和服务端请求验证后才能进行进一步操作,oauth也是如此。
打开https://ms.alipay.com/,登陆支付宝,点击签约入口,选择"应用类产品",填写并等待审核,获取商户ID和账户ID。
签约的时候还要向需要提供实名认证和上传应用,所以我建议先把应用做好了,最后再集成支付宝。


我大概等了1-2天审核,审核是失败的,回复是应用类型啥的应该是"虚拟货币",我改成那个马上自动就审核通过了。

3.密钥配置
解压openssl-0.9.8k_WIN32(RSA密钥生成工具).zip,打开cmd,命令行进入openssl-0.9.8k_WIN32(RSA密钥生成工具)\bin目录下,
(1).执行

?
1 openssl genrsa -out rsa_private_key.pem 1024

生成rsa_private_key.pem文件。
(2).再执行

?
1 openssl rsa -in rsa_private_key.pem -pubout -out rsa_public_key.pem

生成rsa_public_key.pem 文件。
(3).在执行

?
1 openssl pkcs8 -topk8 -inform PEM -in rsa_private_key.pem -outform PEM -nocrypt

将RSA私钥转换成 PKCS8 格式,去掉begin和end那两行,把里面的内容拷贝出来,保存到某个txt中,如rsa_private_pkcs8_key.txt中(我好像没用到这个)。
打开rsa_public_key.pem,即商户的公钥,复制到一个新的TXT中,删除文件头”-----BEGIN PUBLIC KEY-----“与文件尾”-----END PUBLIC KEY-----“还有空格、换行,变成一行字符串并保存该 TXT 文件,然后在网站的“我的商家服务”切换卡下的右边点击“密钥管理”,然后有个"上传商户公钥(RSA)"项,选择上传刚才的TXT文件.
好了,服务器配置OK,因为这一段之前没有截图,现在弄好了又不好截图,如果有不明白的地方请大家参考官方文档。

4.引用jar和包含安装包
(1).新建android工程;
(2).copy上面说的alipay_plugin.jar到工程的libs目录下,并在java build path中通过Add External JARs找到并引用该jar;
(3).copy上面说的alipay_plugin223_0309.apk安装包到assets目录下,后面配置路径用到。

如果libs和assets目录没有,手动建立者两个目录。

5.调用代码整理
这里我们要严重的参考文档中AppDemo,我们建一个包com.tianxia.lib.baseworld.alipay,把AppDemo的com.alipay.android.appDemo4包下的源码全部copy到刚才我们自己的包下,还有res目录下的资源文件也合并到我们工程res下。
其中AlixDemo.java,ProductListAdapter.java,Products.java是示例类,我们借鉴完后可以删除。
PartnerConfig.java是配置类,配置商户的一些配置参数。
其他的类是严重参考类,直接留下使用。
PartnerConfig.java代码如下:

?
1 2 3 4 5 6 7 8 9 10 11 12 public class PartnerConfig { //合作商户ID。用签约支付宝账号登录ms.alipay.com后,在账户信息页面获取。 public static final String PARTNER = "xxx" ; //账户ID。用签约支付宝账号登录ms.alipay.com后,在账户信息页面获取。 public static final String SELLER = "xxx" ; //商户(RSA)私钥 ,即rsa_private_key.pem中去掉首行,最后一行,空格和换行最后拼成一行的字符串 public static final String RSA_PRIVATE = "xxx" ; //支付宝(RSA)公钥 用签约支付宝账号登录ms.alipay.com后,在密钥管理页面获取。 public static final String RSA_ALIPAY_PUBLIC = "xxx" ; //下面的配置告诉应用去assets目录下找安装包 public static final String ALIPAY_PLUGIN_NAME = "alipay_plugin223_0309.apk" ; }

AlixDemo中代码是最终的调用代码在onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) {}中,下面我们提取其中的核心代码。

6.提取核心调用代码
在AlixDemo.java同目录下新建AlixPay.java,来提取AlixDemo.java的核心代码:

[java] view plain copy
  1. packagecom.tianxia.lib.baseworld.alipay;
  2. importjava.net.URLEncoder;
  3. importjava.text.SimpleDateFormat;
  4. importjava.util.Date;
  5. importcom.tianxia.lib.baseworld.R;
  6. importandroid.app.Activity;
  7. importandroid.app.ProgressDialog;
  8. importandroid.content.DialogInterface;
  9. importandroid.os.Handler;
  10. importandroid.os.Message;
  11. importandroid.view.KeyEvent;
  12. importandroid.widget.Toast;
  13. publicclassAlixPay{
  14. staticStringTAG="AlixPay";
  15. privateActivitymActivity;
  16. publicAlixPay(Activityactivity){
  17. mActivity=activity;
  18. }
  19. privateProgressDialogmProgress=null;
  20. //thehandlerusetoreceivethepayresult.
  21. privateHandlermHandler=newHandler(){
  22. publicvoidhandleMessage(Messagemsg){
  23. try{
  24. StringstrRet=(String)msg.obj;
  25. switch(msg.what){
  26. caseAlixId.RQF_PAY:{
  27. closeProgress();
  28. BaseHelper.log(TAG,strRet);
  29. try{
  30. Stringmemo="memo=";
  31. intimemoStart=strRet.indexOf("memo=");
  32. imemoStart+=memo.length();
  33. intimemoEnd=strRet.indexOf(";result=");
  34. memo=strRet.substring(imemoStart,imemoEnd);
  35. ResultCheckerresultChecker=newResultChecker(strRet);
  36. intretVal=resultChecker.checkSign();
  37. if(retVal==ResultChecker.RESULT_CHECK_SIGN_FAILED){
  38. BaseHelper.showDialog(
  39. mActivity,
  40. "提示",
  41. mActivity.getResources().getString(
  42. R.string.check_sign_failed),
  43. android.R.drawable.ic_dialog_alert);
  44. }else{
  45. BaseHelper.showDialog(mActivity,"提示",memo,
  46. R.drawable.infoicon);
  47. }
  48. }catch(Exceptione){
  49. e.printStackTrace();
  50. BaseHelper.showDialog(mActivity,"提示",strRet,
  51. R.drawable.infoicon);
  52. }
  53. }
  54. break;
  55. }
  56. super.handleMessage(msg);
  57. }catch(Exceptione){
  58. e.printStackTrace();
  59. }
  60. }
  61. };
  62. //closetheprogressbar
  63. voidcloseProgress(){
  64. try{
  65. if(mProgress!=null){
  66. mProgress.dismiss();
  67. mProgress=null;
  68. }
  69. }catch(Exceptione){
  70. e.printStackTrace();
  71. }
  72. }
  73. publicvoidpay(){
  74. MobileSecurePayHelpermspHelper=newMobileSecurePayHelper(mActivity);
  75. booleanisMobile_spExist=mspHelper.detectMobile_sp();
  76. if(!isMobile_spExist)
  77. return;
  78. if(!checkInfo()){
  79. BaseHelper.showDialog(mActivity,"提示",
  80. "缺少partner或者seller,",R.drawable.infoicon);
  81. return;
  82. }
  83. try{
  84. //preparetheorderinfo.
  85. StringorderInfo=getOrderInfo();
  86. StringsignType=getSignType();
  87. Stringstrsign=sign(signType,orderInfo);
  88. strsign=URLEncoder.encode(strsign);
  89. Stringinfo=orderInfo+"&sign="+"\""+strsign+"\""+"&"
  90. +getSignType();
  91. //startthepay.
  92. MobileSecurePayermsp=newMobileSecurePayer();
  93. booleanbRet=msp.pay(info,mHandler,AlixId.RQF_PAY,mActivity);
  94. if(bRet){
  95. //showtheprogressbartoindicatethatwehavestartedpaying.
  96. closeProgress();
  97. mProgress=BaseHelper.showProgress(mActivity,null,"正在支付",false,
  98. true);
  99. }else
  100. ;
  101. }catch(Exceptionex){
  102. Toast.makeText(mActivity,R.string.remote_call_failed,
  103. Toast.LENGTH_SHORT).show();
  104. }
  105. }
  106. privatebooleancheckInfo(){
  107. Stringpartner=PartnerConfig.PARTNER;
  108. Stringseller=PartnerConfig.SELLER;
  109. if(partner==null||partner.length()<=0||seller==null
  110. ||seller.length()<=0)
  111. returnfalse;
  112. returntrue;
  113. }
  114. //gettheselectedorderinfoforpay.
  115. StringgetOrderInfo(){
  116. StringstrOrderInfo="partner="+"\""+PartnerConfig.PARTNER+"\"";
  117. strOrderInfo+="&";
  118. strOrderInfo+="seller="+"\""+PartnerConfig.SELLER+"\"";
  119. strOrderInfo+="&";
  120. strOrderInfo+="out_trade_no="+"\""+getOutTradeNo()+"\"";
  121. strOrderInfo+="&";
  122. //这笔交易价钱
  123. strOrderInfo+="subject="+"\""+mActivity.getString(R.string.donate_subject)+"\"";
  124. strOrderInfo+="&";
  125. //这笔交易内容
  126. strOrderInfo+="body="+"\""+mActivity.getString(R.string.donate_body)+"\"";
  127. strOrderInfo+="&";
  128. //这笔交易价钱
  129. strOrderInfo+="total_fee="+"\""+"10.00"+"\"";
  130. strOrderInfo+="&";
  131. strOrderInfo+="notify_url="+"\""
  132. +"http://notify.java.jpxx.org/index.jsp"+"\"";
  133. returnstrOrderInfo;
  134. }
  135. //gettheout_trade_noforanorder.
  136. StringgetOutTradeNo(){
  137. SimpleDateFormatformat=newSimpleDateFormat("MMddHHmmss");
  138. Datedate=newDate();
  139. StringstrKey=format.format(date);
  140. java.util.Randomr=newjava.util.Random();
  141. strKey=strKey+r.nextInt();
  142. strKey=strKey.substring(0,15);
  143. returnstrKey;
  144. }
  145. //getthesigntypeweuse.
  146. StringgetSignType(){
  147. StringgetSignType="sign_type="+"\""+"RSA"+"\"";
  148. returngetSignType;
  149. }
  150. //signtheorderinfo.
  151. Stringsign(StringsignType,Stringcontent){
  152. returnRsa.sign(content,PartnerConfig.RSA_PRIVATE);
  153. }
  154. //theOnCancelListenerforlephoneplatform.
  155. staticclassAlixOnCancelListenerimplements
  156. DialogInterface.OnCancelListener{
  157. Activitymcontext;
  158. AlixOnCancelListener(Activitycontext){
  159. mcontext=context;
  160. }
  161. publicvoidonCancel(DialogInterfacedialog){
  162. mcontext.onKeyDown(KeyEvent.KEYCODE_BACK,null);
  163. }
  164. }
  165. }

这个类的pay方法就是支付的方法,最简单的不设置的话,调用方法如下:

?
1 2 AlixPay alixPay = new AlixPay(SettingTabActivity. this ); alixPay.pay();

如果没有安装支付宝,它会提示你安装,如果已经安装,它直接让你选择付款:

这说明已经配置成功了。
然后可以删掉那些示例java文件了:AlixDemo.java,ProductListAdapter.java,Products.java。
你也可以通过调整参数来修改订单信息,如主题,价格等。
另外在BaseHelper的94行:

?
1 dialog.setOnCancelListener( new AlixDemo.AlixOnCancelListener( (Activity)context ) );

需要修改为:

?
1 dialog.setOnCancelListener( new AlixPay.AlixOnCancelListener( (Activity)context ) );

7.注意
我在测试的时候,调用的activity是框在一个ActivityGroup里的(与tabhost类似,据说tabhost也有这个问题),导致MobileSecurePayer.java的pay方法中调用服务的两行代码:

?
mActivity.bindService( new Intent(IAlixPay. class .getName()), mAlixPayConnection, Context.BIND_AUTO_CREATE); mActivity.unbindService(mAlixPayConnection);

需要修改为:

?
1 2 mActivity.getApplicationContext().bindService( new Intent(IAlixPay. class .getName()), mAlixPayConnection, Context.BIND_AUTO_CREATE); mActivity.getApplicationContext().unbindService(mAlixPayConnection);

不然会报错java.lang.ClassCastException: android.os.BinderProxy cannot be cast to com.android.server.am.ActivityRecord$Token...

8.小结
支付宝的集成比我想象的要复杂一些,比较麻烦,首先需要审核,然后代码需要提取,所以写出来与大家分享。
在做集成配置的时候,一定要仔细认真,一个地方出错,可能要导致后面查错查很长时间。
因为本人是先集成成功后才写的这篇文章,难免会漏掉一些重要的细节或者步骤,如有不对,请留言指正。

更多相关文章

  1. Android获取双卡双待手机的SIM卡信息示例代码
  2. Android个推php后台开发
  3. Android通过WebView实现原生Java与JS交互
  4. Android神技之 使用SVG以及自定义IconFont字体库
  5. Android中java反射(Reflection)实战
  6. MUI如何安卓离线打包,在Android(安卓)Studio创建Hello World离线
  7. 宏锦软件 Android(安卓)的 ListView 使用详解
  8. Android的桌面上的应用介绍
  9. react-native启动流程(android端)

随机推荐

  1. grid仿PHP中文网首页
  2. JS的引入方式
  3. 懒加载及轮播图
  4. JQuery 常用方法
  5. 定位原理、模态框以及三列布局
  6. 作业_0706
  7. 前端作业-07-09
  8. 通过canvas作图片缓存
  9. uniapp H5图片上传压缩自动旋转
  10. canvas图片设置crossOrigin属性