【Java EE 学习 73】【数据采集系统第五天】【参与调查】【导航处理】【答案回显】【保存答案】

时间:2022-04-11 09:41:35

一、参与调查的流程

  单击导航栏上的“参与调查”按钮->EntrySurveyAction做出相应,找到所有的Survey对象并转发到显示所有survey对象的页面上供用户选择->用户单击其中一个调查进入调查页面->页面提供上一页、下一页、退出、保存等按钮供用户选择->用户单击退出直接返回到显示所有Survey对象的页面上去或者单击提交完成调查返回到参与调查的页面上去。

二、显示所有调查

  请求的方法是Action中的toSurveyPage方法

     public String toSurveyPage() throws Exception{
//重新获取survey对象
this.model=this.surveyService.getModelById(this.model.getSurveyId());
float requestPage;
//得到请求页,实际上就是orderNo
if(parameters.get("requestPage")!=null){
requestPage=Float.parseFloat(parameters.get("requestPage")[0]);
}else{
requestPage=this.model.getMinOrderNo();
}
Page responsePage=this.surveyService.getResponsePage(this.model,requestPage);
ServletActionContext.getContext().put("page",responsePage); //将两个参数保存到Session中去
this.session.put(this.CURRENT_SURVEY, this.model);
this.session.put(ALL_PARAMETERS, new HashMap<String,Map<String,String[]>>());
return "surveyPage";
}

  这里需要注意的是上面代码中的粗体部分,该部分实际上是为了方便调用以后回显功能、保存答案功能使用的。

  显示所有调查是以图表配以文字的形式进行说明的,使用的图标是之前说过的上传的logo,如果没有设置logo,则使用默认的logo。

 <s:iterator value="%{#surveys}" var="survey">
<s:a cssClass="survey" action="EntrySurveyAction_toSurveyPage.action" namespace="/">
<s:param name="surveyId" value="#survey.surveyId"></s:param>
<img width="200px" alt="<s:property value='%{#survey.title}'/>" src='<s:property value="getLogoPath(#survey.logoPath)"/>'/><br/>
<s:property value="#survey.title"/>
</s:a>
</s:iterator>

  这里还需要调用Action中的一个方法getLogoPath,该方法用于获取logo的时机url地址,方法如下:

     //判断logo是否存在如果存在返回路径,如果不存在,返回默认路径
public String getLogoPath(String logoPath){
System.out.println(logoPath);
if(logoPath!=null){
String realPath=this.servletConetxt.getRealPath(logoPath);
File file=new File(realPath);
if(file.exists()){
return servletConetxt.getContextPath()+logoPath;
}
}
return servletConetxt.getContextPath()+"/upload/defaultLogo.jpg";
}

三、参与调查的页面设计

  参与调查的页面一直是重用的,无论点击下一步还是上一步,返回的页面还是这一个页面,这肯定是请求了一个参数,这个参数就是页面的标识id。Action根据请求的id获取该Page对象的所有属性值,并将该Page对象返回到前端显示。根据该page的orderNo属性判断是否还有上一页、下一页并根据不同的结果显示出上一步、下一步、提交按钮,但是在每一页上都有“退出”按钮。

  上一篇笔记中提到的Page兑现的minOrderNo和maxOrderNo就是在这个时候用到的。

  导航的代码使用strus2标签控制判断是否显示上一页、下一页、完成这三个按钮:

 <!-- 导航栏界面的问题,包括上一题、下一题、退出按钮的设计 -->
