微信公眾平台開發—成語接龍


因為申請的是個人未認證的訂閱號,開啟開發者模式以后沒有菜單,只能使用接收消息—回復消息的方式來開發

 

一、實現思路

1、成語接龍題庫

成語接龍的題庫比24點的復雜一些,數量也很大,我建的表字段是:"四字成語"、"首字"、"首字拼音"、"尾字"、"尾字拼音"、"全字拼音"、"含義"、"出自"、"例句"這幾個,未來可以支持查詢讀音、含義、出自和例句的功能

題庫來自http://download.csdn.net/index.php/mobile/source/download/opensrc/8385371

找了很久的,有3w多個成語信息,我把內容拷到excel里,截取了上面需要的字段,然后復制到數據庫表里,去掉了重復的和不是四個字的成語,最后數據3w不到一點

2、接龍方式

自己試了一下正常的成語接龍(就是首尾字完全一樣的接),感覺沒幾個就接不下去了,所以加入了諧音接龍(首尾字諧音相同)和包含接龍(包含有那個字,不一定是第一個),這樣接龍連勝可以長一點

3、出題

從29543中取一個隨機數,作為成語id,查詢成語信息然后返回

4、計算答案

(1)首先判斷答案是否在題庫里面出現(是一個成語)

(2)判斷題目尾字是否和答案首字一樣,是就相同匹配,加3分

(3)判斷題目尾字拼音是否和答案首字拼音一樣,是就諧音匹配,加1分

(4)判斷題目尾字是否在答案4字中存在,而且題目和答案不是同一個成語,是就包含匹配,加1分

(5)完成判斷以后更新用戶的"積分、答題數、答對數、命中率、連勝次數、最長連勝"

5、再次出題

根據答案尾字在題庫表中查詢成語,如果得到一個列表,隨機取出一個成語返回(防止每次返回匹配的第一個成語),如果沒有結果,就提示"我也接不下去了",用戶積分加5..然后重新走開始的出題步驟出下一題

6、排行榜

按用戶表的積分大小倒序輸出排名即可,需要注意的是如果前10名沒有當前用戶,要再輸出一行當前用戶的排名。還可以支持最長連勝排行榜

7、答題狀態

出題后用戶會切換成答題狀態,回復的內容除了關鍵字都會當做答案來計算,回復"開始"、"答案"、"公告"答案后會再切換成正常狀態

和24點游戲不同的是這個是一直接下去的(所以增加了連勝次數和最長連勝字段)

 

二、核心源碼

package org.xs.dntown.wx.idiom.modules.web;

import java.util.Date;
import java.util.List;
import java.util.Random;

import javax.servlet.http.HttpServletRequest;

import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.xs.dntown.wx.common.utils.MessageUtil;
import org.xs.dntown.wx.common.web.BaseController;
import org.xs.dntown.wx.core.modules.entity.UserInfo;
import org.xs.dntown.wx.core.modules.entity.enums.ModuleEnum;
import org.xs.dntown.wx.core.modules.entity.req.BaseMsgReq;
import org.xs.dntown.wx.core.modules.service.LogService;
import org.xs.dntown.wx.core.modules.service.UserService;
import org.xs.dntown.wx.idiom.modules.entity.IdiomListInfo;
import org.xs.dntown.wx.idiom.modules.entity.IdiomUserInfo;
import org.xs.dntown.wx.idiom.modules.entity.enums.StepEnum;
import org.xs.dntown.wx.idiom.modules.service.IdiomListService;
import org.xs.dntown.wx.idiom.modules.service.IdiomUserService;

/**
 * 核心Controller
 */
@Controller(value = "idiomCoreController")
@RequestMapping("idiom")
public class CoreController extends BaseController {
	
	@Autowired
	private IdiomUserService idiomUserService;
	
	@Autowired
	private IdiomListService idiomListService;
	
	@Autowired
	private UserService userService;
	
	@Autowired
	private LogService logService;
	
