`
空指针异常
  • 浏览: 22600 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类

微信开发

wx 
阅读更多
参考文章
http://www.cnblogs.com/txw1958/p/weixin-qrcode-with-parameters.html
http://tuposky.iteye.com/blog/2082230
https://github.com/caspar-chen/WechatSDK

@Action("wxAction")
public class WeixinAction extends ActionSupport {
	public static final Logger logger = Logger.getLogger(WeixinAction.class);
	
	@Autowired
	private WXUserService wxUserService;

	/**
	 * 
	 */
	private static final long serialVersionUID = 1L;

	@Override
	public String execute() throws Exception {
		HttpServletRequest req = ServletActionContext.getRequest();
		PrintWriter out = ServletActionContext.getResponse().getWriter();
		String method = req.getMethod();
		if (WXConstants.METHOD_GET.equals(method)) {
			// 微信加密签名,signature结合了开发者填写的token参数和请求中的timestamp参数、nonce参数
			String signature = req.getParameter("signature");
			// 时间戳
			String timestamp = req.getParameter("timestamp");
			// 随机数
			String nonce = req.getParameter("nonce");
			// 随机字符串
			String echostr = req.getParameter("echostr");

			// 通过检验signature对请求进行校验,若校验成功则原样返回echostr,表示接入成功,否则接入失败
			if (SignUtil.checkSignature(signature, timestamp, nonce)) {
				out.print(echostr);
			}
		} else if (WXConstants.METHOD_POST.equals(method)) {
			// xml请求解析
			Map<String, String> reqMap = MessageUtil.parseXml(req);

			// 消息类型
			String msgType = reqMap.get("MsgType");
			if (msgType.equals(WXConstants.REQ_MESSAGE_TYPE_EVENT)) {
				String eventType = reqMap.get("Event");
				String openID = reqMap.get("FromUserName");
				String ticket = reqMap.get("Ticket");
				// 获取用户基本信息
				String nickname = WXInterfaceUtil.getNickname(openID);
				// 用户未关注时
				if (eventType.equals(WXConstants.EVENT_TYPE_SUBSCRIBE)) {
					logger.info("新关注用户openID:" + openID);
				}
				// 用户已关注
				else if (eventType.equals(WXConstants.EVENT_TYPE_SCAN)) {
					logger.info("已关注用户openID:" + openID);
				}
				// 更新数据库
				wxUserService.updateUserInfoByTicket(ticket, nickname, openID);
			}
		}

		out.close();
		out = null;
		return null;
	}
}


@Action("qrcodeCreateAction")
public class QrcodeCreateAction extends ActionSupport {
	@Autowired
	private WXUserService wxUserService;
	
	/**
	 * 
	 */
	private static final long serialVersionUID = -7930634700220791363L;
	
	@Override
	public String execute() throws Exception {
		PrintWriter out = ServletActionContext.getResponse().getWriter();
		// 获取传输的参数SN
		String sn = ServletActionContext.getRequest().getParameter("sn");
		
		// 生成临时二维码,每次生成二维码时ticket不一样,当用户扫描二维码时,可根据ticket更新关联的记录
		// 不能用永久二维码,因为永久二维码每次产生的ticket每次都一样
		String accessToken = WXInterfaceUtil.getAccessToken();
		String ticket = WXInterfaceUtil.createTempQrcode(accessToken);
		if(null == ticket){
			WXInterfaceUtil.access_token = null;
			accessToken = WXInterfaceUtil.getAccessToken();
			ticket = WXInterfaceUtil.createTempQrcode(accessToken);
		}
		
		//将当前请求的数据存入数据库,如果当前sn在数据库中已有记录,更新ticket,如果没有,新增一条记录
		wxUserService.saveOrUpdateUserInfo(sn, ticket);
		
		JSONObject obj = new JSONObject();
		//如果取ticket失败
		if(null == ticket){
			obj.put("errcode", "99");
			obj.put("errmsg", "failure");
		}else{
			//组装生成二维码URL,get请求
			String qrCodeUrl = "https://mp.weixin.qq.com/cgi-bin/showqrcode?ticket=" + ticket;
			obj.put("qrCodeUrl", qrCodeUrl);
			obj.put("errcode", "0");
			obj.put("errmsg", "success");
		}
		out.print(obj.toString());
		return null;
	}
}


@Service
public class WXUserServiceImpl implements WXUserService {
	
	@Autowired
	private WXUserDao wxUserDao;
	/**
	 * 当安卓客户端请求二维码时,如果当前请求的盒子信息没有保存在数据库中,则新增数据至数据库。
	 * 如果当前盒子已经请求过二维码,将ticket更新,并且清空用户信息
	 * 
	 * @param sn
	 * @param ticket
	 */
	public void saveOrUpdateUserInfo(String sn,String ticket){
		WXUser user = wxUserDao.getBySN(sn);
		//如果用户不存在
		if(null == user){
			WXUser u = new WXUser();
			u.setSn(sn);
			u.setTicket(ticket);
			u.setCreateTime(new Date());
			wxUserDao.save(u);
		}
		//用户已存在,更新ticket,清空用户信息
		else{
			user.setTicket(ticket);
			user.setCreateTime(new Date());
			user.setNickname("");
			user.setOpenId("");
			user.setLastLoginTime(null);
			user.setFlag(0);
			wxUserDao.update(user);
		}
	}
	
	/**
	 * 用户扫描二维码,用户根据ticket更新用户信息
	 * 
	 * @param ticket
	 * @param nickname
	 * @param openId
	 */
	public void updateUserInfoByTicket(String ticket,String nickname,String openId){
		WXUser user = wxUserDao.getByTicket(ticket);
		user.setNickname(nickname);
		user.setOpenId(openId);
		user.setLastLoginTime(new Date());
		wxUserDao.update(user);
	}
}


public class MessageUtil {
    /**
     * 解析微信发来的请求(XML)
     * 
     * @param request
     * @return
     * @throws Exception
     */
    @SuppressWarnings("unchecked")
    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();
        inputStream = null;

        return map;
    }
}

public class ReqUtil {
	private static final Logger logger = Logger.getLogger(ReqUtil.class);
	
	/**
	 * POST 请求
	 * 
	 * @param reqUrl
	 * @param reqParams
	 * @return String
	 */
	public static String post(String reqUrl, String params) {
		logger.info("POST Request,Url:" + reqUrl);
		DataOutputStream outStream = null;
		HttpURLConnection conn = null;
		String result = "";
		try {
			byte[] data = params.toString().getBytes();
			URL realUrl = new URL(reqUrl);
			conn = (HttpURLConnection) realUrl
					.openConnection();
			conn.setDoOutput(true);// 发送POST请求必须设置允许输出
			conn.setUseCaches(false);// 不使用Cache
			conn.setRequestMethod(WXConstants.METHOD_POST);
			conn.setRequestProperty("Connection", "Keep-Alive");// 维持长连接
			conn.setRequestProperty("Charset", "UTF-8");
			conn.setRequestProperty("Content-Length",
					String.valueOf(data.length));
			conn.setRequestProperty("Content-Type",
					"application/x-www-form-urlencoded");
			outStream = new DataOutputStream(
					conn.getOutputStream());
			outStream.write(data);
			outStream.flush();
			if (conn.getResponseCode() == 200) {
				result = readData(conn.getInputStream(), "UTF-8");
			}
		} catch (Exception e) {
			e.printStackTrace();
		} finally{
			if(null != outStream){
				try {
					outStream.close();
				} catch (IOException e) {
					e.printStackTrace();
				}
			}
			
			if(null != conn){
				conn.disconnect();
			}
		}
		return result;
	}

	/**
	 * Get 请求
	 * 
	 * @param reqUrl
	 * @return String
	 */
	public static String get(String reqUrl) {
		logger.info("GET Request,Url:" + reqUrl);
		InputStream is = null;
		HttpURLConnection conn = null;
		String result = "";
		try {
			URL url = new URL(reqUrl);
			conn = (HttpURLConnection) url.openConnection();
			conn.setConnectTimeout(5 * 1000);// 设置连接超时
			conn.setRequestMethod(WXConstants.METHOD_GET);// 以get方式发起请求
			if (conn.getResponseCode() != 200)
				throw new RuntimeException("请求url失败");
			is = conn.getInputStream();// 得到网络返回的输入流
			result = readData(is, "UTF-8");
			conn.disconnect();
		} catch (Exception e) {
			e.printStackTrace();
		} finally{
			if(null != is){
				try {
					is.close();
				} catch (IOException e) {
					e.printStackTrace();
				}
			}
			
			if(null != conn){
				conn.disconnect();
			}
		}
		return result;
	}

	/**
	 * 第一个参数为输入流,第二个参数为字符集编码
	 * 
	 * @param inSream
	 * @param charsetName
	 * @return String
	 * @throws Exception
	 */
	private static String readData(InputStream inSream, String charsetName)
			throws Exception {
		ByteArrayOutputStream outStream = new ByteArrayOutputStream();
		byte[] buffer = new byte[1024];
		int len = -1;
		while ((len = inSream.read(buffer)) != -1) {
			outStream.write(buffer, 0, len);
		}
		byte[] data = outStream.toByteArray();
		outStream.close();
		inSream.close();
		return new String(data, charsetName);
	}
}


public class SignUtil {
	// Token可由开发者任意填写,用作生成签名
	private static final String token = "test";
	
	/**
	 * 检验signature对请求进行校验,确认此次GET请求来自微信服务器
	 * 
	 * @param signature
	 * @param token
	 * @param timestamp
	 * @param nonce
	 * @return boolean
	 */
	public static boolean checkSignature(String signature,String timestamp,String nonce){
		String[] arr = new String[]{token, timestamp, nonce};
		// 将token、timestamp、nonce三个参数进行字典序排序
		Arrays.sort(arr);
		// 拼装成字符串
		StringBuilder content = new StringBuilder();
		for(String str : arr){
			content.append(str);
		}
		
		// sha1加密
		MessageDigest md = null;
		String tmpStr = null;
		try {
			md = MessageDigest.getInstance("SHA-1");
			// 将三个参数字符串拼接成一个字符串进行sha1加密
			byte[] digest = md.digest(content.toString().getBytes());
			tmpStr = byteToStr(digest);
		} catch (NoSuchAlgorithmException e) {
			e.printStackTrace();
		}
		
		content = null;
		// 将sha1加密后的字符串可与signature对比,标识该请求来源于微信
		return tmpStr != null ? tmpStr.equals(signature.toUpperCase()) : false;
	}
	
	/**
	 * 将字节数组转换为十六进制字符串
	 * 
	 * @param byteArray
	 * @return String
	 */
	private static String byteToStr(byte[] byteArray) {
		StringBuilder strDigest = new StringBuilder();
		for (int i = 0; i < byteArray.length; i++) {
			strDigest.append(byteToHexStr(byteArray[i]));
		}
		return strDigest.toString();
	}
	
	/**
	 * 将字节转换为十六进制字符串
	 * 
	 * @param mByte
	 * @return String
	 */
	private static String byteToHexStr(byte mByte) {
		char[] Digit = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };
		char[] tempArr = new char[2];
		tempArr[0] = Digit[(mByte >>> 4) & 0X0F];
		tempArr[1] = Digit[mByte & 0X0F];

		String s = new String(tempArr);
		return s;
	}
}

public class WXInterfaceUtil {
	private static final Logger logger = Logger
			.getLogger(WXInterfaceUtil.class);

	/**
	 * access_token有效期为7200秒,不需要每次都去调用
	 */
	public static String access_token = null;

	/**
	 * 获取access_token
	 * 
	 * @return String
	 */
	public static String getAccessToken() {
		if (null == access_token) {
			// 获取ACCESS_TOKEN
			String reqUrl = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid="
					+ WXConstants.APPID + "&secret=" + WXConstants.APPSECRET;
			String result = ReqUtil.get(reqUrl);
			logger.info("get access token,result:" + result);
			JSONObject obj = JSONObject.fromObject(result);
			access_token = (String) obj.get("access_token");
			logger.info("access_token: " + access_token);
		}
		return access_token;
	}

	/**
	 * 获取用户呢称
	 * 
	 * @param openID
	 * @return String
	 */
	public static String getNickname(String openID) {
		String reqUrl = "https://api.weixin.qq.com/cgi-bin/user/info?access_token="
				+ getAccessToken() + "&openid=" + openID + "&lang=zh_CN";
		String result = ReqUtil.get(reqUrl);
		logger.info("get nickname,result:" + result);
		JSONObject obj = JSONObject.fromObject(result);
		String nickname = (String) obj.get("nickname");
		
		logger.info("nickname: " + nickname);
		return nickname;
	}

	/**
	 * 创建临时二维码
	 * 
	 * @return String
	 */
	public static String createTempQrcode(String accessToken) {
		String reqUrl = "https://api.weixin.qq.com/cgi-bin/qrcode/create?access_token="
				+ accessToken;
		String params = "{\"expire_seconds\": " + WXConstants.EXPIRE_SECONDS
				+ ", \"action_name\": \"" + WXConstants.ACTION_NAME
				+ "\", \"action_info\": {\"scene\": {\"scene_id\": "
				+ WXConstants.SCENE_ID + "}}}";
		String result = ReqUtil.post(reqUrl, params);
		logger.info("create temp qrcode,result:" + result);
		JSONObject obj = JSONObject.fromObject(result);
		String ticket = (String)obj.get("ticket");
		logger.info("ticket: " + ticket);
		return ticket;
	}
}
分享到:
评论

相关推荐

    [net毕业设计]ASP.NET基于BS结构的实验室预约模型系统(源代码+论文).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    中医诊所系统,WPF.zip

    中医诊所系统,WPF.zip

    [net毕业设计]ASP.NET淘宝店主交易管理系统的设计与实现(源代码+论文).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    1-全国各省、297个地级市公路里程面板数据1999-2021年-社科数据.zip

    全国各省、297个地级市公路里程面板数据1999-2021年涵盖了中国各地区公路建设的详细情况,是衡量地区基础设施水平的重要指标。这些数据不仅包括了全国31个省份的公路里程,还深入到了297个地级市的层面,提供了从1999年至2021年的连续年份数据。这些数据来源于各省统计年鉴、经济社会发展统计数据库、地级市统计年鉴以及地级市发展统计公报,确保了数据的准确性和权威性。通过这些数据,可以观察到中国公路交通建设的发展不平衡性,沿海地区和长江中下游地区公路交通密度较高,而西部地区相对较低。这些面板数据为研究中国城市化进程、区域经济发展以及交通基础设施建设提供了宝贵的信息资源。

    技术处工作事项延期完成申请单.docx

    技术处工作事项延期完成申请单.docx

    数据库详细设计说明书中文最新版本

    本文为图书馆管理课程设计SQL Server功能规范说明书。本说明书将: 描述数据库设计的目的; 说明数据库设计中的主要组成部分; 说明数据库设计中各功能的实现。 本文档主要内容包括对数据库设计结构的总体描述,对数据库中各种对象的描述(包括对象的名称、对象的属性、对象和其他对象直接的关系);在数据库主要对象之外,本文还将描述数据库安全性设置、数据库属性设置和数据库备份策略,为数据库管理员维护数据库安全稳定地运行提供参考;有需要的朋友可以下载看看

    WebSocketError(解决方案).md

    项目中常见的问题,记录一下解决方案

    octopart(样本).csv

    octopart数据格式样例

    [net毕业设计]ASP.NET通用作业批改系统设计(源代码+论文).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    Oracle11gRAC安装与配置forLinux中文最新版本

    本文档主要讲述的是Oracle 11g RAC安装与配置for Linux;希望对大家的学习会有帮助 文档结构 第一部分:Oracle Grid Infrastructure安装 第二部分:Oracle Clusterware与Oracle Real Application Clusters安装前准备规程 第三部分:安装Oracle Clusterware与Oracle Real Application Clusters 第四部分:Oracle Real Application Clusters环境配置 第五部分:Oracle Clusterware与Oracle Real Application Clusters参考资料

    python教程.txt

    python教程.txt

    脸部痤疮检测数据集VOC+YOLO格式3763张7类别.zip

    文件太大放服务器下请务必到资源详情查看后然后下载 样本图:blog.csdn.net/2403_88102872/article/details/143979016 重要说明:数据集为小目标检测,训练map精度偏低属于正常现象,只要能检测出来即可。如果map低于0.5请勿奇怪,因为小目标检测是业界公认难检测的研究方向之一。 数据集格式:Pascal VOC格式+YOLO格式(不包含分割路径的txt文件,仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数):3763 标注数量(xml文件个数):3763 标注数量(txt文件个数):3763 标注类别数:7 标注类别名称:["blackheads","cyst","fore","nodule","papule","pustule","whiteheads"]

    ASP+ACCESS基于WEB社区论坛设计与实现(源代码+论文)(源代码+论文+说明文档).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    1-全国各地级市固定资产投资统计数据(附省、区县、行业)1996-2020年-社科数据.zip

    全国各地级市固定资产投资统计数据集覆盖了1996至2020年的时间跨度,提供了详尽的年度固定资产投资金额,单位为百万人民币。这些数据不仅包括了地级市级别的投资情况,还涵盖了省、区县以及行业等多个维度,为研究区域经济增长、投资结构和发展趋势提供了宝贵的数据支持。固定资产投资作为衡量一个地区经济发展活力和潜力的重要指标,反映了社会固定资产在生产、投资额的规模和速度。通过这些数据,研究人员可以深入分析不同地区、不同行业的投资特点,以及随时间变化的趋势,进而为政策制定和经济预测提供科学依据。

    training_plan_db.sql

    training_plan_db.sql

    [net毕业设计]ASP.NET多语种网络硬盘系统的设计(源代码+论文).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    5.html

    5

    1-全国各省地区城乡收入差距、泰尔指数、城镇农村居民可支配收入统计数据1990-2021年-社科数据.zip

    全国各省地区城乡收入差距、泰尔指数、城镇农村居民可支配收入统计数据集提供了1990至2021年间的详细数据,覆盖全国31个省份。该数据集不仅包括城镇居民和农村居民的人均可支配收入,还涵盖了乡村人口、全体居民人均可支配收入、城镇人口以及年末常住人口等关键指标。泰尔指数作为衡量收入不平等的重要工具,通过计算城镇收入与农村收入之比,为研究者提供了一个量化城乡收入差距的科学方法。这些数据不仅有助于分析中国城乡之间的经济差异,还能为政策制定者提供决策支持,以缩小城乡差距、促进区域均衡发展。数据集的丰富性使其成为社会科学领域研究城乡发展、收入分配不平等等问题的宝贵资源。

    FileName.zip

    FileName.zip

    java面向对象 - 类与对象代码.zip

    java面向对象 - 类与对象java面向对象 - 类与对象代码.zip

Global site tag (gtag.js) - Google Analytics