<div
style="width: 30%; margin: 0 auto; text-align: center; margin-top: 30px; font-size: 20px;">
<s:if test="#page.orderNo!=#page.survey.minOrderNo">
<input type="submit"
value="<s:property value='#page.survey.submit_pre'/>"
name="submit_pre" />
</s:if>
<s:if test="#page.orderNo!=#page.survey.maxOrderNo">
<input type="submit"
value="<s:property value='#page.survey.submit_next'/>"
name="submit_next">
</s:if>
<s:if test="#page.orderNo==#page.survey.maxOrderNo">
<%-- <input type="submit"
value="<s:property value='#page.survey.submit_done'/>"
name="submit_done"> --%>
<s:submit name="submit_done" value="%{#page.survey.submit_done}"></s:submit>
</s:if>
<input type="submit"
value="<s:property value='#page.survey.submit_exit'/>"
name="submit_exit">
</div>

  关键是Survey对象的minOrderNo和maxOrderNo是什么时候赋值的,又是怎么获取到的。这里使用了hibernate的功能。在Survey的hibernate映射文件中使用一种新型的配置方式解决该问题。

<property name="maxOrderNo" type="float">
  <formula>(select max(p.orderNo) from page p where p.surveyId=surveyId)</formula>
</property>
<property name="minOrderNo" type="float">
  <formula>(select min(p.orderNo) from page p where p.surveyId=surveyId)</formula>
</property>

这是根据数据库中的相应信息将查询结果赋值到Bean对象中相关属性的一种方法。这些属性在数据库中并没有相关字段,但是在前端页面中会使用到。formula(表达式)标签中使用的是原生的sql语句。

  其余问题的显示可以直接拷贝设计调查页面中的设计代码稍微修改一下就好了。 

  参与调查页面的完整代码:

 <%@ page language="java" pageEncoding="utf-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<!--
