微信支付对账单下载

微信支付对账单下载

以上是官网文档说明:

下面是根据文档 梳理下思路:

/**
关于签名的算法,api提供的原文是:

1.签名算法

签名生成的通用步骤如下:

第一步,设所有发送或者接收到的数据为集合M,将集合M内非空参数值的参数按照参数名ASCII码从小到大排序(字典序),使用URL键值对的格式(即key1=value1&key2=value2…)拼接成字符串stringA。

特别注意以下重要规则:
 参数名ASCII码从小到大排序(字典序);
 如果参数的值为空不参与签名;
   参数名区分大小写;
 验证调用返回或微信主动通知签名时,传送的sign参数不参与签名,将生成的签名与该sign值作校验。
 微信接口可能增加字段,验证签名时必须支持增加的扩展字段

第二步,在stringA最后拼接上key=(API密钥的值)得到stringSignTemp字符串,并对stringSignTemp进行MD5运算,再将得到的字符串所有字符转换为大写,得到sign值signValue。

举例:

假设传送的参数如下:

appid: wx8888888888888888

mch_id: 1900000109

device_info: 013467007045764

body: test

nonce_str: 5K8264ILTKCH16CQ2502SI8ZNMTM67VS

第一步:对参数按照key=value的格式,并按照参数名ASCII字典序排序如下:

stringA="appid=wx8888888888888888&body=test&device_info=013467007045764&mch_id=1900000109&nonce_str=5K8264ILTKCH16CQ2502SI8ZNMTM67VS";

第二步:拼接API密钥:

stringSignTemp="stringA&key=332F17B766FC787203EBE9D6E40457A1"

sign=MD5(stringSignTemp).toUpperCase()="332F17B766FC787203EBE9D6E40457A1"

*/

 微信支付签名算法sign:

String keywx = "9e4b63793722c61a555696be897d61f8";
	/**
	 * 微信支付签名算法sign
	 * 创建md5摘要,规则是:按参数名称a-z排序,遇到空值的参数不参加签名。
                 创建签名 签名 sign 	 
	 */
	protected void  createSign(SortedMap<Object,Object> parameters) {
		StringBuffer sb = new StringBuffer();
		//parameters 参数集合值
		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+ "&" );
			}
		}
		//key就是微信支付账户密钥
		sb.append("key=" + keywx);
		//sb.append("key=" + this.getKey());
		System.out.println("字符串拼接后是:"+sb.toString());  
		//字体编码 生成MD5的时候,需要统一编码,这里微信api要求是UTF-8
		String enc = getCharacterEncoding(this.request, this.response);
		//生成MD5签名验证 注意转换成大写  在此是小写
		String sign = MD5Util.MD5Encode(sb.toString(), enc).toUpperCase();
		
		this.setParameter("sign", sign);
		
		//debug信息
		this.setDebugInfo(sb.toString() + " => sign:" + sign);
		return ;
	}
		/**
	*设置debug信息
	*/
	protected void setDebugInfo(String debugInfo) {
		this.debugInfo = debugInfo;
	}
	/**
	 * 设置参数值
	 * @param parameter 参数名称
	 * @param parameterValue 参数值
	 */
	public void setParameter(String parameter, String parameterValue) {
		String v = "";
		if(null != parameterValue) {
			v = parameterValue.trim();
		}
		this.parameters.put(parameter, v);
	}
	/**
	 * 获取编码字符集
	 * @param request
	 * @param response
	 * @return String
	 */
	public static String getCharacterEncoding(HttpServletRequest request,
			HttpServletResponse response) {
		
		if(null == request || null == response) {
			return "gbk";
		}
		
		String enc = request.getCharacterEncoding();
		if(null == enc || "".equals(enc)) {
			enc = response.getCharacterEncoding();
		}
		
		if(null == enc || "".equals(enc)) {
			enc = "gbk";
		}
		
		return enc;
	}

 参数包装成XML格式丢给微信:

 /**
     * 参数包装成XML格式丢给微信
    * */
	public  String GetpayXmlRequest(String appid,String spid,String getNonce_str,String sign, String trans_time, String bill_type)
	{  
		/*
	   <xml>
		   <appid>wx8888888888888888</appid>
		   <bill_date>20160426</bill_date>
		   <bill_type>ALL</bill_type>
		   <mch_id>1900000109</mch_id>
		   <nonce_str>21df7dc9cd8616b56919f20d9f679233</nonce_str>
		   <sign>E8778F1C4C953C0BF37AA45C3BC379D0</sign>
		 </xml> 
	  */
		
		       StringBuffer sb = new StringBuffer();  
			   sb.append("<xml>");  
			   sb.append("<appid><![CDATA[");  
			   sb.append(appid);  
			   sb.append("]]></appid>");  
			
			   sb.append("<mch_id><![CDATA[");  
			   sb.append(spid);  
			   sb.append("]]></mch_id>");  
			 
			   sb.append("<nonce_str><![CDATA[");  
			   sb.append(getNonce_str);  
			   sb.append("]]></nonce_str>");  
			 
			   sb.append("<sign><![CDATA[");  
			   sb.append(sign);  
			   sb.append("]]></sign>");  
			  
			   sb.append("<bill_date><![CDATA[");  
			   sb.append(trans_time);  
			   sb.append("]]></bill_date>");  
			 
			   sb.append("<bill_type><![CDATA[");  
			   sb.append(bill_type);  
			   sb.append("]]></bill_type>");  
			 
			   sb.append("</xml>");  
			   return sb.toString();  

		        /*
		         
		        sb.append("<appid>");
		        sb.append(appid);  
		        sb.append("</appid>");
		        sb.append("<bill_date>");  
		        sb.append(trans_time);  
		        sb.append("</bill_date>");
		        sb.append("<bill_type>");  
		        sb.append(bill_type);  
		        sb.append("</bill_type>");
		        sb.append("<mch_id>");
		        sb.append(spid);
		        sb.append("</mch_id>");
		        sb.append("<nonce_str>"); 
		        sb.append(stamp);  
		        sb.append("</nonce_str>");
		        sb.append("<sign>"); 	
		        sb.append(sign); 
		        sb.append("</sign>");
		        sb.append("</xml>");  
		       return sb.toString();  
		       */

		    }  

 post发送请求数据:

