java对接微信小程序客服(超详细)

1.https://developers.weixin.qq.com,先注册登录微信小程序,然后把认证信息填写了。
2.找到 开发-开发管理-开发设置,获取小程序的appID和生成小程序密钥AppSecret。
在这里插入图片描述

3.把你的服务器配置一个域名,可以通过http或者https访问的地址。
4.在开发设置下面开通消息推送服务。URL就是可以访问消息推送的接口地址(后续贴出来),令牌自己填的,要在接口校验的。(具体流程就是通过这个接口URL,微信会发一个get请求,然后返回信息,微信就让你填写令牌,然后接下来的发送消息,是通过post请求,校验令牌,就可以给客服人员微信发送信息)
在这里插入图片描述
5.在功能-客服-小程序客服,添加客服人员的微信号

前面都是准备,下面是详细代码和说明。

import com.donwait.common.annotation.AccessControl;
import com.donwait.common.service.applet.AutoResponseService;
import com.donwait.common.utils.MessageUtils;
import com.donwait.common.utils.StringUtil;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.*;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.PrintWriter;
import java.security.MessageDigest;
import java.util.*;

/**
 * @Description: 用户
 * @Author: zjh
 * @Date: 2021/6/7 14:49
 **/
@Api(tags = "用户信息")
@RequestMapping("/api/applet/member")
@RestController
@Slf4j
public class MemberApi {


    private final AutoResponseService autoResponseService;

    private final String token = "你的令牌";

    public MemberApi(AutoResponseService autoResponseService) {
        this.autoResponseService = autoResponseService;
    }


    @ApiOperation("微信消息通知-GET请求校验(确认请求来自微信服务器)")
    @RequestMapping(value = "/getSignature", method = {RequestMethod.GET, RequestMethod.POST})
    @AccessControl
    public String signature(HttpServletRequest request, HttpServletResponse response) {
        //微信加密签名,signature结合了开发者填写的token参数和请求中的timestamp参数、nonce参数
        boolean isGet = request.getMethod().toLowerCase().equals("get");
        if (isGet) {
            String signature = request.getParameter("signature");
            // 时间戳
            String timestamp = request.getParameter("timestamp");
            // 随机数
            String nonce = request.getParameter("nonce");
            // 随机字符串
            String echostr = request.getParameter("echostr");
            log.info("echostr>>>>>>>>>>" + echostr);
            //通过检验signature对请求进行校验,若校验成功则原样返回echostr,否则接入失败
            if (checkSignature(signature, timestamp, nonce)) {
                log.info("echostr>>>>>>>>>>>>" + echostr);
                return echostr;
            }
            log.info("客服消息验证url验证结果:{}", echostr);
        } else {
            response(request, response);
        }
        return null;
    }

    /*@ApiOperation("微信消息通知-POST请求校验(确认请求来自微信服务器)")
    @PostMapping("/postSignature")
    @AccessControl*/
    public String response(HttpServletRequest request, HttpServletResponse response) {
        try {
            // 接收消息并返回消息
            String result = acceptMessage(request, response);
            log.info("接受消息并返回消息result>>>>>>" + result);
            // 响应消息
            PrintWriter out = response.getWriter();
            log.info("out>>>>>>>>" + out);
            out.print(result);
            out.close();
        } catch (Exception ex) {
            log.error("微信帐号接口配置失败!" + ex.getMessage());
            ex.printStackTrace();
        }
        return null;
    }