天哪,简直不能认了,必须将struts2标签全部换成html标签才能正常
否则不能正常解析动态指定的name属性,怪不得那个老师不使用struts2标签,我还以为他是傻X呢 最后一个问题:下拉列表类型的问题,是不是必须要有个默认值才行,不是默认选择第一个吗。
但是通过统计答案的过程中发现在下拉列表框中如果没有给他一个默认值的话,就默认没有选择。
-->
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<title>Insert title here</title>
<link rel="stylesheet"
href="${pageContext.servletContext.contextPath}/css/header.css"
type="text/css">
<style type="text/css">
table {
border: 1px solid black;
border-collapse: collapse;
width: 100%;
}
tr td {
border: 1px solid black;
border-collapse: collapse;
} a {
color: gray;
text-decoration: none;
} a:HOVER {
color: red;
} .tdHL {
text-align: left;
border-width: 0px;
} .tdHR {
text-align: right;
border-width: 0px;
} .pageTitle {
background-color: #CCF;
} .questionTitle {
background-color: #CCC;
}
</style>
</head>
<body>
<%@ include file="/header.jsp"%>
这里是答题界面!
<s:form namespace="/" action="EntrySurveyAction_doEntrySurvey.action">
<div>
<s:set value="%{#page.survey.surveyId}" name="id" />
<!-- 当前页的页码 -->
<s:hidden name="pageId" value="%{#page.pageId}"></s:hidden>
<table>
<s:set var="pId" value="%{#page.pageId}"></s:set>
<tr>
<td>
<table style="display: inline-table;">
<tr class="pageTitle">
<!-- 页面标题 -->
<td colspan="2" width="40%" class="tdHL"><s:property
value="%{#page.title}" /></td>
</tr>
<tr>
<td colspan="2">
<table style="display: inline-table;">
<tr>
<td width="20px"></td>
<td>
<!-- 迭代问题的集合 -->
<table style="display: inline-table;">
<s:iterator value="%{#page.questions}" var="question">
<s:set var="qid" value="%{#question.questionId}"></s:set>
<!-- 问题题干 -->
<tr class="questionTitle">
<td colspan="2" class="tdHL"><s:property
value="%{#question.title}" /></td>
</tr>
<!-- 问题主体,主要涉及的问题就是问题的分类 -->
<tr>
<td colspan="2">
<!-- 定义变量,为当前类型的题型 --> <s:set
value="%{#question.questionType}" var="qt"></s:set> <!-- 第一种题型:带有单选框或者复选框的
题目标识就是题号小于4,0-3
--> <s:if test="#qt lt 4">
<s:iterator value="#question.optionTextArr" status="st">
<input name="q<s:property value='#qid'/>"
value='<s:property value="#st.index"/>'
type='<s:property value="#qt<2?'radio':'checkbox'"/>'
<s:property value="setTag(#pId+'','q'+#qid,#st.index,'checked')"/>
>
<s:property />
<s:if test="#qt==1 || #qt==3">
<br />
</s:if>
</s:iterator>
<!-- 处理other的情况 -->
<s:if test="#question.other">
<input name="q<s:property value='#qid'/>"
value="other"
<s:property value="setTag(#pId+'','q'+#qid,'other','checked')"/>
type='<s:property value="#qt<2?'radio':'checkbox'"/>'>其它
<s:if test="#question.otherType==1">
<input type="text" name='q<s:property value="#qid"/>other'
<s:property value="setText(#pId,'q'+#qid+'other')"/>
/>
</s:if>
<s:elseif test="#question.otherType==2">
<select name="q<s:property value='#qid'/>other">
<s:iterator value="#question.otherSelectOptionArr" status="st">
<option
<s:property value="setTag(#pId+'','q'+#qid+'other',#st.index,'selected')"/>
value='<s:property value="#st.index"/>'>
<s:property/>
</option>
</s:iterator>
</select>
</s:elseif>
</s:if>
</s:if>
<!-- 第二种题型,是下拉列表类型的题型 -->
<s:elseif test="#qt==4">
<select name="q<s:property value='#qid'/>">
<s:iterator value="#question.optionTextArr" status="st">
<option
<s:property value="setTag(#pId+'','q'+#qid,#st.index,'selected')"/>
value='<s:property value="#st.index"/>'><s:property/></option>
</s:iterator>
</select>
</s:elseif>
<s:elseif test="#qt==5">
<input type="text" name="q<s:property value='#qid'/>"
<s:property value="setText(#pId,'q'+#qid)"/>
/>
</s:elseif> <!-- 第三种题型,矩阵问题,6,7,8 --> <s:else>
<table style="display: inline-table;">
<!-- 列头 -->
<tr>
<td></td>
<s:iterator value="#question.matrixColTitleArr">
<td><s:property /></td>
</s:iterator>
</tr>
<!-- 输出N多行 -->
<s:iterator value="#question.matrixRowTitleArr"
status="rowst">
<tr>
<td><s:property /></td>
<s:iterator value="#question.matrixColTitleArr"
status="colst">
<td><s:if test="#qt==6">
<!-- 对于矩阵式单选按钮的问题,必须将每一行视为一个单独的问题 -->
<input type="radio"
value='<s:property value="#rowst.index+'_'+#colst.index"/>'
name="q<s:property value='#qid+"_"+#rowst.index'/>"
<s:property value="setTag(#pId+'','q'+#qid+'_'+#rowst.index,#rowst.index+'_'+#colst.index,'checked')"/>
>
</s:if> <s:elseif test="#qt==7">
<input type="checkbox"
value='<s:property value="#rowst.index+'_'+#colst.index"/>'
name="q<s:property value='#qid'/>"
<s:property value="setTag(#pId+'','q'+#qid,#rowst.index+'_'+#colst.index,'checked')"/>
>
</s:elseif> <s:elseif test="#qt==8">
<select name="q<s:property value='#qid'/>">
<s:iterator
status="selst"
value="#question.matrixSelectOptionArr">
<option value='<s:property value="#rowst.index+'_'+#colst.index+'_'+#selst.index"/>'
<s:property value="setTag(#pId+'','q'+#qid,#rowst.index+'_'+#colst.index+'_'+#selst.index,'selected')"/>
>
<s:property />
</option>
</s:iterator>
</select>
</s:elseif></td>
</s:iterator>
</tr>
</s:iterator>
</table>
</s:else>
</td>
</tr>
</s:iterator>
</table>
</td>
</tr>
</table>
</td>
</tr>
</table>
</td>
</tr>
</table>
<!-- 导航栏界面的问题,包括上一题、下一题、退出按钮的设计 -->
<div
style="width: 30%; margin: 0 auto; text-align: center; margin-top: 30px; font-size: 20px;">
<s:if test="#page.orderNo!=#page.survey.minOrderNo">
<input type="submit"
value="<s:property value='#page.survey.submit_pre'/>"
name="submit_pre" />
</s:if>
<s:if test="#page.orderNo!=#page.survey.maxOrderNo">
<input type="submit"
value="<s:property value='#page.survey.submit_next'/>"
name="submit_next">
</s:if>
<s:if test="#page.orderNo==#page.survey.maxOrderNo">
<%-- <input type="submit"
value="<s:property value='#page.survey.submit_done'/>"
name="submit_done"> --%>
<s:submit name="submit_done" value="%{#page.survey.submit_done}"></s:submit>
</s:if>
<input type="submit"
value="<s:property value='#page.survey.submit_exit'/>"
name="submit_exit">
</div>
</div>
</s:form>
</body>
</html>

