Java微信公眾平台開發(八)–多媒體消息回復之音樂
- 2019 年 10 月 5 日
- 筆記
我們上一篇寫了關注出發圖片的回復。想著在發送一次音樂,最後基於回復消息分類情況下,實現一個簡單的只能話回復。先附一張大致效果圖。

下面我們進入程式碼階段。
(一)修改消息轉發器MsgDispatcher類,增加識別用戶發出消息功能,這裡實現了回複數字1,2,3,按內容分類回復,大致程式碼如下。
1 String openid = map.get("FromUserName"); // 用戶openid 2 String mpid = map.get("ToUserName"); // 公眾號原始ID 3 // 普通文本消息 4 TextMessage txtmsg = new TextMessage(); 5 txtmsg.setToUserName(openid); 6 txtmsg.setFromUserName(mpid); 7 txtmsg.setCreateTime(new Date().getTime()); 8 txtmsg.setMsgType(MessageUtil.RESP_MESSAGE_TYPE_TEXT); 9 10 if (map.get("MsgType").equals(MessageUtil.REQ_MESSAGE_TYPE_TEXT)) { // 文本消息 11 StringBuffer sb = new StringBuffer(); 12 String content = map.get("Content"); 13 sb.append("歡迎關注史上最帥公眾號:nn "); 14 sb.append("1、歌德你好 nn "); 15 sb.append("2、聽首歌吧 nn "); 16 sb.append("3、語音回復 nn "); 17 sb.append("回復?調出主菜單哦哦 n "); 18 String context = sb.toString(); 19 txtmsg.setContent(context); 20 21 if ("1".equals(content)) { 22 txtmsg.setContent("歌德你好!"); 23 } else if ("2".equals(content)) { 24 // Test test=new Test(); 25 // test.image(); 26 MusicMessage mucmsg = new MusicMessage(); 27 mucmsg.setToUserName(openid); 28 mucmsg.setFromUserName(mpid); 29 mucmsg.setCreateTime(new Date().getTime()); 30 mucmsg.setMsgType(MessageUtil.RESP_MESSAGE_TYPE_MUSIC); 31 32 Music music = new Music(); 33 HttpPostUploadUtil util = new HttpPostUploadUtil("thumb"); 34 String filepath = "H:\1.jpg"; 35 Map<String, String> textMap = new HashMap<String, String>(); 36 textMap.put("name", "testname"); 37 Map<String, String> fileMap = new HashMap<String, String>(); 38 fileMap.put("userfile", filepath); 39 String mediaidrs = util.formUpload(textMap, fileMap); 40 System.out.println(mediaidrs); 41 String mediaid = JSONObject.fromObject(mediaidrs).getString("thumb_media_id"); 42 System.out.println(mediaid); 43 music.setTitle("十年"); 44 music.setThumbMediaId(mediaid); 45 music.setDescription("十年——陳奕迅"); 46 music.setMusicUrl("http://music.163.com/#/song?id=31877628"); 47 music.setHQMusicUrl("http://music.163.com/#/song?id=31877628"); 48 mucmsg.setMusic(music); 49 return MessageUtil.musicMessageToXml(mucmsg); 50 } else if ("3".equals(content)) { 51 txtmsg.setContent("語音回復!"); 52 } else if ("?".equals(content)) { 53 txtmsg.setContent(context); 54 } else { 55 txtmsg.setContent("你好,歡迎來到gede部落格!"); 56 } 57 return MessageUtil.textMessageToXml(txtmsg); 58 }
簡單的來說就是對music類的封裝。開發過程中,HttpPostUploadUtil util = new HttpPostUploadUtil("thumb");是我後來重寫過的,需要什麼類型就將類型以字元串的方式插入。程式碼如下:
1 public HttpPostUploadUtil(String type) { 2 urlStr = "http://file.api.weixin.qq.com/cgi-bin/media/upload?access_token=" 3 + GlobalConstants.getInterfaceUrl("access_token") + "&type=" + type; 4 }
但是在後來的測試中發現,當類型不是image時,返回的不再是media_id這個屬性。而是將其類型作為前綴返回,我們直接看返回結果:
{"type":"thumb","thumb_media_id":"jooBl6SX9svP1rw4EA7ZnSX49QD4xoj_z59V8-dm3pXcSb-e4EXn-QnwL3jbf4nh","created_at":1558938915}
在我們上面的程式碼中,我直接將thumb_media_id設置成了定值,String mediaid = JSONObject.fromObject(mediaidrs).getString("thumb_media_id");為了j加強程式碼的寵用性,我在這裡重寫了HttpPostUploadUtil中的formUpload方法,大致程式碼如下:
1 public String formUpload(String filePath,String type) throws IOException { 2 File file = new File(filePath);// 判斷文件是否存在 3 if (!file.exists() || !file.isFile()) { 4 throw new IOException("文件不存在"); 5 } 6 System.out.println(urlStr); 7 URL urlobj = new URL(urlStr); 8 // 使用HttpURLConnection連接 9 HttpURLConnection con = (HttpURLConnection) urlobj.openConnection(); 10 con.setRequestMethod("POST"); 11 con.setDoInput(true); 12 con.setDoOutput(true); 13 con.setUseCaches(false); // 使用post提交需 要設置忽略快取 14 // 設置請求頭資訊 15 con.setRequestProperty("Connection", "Keep- Alive"); 16 con.setRequestProperty("Charset", "UTF-8"); 17 // 設置邊界 18 long currentTime = System.currentTimeMillis(); 19 String BOUNDARY = "---------" + currentTime; 20 con.setRequestProperty("Content -Type", "multipart/ form-data; boundary=" + BOUNDARY); 21 StringBuilder sb = new StringBuilder(); 22 sb.append("--"); 23 sb.append(BOUNDARY); 24 sb.append("rn"); 25 sb.append("Content -Di sposition: form-data ;name="file";filename="" + file.getName() + ""rn"); 26 sb.append("Content - Type:applicat ion/octet-streamrnrn"); 27 byte[] head = sb.toString().getBytes("utf-8"); 28 // 獲得輸出流 29 OutputStream out = new DataOutputStream(con.getOutputStream());// 輸出表頭 30 out.write(head); 31 // 文件正文部分 32 // 把文件已流文件的方式推入到ur1中 33 DataInputStream in = new DataInputStream(new FileInputStream(file)); 34 int bytes = 0; 35 byte[] bufferOut = new byte[1024]; 36 while ((bytes = in.read(bufferOut)) != -1) { 37 out.write(bufferOut, 0, bytes); 38 } 39 in.close(); 40 // 結尾部分 41 byte[] foot = ("rn--" + BOUNDARY + "--rn").getBytes("utf-8");// 定義最後數據分隔線 42 out.write(foot); 43 out.flush(); 44 out.close(); 45 StringBuffer buffer = new StringBuffer(); 46 BufferedReader reader = null; 47 String result = null; 48 try { 49 reader = new BufferedReader(new InputStreamReader(con.getInputStream())); 50 String line = null; 51 while ((line = reader.readLine()) != null) { 52 buffer.append(line); 53 } 54 if (result == null) { 55 result = buffer.toString(); 56 } 57 } catch (Exception e) { 58 // TODO: handle exception 59 e.printStackTrace(); 60 } finally { 61 // TODO: handle finally clause 62 if (reader != null) { 63 reader.close(); 64 } 65 } 66 JSONObject jsonObj = JSONObject.fromObject(result); 67 System.out.println(jsonObj); 68 String typeName = "media_id"; 69 if (!"image".equals(type)) { 70 typeName = type + "_mediaid"; 71 } 72 String mediaId = jsonObj.getString(typeName);// 從json中獲取media_id 73 return mediaId; 74 75 }
重寫後的方法,直接反回我們需要的media_id。
最後今天開發中遇到的一些莫名的bug記錄一下:
1、一開始我的項目啟動直接初始化失敗,quartz連接超時。搞了半天沒有發現為什麼,因為配置和jar包都沒有修改過,所以我最後刪除了tomcat,反覆清理伺服器,最終跑起來了。以後再遇到補充完整錯誤吧。只找到這一條錯誤搜索記錄。
parsing XML document from class path resource [spring-quartz.xml]; nested ex
2、微信伺服器允許上傳最大的thumb文件好像時32k,我沒有找到官方給出的大小。當圖片過大時,報以下錯誤。
40006錯誤程式碼
{"errcode":40006,"errmsg":"invalid meida size hint: [nHa_JA0174e298]"}
3、一個比較穩定的音樂解析網站給大家
地址:https://music.liuzhijin.cn/
4、程式碼又抽風了
報了不知道什麼的錯,個人感覺應該是access_token次數上限了。中午睡覺忘了關項目。
[ERROR][2019-05-27 15:09:00,495][com.gede.wechat.quartz.QuartzJob]net.sf.json.JSONException: JSONObject["access_token"] not found. net.sf.json.JSONException: JSONObject["access_token"] not found. at net.sf.json.JSONObject.getString(JSONObject.java:2092) at com.gede.wechat.common.WeChatTask.getToken_getTicket(WeChatTask.java:30) at com.gede.wechat.quartz.QuartzJob.workForToken(QuartzJob.java:19) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.lang.reflect.Method.invoke(Method.java:497) at org.springframework.util.MethodInvoker.invoke(MethodInvoker.java:269) at org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean$MethodInvokingJob.executeInternal(MethodInvokingJobDetailFactoryBean.java:311) at org.springframework.scheduling.quartz.QuartzJobBean.execute(QuartzJobBean.java:113) at org.quartz.core.JobRunShell.run(JobRunShell.java:223) at org.quartz.simpl.SimpleThreadPool$WorkerThread.run(SimpleThreadPool.java:549)