java后台提供ios微信支付接口
提供微信支付接口,需要先在微信開放平臺(https://open.weixin.qq.com/)申請賬號,進行開發者資格認證通過后創建應用。
創建成功后會有對應應用的AppID和AppSecret這個ios需要。我們需要的是商戶平臺的賬號、初始密碼、對應應用的AppID等,在開發者資格認證成功后微信會發到綁定的郵箱上。收到郵箱后需要登錄到微信商戶平臺(https://pay.weixin.qq.com/index.php/core/home/login?return_url=%2F)修改初始密碼并進行設置api密鑰。
關于登錄微信商戶平臺,我在進行登錄時候火狐和谷歌經常不行,證書安裝不好,后來試用qq瀏覽器才能登上,但是退款時候下載證書又不行,后來用了uc才可以的。配置好后我們就可以進行編碼。首先看一下微信官方的文檔上app支付需要的參數。
ios要在app中調起微信支付,需要一下幾個參數:
這些參數由后臺傳送給ios,后臺需要先傳入以下必填參數進行第一次簽名:
注意:對于同一帳號的不同應用,每次申請均需填寫資料,均需簽署合同。
接下來就是代碼部分:微信的配置文件:
提供給ios接口代碼:
import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.PrintWriter; import java.io.UnsupportedEncodingException; import java.lang.reflect.Field; import java.math.BigDecimal; import java.math.RoundingMode; import java.security.Security; import java.text.DecimalFormat; import java.util.HashMap; import java.util.Map; import java.util.SortedMap; import java.util.TreeMap; import java.util.UUID;import javax.annotation.Resource; import javax.crypto.Cipher; import javax.crypto.SecretKey; import javax.crypto.spec.SecretKeySpec; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse;import org.apache.commons.codec.binary.Base64; import org.apache.commons.lang3.StringUtils; import org.apache.http.client.methods.CloseableHttpResponse; import org.apache.http.util.EntityUtils; import org.apache.struts2.convention.annotation.Action; import org.apache.struts2.convention.annotation.Namespace; import org.apache.struts2.convention.annotation.Result; import org.apache.struts2.convention.annotation.Results; import org.bouncycastle.jce.provider.BouncyCastleProvider;import com.alibaba.fastjson.JSONObject; import com.alibaba.fastjson.serializer.SerializerFeature;/*** 微信支付的接口* */ @Namespace("/out/weixinPay") @Results({ @Result(name = "500", location = "/WEB-INF/views/500.jsp") }) @SuppressWarnings("unused") public class WeixinPayAction extends OutInterfaceBasicAction<Consult> {private static final long serialVersionUID = 1L;private Map<String, Object> msg = new HashMap<String, Object>();/** @Resource private UnifiedOrderService unifiedOrderService;*/@Resourceprivate MemberService memberService;@Resourceprivate ALiPayService aLiPayService;private ALiPay aLiPay = new ALiPay();@Resourceprivate IMessageService iMessageService;@Resourceprivate RefundOrderService refundOrderService;private static DecimalFormat df = new DecimalFormat("0.00");// 構造簽名的mapprivate SortedMap<Object, Object> parameters = new TreeMap<Object, Object>();// 構造向微信發送參數的實體類private Unifiedorder unifiedorder = new Unifiedorder();// 微信的參數private WeixinConfigUtils config = new WeixinConfigUtils();public static final String ALGORITHM = "AES/ECB/PKCS7Padding"; private RefundOrder refundOrder = new RefundOrder();@Resourceprivate AESDecodeUtil aesDecodeUtil;@Resourceprivate OrderService orderService;/*** 微信充值*/@Action("weChatPayRecharge")public String weChatPayRecharge() {try {df.setRoundingMode(RoundingMode.HALF_UP);// 獲取用戶IDString memberId = etRequest().getParameter("memberId");String integer = getRequest().getParameter("integer");if (StringUtils.isNotBlank(memberId)&& StringUtils.isNotBlank(integer)) {Integer score = Integer.valueOf(integer);// 產生訂單號String outTradeNo = UuIdUtils.getUUID();// 參數組String appid = config.appid;String mch_id = config.mch_id;String nonce_str = RandCharsUtils.getRandomString(16);String body = "";body = "購買" + score + "積分,支付" + df.format(score / 100)+ "元";String detail = score / 100 + "元支付開始";String attach = "暫時無用(這個用于區分不同版本)";String out_trade_no = outTradeNo;int total_fee = score;// 單位是分,現在按照ios傳遞過來的參數進行String spbill_create_ip = "127.0.0.1";String time_start = RandCharsUtils.timeStart();String time_expire = RandCharsUtils.timeExpire();String notify_url = config.notify_url;String trade_type = "APP";// 參數:開始生成第一次簽名parameters.put("appid", appid);parameters.put("mch_id", mch_id);parameters.put("body", body);parameters.put("nonce_str", nonce_str);parameters.put("detail", detail);parameters.put("attach", attach);parameters.put("out_trade_no", out_trade_no);parameters.put("total_fee", total_fee);parameters.put("time_start", time_start);parameters.put("time_expire", time_expire);parameters.put("notify_url", notify_url);parameters.put("trade_type", trade_type);parameters.put("spbill_create_ip", spbill_create_ip);String sign = WXSignUtils.createSign("UTF-8", parameters);// 微信統一下單unifiedorder.setAppid(appid);unifiedorder.setMch_id(mch_id);unifiedorder.setNonce_str(nonce_str);unifiedorder.setSign(sign);unifiedorder.setBody(body);unifiedorder.setDetail(detail);unifiedorder.setAttach(attach);unifiedorder.setOut_trade_no(out_trade_no);unifiedorder.setTotal_fee(total_fee);unifiedorder.setSpbill_create_ip(spbill_create_ip);unifiedorder.setTime_start(time_start);unifiedorder.setTime_expire(time_expire);unifiedorder.setNotify_url(notify_url);unifiedorder.setTrade_type(trade_type);try {// 微信第二次簽名SortedMap<Object, Object> parameter2 = new TreeMap<Object, Object>();parameter2.put("appid", appid);parameter2.put("noncestr", nonce_str);parameter2.put("package", "Sign=WXPay");parameter2.put("partnerid", mch_id);String prepareId = HttpXmlUtils.getPrepareId(unifiedorder);parameter2.put("prepayid", prepareId);long timestamp = System.currentTimeMillis() / 1000;parameter2.put("timestamp", timestamp);String sign2 = WXSignUtils.createSign("UTF-8", parameter2);unifiedorder = null;// 訂單信息存入數據庫aLiPay.setApp_id(appid);aLiPay.setSubject(body);aLiPay.setScore(integer);aLiPay.setGmt_create(time_start);if (null != memberId && !"".equals(memberId)) {aLiPay.setMember(memberService.getMemberById(Integer.valueOf(memberId)));}aLiPay.setOut_trade_no(out_trade_no);aLiPay.setPayStatus(PayStatusEnum.WAITPAY.getComment());aLiPay.setPayType(PayTypeEnum.WEIXIN.getComment());aLiPay.setState(0);aLiPay.setTotal_amount(String.valueOf(total_fee / 100));aLiPay.setTrade_no(prepareId);aLiPay.setSign(sign);aLiPay.setTotalFee(total_fee);aLiPayService.addALiPay(aLiPay);// 傳遞給ios的參數msg.put("prepay_id", prepareId);msg.put("appid", appid);msg.put("mch_id", mch_id);msg.put("timestamp", timestamp);msg.put("nonce_str", nonce_str);msg.put("sign", sign2);msg.put("stateCode", MsgCode.SUCCESS);msg.put("message ", "請求成功并返回數據!");} catch (Exception e) {msg.put("stateCode", MsgCode.ERROR);msg.put("message ", "請求失敗");}} else {msg.put("stateCode", MsgCode.GET_PARAM_ERROR);msg.put("message ", "獲取參數失敗!");}Struts2Utils.renderJson(JSONObject.toJSONString(msg,SerializerFeature.DisableCircularReferenceDetect));return NONE;} catch (Exception e) {log.error(e.getMessage());e.printStackTrace();return "500";}}*//*** 微信回調url* * @throws IOException*/@Action("notifyUrl")public String notifyUrl() throws IOException {try {Map<String, String> resultMap = JdomParseXmlUtils.getWeixinResult(Struts2Utils.getRequest(), Struts2Utils.getResponse());aLiPay = aLiPayService.searchALiPayByOutTradeNo(resultMap.get("out_trade_no"));String resutlTotal = String.valueOf(Integer.valueOf(resultMap.get("total_fee")) / 100);if (aLiPay.getState() == 0) {if (aLiPay.getOut_trade_no().equals(resultMap.get("out_trade_no"))&& aLiPay.getTotal_amount().equals(resutlTotal)) {aLiPay.setGmt_payment(resultMap.get("time_end"));aLiPay.setPayStatus(PayStatusEnum.PAYOVER.getComment());aLiPay.setState(1);aLiPayService.updateALiPay(aLiPay);iMessageService.sendUserscoreMsg(new IntegrationContent(aLiPay.getMember().getMemberId(), Integer.valueOf(aLiPay.getScore()), "微信充值"+ aLiPay.getScore() + "積分", 1));unifiedorder.setReturn_code("SUCCESS");}} else {unifiedorder.setReturn_code("FAIL");}String refundXml = HttpXmlUtils.refundXml(unifiedorder);Struts2Utils.getResponse().getWriter().write(refundXml);Struts2Utils.getResponse().getWriter().flush();return null;} catch (Exception e) {log.error(e.getMessage());e.printStackTrace();return "500";}} }使用到的工具類代碼:
uuidUtis用于生成訂單號:
WeixinConfigUtils加載weixin.properties文件。
import java.io.IOException; import java.io.InputStream; import java.util.Properties; import java.util.ResourceBundle;import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory;/*** 微信的配置參數* @author * @date 2017/08/10*/ @SuppressWarnings("unused") public class WeixinConfigUtils {private static final Log log = LogFactory.getLog(WeixinConfigUtils.class);public static String appid;public static String mch_id;public static String notify_url;public static String order_notify_url;public static String doctor_notify_url;static {try{InputStream is = WeixinConfigUtils.class.getResourceAsStream("/weixin.properties");Properties properties = new Properties();properties.load(is);appid = properties.getProperty("weixin.appid");mch_id = properties.getProperty("weixin.mch_id");notify_url = properties.getProperty("weixin.notify_url");order_notify_url = properties.getProperty("weixin.order_notify_url");doctor_notify_url = properties.getProperty("weixin.doctor_notify_url");}catch(Exception ex){log.debug("加載配置文件:"+ex.getMessage());}} } import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; import java.util.Random;/*** nonce_str隨即字符串 * @author * @date 2017/08/10*/ public class RandCharsUtils {private static SimpleDateFormat df = new SimpleDateFormat("yyyyMMddHHmmss");public static String getRandomString(int length) { //length表示生成字符串的長度String base = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz"; Random random = new Random(); StringBuffer sb = new StringBuffer();int number = 0;for (int i = 0; i < length; i++) { number = random.nextInt(base.length()); sb.append(base.charAt(number)); } return sb.toString(); } /** 訂單開始交易的時間*/public static String timeStart(){return df.format(new Date());}/** 訂單開始交易的時間*/public static String timeExpire(){Calendar now=Calendar.getInstance();now.add(Calendar.MINUTE,30);return df.format(now.getTimeInMillis());}} import java.util.Iterator; import java.util.Map; import java.util.Set; import java.util.SortedMap;/*** 微信支付簽名* @author yangfuren* @date 2017/08/10*/ public class WXSignUtils {/*** 微信支付簽名算法sign* @param characterEncoding* @param parameters* @return*/@SuppressWarnings("rawtypes")public static String createSign(String characterEncoding,SortedMap<Object,Object> parameters){StringBuffer sb = new StringBuffer();Set es = parameters.entrySet();//所有參與傳參的參數按照accsii排序(升序)Iterator it = es.iterator();while(it.hasNext()) {Map.Entry entry = (Map.Entry)it.next();String k = (String)entry.getKey();Object v = entry.getValue();if(null != v && !"".equals(v) && !"sign".equals(k) && !"key".equals(k)) {sb.append(k + "=" + v + "&");}}sb.append("key=" + weixinConstant.KEY);String sign = MD5Util.MD5Encode(sb.toString(), characterEncoding).toUpperCase();return sign;} } import java.io.BufferedReader; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.io.OutputStream; import java.io.OutputStreamWriter; import java.io.StringReader; import java.net.HttpURLConnection; import java.net.MalformedURLException; import java.net.URL; import java.util.HashMap; import java.util.List; import java.util.Map;import javax.annotation.Resource; import javax.net.ssl.HttpsURLConnection;import org.jdom.JDOMException; import org.jdom.input.SAXBuilder; import org.xml.sax.InputSource;import com.thoughtworks.xstream.XStream; import com.thoughtworks.xstream.io.xml.DomDriver; import com.thoughtworks.xstream.io.xml.XmlFriendlyNameCoder; /*** post提交xml格式的參數* @author * @date 2017/08/10*/ public class HttpXmlUtils {@Resourceprivate RefundOrderService refundOrderService;public static XStream xStream = new XStream(new DomDriver("UTF-8",new XmlFriendlyNameCoder("-_", "_")));/*** 開始post提交參數到接口* 并接受返回* @param url* @param xml* @param method* @param contentType* @return*/public static String xmlHttpProxy(String url,String xml,String method,String contentType){InputStream is = null;OutputStreamWriter os = null;try {URL _url = new URL(url);HttpURLConnection conn = (HttpURLConnection) _url.openConnection();conn.setDoInput(true); conn.setDoOutput(true); conn.setRequestProperty("Content-type", "text/xml");conn.setRequestProperty("Pragma:", "no-cache"); conn.setRequestProperty("Cache-Control", "no-cache"); conn.setRequestMethod("POST");os = new OutputStreamWriter(conn.getOutputStream());os.write(new String(xml.getBytes(contentType)));os.flush();//返回值is = conn.getInputStream();return getContent(is, "utf-8");} catch (MalformedURLException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally{try {if(os!=null){os.close();}if(is!=null){is.close();}} catch (IOException e) {e.printStackTrace();}}return null;}/*** 解析返回的值* @param is* @param charset* @return*/public static String getContent(InputStream is, String charset) {String pageString = null;InputStreamReader isr = null;BufferedReader br = null;StringBuffer sb = null;try {isr = new InputStreamReader(is, charset);br = new BufferedReader(isr);sb = new StringBuffer();String line = null;while ((line = br.readLine()) != null) {sb.append(line + "\n");}pageString = sb.toString();} catch (Exception e) {e.printStackTrace();} finally {try {if (is != null){is.close();}if(isr!=null){isr.close();}if(br!=null){br.close();}} catch (IOException e) {e.printStackTrace();}sb = null;}return pageString;}/*** 解析申請退款之后微信返回的值并進行存庫操作* @throws IOException * @throws JDOMException */public static Map<String, String> parseRefundXml(String refundXml) throws JDOMException, IOException{ParseXMLUtils.jdomParseXml(refundXml);StringReader read = new StringReader(refundXml);// 創建新的輸入源SAX 解析器將使用 InputSource 對象來確定如何讀取 XML 輸入InputSource source = new InputSource(read);// 創建一個新的SAXBuilderSAXBuilder sb = new SAXBuilder();// 通過輸入源構造一個Documentorg.jdom.Document doc;doc = (org.jdom.Document) sb.build(source);org.jdom.Element root = doc.getRootElement();// 指向根節點List<org.jdom.Element> list = root.getChildren();Map<String, String> refundOrderMap = new HashMap<String, String>();if(list!=null&&list.size()>0){for (org.jdom.Element element : list) {refundOrderMap.put(element.getName(), element.getText());}return refundOrderMap;}return null;}/*** 解析申請退款之后微信退款回調返回的字符串中內容* @throws IOException * @throws JDOMException */public static Map<String, String> parseRefundNotifyXml(String refundXml) throws JDOMException, IOException{ParseXMLUtils.jdomParseXml(refundXml);StringReader read = new StringReader(refundXml);// 創建新的輸入源SAX 解析器將使用 InputSource 對象來確定如何讀取 XML 輸入InputSource source = new InputSource(read);// 創建一個新的SAXBuilderSAXBuilder sb = new SAXBuilder();// 通過輸入源構造一個Documentorg.jdom.Document doc;doc = (org.jdom.Document) sb.build(source);org.jdom.Element root = doc.getRootElement();// 指向根節點List<org.jdom.Element> list = root.getChildren();Map<String, String> resultMap = new HashMap<>();if(list!=null&&list.size()>0){for (org.jdom.Element element : list){resultMap.put(element.getName(), element.getText());}return resultMap;}return null;}/*** h5支付時 解析返回的值并返回prepareid* @throws IOException * @throws JDOMException */public static Map<String, String> getUrl(Unifiedorder unifiedorder) throws JDOMException, IOException{String xmlInfo = HttpXmlUtils.xmlH5Info(unifiedorder);String wxUrl = weixinConstant.URL;String method = "POST";String weixinPost = HttpXmlUtils.httpsRequest(wxUrl, method, xmlInfo).toString();ParseXMLUtils.jdomParseXml(weixinPost);StringReader read = new StringReader(weixinPost);// 創建新的輸入源SAX 解析器將使用 InputSource 對象來確定如何讀取 XML 輸入InputSource source = new InputSource(read);// 創建一個新的SAXBuilderSAXBuilder sb = new SAXBuilder();// 通過輸入源構造一個Documentorg.jdom.Document doc;doc = (org.jdom.Document) sb.build(source);org.jdom.Element root = doc.getRootElement();// 指向根節點List<org.jdom.Element> list = root.getChildren();String prepayId =null;Map<String, String> msg = new HashMap<String, String>();if(list!=null&&list.size()>0){for (org.jdom.Element element : list) {msg.put(element.getName(), element.getText());}}return msg;}/*** 解析返回的值并返回prepareid* @throws IOException * @throws JDOMException */public static String getPrepareId(Unifiedorder unifiedorder) throws JDOMException, IOException{String xmlInfo = HttpXmlUtils.xmlInfo(unifiedorder);String wxUrl = weixinConstant.URL;String method = "POST";String weixinPost = HttpXmlUtils.httpsRequest(wxUrl, method, xmlInfo).toString();ParseXMLUtils.jdomParseXml(weixinPost);StringReader read = new StringReader(weixinPost);// 創建新的輸入源SAX 解析器將使用 InputSource 對象來確定如何讀取 XML 輸入InputSource source = new InputSource(read);// 創建一個新的SAXBuilderSAXBuilder sb = new SAXBuilder();// 通過輸入源構造一個Documentorg.jdom.Document doc;doc = (org.jdom.Document) sb.build(source);org.jdom.Element root = doc.getRootElement();// 指向根節點List<org.jdom.Element> list = root.getChildren();String prepayId =null;if(list!=null&&list.size()>0){for (org.jdom.Element element : list) {if ( "prepay_id".equals(element.getName())) {prepayId= element.getText();break;}}}return prepayId;}/*** 向微信發送企業付款請求并解析返回結果* @throws IOException * @throws JDOMException */public static Map<String, String> getTransfersMap(Transfers transfers) throws JDOMException, IOException{String xmlInfo = HttpXmlUtils.xmlTransfer(transfers);String wxUrl = weixinConstant.WITHDRAW_URL;String method = "POST";String weixinPost = HttpXmlUtils.httpsRequest(wxUrl, method, xmlInfo).toString();ParseXMLUtils.jdomParseXml(weixinPost);StringReader read = new StringReader(weixinPost);// 創建新的輸入源SAX 解析器將使用 InputSource 對象來確定如何讀取 XML 輸入InputSource source = new InputSource(read);// 創建一個新的SAXBuilderSAXBuilder sb = new SAXBuilder();// 通過輸入源構造一個Documentorg.jdom.Document doc;doc = (org.jdom.Document) sb.build(source);org.jdom.Element root = doc.getRootElement();// 指向根節點List<org.jdom.Element> list = root.getChildren();Map<String, String> transferMap=new HashMap<>();if(list!=null&&list.size()>0){for (org.jdom.Element element : list) {transferMap.put(element.getName(), element.getText());}}return transferMap;}/*** 構造退款xml參數* @param xml* @return*/public static String refundXml(Unifiedorder unifiedorder){xStream.autodetectAnnotations(true);xStream.alias("xml", Unifiedorder.class);return xStream.toXML(unifiedorder);}/*** 構造企業付款xml參數* @param xml* @return*/public static String transferXml(Transfers transfers){xStream.autodetectAnnotations(true);xStream.alias("xml", Transfers.class);return xStream.toXML(transfers);}/*** 構造xml參數* @param xml* @return*/public static String xmlInfo(Unifiedorder unifiedorder){if(unifiedorder!=null){StringBuffer bf = new StringBuffer();bf.append("<xml>");bf.append("<appid><![CDATA[");bf.append(unifiedorder.getAppid());bf.append("]]></appid>");bf.append("<mch_id><![CDATA[");bf.append(unifiedorder.getMch_id());bf.append("]]></mch_id>");bf.append("<nonce_str><![CDATA[");bf.append(unifiedorder.getNonce_str());bf.append("]]></nonce_str>");bf.append("<sign><![CDATA[");bf.append(unifiedorder.getSign());bf.append("]]></sign>");bf.append("<body><![CDATA[");bf.append(unifiedorder.getBody());bf.append("]]></body>");bf.append("<detail><![CDATA[");bf.append(unifiedorder.getDetail());bf.append("]]></detail>");bf.append("<attach><![CDATA[");bf.append(unifiedorder.getAttach());bf.append("]]></attach>");bf.append("<out_trade_no><![CDATA[");bf.append(unifiedorder.getOut_trade_no());bf.append("]]></out_trade_no>");bf.append("<total_fee><![CDATA[");bf.append(unifiedorder.getTotal_fee());bf.append("]]></total_fee>");bf.append("<spbill_create_ip><![CDATA[");bf.append(unifiedorder.getSpbill_create_ip());bf.append("]]></spbill_create_ip>");bf.append("<time_start><![CDATA[");bf.append(unifiedorder.getTime_start());bf.append("]]></time_start>");bf.append("<time_expire><![CDATA[");bf.append(unifiedorder.getTime_expire());bf.append("]]></time_expire>");bf.append("<notify_url><![CDATA[");bf.append(unifiedorder.getNotify_url());bf.append("]]></notify_url>");bf.append("<trade_type><![CDATA[");bf.append(unifiedorder.getTrade_type());bf.append("]]></trade_type>");bf.append("</xml>");return bf.toString();}return "";}/*** 構造xml參數* @param xml* @return*/public static String xmlH5Info(Unifiedorder unifiedorder){if(unifiedorder!=null){StringBuffer bf = new StringBuffer();bf.append("<xml>");bf.append("<appid><![CDATA[");bf.append(unifiedorder.getAppid());bf.append("]]></appid>");bf.append("<mch_id><![CDATA[");bf.append(unifiedorder.getMch_id());bf.append("]]></mch_id>");bf.append("<nonce_str><![CDATA[");bf.append(unifiedorder.getNonce_str());bf.append("]]></nonce_str>");bf.append("<sign><![CDATA[");bf.append(unifiedorder.getSign());bf.append("]]></sign>");bf.append("<body><![CDATA[");bf.append(unifiedorder.getBody());bf.append("]]></body>");bf.append("<attach><![CDATA[");bf.append(unifiedorder.getAttach());bf.append("]]></attach>");bf.append("<out_trade_no><![CDATA[");bf.append(unifiedorder.getOut_trade_no());bf.append("]]></out_trade_no>");bf.append("<total_fee><![CDATA[");bf.append(unifiedorder.getTotal_fee());bf.append("]]></total_fee>");bf.append("<spbill_create_ip><![CDATA[");bf.append(unifiedorder.getSpbill_create_ip());bf.append("]]></spbill_create_ip>");bf.append("<notify_url><![CDATA[");bf.append(unifiedorder.getNotify_url());bf.append("]]></notify_url>");bf.append("<trade_type><![CDATA[");bf.append(unifiedorder.getTrade_type());bf.append("]]></trade_type>");bf.append("<scene_info><![CDATA[");bf.append(unifiedorder.getScene_info());bf.append("]]></scene_info>");bf.append("</xml>");return bf.toString();}return "";}/*** 構造退款xml參數* @param xml* @return*/public static String xmlTransfer(Transfers transfers){if(transfers!=null){StringBuffer bf = new StringBuffer();bf.append("<xml>");bf.append("<mch_appid><![CDATA[");bf.append(transfers.getMch_appid());bf.append("]]></mch_appid>");bf.append("<mchid><![CDATA[");bf.append(transfers.getMchid());bf.append("]]></mchid>");bf.append("<nonce_str><![CDATA[");bf.append(transfers.getNonce_str());bf.append("]]></nonce_str>");bf.append("<sign><![CDATA[");bf.append(transfers.getSign());bf.append("]]></sign>");bf.append("<partner_trade_no><![CDATA[");bf.append(transfers.getPartner_trade_no());bf.append("]]></partner_trade_no>");bf.append("<openid><![CDATA[");bf.append(transfers.getOpenid());bf.append("]]></openid>");bf.append("<check_name><![CDATA[");bf.append(transfers.getCheck_name());bf.append("]]></check_name>");bf.append("<amount><![CDATA[");bf.append(transfers.getAmount());bf.append("]]></amount>");bf.append("<desc><![CDATA[");bf.append(transfers.getDesc());bf.append("]]></desc>");bf.append("<spbill_create_ip><![CDATA[");bf.append(transfers.getSpbill_create_ip());bf.append("]]></spbill_create_ip>");bf.append("</xml>");return bf.toString();}return "";}/*** post請求并得到返回結果* @param requestUrl* @param requestMethod* @param output* @return*/public static String httpsRequest(String requestUrl, String requestMethod, String output) {try{URL url = new URL(requestUrl);HttpsURLConnection connection = (HttpsURLConnection) url.openConnection();connection.setDoOutput(true);connection.setDoInput(true);connection.setUseCaches(false);connection.setRequestMethod(requestMethod);if (null != output) {OutputStream outputStream = connection.getOutputStream();outputStream.write(output.getBytes("UTF-8"));outputStream.close();}// 從輸入流讀取返回內容InputStream inputStream = connection.getInputStream();InputStreamReader inputStreamReader = new InputStreamReader(inputStream, "utf-8");BufferedReader bufferedReader = new BufferedReader(inputStreamReader);String str = null;StringBuffer buffer = new StringBuffer();while ((str = bufferedReader.readLine()) != null) {buffer.append(str);}bufferedReader.close();inputStreamReader.close();inputStream.close();inputStream = null;connection.disconnect();return buffer.toString();}catch(Exception ex){ex.printStackTrace();}return "";} } /*** 微信支付相關的常量* @author yangfuren* @since 2017/08/16*/ public class weixinConstant {/*** 微信支付API秘鑰 第一次登錄時設置的*/public static final String KEY = "";/*** url*/public static final String URL = "https://api.mch.weixin.qq.com/pay/unifiedorder";/*** 退款url*/public static final String REFUND_URL = "https://api.mch.weixin.qq.com/secapi/pay/refund";/*** 證書地址 退款及h5支付使用*/public static final String PATH ="";/*** 付款url*/public static final String WITHDRAW_URL="https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers"; } import java.security.MessageDigest;public class MD5Util {private static String byteArrayToHexString(byte b[]) {StringBuffer resultSb = new StringBuffer();for (int i = 0; i < b.length; i++)resultSb.append(byteToHexString(b[i]));return resultSb.toString();}private static String byteToHexString(byte b) {int n = b;if (n < 0)n += 256;int d1 = n / 16;int d2 = n % 16;return hexDigits[d1] + hexDigits[d2];}public static String MD5Encode(String origin, String charsetname) {String resultString = null;try {resultString = new String(origin);MessageDigest md = MessageDigest.getInstance("MD5");if (charsetname == null || "".equals(charsetname))resultString = byteArrayToHexString(md.digest(resultString.getBytes()));elseresultString = byteArrayToHexString(md.digest(resultString.getBytes(charsetname)));} catch (Exception exception) {}return resultString;}private static final String hexDigits[] = { "0", "1", "2", "3", "4", "5","6", "7", "8", "9", "a", "b", "c", "d", "e", "f" };}統一下單提交為微信的參數
import java.io.Serializable;/*** 統一下單提交為微信的參數* @author * @date 2017年08月11日*/ public class Unifiedorder implements Serializable{private static final long serialVersionUID = 1L;//微信支付表idprivate Integer weixinId;//微信分配的公眾賬號ID(企業號corpid即為此appId)private String appid;//商戶idprivate String mch_id;//終端設備號(門店號或收銀設備ID),注意:PC網頁或公眾號內支付請傳"WEB"private String device_info;//隨機字符串:數字+大寫字母的組合,32位private String nonce_str;//簽名private String sign;//商品或支付單簡要描述private String body;//商品名稱明細列表private String detail;//附加參數(例如:用于區別本商戶不同的分店)private String attach;//商戶系統內部的訂單號private String out_trade_no;//貨幣類型:符合ISO 4217標準的三位字母代碼,默認人民幣:CNYprivate String fee_type;//總金額private int total_fee;//APP和網頁支付提交[用戶端ip],Native支付填調用微信支付API的機器IP。private String spbill_create_ip;//訂單生成時間,格式為yyyyMMddHHmmss,private String time_start;//訂單失效時間,格式為yyyyMMddHHmmss,最短失效時間間隔必須大于5分鐘[支付寶是30分鐘,同樣30分鐘]private String time_expire;//商品標記,代金券或立減優惠功能的參數private String goods_tag;//接收微信支付異步通知回調地址private String notify_url;//交易類型:JSAPI,NATIVE,APP h5為 MWEBprivate String trade_type;//trade_type=NATIVE,此參數必傳。此id為二維碼中包含的商品ID,商戶自行定義。private String product_id;//no_credit--指定不能使用信用卡支付private String limit_pay;//trade_type=JSAPI,此參數必傳,用戶在商戶appid下的唯一標識private String openid;//商戶內部自己的退款單號private String out_refund_no;//退款總金額單位為分private int refund_fee;//操作員的id默認為mch_idprivate String op_user_id;//微信官方提供的訂單號private String prepayid;//記錄所對應的memberprivate Member member;//返回給微信的狀態碼(用于支付回調時)public String return_code;//微信h5支付時候的場景信息官方的信息模板 {"h5_info"://h5支付固定傳"h5_info" //{"type":"",//場景類型 "wap_url":"",//WAP網站URL地址"wap_name": ""//WAP 網站名}}public String scene_info;public String getScene_info() {return scene_info;}public void setScene_info(String scene_info) {this.scene_info = scene_info;}public String getReturn_code() {return return_code;}public void setReturn_code(String return_code) {this.return_code = return_code;}public String getAppid() {return appid;}public String getMch_id() {return mch_id;}public String getDevice_info() {return device_info;}public String getNonce_str() {return nonce_str;}public String getSign() {return sign;}public String getBody() {return body;}public String getDetail() {return detail;}public String getAttach() {return attach;}public String getOut_trade_no() {return out_trade_no;}public String getFee_type() {return fee_type;}public int getTotal_fee() {return total_fee;}public String getSpbill_create_ip() {return spbill_create_ip;}public String getTime_start() {return time_start;}public String getTime_expire() {return time_expire;}public String getGoods_tag() {return goods_tag;}public String getNotify_url() {return notify_url;}public String getTrade_type() {return trade_type;}public String getProduct_id() {return product_id;}public String getLimit_pay() {return limit_pay;}public String getOpenid() {return openid;}public void setAppid(String appid) {this.appid = appid;}public void setMch_id(String mch_id) {this.mch_id = mch_id;}public void setDevice_info(String device_info) {this.device_info = device_info;}public void setNonce_str(String nonce_str) {this.nonce_str = nonce_str;}public void setSign(String sign) {this.sign = sign;}public void setBody(String body) {this.body = body;}public void setDetail(String detail) {this.detail = detail;}public void setAttach(String attach) {this.attach = attach;}public void setOut_trade_no(String out_trade_no) {this.out_trade_no = out_trade_no;}public void setFee_type(String fee_type) {this.fee_type = fee_type;}public void setTotal_fee(int total_fee) {this.total_fee = total_fee;}public void setSpbill_create_ip(String spbill_create_ip) {this.spbill_create_ip = spbill_create_ip;}public void setTime_start(String time_start) {this.time_start = time_start;}public void setTime_expire(String time_expire) {this.time_expire = time_expire;}public void setGoods_tag(String goods_tag) {this.goods_tag = goods_tag;}public void setNotify_url(String notify_url) {this.notify_url = notify_url;}public void setTrade_type(String trade_type) {this.trade_type = trade_type;}public void setProduct_id(String product_id) {this.product_id = product_id;}public void setLimit_pay(String limit_pay) {this.limit_pay = limit_pay;}public void setOpenid(String openid) {this.openid = openid;}public String getOut_refund_no() {return out_refund_no;}public void setOut_refund_no(String out_refund_no) {this.out_refund_no = out_refund_no;}public int getRefund_fee() {return refund_fee;}public void setRefund_fee(int refund_fee) {this.refund_fee = refund_fee;}public Integer getWeixinId() {return weixinId;}public void setWeixinId(Integer weixinId) {this.weixinId = weixinId;}public Member getMember() {return member;}public void setMember(Member member) {this.member = member;}public String getPrepayid() {return prepayid;}public void setPrepayid(String prepayid) {this.prepayid = prepayid;}public String getOp_user_id() {return op_user_id;}public void setOp_user_id(String op_user_id) {this.op_user_id = op_user_id;}}將參數傳遞給ios,ios會通過我們傳遞的參數調起微信支付。在生成簽名時,可以將生成的sign和傳入的appid,隨機字符串,時間戳,微信官方的訂單號,輸入到https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=20_1中按照順序輸入參數并生成sign看和你的sign是否一致。
總結
以上是生活随笔為你收集整理的java后台提供ios微信支付接口的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 5分钟自建数据库可视化平台,在线管理数据
- 下一篇: day64 url用法以及django的