手机的在线支付,被认为是2012年最看好的功能,我个人认为这也是移动互联网较传统互联网将会大放光彩的一个功能。

人人有手机,人人携带手机,花钱买东西,不再需要取钱付现,不再需要回家上网银,想买什么,扫描一下,或者搜索一下,然后下单,不找零,直接送到你家,这将是手机支付给我们带来的全新交易体验。

谷歌刚推出了谷歌钱包,这必是我们后面要使用的主要手段,但是鉴于当前国情,我觉得有必要介绍一下android手机集成支付宝功能。

1.下载官方架包和说明文档

其实官方已经提供了安装指南,下载地址:

https://b.alipay.com/order/productIndex.htm

里面有有个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的核心代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
package com.tianxia.lib.baseworld.alipay;
 
import java.net.URLEncoder;
import java.text.SimpleDateFormat;
import java.util.Date;
 
import com.tianxia.lib.baseworld.R;
 
import android.app.Activity;
import android.app.ProgressDialog;
import android.content.DialogInterface;
import android.os.Handler;
import android.os.Message;
import android.view.KeyEvent;
import android.widget.Toast;
 
public class AlixPay
{
 
    static String
TAG =
"AlixPay";
 
    private Activity
mActivity;
    public AlixPay(Activity
activity) {
        mActivity
= activity;
    }
 
    private ProgressDialog
mProgress =
null;
 
    //
the handler use to receive the pay result.
    private Handler
mHandler =
new Handler()
{
        public void handleMessage(Message
msg) {
            try {
                String
strRet = (String) msg.obj;
 
                switch (msg.what)
{
                case AlixId.RQF_PAY:
{
 
                    closeProgress();
 
                    BaseHelper.log(TAG,
strRet);
 
                    try {
                        String
memo =
"memo=";
                        int imemoStart
= strRet.indexOf(
"memo=");
                        imemoStart
+= memo.length();
                        int imemoEnd
= strRet.indexOf(
";result=");
                        memo
= strRet.substring(imemoStart, imemoEnd);
 
                        ResultChecker
resultChecker =
new ResultChecker(strRet);
 
                        int retVal
= resultChecker.checkSign();
                        if (retVal
== ResultChecker.RESULT_CHECK_SIGN_FAILED) {
                            BaseHelper.showDialog(
                                    mActivity,
                                    "提示",
                                    mActivity.getResources().getString(
                                            R.string.check_sign_failed),
                                    android.R.drawable.ic_dialog_alert);
                        }
else {
                            BaseHelper.showDialog(mActivity,
"提示",
memo,
                                    R.drawable.infoicon);
                        }
                         
                    }
catch (Exception
e) {
                        e.printStackTrace();
 
                        BaseHelper.showDialog(mActivity,
"提示",
strRet,
                                R.drawable.infoicon);
                    }
                }
                    break;
                }
 
                super.handleMessage(msg);
            }
catch (Exception
e) {
                e.printStackTrace();
            }
        }
    };
 
    //
close the progress bar
    void closeProgress()
{
        try {
            if (mProgress
!=
null)
{
                mProgress.dismiss();
                mProgress
=
null;
            }
        }
catch (Exception
e) {
            e.printStackTrace();
        }
    }
 
    public void pay()
{
        MobileSecurePayHelper
mspHelper =
new MobileSecurePayHelper(mActivity);
        boolean isMobile_spExist
= mspHelper.detectMobile_sp();
        if (!isMobile_spExist)
            return;
 
        if (!checkInfo())
{
            BaseHelper.showDialog(mActivity,
"提示",
                    "缺少partner或者seller,",
R.drawable.infoicon);
            return;
        }
 
        try {
            //
prepare the order info.
            String
orderInfo = getOrderInfo();
            String
signType = getSignType();
            String
strsign = sign(signType, orderInfo);
            strsign
= URLEncoder.encode(strsign);
            String
info = orderInfo +
"&sign=" +
"\"" +
strsign +
"\"" +
"&"
                    +
getSignType();
             
            //
start the pay.
            MobileSecurePayer
msp =
new MobileSecurePayer();
            boolean bRet
= msp.pay(info, mHandler, AlixId.RQF_PAY, mActivity);
             
            if (bRet)
{
                //
show the progress bar to indicate that we have started paying.
                closeProgress();
                mProgress
= BaseHelper.showProgress(mActivity,
null,
"正在支付",
false,
                        true);
            }
else
                ;
        }
catch (Exception
ex) {
            Toast.makeText(mActivity,
R.string.remote_call_failed,
                    Toast.LENGTH_SHORT).show();
        }
         
    }
 