	/**
	 * 文本消息
	 */
	@ResponseBody
	@RequestMapping(value = "msg", produces = { "text/plain;charset=utf-8" })
	public String textMsg(HttpServletRequest request) throws Exception {
		String result = "";
		String openId = "";
		String userName = "";
		String content = "";
		try {
			
			//得到接收消息實體
			BaseMsgReq msgReq = (BaseMsgReq)request.getAttribute("msgReq");
			openId = msgReq.getFromUserName();
			content = msgReq.getContent();
			
			//得到用戶信息實體
			UserInfo userInfo = (UserInfo)request.getAttribute("userInfo");
			userName = userInfo.getUserName();
			
			log.info("收到消息:" + msgReq.getContent());
			
			//更新模塊
			userService.setModule(msgReq.getFromUserName(), ModuleEnum.idiom.getValue());
			
			//得到接龍用戶信息實體
			IdiomUserInfo iUserInfo = idiomUserService.getByOpenId(userInfo.getOpenId());
			if(iUserInfo == null) {
				log.info("新增接龍用戶");
				
				//如果沒有就新增
				iUserInfo = new IdiomUserInfo();
				iUserInfo.setOpenId(userInfo.getOpenId());
				iUserInfo.setModuleStep(StepEnum.normal.getValue());
				iUserInfo.setStepTime(new Date());
				iUserInfo.setIsAnswered(false);
				idiomUserService.add(iUserInfo);
			}
			
			//刷新模塊(1小時)
			long timeDiff = new Date().getTime() - userInfo.getModuleTime().getTime();
			long hoursDiff = timeDiff / (1000 * 60 * 60);
			if(hoursDiff >= 1) {
				idiomUserService.setStep(userInfo.getOpenId(), StepEnum.normal.getValue());
				iUserInfo.setModuleStep(StepEnum.normal.getValue());
			}
			
			log.info("開始轉入步驟");
			//出題
			if(msgReq.getContent().equals("開始")) {
				
				//完成上一個成語
				if(!StringUtils.isEmpty(iUserInfo.getIdiomId()) && iUserInfo.getModuleStep().equals(StepEnum.answer.getValue())) {
					//處理答題結果
					disposeAnswer(iUserInfo, msgReq);
					
					result += finishAnswer(iUserInfo, msgReq, false) + "\n";
				}
				
				log.info("開始下一題");
				
				//下一題
				result = nextIdiom(iUserInfo, result);
				
				logService.addInfoLog(openId, userName, content, result, ModuleEnum.idiom.getValue(), StepEnum.answer.getValue());
				result += MessageUtil.msgReqToXml(msgReq, result);
				log.info(result);
				return result;
			}
			//答案
			else if(msgReq.getContent().equals("答案") && iUserInfo.getModuleStep().equals(StepEnum.answer.getValue())) {
				
				//返回答案
				if(!StringUtils.isEmpty(iUserInfo.getIdiomId())) {
					
					log.info("返回答案");
					
					//完成答題
					result += finishAnswer(iUserInfo, msgReq, true);
					
					//設置步驟
					iUserInfo.setModuleStep(StepEnum.normal.getValue());
					iUserInfo.setStepTime(new Date());
					idiomUserService.update(iUserInfo);
					
					logService.addInfoLog(openId, userName, content, result, ModuleEnum.game24.getValue(), StepEnum.answer.getValue());
					result = MessageUtil.msgReqToXml(msgReq, result);
					log.info(result);
					return result;
				}
			}
			//排行榜
			else if(msgReq.getContent().equals("排行") || msgReq.getContent().equals("排行榜")) {
				
				log.info("返回排行榜");
				
				result += "【排行榜】\n";
				
				boolean flag = false;
				List<IdiomUserInfo> list = idiomUserService.getTopScope(1000);
				if(list != null && list.size() > 0) {
					for (int i = 0; i < list.size(); i++) {
						if(list.get(i).getOpenId().equals(iUserInfo.getOpenId())) {
							flag = true;
						}
						//TODO 優化
						UserInfo sUserInfo = userService.getByOpenId(list.get(i).getOpenId());
						if(sUserInfo != null) {
							result += "No." + (i + 1) + " 『" + sUserInfo.getUserName() + "』(" + list.get(i).getScore() + "分)\n";
						}
					}
				}
				
				if(!flag) {
					result += "No." + (Integer.parseInt(idiomUserService.getRankScope(iUserInfo.getScore()))) + " 『" + userInfo.getUserName() + "』(" + iUserInfo.getScore() + "分)";
				} else {
					result = result.substring(0, result.length() - 1);
				}
				
				logService.addInfoLog(openId, userName, content, result, ModuleEnum.idiom.getValue(), StepEnum.normal.getValue());
				result = MessageUtil.msgReqToXml(msgReq, result);
				log.info(result);
				return result;
			}
			
			//如果是新進來就刷新狀態
			if(msgReq.getContent().equals("接龍")) {
				
				//完成答題
				if(!StringUtils.isEmpty(iUserInfo.getIdiomId())) {
					finishAnswer(iUserInfo, msgReq, true);
				}
				
				//設置步驟
				iUserInfo.setModuleStep(StepEnum.normal.getValue());
				iUserInfo.setStepTime(new Date());
				idiomUserService.update(iUserInfo);
			}
			
			//如果是答題狀態
			if(iUserInfo.getModuleStep().equals(StepEnum.answer.getValue())) {
				
				//返回處理答題結果
				result = disposeAnswer(iUserInfo, msgReq);
				logService.addInfoLog(openId, userName, content, result, ModuleEnum.idiom.getValue(), StepEnum.answer.getValue());
				result = MessageUtil.msgReqToXml(msgReq, result);
				log.info(result);
				return result;
			}
			
			//返回接龍公告
			result += "『成語接龍』\n";
			result += "【規則】\n";
			result += "有3種接龍方式:\n";
			result += "與尾字相同接龍,積分+3\n";
			result += "與尾字諧音接龍,積分+1\n";
			result += "包含尾字接龍,積分+1\n";
			result += "連勝有積分獎勵\n";
			result += "【說明】\n";
			result += "1、輸入“開始”:開始答題\n";
			result += "2、輸入“答案”:完成答題\n";
			result += "3、輸入“排行”:查看排行\n";
			result += "4、輸入“公告”:返回首頁";
			logService.addInfoLog(openId, userName, content, result, ModuleEnum.idiom.getValue(), StepEnum.normal.getValue());
			result = MessageUtil.msgReqToXml(msgReq, result);
		} catch (Exception e) {
			log.debug("執行失敗");
			logService.addErrorLog(openId, userName, content, "", ModuleEnum.idiom.getValue(), StepEnum.normal.getValue(), e);
			e.printStackTrace();
		}
		
		//返回消息
		log.info(result);
		return result;
	}
	
