1、將支付寶Demo中的相關(guān)文件復(fù)制到我們的項目中:
首先疼蛾,我們將Demo中src中的包及里面的文件復(fù)制到我們項目中
image.png
同樣肛跌,我們也要講
zhifubao.properties
這個配置文件方法我們項目中的 resources
目錄下:
image.png
加下來就是我們的
jar
包了~首先我們先在
webapp
目錄下面新建lib
文件夾:然后將Demo中如圖所示的四個jar
包復(fù)制到lib
目錄下。image.png
至于剩下的jar
我們只用Maven
配置即可
<!-- alipay -->
<dependencies>
<dependency>
<groupId>commons-codec</groupId>
<artifactId>commons-codec</artifactId>
<version>1.10</version>
</dependency>
<dependency>
<groupId>commons-configuration</groupId>
<artifactId>commons-configuration</artifactId>
<version>1.10</version>
</dependency>
<dependency>
<groupId>commons-lang</groupId>
<artifactId>commons-lang</artifactId>
<version>2.6</version>
</dependency>
<dependency>
<groupId>commons-logging</groupId>
<artifactId>commons-logging</artifactId>
<version>1.1.1</version>
</dependency>
<dependency>
<groupId>com.google.zxing</groupId>
<artifactId>core</artifactId>
<version>2.1</version>
</dependency>
<dependency>
<groupId>com.google.code.gson</groupId>
<artifactId>gson</artifactId>
<version>2.3.1</version>
</dependency>
<dependency>
<groupId>org.hamcrest</groupId>
<artifactId>hamcrest-core</artifactId>
<version>1.3</version>
</dependency>
</dependencies>
上面是jar
包的配置,接下來我們在pom.xml
里面配置一個插件衍慎,使項目編譯需要的jar
在項目部署的時候自動配置到上面的lib
文件夾下:
<build>
<!-- geelynote maven的核心插件之-complier插件默認只支持編譯Java 1.4转唉,因此需要加上支持高版本jre的配置,在pom.xml里面加上 增加編譯插件 -->
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<configuration>
<source>1.8</source>
<target>1.8</target>
<encoding>UTF-8</encoding>
<compilerArguments>
<extdirs>${project.basedir}/src/main/webapp/WEB-INF/lib</extdirs>
</compilerArguments>
</configuration>
</plugin>
</plugins>
</build>
繼而我們運行一下復(fù)制過來的Main.class
稳捆,發(fā)現(xiàn)Demo是可以跑起來的~
image.png
2赠法、編寫我們的支付接口
1、在portal
下新建OrderController
類:
image.png
在
OrderController
類上添加這兩行注解:并且添加后面要用到的日志配置乔夯。
@Controller
@RequestMapping("/order/")
public class OrderController {
private static final Logger logger = LoggerFactory.getLogger(OrderController.class);
}
*Controller
:
//支付寶接口
@RequestMapping("pay.do")
@ResponseBody
public ServerResponse pay(HttpSession session, Long orderNo, HttpServletRequest request) {
User user = (User) session.getAttribute(Const.CURRENT_USER);
if (user == null) {
return ServerResponse.createByErrorCodeMessage(ResponseCode.NEED_LOGIN.getCode(), ResponseCode.NEED_LOGIN.getDesc());
}
//這是一個文件目錄砖织,用戶存放支付寶中生成的二維碼存放目錄,對應(yīng)的目錄路徑是tomcat 服務(wù)器目錄
String path = request.getSession().getServletContext().getRealPath("upload");
return iOrderService.pay(orderNo, user.getId(), path);
}
在service
包下新建IOrderService
接口類
*Service
:
public interface IOrderService {
//支付接口的實現(xiàn)
ServerResponse pay(Long orderNo, Integer userId, String path);
}
同理在Impl
包下新建對應(yīng)的實現(xiàn)類:
*ServiceImpl
:
//支付接口的實現(xiàn)
public ServerResponse pay(Long orderNo,Integer userId,String path){
Map<String, String> resultMap= Maps.newHashMap();
Order order = orderMapper.selectByUserIdAndOrderNo(userId,orderNo);
if(order == null){
return ServerResponse.createByErrorMessage("用戶沒有該訂單");
}
resultMap.put("orderNo",String.valueOf(order.getOrderNo()));
// (必填) 商戶網(wǎng)站訂單系統(tǒng)中唯一訂單號末荐,64個字符以內(nèi)侧纯,只能包含字母、數(shù)字甲脏、下劃線茂蚓,
// 需保證商戶系統(tǒng)端不能重復(fù),建議通過數(shù)據(jù)庫sequence生成剃幌,
String outTradeNo = order.getOrderNo().toString();
// (必填) 訂單標題聋涨,粗略描述用戶的支付目的。如“xxx品牌xxx門店當面付掃碼消費”
String subject = new StringBuilder().append("happymmall掃碼支付").append(outTradeNo).toString();
// (必填) 訂單總金額负乡,單位為元牍白,不能超過1億元
// 如果同時傳入了【打折金額】,【不可打折金額】,【訂單總金額】三者,則必須滿足如下條件:【訂單總金額】=【打折金額】+【不可打折金額】
String totalAmount = order.getPayment().toString();
// (可選) 訂單不可打折金額,可以配合商家平臺配置折扣活動抖棘,如果酒水不參與打折茂腥,則將對應(yīng)金額填寫至此字段
// 如果該值未傳入,但傳入了【訂單總金額】,【打折金額】,則該值默認為【訂單總金額】-【打折金額】
String undiscountableAmount = "0";
// 賣家支付寶賬號ID,用于支持一個簽約賬號下支持打款到不同的收款賬號切省,(打款到sellerId對應(yīng)的支付寶賬號)
// 如果該字段為空最岗,則默認為與支付寶簽約的商戶的PID,也就是appid對應(yīng)的PID
String sellerId = "";
// 訂單描述朝捆,可以對交易或商品進行一個詳細地描述般渡,比如填寫"購買商品2件共15.00元"
String body = new StringBuilder().append("訂單").append(outTradeNo).append("購買商品共").append(totalAmount).append("元").toString();
// 商戶操作員編號,添加此參數(shù)可以為商戶操作員做銷售統(tǒng)計
String operatorId = "test_operator_id";
// (必填) 商戶門店編號芙盘,通過門店號和商家后臺可以配置精準到門店的折扣信息驯用,詳詢支付寶技術(shù)支持
String storeId = "test_store_id";
// 業(yè)務(wù)擴展參數(shù),目前可添加由支付寶分配的系統(tǒng)商編號(通過setSysServiceProviderId方法)儒老,詳情請咨詢支付寶技術(shù)支持
ExtendParams extendParams = new ExtendParams();
extendParams.setSysServiceProviderId("2088100200300400500");
// 支付超時蝴乔,定義為120分鐘
String timeoutExpress = "120m";
// 商品明細列表,需填寫購買商品詳細信息驮樊,
List<GoodsDetail> goodsDetailList = new ArrayList<GoodsDetail>();
List<OrderItem> orderItemList= orderItemMapper.getByOrderNoUserId(orderNo,userId);
for (OrderItem orderItem : orderItemList){
// 創(chuàng)建一個商品信息薇正,參數(shù)含義分別為商品id(使用國標)片酝、名稱、單價(單位為分)挖腰、數(shù)量钠怯,如果需要添加商品類別,詳見GoodsDetail
GoodsDetail goods = GoodsDetail.newInstance(orderItem.getProductId().toString(), orderItem.getProductName(),
BigDecimalUtil.mul(orderItem.getCurrentUnitPrice().doubleValue(),
new Double(100).doubleValue()).longValue(), orderItem.getQuantity());
// 創(chuàng)建好一個商品后添加至商品明細列表
goodsDetailList.add(goods);
}
// // 創(chuàng)建一個商品信息曙聂,參數(shù)含義分別為商品id(使用國標)、名稱鞠鲜、單價(單位為分)宁脊、數(shù)量,如果需要添加商品類別贤姆,詳見GoodsDetail
// GoodsDetail goods1 = GoodsDetail.newInstance("goods_id001", "xxx小面包", 1000, 1);
// // 創(chuàng)建好一個商品后添加至商品明細列表
// goodsDetailList.add(goods1);
// 繼續(xù)創(chuàng)建并添加第一條商品信息榆苞,用戶購買的產(chǎn)品為“黑人牙刷”,單價為5.00元霞捡,購買了兩件
// GoodsDetail goods2 = GoodsDetail.newInstance("goods_id002", "xxx牙刷", 500, 2);
// goodsDetailList.add(goods2);
// 創(chuàng)建掃碼支付請求builder坐漏,設(shè)置請求參數(shù)
AlipayTradePrecreateRequestBuilder builder = new AlipayTradePrecreateRequestBuilder()
.setSubject(subject).setTotalAmount(totalAmount).setOutTradeNo(outTradeNo)
.setUndiscountableAmount(undiscountableAmount).setSellerId(sellerId).setBody(body)
.setOperatorId(operatorId).setStoreId(storeId).setExtendParams(extendParams)
.setTimeoutExpress(timeoutExpress)
.setNotifyUrl(PropertiesUtil.getProperty("alipay.callback.url"))//支付寶服務(wù)器主動通知商戶服務(wù)器里指定的頁面http路徑,根據(jù)需要設(shè)置
//.setNotifyUrl("http://kxkc7h.natappfree.cc/order/alipay_callback.do")
.setGoodsDetailList(goodsDetailList);
/** 一定要在創(chuàng)建AlipayTradeService之前調(diào)用Configs.init()設(shè)置默認參數(shù)
* Configs會讀取classpath下的zfbinfo.properties文件配置信息,如果找不到該文件則確認該文件是否在classpath目錄
*/
Configs.init("zfbinfo.properties");
/** 使用Configs提供的默認參數(shù)
* AlipayTradeService可以使用單例或者為靜態(tài)成員對象碧信,不需要反復(fù)new
*/
AlipayTradeService tradeService = new AlipayTradeServiceImpl.ClientBuilder().build();
AlipayF2FPrecreateResult result = tradeService.tradePrecreate(builder);
switch (result.getTradeStatus()) {
case SUCCESS:
logger.info("支付寶預(yù)下單成功: )");
AlipayTradePrecreateResponse response = result.getResponse();
dumpResponse(response);
//檢查是否存在上傳文件夾赊琳,如果不存在就賦予創(chuàng)建文件夾權(quán)限
File folder = new File(path);
if(!folder.exists()){
folder.setWritable(true);
folder.mkdirs();
}
// 需要修改為運行機器上的路徑 C:\ /Users/sudo/Desktop
String qrpath = String.format(path+"/qr-%s.png",response.getOutTradeNo());
String qrFileName = String.format("/qr-%s.png",response.getOutTradeNo());
ZxingUtils.getQRCodeImge(response.getQrCode(),256,qrpath);
File targetFile = new File(path,qrFileName);
try {
//將生成的二維碼上傳到FTP服務(wù)器上
FTPUtil.uploadFile(Lists.newArrayList(targetFile));
} catch (IOException e) {
logger.info("上傳二維碼異常",e);
}
logger.info("qrpath:" + qrpath);
String qrUrl = PropertiesUtil.getProperty("ftp.server.http.prefix")+targetFile.getName();
resultMap.put("qrUrl",qrUrl);
return ServerResponse.createBySuccess(resultMap);
case FAILED:
logger.error("支付寶預(yù)下單失敗!!!");
return ServerResponse.createByErrorMessage("支付寶預(yù)下單失敗!!!");
case UNKNOWN:
logger.error("系統(tǒng)異常,預(yù)下單狀態(tài)未知!!!");
return ServerResponse.createByErrorMessage("系統(tǒng)異常砰碴,預(yù)下單狀態(tài)未知!!!");
default:
logger.error("不支持的交易狀態(tài)躏筏,交易返回異常!!!");
return ServerResponse.createByErrorMessage("不支持的交易狀態(tài),交易返回異常!!!");
}
}
// 簡單打印應(yīng)答
private void dumpResponse(AlipayResponse response) {
if (response != null) {
logger.info(String.format("code:%s, msg:%s", response.getCode(), response.getMsg()));
if (StringUtils.isNotEmpty(response.getSubCode())) {
logger.info(String.format("subCode:%s, subMsg:%s", response.getSubCode(),
response.getSubMsg()));
}
logger.info("body:" + response.getBody());
}
}
我們會發(fā)現(xiàn)呈枉,上面的大多數(shù)是直接從支付寶Demo中的Main
類中復(fù)制過來的趁尼,
我們只是更改了對應(yīng)的參數(shù)
上面第四行代碼我們用到了自己封裝了一個根據(jù)訂單號和用戶Id來查詢訂單的方法:
*Mapper
:
//根據(jù)訂單號和用戶的Id來查詢訂單
Order selectByUserIdAndOrderNo(@Param("userId") Integer userId,@Param("orderNo") Long orderNo);
*Mapper.xml
<select id="selectByUserIdAndOrderNo" parameterType="map" resultMap="BaseResultMap">
select
<include refid="Base_Column_List"/>
from mmall_order
where order_no = #{orderNo}
and user_id = #{userId}
</select>
2、支付接口測試:
image.png
用沙箱版支付寶掃描支付我們生成的二維碼猖辫,發(fā)現(xiàn)可以啦~
image.png