支付宝微信支付
大家好,我是一个萌新,由于才接触java没多久,项目中需要使用支付宝和微信中的支付功能,在此做个记录,方便以后查看,希望也能帮助你们!!!话不多说,直接上代码。
我是用的spring boot 和 maven框架搭建的项目。
1、支付宝
1.1、首先需要导入支付宝和微信的相关依赖
<!-- 支付宝依赖jar -->
<dependency>
<groupId>com.alipay.sdk</groupId>
<artifactId>alipay-sdk-java</artifactId>
</dependency>
<!--微信依赖jar-->
<dependency>
<groupId>com.github.wxpay</groupId>
<artifactId>wxpay-sdk</artifactId>
</dependency>
1.2、生成支付宝的帮助类
/**
* @author ???
* 支付宝配置信息
*/
public class AlipayConfig {
//卖家用户id
public static String seller_id = "卖家的用户id";
//应用的appid
public static String appId = "应用appid";
//商户公钥
public static String publicKey ="商户的公钥"; //公钥生成方式查看支付宝官方文档
//
public static String charset = "UTF-8"; //不许要更改
//正式环境支付宝网管
public static String wanggaun = "https://openapi.alipay.com/gateway.do"; //不需要更改
//沙箱支付宝网管
// public static String wanggaun = "https://openapi.alipaydev.com/gateway.do"; //不需要更改
//signType
public static String signtype = "RSA2"; //不需要更改
//format
public static String format = "json"; //不需要更改
//异步处理返回结果
public static String notify_url = "支付宝异步返回到自己后台服务的地址";
}
1.3、支付宝支付接口调用
import com.alipay.api.AlipayApiException;
import com.alipay.api.AlipayClient;
import com.alipay.api.DefaultAlipayClient;
import com.alipay.api.domain.AlipayTradeAppPayModel;
import com.alipay.api.request.AlipayTradeAppPayRequest;
import com.alipay.api.response.AlipayTradeAppPayResponse;
/**
* @author ???
* @version 1.0
* 支付宝支付接口调用
*/
public class Zhifubao {
AlipayClient alipayClient = new DefaultAlipayClient(AlipayConfig.wanggaun,AlipayConfig.appId, AlipayConfig.privateKey, AlipayConfig.format, AlipayConfig.charset, AlipayConfig.publicKey, AlipayConfig.signtype); //只需要调用一次
public String pay(Order order){
/**
* 获得初始化的AlipayClient
*/
AlipayTradeAppPayRequest request = new AlipayTradeAppPayRequest();//创建API对应的request类
AlipayTradeAppPayModel Model = new AlipayTradeAppPayModel();
Model.setSubject(order.getSubject()); // 商品名称
Model.setBody(order.getBody());
Model.setOutTradeNo(order.getOut_trade_no());//商品编号
Model.setTotalAmount(order.getTotal_amount()); // 支付金额
Model.setProductCode("QUICK_MSECURITY_PAY"); // 销售产品码
request.setBizModel(Model);//设置业务参数
request.setNotifyUrl(AlipayConfig.notify_url);
AlipayTradeAppPayResponse response;//通过alipayClient调用API,获得对应的response
try {
response = alipayClient.sdkExecute(request);
return response.getBody();
} catch (AlipayApiException e) {
e.printStackTrace();
}
return null;
}
}
1.4 创建一个订单类
/**
* @author ???
* @version 1.0
* 订单对象
*/
public class Order {
//订单编号
private String out_trade_no;
//支付宝订单价格
private String total_amount;
//微信支付总价格
private String total_fee;
//订单介绍
private String subject;
//订单主体
private String body;
//订单附带数据
private String attach;
//订单code固定值
private String product_code = "QUICK_WAP_PAY";
}
1.5 生成订单签名
/**
* 生成订单
*/
public String pay(Order order) {
Zhifubao zhifubao = new Zhifubao();
String pay = zhifubao.pay(order);
return pay;
}
//把得到的加签字符串直接返回给前端
1.6 获取支付宝的异步回调信息
/**
* 获取支付宝异步返回的支付信息
* @param request
*/
@RequestMapping("/notify_url")
public String notify_url(HttpServletRequest requests){
// 1.从支付宝回调的request域中取值
Map<String, String[]> requestParams = requests.getParameterMap();
HashMap<String, String> params = new HashMap<>();
System.out.println("从支付宝回调的request域中取值:requestParams===================================" + requestParams);
for (Iterator<String> iter = requestParams.keySet().iterator(); iter.hasNext();) {
String name = iter.next();
if(name.equals("sign") || name.equals("sign_type")){
continue;
}
String[] values = requestParams.get(name);
String valueStr = "";
for (int i = 0; i < values.length; i++) {
valueStr = (i == values.length - 1) ? valueStr + values[i] : valueStr + values[i] + ",";
}
// 乱码解决,这段代码在出现乱码时使用。如果mysign和sign不相等也可以使用这段代码转化
try {
valueStr = new String(valueStr.getBytes("gb2312"), "gbk");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
params.put(name, valueStr);
}
// 2.封装必须参数
String out_trade_no = requests.getParameter("out_trade_no"); // 商户订单号
System.err.println("out_trade_no==================================" + out_trade_no);
String orderType = requests.getParameter("body"); // 订单内容
System.out.println("orderType==================================" + orderType);
String tradeStatus = requests.getParameter("trade_status"); // 交易状态
System.err.println("tradeStatus=================================" + tradeStatus);
String total_amount = requests.getParameter("total_amount"); //交易状态实际金额
String seller_id = requests.getParameter("seller_id"); //
String app_id = requests.getParameter("app_id"); //得到商户id
// 3.签名验证(对支付宝返回的数据验证,确定是支付宝返回的)
// boolean signVerified = false;
// try {
// // 3.1调用SDK验证签名
// signVerified = AlipaySignature.rsaCheckV1(params, AlipayConfig.publicKey, AlipayConfig.charset);
// System.out.println(signVerified);
// } catch (Exception e) {
// e.printStackTrace();
// }
// 4.对验签进行处理
// if (signVerified){ // 验签通过
if (tradeStatus.equals("TRADE_SUCCESS")) {// 只处理支付成功的订单:修改交易表状态,支付成功
//查看订单信息
if(app_id.equals(AlipayConfig.appId) && seller_id.equals(AlipayConfig.seller_id)){
//自己的业务逻辑。。。。。。。。
}
}
return "failure";
}
return "failure";
}
2、微信支付
2.1、生成微信配置文件
import com.github.wxpay.sdk.WXPayConfig;
import java.io.ByteArrayInputStream;
import java.io.InputStream;
/**
* @author ???
* @version 1.0
* 微信配置模块
*/
public class WXConfigUtil implements WXPayConfig {
private byte[] certData;
public static final String APP_ID = "应用appid";
public static final String KEY = "key";
public static final String MCH_ID = "MCH_ID";
public WXConfigUtil() throws Exception {
// String certPath = "证书地址";//从微信商户平台下载的安全证书存放的路径
// File file = new File(certPath);
// InputStream certStream = new FileInputStream(file);
// this.certData = new byte[(int) file.length()];
// certStream.read(this.certData);
// certStream.close();
}
@Override
public String getAppID() {
return APP_ID;
}
@Override
public String getMchID() {
return MCH_ID;
}
@Override
public String getKey() {
return KEY;
}
@Override
public InputStream getCertStream() {
ByteArrayInputStream certBis = new ByteArrayInputStream(this.certData);
return certBis;
}
@Override
public int getHttpConnectTimeoutMs() {
return 8000;
}
@Override
public int getHttpReadTimeoutMs() {
return 10000;
}
}
2.2 创建md5加签模块
import com.github.wxpay.sdk.WXPayConstants;
import java.io.UnsupportedEncodingException;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;
import java.util.Map;
import java.util.Set;
/**
* @author ???
* @version 1.0
* MD5加签吗模块
*/
public class WxMD5Util {
public String getSign(Map<String, String> data) throws Exception {
WXConfigUtil config = new WXConfigUtil();
Set<String> keySet = data.keySet();
String[] keyArray = keySet.toArray(new String[keySet.size()]);
Arrays.sort(keyArray);
StringBuilder sb = new StringBuilder();
for (String k : keyArray) {
if (k.equals(WXPayConstants.FIELD_SIGN)) {
continue;
}
if (data.get(k).trim().length() > 0) // 参数值为空,则不参与签名
sb.append(k).append("=").append(data.get(k).trim()).append("&");
}
sb.append("key=").append(config.getKey());
MessageDigest md = null;
try {
md = MessageDigest.getInstance("MD5");
} catch (NoSuchAlgorithmException e) {
e.printStackTrace();
}
byte[] array = new byte[0];
try {
array = md.digest(sb.toString().getBytes("UTF-8"));
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
StringBuilder sb2 = new StringBuilder();
for (byte item : array) {
sb2.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3));
}
return sb2.toString().toUpperCase();
}
}
2.3 创建一个微信map集合
import java.util.HashMap;
import java.util.Map;
/**
* @author nonpool
*/
public class WXMap {
public static Map map(Map<String, String> result,String attach) throws Exception {
if(result == null){
return null;
}
Map<String, String> map = new HashMap<String, String>();
WxMD5Util md5Util = new WxMD5Util();
//返回APP端的数据
//参加调起支付的签名字段有且只能是6个,分别为appid、partnerid、prepayid、package、noncestr和timestamp,而且都必须是小写
//参加调起支付的签名字段有且只能是6个,分别为appid、partnerid、prepayid、package、noncestr和timestamp,而且都必须是小写
//参加调起支付的签名字段有且只能是6个,分别为appid、partnerid、prepayid、package、noncestr和timestamp,而且都必须是小写
map.put("appid", result.get("appid"));
map.put("partnerid", result.get("mch_id"));
map.put("prepayid", result.get("prepay_id"));
map.put("package", "Sign=WXPay");
map.put("noncestr", result.get("nonce_str"));
map.put("timestamp", String.valueOf(System.currentTimeMillis() / 1000));//单位为秒
// 这里不要使用请求预支付订单时返回的签名
// 这里不要使用请求预支付订单时返回的签名
// 这里不要使用请求预支付订单时返回的签名
map.put("sign", md5Util.getSign(map));
map.put("extdata", attach);
//使用返回的签名
// map.put("sign",result.get("sign"));
return map;
}
}
2.4 创建微信service
import java.util.Map;
/**
* @author ???
* @version 1.0
* service层
*/
public interface WXservice {
Map<String, String> dounifiedOrder(Order order) throws Exception;
String payBack(String notifyData , OrderssServiceImpl orderssService);
}
2.5 调动官方的SDK,获取预支付订单等参数
import com.baomidou.mybatisplus.mapper.EntityWrapper;
import com.github.wxpay.sdk.WXPay;
import com.github.wxpay.sdk.WXPayUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Service;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* @author nonpool
* @version 1.0
*/
@Service
public class WXserviceImpl implements WXservice {
private static final Logger logger = LoggerFactory.getLogger("MainLogger");
public static final String SPBILL_CREATE_IP = "";
public static final String NOTIFY_URL = "微信异步返回通地址";
public static final String TRADE_TYPE_APP = "APP";
/**
* 调动官方的SDK,获取预支付订单等参数
* @param total_fee
* @param attach
* @return
* @throws Exception
*/
public Map<String, String> dounifiedOrder(Order order) throws Exception {
WxMD5Util md5Util = new WxMD5Util();
Map<String, String> returnMap = new HashMap<String, String>();
WXConfigUtil config = new WXConfigUtil();
WXPay wxpay = new WXPay(config);
//生成订单数据的集合
Map<String, String> data = new HashMap<String, String>();
//生成商户订单号,不可重复
String out_trade_no = "wxpay" + order.getOut_trade_no();
data.put("appid", config.getAppID());
data.put("mch_id", config.getMchID());
data.put("nonce_str", WXPayUtil.generateNonceStr());
data.put("body", order.getSubject());
//商户订单编号
data.put("out_trade_no", out_trade_no);
//订单金额
data.put("total_fee", order.getTotal_fee());
//自己的服务器IP地址
data.put("spbill_create_ip", SPBILL_CREATE_IP);
//异步通知地址(请注意必须是外网)
data.put("notify_url", NOTIFY_URL);
//交易类型
data.put("trade_type", TRADE_TYPE_APP);
//附加数据,在查询API和支付通知中原样返回,该字段主要用于商户携带订单的自定义数据
data.put("attach", order.getAttach());
String sign1 = md5Util.getSign(data);
System.out.println(sign1);
data.put("sign", sign1);
try {
//使用官方API请求预付订单
Map<String, String> response = wxpay.unifiedOrder(data);
System.out.println(response);
String returnCode = response.get("return_code"); //获取返回码
//若返回码为SUCCESS,则会返回一个result_code,再对该result_code进行判断
if (returnCode.equals("SUCCESS")) {//主要返回以下5个参数
String resultCode = response.get("result_code");
returnMap.put("appid", response.get("appid"));
returnMap.put("mch_id", response.get("mch_id"));
returnMap.put("nonce_str", response.get("nonce_str"));
returnMap.put("sign", response.get("sign"));
if ("SUCCESS".equals(resultCode)) {//resultCode 为SUCCESS,才会返回prepay_id和trade_type
//获取预支付交易回话标志
returnMap.put("trade_type", response.get("trade_type"));
returnMap.put("prepay_id", response.get("prepay_id"));
return returnMap;
} else {
//此时返回没有预付订单的数据
return returnMap;
}
} else {
return returnMap;
}
} catch (Exception e) {
System.out.println(e);
//系统等其他错误的时候
}
return returnMap;
}
/**
* 异步通知后的Xml数据
* @param notifyData
* @return
*/
public String payBack(String notifyData , OrderssServiceImpl orderssService) {
WXConfigUtil config = null;
try {
config = new WXConfigUtil();
} catch (Exception e) {
e.printStackTrace();
}
WXPay wxpay = new WXPay(config);
String xmlBack = "";
Map<String, String> notifyMap = null;
try {
notifyMap = WXPayUtil.xmlToMap(notifyData); // 调用官方SDK转换成map类型数据
if (wxpay.isPayResultNotifySignatureValid(notifyMap)) {//验证签名是否有效,有效则进一步处理
String return_code = notifyMap.get("return_code");//状态
String out_trade_no = notifyMap.get("out_trade_no");//商户订单号
if (return_code.equals("SUCCESS")) {
if (out_trade_no != null) {
// 注意特殊情况:订单已经退款,但收到了支付结果成功的通知,不应把商户的订单状态从退款改成支付成功
// 注意特殊情况:微信服务端同样的通知可能会多次发送给商户系统,所以数据持久化之前需要检查是否已经处理过了,处理了直接返回成功标志
//业务数据持久化
//查询订单状态
String substring = out_trade_no.substring(5);
EntityWrapper<Orderss> orderssEntityWrapper = new EntityWrapper<>();
orderssEntityWrapper.where("order_name={0}",substring);
List<Orderss> ordersses = orderssService.selectList(orderssEntityWrapper);
if(ordersses.size() > 0){
Orderss orderss = new Orderss();
orderss.setOrderStatus(1);
EntityWrapper<Orderss> orderssEntityWrapper1 = new EntityWrapper<>();
orderssEntityWrapper1.where("order_id={0}",ordersses.get(0).getOrderId());
orderssService.update(orderss, orderssEntityWrapper1);
}
System.err.println("支付成功");
logger.info("微信手机支付回调成功订单号:{}", out_trade_no);
xmlBack = "<xml>" + "<return_code><![CDATA[SUCCESS]]></return_code>" + "<return_msg><![CDATA[OK]]></return_msg>" + "</xml> ";
} else {
logger.info("微信手机支付回调失败订单号:{}", out_trade_no);
xmlBack = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>" + "<return_msg><![CDATA[报文为空]]></return_msg>" + "</xml> ";
}
}
return xmlBack;
} else {
// 签名错误,如果数据里没有sign字段,也认为是签名错误
//失败的数据要不要存储?
logger.error("手机支付回调通知签名错误");
xmlBack = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>" + "<return_msg><![CDATA[报文为空]]></return_msg>" + "</xml> ";
return xmlBack;
}
} catch (Exception e) {
logger.error("手机支付回调通知失败", e);
xmlBack = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>" + "<return_msg><![CDATA[报文为空]]></return_msg>" + "</xml> ";
}
return xmlBack;
}
}
2.6 微信Controller
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.Map;
/**
* @author ???
* @version 1.0
* 微信支付功能接口模块
*/
@RestController
@RequestMapping("/weixin")
public class WXConttoller {
@Autowired
private WXserviceImpl wXservice;
@Autowired
private OrderssServiceImpl orderssService;
/**
*
* 统一下单
* 官方文档:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_1
* @param user_id
* @param total_fee
* @return
* @throws Exception
*/
public Map wxPay(Order order , WXserviceImpl wXservice) throws Exception {
//请求预支付订单
Map<String, String> result = wXservice.dounifiedOrder(order);
Map map = WXMap.map(result, order.getAttach());
return map;
}
/**
* 支付异步结果通知,我们在请求预支付订单时传入的地址
* 官方文档 :https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_7&index=3
*/
@RequestMapping(value = "/notify.json", method = {RequestMethod.GET, RequestMethod.POST})
public String wxPayNotify(HttpServletRequest request, HttpServletResponse response) {
String resXml = "";
try {
InputStream inputStream = request.getInputStream();
//将InputStream转换成xmlString
BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
StringBuilder sb = new StringBuilder();
String line = null;
try {
while ((line = reader.readLine()) != null) {
sb.append(line + "\n");
}
} catch (IOException e) {
System.out.println(e.getMessage());
} finally {
try {
inputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
resXml = sb.toString();
String result = wXservice.payBack(resXml,orderssService);
return result;
} catch (Exception e) {
System.out.println("微信手机支付失败:" + e.getMessage());
String result = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>" + "<return_msg><![CDATA[报文为空]]></return_msg>" + "</xml> ";
return result;
}
}
}
!!!!!!!代码结束。。。感谢观看