	/**
	 * 獲得下一題
	 */
	private String nextIdiom(IdiomUserInfo iUserInfo, String result) {
		//下一題
		Random r = new Random();
		String questionId = String.valueOf(r.nextInt(29543) + 1);
		iUserInfo.setIdiomId(questionId);
		iUserInfo.setModuleStep(StepEnum.answer.getValue());
		iUserInfo.setStepTime(new Date());
		iUserInfo.setIsAnswered(false);
		//更新模塊
		idiomUserService.update(iUserInfo);
		//返回題目
		IdiomListInfo idiomListInfo = idiomListService.getById(questionId);
		if(!StringUtils.isEmpty(result)) {
			result += "\n";
		}
		result += idiomListInfo.getWord();
		return result;
	}
	
	/**
	 * 處理答題結果
	 */
	private String disposeAnswer(IdiomUserInfo iUserInfo, BaseMsgReq msgReq) {
		
		log.info("處理答題結果");
		
		String result = "";
		boolean flag = true;
		int score = 0;
		
		//上一個成語
		IdiomListInfo idiomListInfo = idiomListService.getById(iUserInfo.getIdiomId());
		
		//獲得回答的成語
		IdiomListInfo idiomAnswerInfo = idiomListService.getByWord(msgReq.getContent());
		
		//沒有這個成語
		if(idiomAnswerInfo == null) {
			result = "這不是成語吧";
			flag = false;
		} else {
			
			//判斷是否匹配
			//如果是相同匹配
			if(idiomListInfo.getWordEnd().equals(idiomAnswerInfo.getWordBegin())) {
				result = "答對了,積分+3\n";
				flag = true;
				score = 3;
			}
			//如果是諧音匹配
			else if(idiomListInfo.getWordEndSpell().equals(idiomAnswerInfo.getWordBeginSpell())) {
				result = "答對了,諧音,積分+1\n";
				flag = true;
				score = 1;
			}
			//如果是包含匹配
			else if(idiomAnswerInfo.getWord().indexOf(idiomListInfo.getWordEnd()) > -1) {
				if(idiomListInfo.getId().equals(idiomAnswerInfo.getId())) {
					//不能是同一個成語
					result = "不能是同一個成語";
					flag = false;
				} else {
					result = "答對了,包含,積分+1\n";
					flag = true;
					score = 1;
				}
			}
			//不匹配
			else {
				result = "成語沒接上";
				flag = false;
			}
		}
		
		//回答正確
		if(flag) {
			
			log.info("回答正確");
			
    		//下一題
    		idiomListInfo = idiomListService.getNextIdiom(idiomAnswerInfo.getWordEnd());
    		if(idiomListInfo != null) {
    			iUserInfo.setIdiomId(idiomListInfo.getId());
    			iUserInfo.setStepTime(new Date());
    			iUserInfo.setIsAnswered(false);
    			
    			//返回題目
    			if(!StringUtils.isEmpty(result)) {
    				result += "\n";
    			}
    			result += idiomListInfo.getWord();
    		} else {
    			result += "我接不下去了,積分+5\n";
    			score += 5;
    			result = nextIdiom(iUserInfo, result);
    		}
    		
    		//更新用戶積分、次數、答對次數、連勝次數、最長連勝
			iUserInfo.setScore(iUserInfo.getScore() + score);
			iUserInfo.setRightTimes(iUserInfo.getRightTimes() + 1);
			iUserInfo.setTimes(iUserInfo.getTimes() + 1);
			iUserInfo.setComboTimes(iUserInfo.getComboTimes() + 1);
    		if(iUserInfo.getComboTimes() > iUserInfo.getBestTimes()) {
    			iUserInfo.setBestTimes(iUserInfo.getComboTimes());
    		}
    		iUserInfo.setStepTime(new Date());
    		iUserInfo.setIsAnswered(false);
		}
		//回答錯誤
		else {
			
			log.info("回答錯誤");
			
			//如果第一次回答
        	if(!iUserInfo.getIsAnswered()) {
        	
	        	//更新用戶積分、次數、答對次數、連勝次數、最長連勝
        		iUserInfo.setTimes(iUserInfo.getTimes() + 1);
        		iUserInfo.setIsAnswered(true);
        	}
		}
		
		//更新接龍用戶
		idiomUserService.update(iUserInfo);
		
		return result;
	}
	