    private boolean checkInfo()
{
        String
partner = PartnerConfig.PARTNER;
        String
seller = PartnerConfig.SELLER;
        if (partner
==
null ||
partner.length() <=
0 ||
seller ==
null
                ||
seller.length() <=
0)
            return false;
 
        return true;
    }
 
 
    //
get the selected order info for pay.
    String
getOrderInfo() {
        String
strOrderInfo =
"partner=" +
"\"" +
PartnerConfig.PARTNER +
"\"";
        strOrderInfo
+=
"&";
        strOrderInfo
+=
"seller=" +
"\"" +
PartnerConfig.SELLER +
"\"";
        strOrderInfo
+=
"&";
        strOrderInfo
+=
"out_trade_no=" +
"\"" +
getOutTradeNo() +
"\"";
        strOrderInfo
+=
"&";
        //这笔交易价钱
        strOrderInfo
+=
"subject=" +
"\"" +
mActivity.getString(R.string.donate_subject) +
"\"";
        strOrderInfo
+=
"&";
        //这笔交易内容
        strOrderInfo
+=
"body=" +
"\"" +
mActivity.getString(R.string.donate_body) +
"\"";
        strOrderInfo
+=
"&";
        //这笔交易价钱
        strOrderInfo
+=
"total_fee=" +
"\"" +
"10.00" +
"\"";
        strOrderInfo
+=
"&";
        strOrderInfo
+=
"notify_url=" +
"\""
                +
"http://notify.java.jpxx.org/index.jsp" +
"\"";
 
        return strOrderInfo;
    }
 
    //
get the out_trade_no for an order.
    String
getOutTradeNo() {
        SimpleDateFormat
format =
new SimpleDateFormat("MMddHHmmss");
        Date
date =
new Date();
        String
strKey = format.format(date);
 
        java.util.Random
r =
new java.util.Random();
        strKey
= strKey + r.nextInt();
        strKey
= strKey.substring(
0,
15);
        return strKey;
    }
 
    //
get the sign type we use.
    String
getSignType() {
        String
getSignType =
"sign_type=" +
"\"" +
"RSA" +
"\"";
        return getSignType;
    }
 
    //
sign the order info.
    String
sign(String signType, String content) {
        return Rsa.sign(content,
PartnerConfig.RSA_PRIVATE);
    }
 
    //
the OnCancelListener for lephone platform.
    static class AlixOnCancelListener
implements
            DialogInterface.OnCancelListener
{
        Activity
mcontext;
 
        AlixOnCancelListener(Activity
context) {
            mcontext
= context;
        }
 
        public void onCancel(DialogInterface
dialog) {
            mcontext.onKeyDown(KeyEvent.KEYCODE_BACK,
null);
        }
    }
}

这个类的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.小结

支付宝的集成比我想象的要复杂一些,比较麻烦,首先需要审核,然后代码需要提取,所以写出来与大家分享。 

在做集成配置的时候,一定要仔细认真,一个地方出错,可能要导致后面查错查很长时间。

因为本人是先集成成功后才写的这篇文章,难免会漏掉一些重要的细节或者步骤,如有不对,请留言指正。