    /**
     * 校验(确认请求来自微信服务器)
     *
     * @param signature
     * @param timestamp
     * @param nonce
     * @return
     */
    public boolean checkSignature(String signature, String timestamp, String nonce) {
        String[] params = {"dw818502", timestamp, nonce};
        log.info("params>>>>>>>>>>>>>" + Arrays.toString(params));
        //1、将token、timestamp、nonce三个参数进行字典序排序
        Arrays.sort(params);
        //2、将三个参数字符串拼接成一个字符串
        StringBuilder sb = new StringBuilder();
        for (String param : params) {
            sb.append(param);
        }
        log.info("字符串拼接>>>>>>>>>>>>>>>>>" + sb);
        MessageDigest messageDigest = null;
        String tpmStr = null;
        try {
            messageDigest = MessageDigest.getInstance("SHA-1");
            log.info("md>>>>>>>>>>>>" + messageDigest);
            //3、将三个参数字符串拼接成一个字符串进行sha1加密
            byte[] digest = messageDigest.digest(sb.toString().getBytes());
            log.info("digest>>>>>>>>>>>" + Arrays.toString(digest));
            tpmStr = StringUtil.byteToHex(digest);
            log.info("加密串>>>>>>>>>>>" + tpmStr);
        } catch (Exception e) {
            log.info("错误信息>>>>>>>>>>>>" + e.getMessage());
            e.printStackTrace();
        }
        //4、将sha1加密后的字符串可与signature对比,标识该请求来源于微信
        return tpmStr != null && tpmStr.equals(signature);
    }

    public String acceptMessage(HttpServletRequest request, HttpServletResponse response) {
        //返回值
        String result = "success";
        try {
            // 将请求、响应的编码均设置为UTF-8(防止中文乱码)
            request.setCharacterEncoding("UTF-8");
            response.setCharacterEncoding("UTF-8");
            Map<String, String> requestMap = MessageUtils.parseXml(request);
            log.info("requestMap>>>>>>>>>>" + requestMap);
            // 发送者的openid
            String fromUserName = requestMap.get("FromUserName");
            // 小程序的原始ID
            String toUserName = requestMap.get("ToUserName");
            // 消息类型
            String msgType = requestMap.get("MsgType");
            // 文本消息内容
            String content = requestMap.get("Content");
            // 事件类型
            String event = requestMap.get("Event");
            log.info("fromUserName=" + fromUserName + ",toUserName=" + toUserName + ",msgType=" + msgType + ",content=" + content + ",event=" + event);
            StringBuffer contentMessage = new StringBuffer();
            autoResponseService.sendCustomerTextMessage(fromUserName, contentMessage.toString(), token);
        } catch (Exception e) {
            log.info("错误信息打印>>>>>>>>>>>" + e.getMessage());
            e.printStackTrace();
        }
        return result;
    }


}

/**
 * 客服功能 - 消息发送请求工具类
 * Created by Lance on 2020/10/10 17:52
 */
public class HttpUtils {

    /**
     * Get 发送的请求
     * @param url
     * @return
     */
    public static String sendGetRequest(String url) {
        RestTemplate restTemplate = new RestTemplate();
        return restTemplate.getForObject(url, String.class);
    }

    public static String sendPostRequest(String url, HttpMethod method, HttpEntity<Map<String, Object>> httpEntity) {
        RestTemplate restTemplate = new RestTemplate();
        ResponseEntity<String> response = restTemplate.exchange(url, method, httpEntity, String.class);
        return response.getBody();
    }
}

/**
 * 客服功能 - 消息工具类
 * Created by Lance on 2020/10/10 17:52
 */
public class MessageUtils {

    public static Map<String, String> parseXml(HttpServletRequest request) throws Exception {
        // 将解析结果存储在HashMap 中
        Map<String, String> map = new HashMap<String, String>();
        // 从request 中取得输入流
        InputStream inputStream = request.getInputStream();
        // 读取输入流
        SAXReader reader = new SAXReader();
        Document document = reader.read(inputStream);
        // 得到xml 根元素
        Element root = document.getRootElement();
        // 得到根元素的所有子节点
        List<Element> elementList = root.elements();
        // 遍历所有子节点
        for (Element e : elementList) {
            map.put(e.getName(), e.getText());
        }
        // 释放资源
        inputStream.close();
        return map;
    }
}
/**
 * 客服功能 - 字符串转换
 * Created by Lance on 2020/10/12 10:52
 */
@Slf4j
public class StringUtil {
    /**
     * 十六进制字节数组转为字符串
     *
     * @param hash
     * @return
     */
    public static String byteToHex(final byte[] hash) {
        Formatter formatter = new Formatter();
        for (byte b : hash) {
            formatter.format("%02x", b);
        }
        String result = formatter.toString();
        formatter.close();
        return result;
    }
}
import java.security.MessageDigest;

