zoukankan      html  css  js  c++  java
  • 问题与成长

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

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

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

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


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

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


    ================================== 赵客缦胡缨,吴钩霜雪明。 银鞍照白马,飒沓如流星。 ==================================
  • 相关阅读:
    System.Web.HttpException: 应用程序已预编译,因此不允许使用目录“/App_Code/”。
    ASP.NET导出Excel文件
    简单易学的OA报表
    input添加邮箱的时候自动显示后缀
    Sql Server中存储过程执行很快,但程序调用时执行非常慢的问题(连接超时)
    C#如何卸载已安装的Windows Service服务
    CSS专题(二):元素大小与位置offsetLeft offsetTop offsetWidth offsetHeight clientWidth clientHeight scrollWidth scrollHeight scrollLeft scrollTop
    Eclipse maven构建springmvc项目
    49个jQuery代码经典片段
    CSS专题(一):Background
  • 原文地址:https://www.cnblogs.com/lucare/p/9312680.html
Copyright © 2011-2022 走看看