Android中集成支付宝的更多相关文章

  1. Android App集成支付宝

    原地址:http://blog.csdn.net/wenbingoon/article/details/7933078 手机的在线支付,被认为是2012年最看好的功能,我个人认为这也是移动互联网较传统 ...

  2. Android中集成第三方库的方法和问题

    Android中集成第三方库的方法和问题 声明: 1. 本文參考了网上同学们的现有成果,在此表示感谢,參考资料在文后有链接. 2. 本文的重点在第三部分,是在开发中遇到的问题及解决的方法.第一,第二部 ...

  3. Android中集成ffmpeg(一):编译ffmpeg

    方案选择 Android中集成ffmpeg的codec功能无非两种方式: JNI直接调用,主要用于App开发(无权限修改系统底层),如EXOPlayer,JPlayer等. 集成ffmpeg到OMX, ...

  4. android app 集成 支付宝支付 微信支付

    项目中部分功能点需要用到支付功能,移动端主要集成支付宝支付和微信支付 支付宝sdk以及demo下载地址:https://doc.open.alipay.com/doc2/detail.htm?spm= ...

  5. Android中集成第三方支付

    常见的第三方支付解决方案 支付宝支付 微信支付 银联支付 Ping++统一支付平台(需要继承服务器端和客户端) 短信支付 支付宝的集成流程 相关资料链接: 支付宝支付指引流程:支付指引流程 支付宝An ...

  6. Android应用集成支付宝接口的简化

    拿到支付宝接口的andriod demo后有点无语,集成一个支付服务而已,要在十几个java类之间引用来引用去,这样不仅容易导致应用本身代码结构的复杂化,调试起来也很累,于是操刀改造之: 该删的删,该 ...

  7. Android中集成QQ登陆和QQ好友分享及QQ空间分享

    extends : http://blog.csdn.net/arjinmc/article/details/38439957 相关官方文档及下载地址: 如果只用分享和登陆,用lite包就可以,体积小 ...

  8. display flex在部分低级android中的支付宝窗口表现

    display flex用在移动端布局 当该元素是inline元素如span的时候回出现无宽高的情况,需要增加display:block: 他的子元素如果是inline元素那么也同样会出现这个问题,需 ...

  9. Android集成支付宝接口 实现在线支付

    手机的在线支付,被认为是2012年最看好的功能,我个人认为这也是移动互联网较传统互联网将会大放光彩的一个功能. 人人有手机,人人携带手机,花钱买东西,不再需要取钱付现,不再需要回家上网银,想买什么,扫 ...

随机推荐

  1. Linux下的lds链接脚本简介(四)

    十一. 表达式 lds中表达式的文法与C语言的表达式文法一致,表达式的值都是整型,如果ld的运行主机和生成文件的目标机都是32位,则表达式是32位数据,否则是64位数据. 以下是一些常用的表达式: _ ...

  2. 开发板 视频04_05 ubuntu的联网及基本设置

    4g内存 如果电脑有两g,只能给1.5g 处理器可以根据实际选 usb3.0 或者 2.0 联网模式:: 桥接模式 启动式连接,,,,网是不固定的 仅主机模式,主机和虚拟机在一个网络 第三种联网,自定 ...

  3. UVA 11624 - Fire! 图BFS

    看题传送门 昨天晚上UVA上不去今天晚上才上得去,这是在维护么? 然后去看了JAVA,感觉还不错昂~ 晚上上去UVA后经常连接失败作死啊. 第一次做图的题~ 基本是照着抄的T T 不过搞懂了图的BFS ...

  4. 【习题5-5 UVA-10391】Compound Words

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 枚举每一个串的分割点. 看看左右两个串在不在字符串中即可. [代码] #include <bits/stdc++.h> ...

  5. POJ2739_Sum of Consecutive Prime Numbers【筛法求素数】【枚举】

    Sum of Consecutive Prime Numbers Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 19350 Ac ...

  6. 《大型网站技术架构》1:概述 分类: C_OHTERS 2014-05-07 20:40 664人阅读 评论(0) 收藏

    参考自<大型网站技术架构>第1~3章 1.大型网站架构演化发展历程 (1)初始阶段的网站架构:一台服务器分别作为应用.数据.文件服务器 (2)应用服务和数据服务分离:三台服务器分别承担上述 ...

  7. vs 2013 常用快捷键及常见问题的解决

    1. 代码编辑 关闭当前文档:ctrl + F4 打开光标所在位置的文档:ctrl + G(shift + g) 返回上次编辑的位置:ctrl + -(键盘数字键 0 后的那个按键) 移动光标所在的行 ...

  8. Android自定义组件系列【3】——自定义ViewGroup实现侧滑

    有关自定义ViewGroup的文章已经很多了,我为什么写这篇文章,对于初学者或者对自定义组件比较生疏的朋友虽然可以拿来主义的用了,但是要一步一步的实现和了解其中的过程和原理才能真真脱离别人的代码,举一 ...

  9. System.Xml.XmlException: 引用了未声明的实体“nbsp”

    在XML文件中<, >,&等有特殊含义,(前两个字符用于链接签,&用于转义),不能直接使用.使用这些个字符时,应使用它们的转义序列,下面是5个在XML文件中预定义好的实体: ...

  10. 网络编程02---HTTP协议

    1.URL简单介绍 1.client怎样找到server 我们都知道网络中部署着各种各样的server.比方腾讯的server.百度的server.那么问题来了.client怎样找到想要连接的serv ...