surveyPage.jsp

四、答案回显

到三未知,已经能够实现页面的显示并通过上一页、下一页翻页了。下一步实现答案的回显。所谓的答案回显就是单击上一步按钮之后能够显示出所回答的问题。

第一步,首先需要解决将答案保存到哪里的问题,很明显,应该将答案保存到Session中,关键是数据结构的设计问题。

  (1)这里我们使用HashMap的数据结构来保存所有的答案:Map<String,Map<String,String[]>> allAnswers=new HashMap<String,HashMap<String,String[]>>();

  这里key值是pageId,value值是对应页面上的所有答案。

  (2)Action中还定义了一个Map<String,String[]>类型的字段,该字段是用于保存当前页中所有请求参数用的,当然也包括所有的答案,该字段和struts2中的params值相同,为此,Action需要实现ParameterAware接口,以方便通过parameterIntercptor拦截器获取到所有参数。

  (3)同时为了方便数据处理,还定义了一个“当前的调查(currentSurvey)”字段保存到了Session。

  这两个字段的初始化是在进入调查之前完成的,一下的代码中的粗体字段是初始化的过程。

     public String toSurveyPage() throws Exception{
//重新获取survey对象
this.model=this.surveyService.getModelById(this.model.getSurveyId());
float requestPage;
//得到请求页,实际上就是orderNo
if(parameters.get("requestPage")!=null){
requestPage=Float.parseFloat(parameters.get("requestPage")[0]);
}else{
requestPage=this.model.getMinOrderNo();
}
Page responsePage=this.surveyService.getResponsePage(this.model,requestPage);
ServletActionContext.getContext().put("page",responsePage); //将两个参数保存到Session中去
this.session.put(this.CURRENT_SURVEY, this.model);
this.session.put(ALL_PARAMETERS, new HashMap<String,Map<String,String[]>>());
return "surveyPage";
}

第二步,在单击下一步的时候保存答案到Session中的Map对象。

  为了解决这个问题,我们给上一步、下一步、退出、完成按钮进行了特殊的命名,

  上一步:submit_pre,下一步:submit_next,退出:submit_exit,完成:submit_done,这样我就能够比较方便的进行判断了。以submit开头是因为这么做比较方便的标记出这四个按钮的一种类型的按钮,方便以后的答案处理。

  一旦单击四个按钮中的任何一个按钮Action都会调用一下方法处理:

 public String doEntrySurvey() throws Exception{
Page oldPage=pageService.getPage(pageId);
String submitType=this.getSubmitType();
//上一页
if(submitType.endsWith("pre")){
//将答案保存到Session中
mergeIntoSession();
Page page=pageService.getPrePage(oldPage);
ActionContext.getContext().put("page", page);
return "surveyPage";
}
//下一页
if(submitType.endsWith("next")){
mergeIntoSession();
Page page=pageService.getNextPage(oldPage);
ActionContext.getContext().put("page", page);
return "surveyPage";
}
//完成
if(submitType.endsWith("done")){
//首先保存住所有的答案到Session
mergeIntoSession();
//然后保存所有的答案到数据库
saveAllAnswers();
return "toEntrySurveyPageAction";
}
//退出
if(submitType.endsWith("exit")){
//首先清除掉session中的所有答案
clearAnswersOnSession();
//返回到前端的调查页面,显示出所有可用的调查
return "toEntrySurveyPageAction";
}
return "";
}

  mergeIntoSession方法,该方法是将答案保存到Session的关键方法:

 //将Parameters中的相关答案选项取出保存到Session中
