使用支付宝支付官方提供了标准的文档,文档地址
并且还提供了demo:
我们下载一下它提供的demo导入到idea看一下:
readme.txt请好好看一下。
只有一个Java配置类,其余都是JSP。我上面红框框的配置你需要上支付宝开发者平台申请一下。
>>注册蚂蚁金服开发者账号(免费,不像苹果会收取费用)
注册地址: ,用你的支付宝账号扫码登录,完善个人信息,选择服务类型(我选的是自研)。
设置app_id和gatewayUrl
沙箱地址我直接给出:
这里关键是密匙支付宝数字证书怎么用,我们点击“设置”密匙
如何生成的文档:
密匙的生成工具可以选择window或者mac版本
下载相应环境工具并安装后即可使用,本步骤指引以 MAC_OSX 界面为例,如上图所示
开发者根据开发语言选择密钥格式和密钥长度,新建应用请务必使用 RSA2密钥长度 即2048 位(目前已使用 RSA 密钥长度即1024 位密钥长度的应用仍然可以正常调用接口)。点击 生成密钥 后,工具会自动生成商户应用公钥(public key)和应用私钥(private key),如上图所示:
开发者点击工具界面下方的 打开文件位置,即可找到生成的公私钥文件,如下图所示
生成密钥后,开发者就可以在应用的开发配置页面进行配置。点击 设置应用公钥 后,复制上一步生成的公钥,点击 保存,即可完成公钥的设置,如下图所示。
生成的私钥需妥善保管,避免遗失,不要泄露。应用私钥需填写到代码中供签名时使用。应用公钥需提供给支付宝账号管理者上传到支付宝开放平台。
再说一下如何生成公匙
若开发者使用公钥证书签名方式,开放平台支持通过上传 CSR 文件的方式给开发者在线签发应用公钥证书,新的开放平台 RSA 验签和签名工具支持生成 CSR 文件(个人用户由于不涉及到资金接口,建议使用普通公钥方式接入,降低接入成本),具体操作步骤如下:
同“普通公钥方式”一样,下载相应环境工具并安装后即可使用,本步骤指引以 MAC_OSX 界面为例:
点击工具界面下方的 点击获取,生成应用公钥证书 CSR 申请文件。
点击进入 获取csr 页面后,根据如下提示完善填写信息,点击 生成CSR文件。请注意“组织/公司”名称一定要和开发者中心门户账号信息的公司名称保持一致,否则会导致后续步骤中上传csr证书文件校验失败。
Tips:沙箱环境下“组织/公司”名称应填写为“沙箱环境”。
在生成 CSR 文件后,点击 打开密钥文件路径,在对应的文件夹里可以看到三个文件:应用公钥 key 串、应用私钥key 串,以及 csr格式的应用公钥证书文件。如下图所示:
生成公钥证书 CSR 申请文件,开发者就可以在应用的开发配置页面/接口加签方式进行配置(如下图 1)。点击 设置 后,选择公钥证书(如下图2)> 上传CSR文件在线生成证书 > 上传CSR文件在线生成,选择上一步骤生成的 .csr文件上传,即可完成公钥证书的设置,如下图所示。
这样公匙和密匙都生成好了,我们将下载的demo替换一下对应得公匙和密匙;
然后:
服务器异步通知页面路径(notify_url)
如果没有改名,修改IP和端口号就可以了,我自己的如下:
http://localhost:8080/alipay.trade.page.pay-JAVA-UTF-8/notify_url.jsp
页面跳转同步通知页面的路径(return_url)
http://localhost:8080/alipay.trade.page.pay-JAVA-UTF-8/return_url.jsp
一切就绪,我们可以测试运行一下了
测试用的支付宝买家账户可以在”沙箱账”这个页面可以找到:
支付成功后,验签结果:
由于我们使用的是沙箱测试环境支付宝数字证书怎么用,测试环境和正式上线的环境的网关是不一样的,如果配置错误,会出现,appid错误的问题,所以记得修改网关地址。
上面的测试环境的项目代码下载地址:
提取码:9b7z
后面我再讲如何将此支付功能整合到ssm框架中;
项目结构:
项目架构:spring+springmvc+mybatis数据库:mysql部署环境:tomcat9.0开发环境:jdk9、idea支付:支付宝、微信
整合到ssm一样,我们需要像沙箱测试环境一样,需要修改支付的配置信息
数据库的sql:
主要包括以下的数据库表:
drop table if exists user;
/*==============================================================*/
/* Table: user */
/*==============================================================*/
create table user
(
id varchar(20) not null,
username varchar(128),
sex varchar(20),
primary key (id)
);
alter table user comment '用户表';
CREATE TABLE `flow` (
`id` varchar(20) NOT NULL,
`flow_num` varchar(20) DEFAULT NULL COMMENT '流水号',
`order_num` varchar(20) DEFAULT NULL COMMENT '订单号',
`product_id` varchar(20) DEFAULT NULL COMMENT '产品主键ID',
`paid_amount` varchar(11) DEFAULT NULL COMMENT '支付金额',
`paid_method` int(11) DEFAULT NULL COMMENT '支付方式
1:支付宝
2:微信',
`buy_counts` int(11) DEFAULT NULL COMMENT '购买个数',
`create_time` datetime DEFAULT NULL COMMENT '创建时间',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='流水表';
CREATE TABLE `orders` (
`id` varchar(20) NOT NULL,
`order_num` varchar(20) DEFAULT NULL COMMENT '订单号',
`order_status` varchar(20) DEFAULT NULL COMMENT '订单状态
10:待付款
20:已付款',
`order_amount` varchar(11) DEFAULT NULL COMMENT '订单金额',
`paid_amount` varchar(11) DEFAULT NULL COMMENT '实际支付金额',
`product_id` varchar(20) DEFAULT NULL COMMENT '产品表外键ID',
`buy_counts` int(11) DEFAULT NULL COMMENT '产品购买的个数',
`create_time` datetime DEFAULT NULL COMMENT '订单创建时间',
`paid_time` datetime DEFAULT NULL COMMENT '支付时间',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='订单表';
CREATE TABLE `product` (
`id` varchar(20) NOT NULL,
`name` varchar(20) DEFAULT NULL COMMENT '产品名称',
`price` varchar(11) DEFAULT NULL COMMENT '价格',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='产品表 ';
dao接口层
这里就不介绍了,这个只包括简单的curd,可以使用通用mapper,或者逆向工程就行。以订单order为例给出:
public interface OrdersMapper {
int countByExample(OrdersExample example);
int deleteByExample(OrdersExample example);
int deleteByPrimaryKey(String id);
int insert(Orders record);
int insertSelective(Orders record);
List selectByExample(OrdersExample example);
Orders selectByPrimaryKey(String id);
int updateByExampleSelective(@Param("record") Orders record, @Param("example") OrdersExample example);
int updateByExample(@Param("record") Orders record, @Param("example") OrdersExample example);
int updateByPrimaryKeySelective(Orders record);
int updateByPrimaryKey(Orders record);
}
再说service层
以订单order为例给出
/**
* 订单操作 service
* @author ibm
*
*/
public interface OrdersService {
/**
* 新增订单
* @param order
*/
public void saveOrder(Orders order);
/**
*
* @Title: OrdersService.java
* @Package com.sihai.service
* @Description: 修改叮当状态,改为 支付成功,已付款; 同时新增支付流水
* Copyright: Copyright (c) 2017
* Company:FURUIBOKE.SCIENCE.AND.TECHNOLOGY
*
* @author sihai
* @date 2017年8月23日 下午9:04:35
* @version V1.0
*/
public void updateOrderStatus(String orderId, String alpayFlowNum, String paidAmount);
/**
* 获取订单
* @param orderId
* @return
*/
public Orders getOrderById(String orderId);
好了,我们启动项目试试;
首先,启动项目后,输入:8080/,会进入到商品页面,如下:
页面的代码在这儿:
代码实现:
<script src="/static/js/jquery.min.js" type="text/javascript">
产品编号
产品名称
产品价格
操作
${p.id }
${p.name }
${p.price }
<a href="/alipay/goConfirm.action?productId=${p.id }">购买
<input type="hidden" id="hdnContextPath" name="hdnContextPath" value=""/>
$(document).ready(function() {
var hdnContextPath = $("#hdnContextPath").val();
});
点击上面的购买,进入到订单页面
填写个数,然后点击生成订单,调用如下代码
/**
* 分段提交
* 第一段:保存订单
* @param order
* @return
* @throws Exception
*/
@RequestMapping(value = "/createOrder")
@ResponseBody
public LeeJSONResult createOrder(Orders order) throws Exception {
Product p = productService.getProductById(order.getProductId());
String orderId = sid.nextShort();
order.setId(orderId);
order.setOrderNum(orderId);
order.setCreateTime(new Date());
order.setOrderAmount(String.valueOf(Float.valueOf(p.getPrice()) * order.getBuyCounts()));
order.setOrderStatus(OrderStatusEnum.WAIT_PAY.key);
orderService.saveOrder(order);
return LeeJSONResult.ok(orderId);
}
根据SID(生成id的工具)等信息生成订单,保存到数据库。
进入到选择支付页面
调用了如下代码:
/**
* 分段提交
* 第二段
* @param orderId
* @return
* @throws Exception
*/
@RequestMapping(value = "/goPay")
public ModelAndView goPay(String orderId) throws Exception {
Orders order = orderService.getOrderById(orderId);
Product p = productService.getProductById(order.getProductId());
ModelAndView mv = new ModelAndView("goPay");
mv.addObject("order", order);
mv.addObject("p", p);
return mv;
}
然后,我们选择支付宝支付,进入到了我们支付的页面了,大功告成!
调用了如下代码:
/**
*
* @Title: AlipayController.java
* @Package com.sihai.controller
* @Description: 前往支付宝第三方网关进行支付
* Copyright: Copyright (c) 2017
* Company:FURUIBOKE.SCIENCE.AND.TECHNOLOGY
*
* @author sihai
* @date 2017年8月23日 下午8:50:43
* @version V1.0
*/
@RequestMapping(value = "/goAlipay", produces = "text/html; charset=UTF-8")
@ResponseBody
public String goAlipay(String orderId, HttpServletRequest request, HttpServletRequest response) throws Exception {
Orders order = orderService.getOrderById(orderId);
Product product = productService.getProductById(order.getProductId());
//获得初始化的AlipayClient
AlipayClient alipayClient = new DefaultAlipayClient(AlipayConfig.gatewayUrl, AlipayConfig.app_id, AlipayConfig.merchant_private_key, "json", AlipayConfig.charset, AlipayConfig.alipay_public_key, AlipayConfig.sign_type);
//设置请求参数
AlipayTradePagePayRequest alipayRequest = new AlipayTradePagePayRequest();
alipayRequest.setReturnUrl(AlipayConfig.return_url);
alipayRequest.setNotifyUrl(AlipayConfig.notify_url);
//商户订单号,商户网站订单系统中唯一订单号,必填
String out_trade_no = orderId;
//付款金额,必填
String total_amount = order.getOrderAmount();
//订单名称,必填
String subject = product.getName();
//商品描述,可空
String body = "用户订购商品个数:" + order.getBuyCounts();
// 该笔订单允许的最晚付款时间,逾期将关闭交易。取值范围:1m~15d。m-分钟,h-小时,d-天,1c-当天(1c-当天的情况下,无论交易何时创建,都在0点关闭)。 该参数数值不接受小数点, 如 1.5h,可转换为 90m。
String timeout_express = "1c";
alipayRequest.setBizContent("{"out_trade_no":""+ out_trade_no +"","
+ ""total_amount":""+ total_amount +"","
+ ""subject":""+ subject +"","
+ ""body":""+ body +"","
+ ""timeout_express":""+ timeout_express +"","
+ ""product_code":"FAST_INSTANT_TRADE_PAY"}");
//请求
String result = alipayClient.pageExecute(alipayRequest).getBody();
return result;
}
这段代码都可以在阿里支付的demo里面找到的,只需要复制过来,然后改改,整合到ssm环境即可。
上面就是将阿里支付宝支付整合到ssm的全过程;
此整合过程在我的github有源码:
喜欢的可以自行fork;如果你整合遇到困难可以在留言区留言,如果你喜欢我的文章记得点击关注我。
彩蛋:
细心的同学可能会发现我的项目里面已经包含微信支付的功能了,代码大家可以自行研究
本文到此结束,希望对大家有所帮助!