/** 请求的参数 */
	protected SortedMap parameters;
	
    protected HttpServletRequest request;
	
	protected HttpServletResponse response;
	/** 密钥 */
	protected String key;
	/** debug信息 */
	protected String debugInfo;
	
	/**
	*获取密钥
	*/
	public String getKey() {
		return key;
	}
	/**
	*设置密钥
	*/
	public void setKey(String key) {
		this.key = key;
	}
	
	/**
	 * post发送请求数据
	 * @param urlStr
	 */
	public  void testPost(String urlStr) {  
		 BufferedReader br= null;
		 HttpURLConnection con=null;
		 OutputStreamWriter out=null;
		        try {  
		            URL url = new URL(urlStr);  
		           // URLConnection con = url.openConnection(); 
		            con = (HttpURLConnection) url.openConnection(); 
		         // 发送POST请求必须设置如下两行
		            con.setDoOutput(true);  // POST方式
		            con.setDoInput(true);
		            con.setUseCaches(false);
		            con.setRequestMethod("POST"); 
		         // 设置通用的请求属性
		            con.setRequestProperty("accept", "*/*");
		            con.setRequestProperty("connection", "Keep-Alive");
		            con.setRequestProperty("user-agent",
		                    "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1;SV1)");
		            
//		            con.setRequestProperty("Pragma:", "no-cache");  
//		            con.setRequestProperty("Cache-Control", "no-cache");  
//		            con.setRequestProperty("Content-Type", "text/xml");  
		            
		            // 输出流,写数据
		             out = new OutputStreamWriter(con  
		                   .getOutputStream(),"UTF-8");    
		            
		            
		         // 发送请求参数
		            String xmlInfo = GetpayXmlRequest("wx8888888888888888","1900000109","1462240367309","56B106AFDDF1802B12DF8634C2E73B92","20160623","ALL");  
		            System.out.println("urlStr=" + urlStr);  
		            System.out.println("xmlInfo=" + xmlInfo);  
		           out.write(new String(xmlInfo.getBytes("UTF-8"))); 
		        // flush输出流的缓冲
		            out.flush();  
		           // out.close();  
		          //读取服务器的响应内容并显示 定义BufferedReader输入流来读取URL的响应
		            br = new BufferedReader(new InputStreamReader(con  
		                    .getInputStream(),"UTF-8"));  
		           String line = "";  
		           StringBuffer buffer = new StringBuffer();
//		            for (line = br.readLine(); line != null; line = br.readLine()) {  
//		               System.out.println(line);  
//		               buffer.append(line);// 将读到的内容添加到 buffer 中
//		               buffer.append("\n"); // 添加换行符
//
//		            } 
		            while ((line = br.readLine()) != null)
		            {   
		            buffer.append(line); // 将读到的内容添加到 buffer 中
		            buffer.append("\n"); // 添加换行符
		            System.out.println(line);  
	                }  
		            
		            //con.getOutputStream().close();
		        } 
		        catch (MalformedURLException e) 
		        {  
		            e.printStackTrace();  
		        } 
		        catch (IOException e)
		        {  
		            e.printStackTrace();  
		        }  
		        // 使用finally块来关闭输入流
		        finally {
		            try {
		            	if(out!=null){
		                    out.close();
		                }

		                if (br != null) 
		                {
		                	br.close();
		                	con.disconnect();
		                }
		            } catch (Exception e2) {
		                e2.printStackTrace();
		            }
		        }

		    }  

 

/**
	 * @param args
	 */
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		String url = "https://api.mch.weixin.qq.com/pay/downloadbill";  
		 
		  new HttpPostTest().testPost(url);  

		    
		  //微信api提供的参数   赋值
		    String appid="wx8888888888888888"; 
		    String spid="1900000109";//       1221024001
		    String getNonce_str="1462240367309";//1462240367309  ibuaiVcKdpRxkhJA
		    String trans_time="20160623";
		    String bill_type="ALL";
		    String sign ="";
	         
		   // 获取值
	        SortedMap<Object,Object> sparameters = new TreeMap<Object,Object>();  
	        sparameters.put("appid", appid);
	        sparameters.put("mch_id", spid); 
	        sparameters.put("nonce_str", getNonce_str);
	        sparameters.put("bill_date", trans_time);  
	        sparameters.put("bill_type", bill_type);  
	         
	        // new HttpPostTest().createSign(sparameters);
	}

 注意 以上单元测试时要先 创建生成签名后,再把签名的值传到参数中去,然后进行httppost请求即可。

 这里商户号 一定都要改成自己申请的appid 和 mch_id 这样就会有微信返回结果数据显示了,结果实例就不在这里展现出来。

微信支付对账单下载

  获取到对账信息后就是后续的业务处理了,根据不同的需求进行不同的业务处理了,主要就是解析数据格式,生成文档,保存到本地以及导入到数据库的逻辑操作了。

 

微信支付对账单下载

上一篇:微信红包的架构设计简介{转}


下一篇:金庸武功之“黯然销魂掌”---zabbix实现微信报警