在枯燥的理论学习中,也没有学到什么值得一提的技术。
后来在一些练手的项目中开始遇到困难,开始有了一个程序员经常遇到的状况:发现问题--尝试解决--尚未解决--换个方法--还是报错--......--原来如此,可能有时候困得更久,但是我却渐渐喜欢这种感觉,因为我知道问题终会解决,而被这个问题困得越久,最后的收获越大,留下的印象也越深,成就感自然更大啦。
我真的是越来越喜欢这种感觉了,享受失败的痛苦和成功的喜悦。但是面对问题可以坚持去解决,千万不能一条路走死,我时常提醒自己要清醒,特别是面对问题的时候,要站在当前的问题点上观察,有时候还要回退到全局的角度,看看你一开始是否走错了路,导致进了死胡同。这就像是走迷宫,你可以沿着一条路走下去,你也可以回到最后一个分叉点,再走第二条路。你更可以站在高处,俯瞰一切,重新定制走路的方案。
这段时间没事,找别人要了一个商城项目的模板,然后根据别人设计的架构开始做,期间遇到了不少问题,但是我还是要说下这个让我印象深刻的问题。
先上原来别人写的代码(也不知这究竟谁写的,总有可以学习的地方):
public class UserLoginAction extends ActionSupport{
private static final Logger log = Logger.getLogger(UserLoginAction.class);
private static final long serialVersionUID = 1L;
protected static MD5 md5 = new MD5();
private String userCode;
private String password;
private String checkCode;
private String keepTime;
private String[] isRemberPass;
private ByteArrayInputStream imageStream;
private String RedirectPath;
private String message;
private UserServiceImpl userService;//这是我之前没有注意的地方
private HashMap<String, String> INFOMAP = new HashMap<String, String>() {
private static final long serialVersionUID = 1L;
{
put("nameNotNull", "用户名不能为空");
put("passNotNull", "密码不能为空");
put("codeNotNull", "验证码不能为空");
put("codeErro", "验证码错误");
put("isBlack", "抱歉,该用户被列为黑名单,不能进行登陆");
put("infoErro", "用户名密码错误");
put("userNotPass", "该代理人信息暂未进行审核,请耐心等待");
put("userNotActive", "该代理人审核不通过或未激活,请重新申请或激活");
put("userNotExist", "用户不存在");
}
};
public String login(){
HttpServletRequest req=ServletActionContext.getRequest();
HttpServletResponse res=ServletActionContext.getResponse();
String backUrl="loginPage";
String flag="";
//数据验证
if (StringUtils.isBlank(userCode)) {
flag="nameNotNull";
}
if (StringUtils.isBlank(password)) {
flag="passNotNull";
}
if (StringUtils.isBlank(checkCode)) {
flag="codeNotNull";
}
if (checkCode!=null&&!checkCode.equals((String)req.getSession(true).getAttribute("adminRand"))) {
flag="codeErro";
}
if(!StringUtils.isBlank(flag)){
message=INFOMAP.get(flag);
return backUrl;
}
String validateString = md5.MD5Encode(password);
if(userService==null)
userService=(UserServiceImpl) SpringBeanFactory.getBean("userService");//这是直接转换为实现类了
User oper = userService.queryByCode(userCode);
if (oper != null) {
flag=userService.login(oper,validateString,req,res);
} else {
flag="userNotExist";
message=INFOMAP.get(flag);
}
if("redirectURL".equals(flag)){
RedirectPath = (String) req.getSession().getAttribute("RedirectPath");
req.getSession().removeAttribute("RedirectPath");
setCookie(res, req, userCode, password, isRemberPass, keepTime);
return "redirectUrl";
}else if("redirectIndex".equals(flag)){
setCookie(res, req, userCode, password, isRemberPass, keepTime);
return "loginSucc";
}else{
message=INFOMAP.get(flag);
return backUrl;
}
}
}
我准备先写后台的角色管理部分,可是每次都要登陆还要输可恶的验证码,于是我稍作修改实现免登陆:
public class UserLoginAction extends ActionSupport{
private static final Logger log = Logger.getLogger(UserLoginAction.class);
private static final long serialVersionUID = 1L;
protected static MD5 md5 = new MD5();
private String userCode = "esteban";
private String password = "fcs";
private String checkCode = "hah";//强行进入 免登陆
private String keepTime;
private String[] isRemberPass;
private ByteArrayInputStream imageStream;
private String RedirectPath;
private String message;
private UserServiceImpl userService;
private HashMap<String, String> INFOMAP = new HashMap<String, String>() {
private static final long serialVersionUID = 1L;
{
put("nameNotNull", "用户名不能为空");
put("passNotNull", "密码不能为空");
put("codeNotNull", "验证码不能为空");
put("codeErro", "验证码错误");
put("isBlack", "抱歉,该用户被列为黑名单,不能进行登陆");
put("infoErro", "用户名密码错误");
put("userNotPass", "该代理人信息暂未进行审核,请耐心等待");
put("userNotActive", "该代理人审核不通过或未激活,请重新申请或激活");
put("userNotExist", "用户不存在");
}
};
public String login(){
HttpServletRequest req=ServletActionContext.getRequest();
HttpServletResponse res=ServletActionContext.getResponse();
String backUrl="loginPage";
String flag="";
//数据验证
if (StringUtils.isBlank(userCode)) {
flag="nameNotNull";
}
if (StringUtils.isBlank(password)) {
flag="passNotNull";
}
if (StringUtils.isBlank(checkCode)) {
flag="codeNotNull";
}
if (checkCode!=null&&!checkCode.equals((String)req.getSession(true).getAttribute("adminRand"))) {
flag="";//改动处
}
if(!StringUtils.isBlank(flag)){
message=INFOMAP.get(flag);
return backUrl;
}
String validateString = md5.MD5Encode(password);
if(userService==null)
userService= (UserServiceImpl) SpringBeanFactory.getBean("userService");
User oper = userService.queryByCode(userCode);
if (oper != null) {
flag=userService.login(oper,validateString,req,res);
} else {
flag="userNotExist";
message=INFOMAP.get(flag);
}
if("redirectURL".equals(flag)){
RedirectPath = (String) req.getSession().getAttribute("RedirectPath");
req.getSession().removeAttribute("RedirectPath");
setCookie(res, req, userCode, password, isRemberPass, keepTime);
return "redirectUrl";
}else if("redirectIndex".equals(flag)){
setCookie(res, req, userCode, password, isRemberPass, keepTime);
return "loginSucc";
}else{
message=INFOMAP.get(flag);
return backUrl;
}
}
}
添加角色写完后一看没有配置事物,于是作如下配置(部分代码):
<bean id="txManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
<property name="sessionFactory" ref="mySessionFactory"></property>
</bean>
<tx:advice id="ta" transaction-manager="txManager">
<tx:attributes>
<tx:method name="add*" propagation="REQUIRED"/>
<tx:method name="del*" propagation="REQUIRED"/>
<tx:method name="update*" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut id="interceptorPointCuts" expression="execution(* com.esteban.*.service.impl.*.*(..))" />
<aop:advisor advice-ref="ta" pointcut-ref="interceptorPointCuts"/>
</aop:config>
可是一加上面的<aop:config>部分,设置了切面和切点, 后台首页就进不去了,由于后台首页是通过jQuery的ajax方法获取树形菜单选项,我在页面上通过开发人员工具找到了响应参数,显示错误发生在 MenuTreeAction中的treeJSON方法:
public void treeJSON(){
HttpServletRequest req = ServletActionContext.getRequest();
HttpServletResponse res = ServletActionContext.getResponse();
if(userService==null)
userService = (UserServiceImpl) SpringBeanFactory.getBean("userService");//报错点
if(menuTreeService==null)
menuTreeService = (MenuTreeService) SpringBeanFactory.getBean("menuTreeService");
/**
* TODO 登录本该放到最后处理 否则不便于测试
* 这里我手动给予权限 为了避开登录界面
*/
UserLoginAction ula = new UserLoginAction();
ula.login();
User oper = WebUtils.getOper(req);
List<String> rights = userService.getOperRights(oper);
String str = "";
if(rights != null){
str="[";
List<TreeMenu> listOne = menuTreeService.queryTreeMenu(nodeGrade, parentNode,rights);
str += getHtmlONE(listOne, rights,menuTreeService);
str += "]";
}
try {
res.setHeader("ContentType", "text/json");
res.setCharacterEncoding("UTF-8");
res.getWriter().print(str);
} catch (IOException e) {
e.printStackTrace();
}
}
使用debug跟踪进去才找到了异常:
java.lang.ClassCastException: com.sun.proxy.$Proxy0 cannot be cast to com.esteban.admin.service.impl.UserServiceImpl
说代理类无法转换为实现类,我仔细一看这行是他原来写的,表面上并没有错误,他直接将获取的bean转换为实现类,可是一旦我添加了AOP来面向切面管理事物,由于实现了service接口,会使用JDK动态代理来产生代理类,这时候获取的bean应该是生成的代理类,而这个代理类无法转换为实现类的类型,必须是他们共同的接口才行。上面肯定是原作者不小心造成的,因为在这个action里面定义的是接口UserService类型,下面转型却转成实现类了。于是我把它换为接口类型。
再回到最上面UserLoginAction的代码一看,红色部分直接定义的UserServiceImpl实现类类型,我想这不会又是失误吧,于是我也把它改为接口类型,这时下面的代码爆红了,一查看发现service实现类中不仅实现了接口中的方法,还自己添加了其他的方法,这时我向上转型后,接口中由于没有定义实现类额外的方法,所以那些方法就无法调用。于是我又把实现类中所有的方法在接口中定义了一遍。原来这不是失误,而是原作者没有意识到他这样的设计会对后面产生什么影响,直接定义实现类类型,这不是面向接口编程,无法实现多态,也让spring的AOP发挥不了功效。
这次遇到的问题可以总结一下:
1.要学会质疑
2.遇到问题要看透本质,不要轻易放弃
3.面向接口编程
4.想要更好的运用框架,还是要熟悉原理