	/**
	 * 完成答題
	 */
	private String finishAnswer(IdiomUserInfo iUserInfo, BaseMsgReq msgReq, boolean flag) {
		
		log.info("完成答題");
		
		String result = "";
		
		//上一個成語
		IdiomListInfo idiomListInfo = idiomListService.getById(iUserInfo.getIdiomId());
		
		//下一題
		if(flag) {
			idiomListInfo = idiomListService.getNextIdiom(idiomListInfo.getWordEnd());
			if(idiomListInfo != null) {
				result += "答案:" + idiomListInfo.getWord() + "\n\n";
			} else {
				result += "答案我也不知道\n\n";
			}
		}
		
		int score = 0;
		
		int comboTimes = iUserInfo.getComboTimes();
		if(comboTimes > 1) {
			int rank = comboTimes / 5 + 1;
			score += Math.pow(2, rank);
		}
		
		//更新用戶積分、次數、答對次數、連勝次數、最長連勝
		iUserInfo.setScore(iUserInfo.getScore() + score);
		iUserInfo.setStepTime(new Date());
		iUserInfo.setIsAnswered(false);
		iUserInfo.setComboTimes(0);
		
		//更新接龍用戶
		idiomUserService.update(iUserInfo);
		
		result += "答對" + comboTimes + "題,連勝獎勵積分+" + score;
		return result;
	}
}

 

三、演示地址

 

四、源碼地址

https://github.com/ctxsdhy/dntown

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM