问题与成长

        在枯燥的理论学习中,也没有学到什么值得一提的技术。

        后来在一些练手的项目中开始遇到困难,开始有了一个程序员经常遇到的状况:发现问题--尝试解决--尚未解决--换个方法--还是报错--......--原来如此,可能有时候困得更久,但是我却渐渐喜欢这种感觉,因为我知道问题终会解决,而被这个问题困得越久,最后的收获越大,留下的印象也越深,成就感自然更大啦。

        我真的是越来越喜欢这种感觉了,享受失败的痛苦和成功的喜悦。但是面对问题可以坚持去解决,千万不能一条路走死,我时常提醒自己要清醒,特别是面对问题的时候,要站在当前的问题点上观察,有时候还要回退到全局的角度,看看你一开始是否走错了路,导致进了死胡同。这就像是走迷宫,你可以沿着一条路走下去,你也可以回到最后一个分叉点,再走第二条路。你更可以站在高处,俯瞰一切,重新定制走路的方案。

        这段时间没事,找别人要了一个商城项目的模板,然后根据别人设计的架构开始做,期间遇到了不少问题,但是我还是要说下这个让我印象深刻的问题。


先上原来别人写的代码(也不知这究竟谁写的,总有可以学习的地方):

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.想要更好的运用框架,还是要熟悉原理


================================== 赵客缦胡缨,吴钩霜雪明。 银鞍照白马,飒沓如流星。 ==================================
原文地址:https://www.cnblogs.com/lucare/p/9312680.html