import com.capinfo.crypt.*;
創(chuàng)新互聯(lián)建站-專業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價(jià)比永康網(wǎng)站開發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫,直接使用。一站式永康網(wǎng)站制作公司更省心,省錢,快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋永康地區(qū)。費(fèi)用合理售后完善,十多年實(shí)體公司更值得信賴。
public class RsaTest
{
public RsaTest(){
}
public static void main(String[] args){
//首信公鑰文件
String publicKey = "d:/Public1024.key";
//簽名信息
String SignString = "3e5671bc4f91c3d055b18c1e5e22dd9db157380c7ee8facf0b1117082fbf398d7113c2df7e3219fc28dd88dd26cb096cabe607f3e397dfc2dcdb3349351a5f025ea0761da6e39e2d2fd311294a6076e777fe2ab8911f22113c435b89d63ae4f2aff2f333f7ebd40cc89601d58fb37b16596b5c94eb8b64cd52e12b9679248e6a";
//原信息
String strSource = "abcde12345";
try{
//公鑰驗(yàn)證
RSA_MD5 rsaMD5 = new RSA_MD5();
int k = rsaMD5.PublicVerifyMD5(publicKey,SignString,strSource);
if(k==0)
System.out.println("驗(yàn)證成功.");
else
System.out.println("驗(yàn)證失敗.");
}catch(Exception e){
System.out.println("驗(yàn)證異常.\n"+e);
}
}
}
1、如果你要做支付功能,技術(shù)上很簡單,支付寶有提供接口,改兩個(gè)參數(shù)就可以用了,非常簡單。2、但是你得網(wǎng)站如果要上線盈利的話,就麻煩了。3、首先你的支付寶收款的賬號(hào)要進(jìn)行商家認(rèn)證,具體怎么認(rèn)證我就不知道了,當(dāng)時(shí)是我老板認(rèn)證的,我就知道這個(gè)商家實(shí)名認(rèn)證比個(gè)人實(shí)名認(rèn)證麻煩的多。認(rèn)證之后支付寶會(huì)給你兩個(gè)參數(shù),一個(gè)是合作身份者ID,以2088開頭由16位純數(shù)字組成的字符串,一個(gè)是商戶的私鑰。你拿到這兩個(gè)參數(shù)之后去下載支付寶提供的接口程序,把這兩個(gè)參數(shù)填進(jìn)去就好了,別的都不用動(dòng)。當(dāng)然訂單價(jià)格和訂單號(hào)的生成規(guī)則你可以自己寫??傊Ц秾毜慕涌陂_發(fā)的非常完善,而且傻瓜式操作,一看就懂。4、然后你的網(wǎng)站域名要備案,申請(qǐng)ICP證,經(jīng)營性ICP(為用戶提供有償?shù)男畔⒎?wù))需要你注冊(cè)公司的,而且營業(yè)執(zhí)照很麻煩,我老板現(xiàn)在還在。如果你只是想加個(gè)功能,看看接口怎么用,那簡單。但是你要做正規(guī)的網(wǎng)站用接口盈利的話,技術(shù)上實(shí)現(xiàn)很簡單,把程序包下下來,然后改兩個(gè)參數(shù),但是合法經(jīng)營的話需要的手續(xù)很多。
JAR(Java ARchive,Java 歸檔)、安裝java軟件就可以了
-------------
JAR(Java ARchive,Java 歸檔)是一種與平臺(tái)無關(guān)的文件格式,可將多個(gè)文件合成一個(gè)文件。用戶可將多個(gè) Java applet 及其所需組件(.class 文件、圖像和聲音)綁定到 JAR 文件中,而后作為單個(gè)的簡單 HTTP(Hypertext Tranfer Protocal,超文本傳輸協(xié)議)事務(wù)下載到瀏覽器中,從而大大提高下載速度。JAR 格式也支持壓縮,從而減小了文件的大小,進(jìn)一步縮短下載時(shí)間。另外,applet 編寫者也可在 JAR 文件中用數(shù)字簽名的方式簽寫各項(xiàng)以確認(rèn)其來源。它用 Java 編寫,可與現(xiàn)有的 applet 代碼完全向后兼容且可充分?jǐn)U展。
本版本的 Java 平臺(tái)擴(kuò)展了 JAR 格式的用途。增強(qiáng)功能包括增加命令行 JAR 工具的功能,可用于創(chuàng)建和更新已簽名的 JAR 文件。同時(shí),它也提供了讀取和寫入 JAR 文件的新的標(biāo)準(zhǔn) API(Application Programming Interface,應(yīng)用程序接口)。另外,Java 擴(kuò)展構(gòu)架 (Extensions Framework) 也提供了一種機(jī)制,用來處理擴(kuò)展相關(guān)性及打包為 JAR 文件的其它第三方庫。
JAR 功能
JAR 指南。
清單和簽名規(guī)范
JAR 文件和擴(kuò)展構(gòu)架
Java 平臺(tái)中的新擴(kuò)展機(jī)制用 JAR 文件格式打包擴(kuò)展類。為了支持?jǐn)U展機(jī)制和相關(guān)功能(如包的密封和版本演變),我們提供了新的清單屬性。有關(guān)詳細(xì)信息,參見擴(kuò)展規(guī)范。
增強(qiáng) Jar 工具
Jar 工具提供了新的 -C 和 u 選項(xiàng),可用于創(chuàng)建和更新 JAR 文件。參見 Jar 工具參考頁:
Jar 工具參考頁 (for Solaris)
Jar 工具參考頁 (for Windows)
讀取和寫入 JAR 文件:API 規(guī)范
包 java.util.jar:用來創(chuàng)建和讀取 JAR 文件的類
類 java.net.JarURLConnection:利用 jar 協(xié)議實(shí)現(xiàn)到 JAR 文件的 URL(Uniform Resource Locators,統(tǒng)一資源定位符)連接的抽象類。
教程
Java Software 站點(diǎn):
Java 教程中的 Java 歸檔 (JAR) 文件格式。
微信支付系列文章
微信支付-java后端實(shí)現(xiàn)
微信支付-vue 前端實(shí)現(xiàn)
java demo: 下載地址文章底部
技術(shù)棧
Spring boot
java
XML (微信在http協(xié)議中數(shù)據(jù)傳輸方案)
MD5 簽名
微信支付術(shù)語
openid (OpenID是公眾號(hào)一對(duì)一對(duì)應(yīng)用戶身份的標(biāo)識(shí))
app_id (公眾號(hào)id,登錄微信公眾號(hào)–開發(fā)–基本配置中獲得;)
key (收款商戶后臺(tái)進(jìn)行配置,登錄微信商戶平臺(tái)–賬戶中心–API安全-設(shè)置秘鑰,設(shè)置32位key值;)
mch_id (收款商家商戶號(hào);)
certPath (API證書, 登錄微信商戶平臺(tái)–賬戶中心-API安全-下載證書)
后端流程
服務(wù)端需要的核心操作, 總共分為以下幾步:
統(tǒng)一下單
前端調(diào)起微信支付必要參數(shù) (需加密)
訂單結(jié)果主動(dòng)通知 (回調(diào)接口)
查詢訂單結(jié)果
結(jié)束訂單支付接口(關(guān)閉訂單,支付訂單關(guān)閉)
代碼
微信總共支持多種語言的sdk, 在官網(wǎng)可以下載例子, java程序也可以引入微信支付的sdk包, 但是github上的sdk已經(jīng)很久沒有更新了, 最好的選擇, 也是我的選擇, 在官網(wǎng)上下載sdk項(xiàng)目, 將其中所有java類copy到自己的項(xiàng)目中.
官網(wǎng)sdk下載目錄
鏈接: 商戶平臺(tái)首頁
#### 根據(jù)微信sdk生成配置類 WXPayConfig
創(chuàng)建IWxPayConfig.class, 繼承sdk WXPayConfig.class, 實(shí)現(xiàn)sdk中部分抽象方法, 讀取本地證書, 加載到配置類中.
package core.com.chidori.wxpay;
import core.com.wxpay.IWXPayDomain;
import core.com.wxpay.WXPayConfig;
import core.com.wxpay.WXPayConstants;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
@Service
public class IWxPayConfig extends WXPayConfig { // 繼承sdk WXPayConfig 實(shí)現(xiàn)sdk中部分抽象方法
private byte[] certData;
@Value("${vendor.wx.config.app_id}")
private String app_id;
@Value("${vendor.wx.pay.key}")
private String wx_pay_key;
@Value("${vendor.wx.pay.mch_id}")
private String wx_pay_mch_id;
public IWxPayConfig() throws Exception { // 構(gòu)造方法讀取證書, 通過getCertStream 可以使sdk獲取到證書
String certPath = "/data/config/chidori/apiclient_cert.p12";
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 wx_pay_mch_id;
}
@Override
public String getKey() {
return wx_pay_key;
}
@Override
public InputStream getCertStream() {
return new ByteArrayInputStream(this.certData);
}
@Override
public IWXPayDomain getWXPayDomain() { // 這個(gè)方法需要這樣實(shí)現(xiàn), 否則無法正常初始化WXPay
IWXPayDomain iwxPayDomain = new IWXPayDomain() {
@Override
public void report(String domain, long elapsedTimeMillis, Exception ex) {
}
@Override
public DomainInfo getDomain(WXPayConfig config) {
return new IWXPayDomain.DomainInfo(WXPayConstants.DOMAIN_API, true);
}
};
return iwxPayDomain;
}
}
發(fā)起統(tǒng)一下單 AND 前端調(diào)起微信支付必要參數(shù)
// 發(fā)起微信支付
WXPay wxpay = null;
Map result = new HashMap();
try {
// ******************************************
//
// 統(tǒng)一下單
//
// ******************************************
wxpay = new WXPay(iWxPayConfig); // *** 注入自己實(shí)現(xiàn)的微信配置類, 創(chuàng)建WXPay核心類, WXPay 包括統(tǒng)一下單接口
Map data = new HashMap ();
data.put("body", "訂單詳情");
data.put("out_trade_no", transOrder.getGlobalOrderId()); // 訂單唯一編號(hào), 不允許重復(fù)
data.put("total_fee", String.valueOf(transOrder.getOrderAmount().multiply(new BigDecimal(100)).intValue())); // 訂單金額, 單位分
data.put("spbill_create_ip", "192.168.31.166"); // 下單ip
data.put("openid", openId); // 微信公眾號(hào)統(tǒng)一標(biāo)示openid
data.put("notify_url", ""); // 訂單結(jié)果通知, 微信主動(dòng)回調(diào)此接口
data.put("trade_type", "JSAPI"); // 固定填寫
logger.info("發(fā)起微信支付下單接口, request={}", data);
Map response = wxpay.unifiedOrder(data); // 微信sdk集成方法, 統(tǒng)一下單接口unifiedOrder, 此處請(qǐng)求 MD5加密 加密方式
logger.info("微信支付下單成功, 返回值 response={}", response);
String returnCode = response.get("return_code");
if (!SUCCESS.equals(returnCode)) {
return null;
}
String resultCode = response.get("result_code");
if (!SUCCESS.equals(resultCode)) {
return null;
}
String prepay_id = response.get("prepay_id");
if (prepay_id == null) {
return null;
}
// ******************************************
//
// 前端調(diào)起微信支付必要參數(shù)
//
// ******************************************
String packages = "prepay_id=" + prepay_id;
Map wxPayMap = new HashMap ();
wxPayMap.put("appId", iWxPayConfig.getAppID());
wxPayMap.put("timeStamp", String.valueOf(Utility.getCurrentTimeStamp()));
wxPayMap.put("nonceStr", Utility.generateUUID());
wxPayMap.put("package", packages);
wxPayMap.put("signType", "MD5");
// 加密串中包括 appId timeStamp nonceStr package signType 5個(gè)參數(shù), 通過sdk WXPayUtil類加密, 注意, 此處使用 MD5加密 方式
String sign = WXPayUtil.generateSignature(wxPayMap, iWxPayConfig.getKey());
// ******************************************
//
// 返回給前端調(diào)起微信支付的必要參數(shù)
//
// ******************************************
result.put("prepay_id", prepay_id);
result.put("sign", sign);
result.putAll(wxPayMap);
return result;
} catch (Exception e) {
}
回調(diào)結(jié)果處理
核心是支付訂單回調(diào)時(shí), 需校驗(yàn)加密簽名是否匹配, 防止出現(xiàn)模擬成功通知
@RequestMapping(value = "/payCallback", method = RequestMethod.POST)
public String payCallback(HttpServletRequest request, HttpServletResponse response) {
logger.info("進(jìn)入微信支付異步通知");
String resXml="";
try{
//
InputStream is = request.getInputStream();
//將InputStream轉(zhuǎn)換成String
BufferedReader reader = new BufferedReader(new InputStreamReader(is));
StringBuilder sb = new StringBuilder();
String line = null;
try {
while ((line = reader.readLine()) != null) {
sb.append(line + " ");
}
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
is.close();
} catch (IOException e) {
e.printStackTrace();
}
}
resXml=sb.toString();
logger.info("微信支付異步通知請(qǐng)求包: {}", resXml);
return wxTicketService.payBack(resXml);
}catch (Exception e){
logger.error("微信支付回調(diào)通知失敗",e);
String result = " ";
return result;
}
}
@Override
public String payBack(String notifyData) {
logger.info("payBack() start, notifyData={}", notifyData);
String xmlBack="";
Map notifyMap = null;
try {
WXPay wxpay = new WXPay(iWxPayConfig);
notifyMap = WXPayUtil.xmlToMap(notifyData); // 轉(zhuǎn)換成map
if (wxpay.isPayResultNotifySignatureValid(notifyMap)) {
// 簽名正確
// 進(jìn)行處理。
// 注意特殊情況:訂單已經(jīng)退款,但收到了支付結(jié)果成功的通知,不應(yīng)把商戶側(cè)訂單狀態(tài)從退款改成支付成功
String return_code = notifyMap.get("return_code");//狀態(tài)
String out_trade_no = notifyMap.get("out_trade_no");//訂單號(hào)
if (out_trade_no == null) {
logger.info("微信支付回調(diào)失敗訂單號(hào): {}", notifyMap);
xmlBack = " ";
return xmlBack;
}
// 業(yè)務(wù)邏輯處理 ****************************
logger.info("微信支付回調(diào)成功訂單號(hào): {}", notifyMap);
xmlBack = " ";
return xmlBack;
} else {
logger.error("微信支付回調(diào)通知簽名錯(cuò)誤");
xmlBack = " ";
return xmlBack;
}
} catch (Exception e) {
logger.error("微信支付回調(diào)通知失敗",e);
xmlBack = " ";
}
return xmlBack;
}
統(tǒng)一下單的簽名和后續(xù)前端拉取微信支付的簽名需要統(tǒng)一, 也就是都采用MD5加密, 如果2者不同, 會(huì)導(dǎo)致前端拉取微信支付fail, 這是一個(gè)巨大的坑, 因?yàn)檫@個(gè)原因調(diào)試了好久, 微信在文檔里沒有明確標(biāo)出統(tǒng)一下單的簽名校驗(yàn)方式 需要和前端拉取微信支付的簽名校驗(yàn)保持一致.
微信sdk里的源碼需要針對(duì)這個(gè)問題調(diào)整一下, 調(diào)整如下:
WXPay類需要修改下加密判斷,在WXPay構(gòu)造方法中,調(diào)整如下
public WXPay(final WXPayConfig config, final String notifyUrl, final boolean autoReport, final boolean useSandbox) throws Exception {
this.config = config;
this.notifyUrl = notifyUrl;
this.autoReport = autoReport;
this.useSandbox = useSandbox;
if (useSandbox) {
this.signType = SignType.MD5; // 沙箱環(huán)境
}
else {
this.signType = SignType.MD5; // 將這里的加密方式修改為SignType.MD5, 保持跟前端吊起微信加密方式保持一致
}
this.wxPayRequest = new WXPayRequest(config);
}
結(jié)束語
做完以后, 微信支付的后端邏輯還是很清晰的, 但是在開發(fā)過程中很煎熬, 不清楚每個(gè)專業(yè)術(shù)語在微信哪里配置, 加密方式亂的很
支付平臺(tái)程序??+[Q]845567777
【程序】?API支付平臺(tái)程序?(適用?第三方?第四方?支付平臺(tái)搭建?等)
【語言】Java
【數(shù)據(jù)庫】mysql
【程序優(yōu)點(diǎn)】
JAVA語言編寫,安全、高效、快捷,通過本API系統(tǒng)程序可快速搭建自己的支付平臺(tái),有支付接口即可發(fā)展商戶。
國內(nèi)支付平臺(tái)程序首批Java程序國內(nèi)外大型企業(yè)、銀行、第三方企業(yè)級(jí)的站?等都用JAVA開發(fā)
JAVA安全性?穩(wěn)定性?極高,跨平臺(tái)等優(yōu)點(diǎn)
其他語言asp?php?net等和JAVA比不是一個(gè)檔次
【更多功能】
★本平臺(tái)程序?接口?兼容?易寶支付接口。只要客戶網(wǎng)站有易寶接口,直接換掉?提交地址?和?ID密鑰?即可使用
★兼容點(diǎn)卡和網(wǎng)銀、支持直連接入、非直連接入、在線充值。直接接入就是在自己的網(wǎng)站上寫入銀行編碼、請(qǐng)求的時(shí)候直接到銀行非直連的意思就是不需要在自己的網(wǎng)站寫網(wǎng)銀編碼、在系統(tǒng)上提交到銀行
★特有功能:扣單、恢復(fù)訂單、銀行拆分、獨(dú)立支付寶、獨(dú)立網(wǎng)銀?等后臺(tái)訂單明細(xì)特別清晰?后臺(tái)明細(xì)多種方式??并非那些小平臺(tái)可比