private void mergeIntoSession() {
Map<String,Map<String,String[]>>allAnswers=(Map<String, Map<String, String[]>>) this.session.get(this.ALL_PARAMETERS);
//将当前页面上的答案集合保存到所有答案的集合中
allAnswers.put(pageId+"", this.parameters); //将传递过来的所有参数全部保存起来
}

  当然代码很简单,这是因为简化处理了,将所有参数都保存了下来,包括非答案的部分。

第三步:回显

  对于非文本框的表单,我们使用setTag(pageId,questionId,propertyName)方法来设置属性值,对于文本框类型的表单,我们使用setText(pageId,questionId)方法来设置属性值。

  核心代码如下

 <!-- 定义变量,为当前类型的题型 --> <s:set
value="%{#question.questionType}" var="qt"></s:set> <!-- 第一种题型:带有单选框或者复选框的
题目标识就是题号小于4,0-3
--> <s:if test="#qt lt 4">
<s:iterator value="#question.optionTextArr" status="st">
<input name="q<s:property value='#qid'/>"
value='<s:property value="#st.index"/>'
type='<s:property value="#qt<2?'radio':'checkbox'"/>'
<s:property value="setTag(#pId+'','q'+#qid,#st.index,'checked')"/>
>
<s:property />
<s:if test="#qt==1 || #qt==3">
<br />
</s:if>
</s:iterator>
<!-- 处理other的情况 -->
<s:if test="#question.other">
<input name="q<s:property value='#qid'/>"
value="other"
<s:property value="setTag(#pId+'','q'+#qid,'other','checked')"/>
type='<s:property value="#qt<2?'radio':'checkbox'"/>'>其它
<s:if test="#question.otherType==1">
<input type="text" name='q<s:property value="#qid"/>other'
<s:property value="setText(#pId,'q'+#qid+'other')"/>
/>
</s:if>
<s:elseif test="#question.otherType==2">
<select name="q<s:property value='#qid'/>other">
<s:iterator value="#question.otherSelectOptionArr" status="st">
<option
<s:property value="setTag(#pId+'','q'+#qid+'other',#st.index,'selected')"/>
value='<s:property value="#st.index"/>'>
<s:property/>
</option>
</s:iterator>
</select>
</s:elseif>
</s:if>
</s:if>
<!-- 第二种题型,是下拉列表类型的题型 -->
<s:elseif test="#qt==4">
<select name="q<s:property value='#qid'/>">
<s:iterator value="#question.optionTextArr" status="st">
<option
<s:property value="setTag(#pId+'','q'+#qid,#st.index,'selected')"/>
value='<s:property value="#st.index"/>'><s:property/></option>
</s:iterator>
</select>
</s:elseif>
<s:elseif test="#qt==5">
<input type="text" name="q<s:property value='#qid'/>"
<s:property value="setText(#pId,'q'+#qid)"/>
/>
</s:elseif> <!-- 第三种题型,矩阵问题,6,7,8 --> <s:else>
<table style="display: inline-table;">
<!-- 列头 -->
<tr>
<td></td>
<s:iterator value="#question.matrixColTitleArr">
<td><s:property /></td>
</s:iterator>
</tr>
<!-- 输出N多行 -->
<s:iterator value="#question.matrixRowTitleArr"
status="rowst">
<tr>
<td><s:property /></td>
<s:iterator value="#question.matrixColTitleArr"
status="colst">
<td><s:if test="#qt==6">
<!-- 对于矩阵式单选按钮的问题,必须将每一行视为一个单独的问题 -->
<input type="radio"
value='<s:property value="#rowst.index+'_'+#colst.index"/>'
name="q<s:property value='#qid+"_"+#rowst.index'/>"
<s:property value="setTag(#pId+'','q'+#qid+'_'+#rowst.index,#rowst.index+'_'+#colst.index,'checked')"/>
>
</s:if> <s:elseif test="#qt==7">
<input type="checkbox"
value='<s:property value="#rowst.index+'_'+#colst.index"/>'
name="q<s:property value='#qid'/>"
<s:property value="setTag(#pId+'','q'+#qid,#rowst.index+'_'+#colst.index,'checked')"/>
>
</s:elseif> <s:elseif test="#qt==8">
<select name="q<s:property value='#qid'/>">
<s:iterator
status="selst"
value="#question.matrixSelectOptionArr">
<option value='<s:property value="#rowst.index+'_'+#colst.index+'_'+#selst.index"/>'
<s:property value="setTag(#pId+'','q'+#qid,#rowst.index+'_'+#colst.index+'_'+#selst.index,'selected')"/>
>
<s:property />
</option>
</s:iterator>
</select>
</s:elseif></td>
</s:iterator>
</tr>
</s:iterator>
</table>
</s:else>

  Action中定义了setTag方法和setText方法:

  setTag方法:

 public String setTag(String pageId,String name,String value,String tag){
Map<String,Map<String,String[]>>allAnswers=(Map<String, Map<String, String[]>>) this.session.get(this.ALL_PARAMETERS);
Map<String,String[]>pageAnswers=allAnswers.get(pageId);
if(pageAnswers!=null){
String []values=pageAnswers.get(name);
if(values!=null){
for(String temp:values){
if(temp.equals(value)){
return tag;
}
}
}
}
return "";
}

  setText方法:

 public String setText(String pageId,String name){
Map<String,Map<String,String[]>>allAnswers=(Map<String, Map<String, String[]>>) this.session.get(this.ALL_PARAMETERS);
Map<String,String[]>pageAnswers=allAnswers.get(pageId);
if(pageAnswers!=null){
String []values=pageAnswers.get(name);
if(values!=null){
return "value='"+values[0]+"'";
}
}
return "";
}

  两个方法中特别要注意的是判空操作一定要有,因为第一次访问页面的时候调用setTag方法由于Session中还没有相对应答案实体(pageId对应的答案)所以一定会报出空指针异常,但是被struts2的异常处理机制捕获到而且没有打印,直接就提示“没有找到相关方法”的提示,但是通过debug调试我却发现确实是找到了,实际上这是由于空指针异常导致的方法执行失败导致的,所以一定要判空,虽然不进行判空操作也没有什么问题,但是一次次的提示找不到方法,实在烦人。

  到这里,回显功能就已经实现了。