public final class SHA1 {

    private static final char[] HEX_DIGITS = {'0', '1', '2', '3', '4', '5',
            '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};

    /**
     * Takes the raw bytes from the digest and formats them correct.
     *
     * @param bytes the raw bytes from the digest.
     * @return the formatted bytes.
     */
    private static String getFormattedText(byte[] bytes) {
        int len = bytes.length;
        StringBuilder buf = new StringBuilder(len * 2);
        // 把密文转换成十六进制的字符串形式
        for (int j = 0; j < len; j++) {
            buf.append(HEX_DIGITS[(bytes[j] >> 4) & 0x0f]);
            buf.append(HEX_DIGITS[bytes[j] & 0x0f]);
        }
        return buf.toString();
    }

    public static String encode(String str) {
        if (str == null) {
            return null;
        }
        try {
            MessageDigest messageDigest = MessageDigest.getInstance("SHA1");
            messageDigest.update(str.getBytes());
            return getFormattedText(messageDigest.digest());
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
}
/**
 * 客服功能 - 自动回复小程序APP链接
 * Created by Lance on 2020/10/10 11:03
 */
@Slf4j
@Service
public class AutoResponseService {

    /**
     * 人工服务
     *
     * @param fromUserName
     * @param toUserName
     * @param requestMap
     * @return
     */
    public String switchCustomerService(String fromUserName, String toUserName) {
        String returnXml = "<xml>\n" +
                "    <ToUserName><![CDATA[" + fromUserName + "]]></ToUserName>\n" +
                "    <FromUserName><![CDATA[" + toUserName + "]]></FromUserName>\n" +
                "    <CreateTime>" + System.currentTimeMillis() / 1000 + "</CreateTime>\n" +
                "    <MsgType><![CDATA[transfer_customer_service]]></MsgType>\n" +
                "</xml>";
        log.info("人工服务result>>>>>>>" + returnXml);
        return returnXml;
    }
    /**
     * 文本消息
     *
     * @param openid
     * @param text
     * @param accessToken
     * @return
     */
    private static String SEND_URL = "https://api.weixin.qq.com/cgi-bin/message/custom/send";//固定发送地址

    public String sendCustomerTextMessage(String openid, String text, String accessToken) throws IOException {
        HashMap<String, Object> map_content = new HashMap<>();
        map_content.put("content", text);
        HashMap<String, Object> map = new HashMap<>();
        map.put("touser", openid);
        map.put("msgtype", "text");
        map.put("text", map_content);
        String content = JSONObject.toJSONString(map);
        log.info("文本消息content" + content);
        HttpHeaders headers = new HttpHeaders();
        headers.setContentType(MediaType.APPLICATION_JSON);
        HttpEntity httpEntity = new HttpEntity(map, headers);
        String str = HttpUtils.sendPostRequest(SEND_URL + "?access_token=" + accessToken, HttpMethod.POST, httpEntity);
        log.info("文本消息str>>>>>>>>>>"+str);
        return str;
    }
}

到这里就已经可以了,然后看一下效果
在这里插入图片描述
在这里插入图片描述

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
要在 Java对接微信小程序支付,你需要完成以下步骤: 1. 获取微信支付的 API 密钥和商户号,以及小程序 appID 和 appSecret。 2. 在小程序开发者工具中创建支付统一下单接口,并记录好接口地址。 3. 在 Java 中编写代码,向微信支付下单接口发送请求,并获取返回的 prepay_id。 4. 将 prepay_id 和其他参数按照微信支付要求的格式进行签名,并返回给前端。 5. 前端通过微信小程序支付 API 调起支付界面,用户完成支付后,微信服务器会向商户服务器发送支付结果通知。 6. 商户服务器接收到支付结果通知后,需要进行签名验证和订单处理。 以下是一份示例代码,可以帮助你完成微信小程序支付的对接: ```java import java.security.MessageDigest; import java.security.NoSuchAlgorithmException; import java.util.HashMap; import java.util.Map; import java.util.SortedMap; import java.util.TreeMap; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import org.w3c.dom.Document; import org.w3c.dom.Element; public class WechatPayment { private String appid; // 小程序 ID private String mch_id; // 商户号 private String key; // API 密钥 public WechatPayment(String appid, String mch_id, String key) { this.appid = appid; this.mch_id = mch_id; this.key = key; } public Map<String, String> unifiedorder(String body, String out_trade_no, int total_fee, String spbill_create_ip, String notify_url) throws Exception { SortedMap<String, String> params = new TreeMap<String, String>(); params.put("appid", appid); params.put("mch_id", mch_id); params.put("nonce_str", getRandomString(32)); params.put("body", body); params.put("out_trade_no", out_trade_no); params.put("total_fee", String.valueOf(total_fee)); params.put("spbill_create_ip", spbill_create_ip); params.put("notify_url", notify_url); params.put("trade_type", "JSAPI"); params.put("openid", "用户的 openid"); String sign = createSign(params); params.put("sign", sign); String xml = mapToXml(params); String responseXml = HttpUtils.post("https://api.mch.weixin.qq.com/pay/unifiedorder", xml); DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); DocumentBuilder builder = factory.newDocumentBuilder(); Document document = builder.parse(new ByteArrayInputStream(responseXml.getBytes())); Element rootElement = document.getDocumentElement(); String return_code = getElementValue(rootElement, "return_code"); String result_code = getElementValue(rootElement, "result_code"); String prepay_id = getElementValue(rootElement, "prepay_id"); if ("SUCCESS".equals(return_code) && "SUCCESS".equals(result_code) && prepay_id != null) { Map<String, String> result = new HashMap<String, String>(); result.put("prepay_id", prepay_id); String nonce_str = getRandomString(32); result.put("nonceStr", nonce_str); result.put("timeStamp", String.valueOf(System.currentTimeMillis() / 1000)); result.put("signType", "MD5"); result.put("package", "prepay_id=" + prepay_id); sign = createSign(result); result.put("paySign", sign); return result; } else { throw new Exception(getElementValue(rootElement, "return_msg")); } } private String createSign(Map<String, String> params) throws NoSuchAlgorithmException { StringBuilder sb = new StringBuilder(); for (Map.Entry<String, String> entry : params.entrySet()) { if (entry.getValue() != null && entry.getValue().length() > 0 && !"sign".equals(entry.getKey()) && !"key".equals(entry.getKey())) { sb.append(entry.getKey()).append("=").append(entry.getValue()).append("&"); } } sb.append("key=").append(key); MessageDigest md = MessageDigest.getInstance("MD5"); md.update(sb.toString().getBytes()); byte[] digest = md.digest(); StringBuilder signBuilder = new StringBuilder(); for (byte b : digest) { String hex = Integer.toHexString(b & 0xff); if (hex.length() == 1) { signBuilder.append("0"); } signBuilder.append(hex); } return signBuilder.toString().toUpperCase(); } private String mapToXml(Map<String, String> params) { StringBuilder sb = new StringBuilder(); sb.append("<xml>"); for (Map.Entry<String, String> entry : params.entrySet()) { sb.append("<").append(entry.getKey()).append(">"); sb.append(entry.getValue()); sb.append("</").append(entry.getKey()).append(">"); } sb.append("</xml>"); return sb.toString(); } private String getElementValue(Element element, String tagName) { Element childElement = (Element) element.getElementsByTagName(tagName).item(0); if (childElement != null) { return childElement.getTextContent(); } return null; } private String getRandomString(int length) { String str = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"; StringBuilder sb = new StringBuilder(); for (int i = 0; i < length; i++) { sb.append(str.charAt((int) (Math.random() * str.length()))); } return sb.toString(); } } ``` 其中,HttpUtils.post() 方法用于发送 HTTP POST 请求,你需要自己实现。在 unifiedorder() 方法中,我们按照微信支付要求的格式构造请求参数,并对参数进行签名。然后向微信支付下单接口发送请求,获取 prepay_id。最后,我们将 prepay_id 和其他参数按照微信支付要求的格式进行签名,并返回给前端。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

随风奔跑的十八岁

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值