五、处理答案

  当单击“提交”的时候,Action会首先调用doEntrySurvey方法保存到所有答案到Session,然后调用saveAllAnswers方法处理所有的答案并保存到数据库,最后重定向到显示所有调查列表的界面。

  其核心是saveAllAnswers方法,该方法会过滤所有的答案选项并将答案分门别类的进行整理最后保存到数据库,处理的核心对象是allAnswers(HashMap<String,HashMap<String,String[]>>)。

  这里将针对矩阵式单选按钮类型的问题集进行单独处理,因为矩阵式单选按钮类型的问题有一个唯一的标识:以q开头,而且包含'_',所以比较容易单独划分出来。

  同时对于非矩阵式单选按钮类型的问题,默认为都带有其他项,这样比较方便处理。

  保存答案的方法:

 private void saveAllAnswers() {
/*
* 使用一个单独的Map封装所有的单选按钮矩阵题型
* Integer类型是questionId
* String类型是该题目中的所有单选按钮选项的集合
*/
Map<Integer,String>matrixRadioMap=new HashMap<Integer,String>();
//使用一个集合保存住所有的答案
List<Answer>answers=new ArrayList<Answer>();
Map<String,Map<String,String[]>>allAnswers=(Map<String, Map<String, String[]>>) this.session.get(this.ALL_PARAMETERS);
for(Map<String,String[]>map:allAnswers.values()){
String key;
String[] values;
for(Map.Entry<String, String[]>entry:map.entrySet()){
key=entry.getKey();
values=entry.getValue();
//首先将所有q开头的参数都挑选出来,表明是答案的选项
if(key.startsWith("q")){
//这是最普通的一种答案,只是q开头,然后后面跟上一个题目的标号id
if(!key.contains("other")&&!key.contains("_")){
Answer answer=new Answer();
answer.setAnswerIndexs(StringUtils.arr2String(values)); //设置所有的答案选项
// answer.setAnswerTime(new Date()); //设置答题时间,这个字段先不要填写
answer.setQuestionId(getQuestionId(key)); //设置问题id
answer.setSurveyId(getSurveyId()); //设置调查id
// answer.setUuid(null); //设置回答批次:回答批次不能乱写,一次回答中的所有回答都必须使用一个回答批次
//处理其他项,默认所有题目都带有其他项,这样方便处理,注意,这里必须在map变量中查找
String[] otherValuesArr=map.get(key+"other");
answer.setOtherAnswer(StringUtils.arr2String(otherValuesArr)); //如果有其它答案的话设置其它答案
answers.add(answer);
}else if(key.contains("_")){ //如果name属性值中带有"_",表名一定是矩阵式单选按钮的问题
Integer questionId=getMatrixRadioQuestionId(key);
String oldValue=matrixRadioMap.get(questionId);
if(oldValue!=null){
matrixRadioMap.put(questionId, oldValue+","+StringUtils.arr2String(values));
}else{
matrixRadioMap.put(questionId,StringUtils.arr2String(values));
}
}
}
}
//最后集中处理单选按钮矩阵式问题,将所有答案保存到anwers中
saveRadioMatrixMapToAnwers(answers,matrixRadioMap);
} /**
* 在这里创建SurveyToken对象并绑定到当前线程
* TODO 使用分库把保存答案:除了修改配置文件之外,还需要修改该处。
*/
SurveyToken surveyToken=new SurveyToken();
Survey survey=this.surveyService.getModelById(getSurveyId());
surveyToken.setSurvey(survey);
SurveyToken.bind(surveyToken); writeAnswersToDB(answers);
}

  由于矩阵式单选按钮类型的问题需要单独处理,所以另外使用方法将该类问题答案保存到Map(新开的只存放答案的Map,数据结构为Map<Integer,String>,key值为questionId,value值为答案列表)。

 //将矩阵式单选按钮的问题保存到answes中
private void saveRadioMatrixMapToAnwers(List<Answer> answers, Map<Integer, String> matrixRadioMap) {
Integer key;
String value;
for(Map.Entry<Integer, String>entry:matrixRadioMap.entrySet()){
key=entry.getKey();
value=entry.getValue();
Answer answer=new Answer();
answer.setAnswerIndexs(value);
// answer.setAnswerTime(new Date());
answer.setQuestionId(key);
answer.setSurveyId(getSurveyId());
answers.add(answer);
}
}

  最后集中保存到数据库中

 /**
* 将答案全部写到数据库
* @param answers
*/
private void writeAnswersToDB(List<Answer> answers) {
for(Answer answer:answers){
answerService.saveAnswer(answer);
}
}

  至此,已经能够将采集到的答案保存到数据库中了,下一步进行数据统计。