?1.在编写代码时,子控制器中初始化代码会有重复问题,于是用一个xml文件代替Map
Cpu(中央核心)
package com.zking.frame;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.beanutils.BeanUtils;
import org.apache.commons.beanutils.PropertyUtils;
import com.zking.web.BookCupi;
import com.zking.web.GoodsCupi;
/**
* 目标: 根据自定义mvc框架的原理图 完成 框架的研发
*
* Cpu中央控制器
*
* @author 花火
*
* 寻找子控制器
*/
@WebServlet("*.action")
public class Cpu extends HttpServlet {
//子控制器不只一个,有专门存放子控制器的地方
//private Map<String, CupiE> ie = new HashMap<String, CupiE>();
private ConfigModel com=null;
//初始化子控制器集合,经过初始化,ie容器就有子控制器集合
//init ,Serviet, destroy
/**
* & 缺陷:指的是需要改变别人的代码,并伴随风险性
* 目标:
* 1.解决子控制器中初始化代码重复问题
* 需求:
* 增加一个商品的增删改查
* 步骤:
* 改动init中代码
* 思考:
* 能不能不改动代码完成这个解决子控制器中初始化需求
* 参考DBAccess的数据源配置文件config.properties
* 1.减少代码改动的风险性
* 2.减少代码的编译次数(对于已经部署到服务器后)
* 解决方案
* 改成子控制器可配置
* 解决步骤:
* 1.必须有配置文件config.xml
* 2.配置文件config.xml中包含处理业务的子控制器
* 3.读取到配置文件config.xml中对应的处理浏览器请求的子控制器
* 编码
*
*/
@Override
public void init() throws ServletException {
// 初试化子控制器
try {
com=ConfigModelFactory.build();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
/*ie.put("/book", new BookCupi());
ie.put("/goods", new GoodsCupi());*/
}
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doPost(req, resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 寻找子控制器
// 浏览器:http//localhost:8080/MVC/book.action?methodName=add
// 目标:BookCupiE.add();
/**
* 思路 1.从浏览器URL中获取"/book"字符串 2.在子控制器容器中拿到Cupie 3.找到方法
*/
// 获取uri
String uri = req.getRequestURI();
// 截取
uri = uri.substring(uri.lastIndexOf("/"), uri.lastIndexOf("."));
/*CupiE c = ie.get(uri);*/
ActionModel acm = com.pop(uri);
String type = acm.getType();
CupiE c=null;
try {
c = (CupiE)Class.forName(type).newInstance();
if(c instanceof ModelDriver) {
ModelDriver m=(ModelDriver) c;
Object obj = m.getModel();
//有对象
//接受所有的前端Jsp传递到后台的参数
Map<String, String[]> pm = req.getParameterMap();
//对对象赋值
//PropertyUtils.getProperty(obj, name)
BeanUtils.populate(obj, pm);
}
String res = c.execute(req, resp);
ForwardModel pc = acm.pop(res);
if(pc.isRedirect()) {
resp.sendRedirect(req.getContextPath()+pc.getPath());
System.out.println(req.getContextPath()+pc.getPath());
}else {
req.getRequestDispatcher(pc.getPath()).forward(req, resp);
}
/**
* 思路
* 1.方法执行完毕,必须要有返回值
* 2.通过返回值是否要重定向还是转发
* 3.通过返回值决定要跳那个页面
*/
} catch (Exception e) {
e.printStackTrace();
}
}
//YUXIG
}
2.实现实体类自动化实例化 &解决转发req和重定向resp代码重复问题
package com.zking.web;
import java.io.IOException;
import java.lang.reflect.Field;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import com.zking.entity.Book;
import com.zking.frame.CupiE;
import com.zking.frame.ModelDriver;
public class BookCupi extends CupiE implements ModelDriver<Book> {
private Book b = new Book();
private void add(HttpServletRequest req, HttpServletResponse resp) {
// TODO Auto-generated method stub
System.out.println("bookDao.add()..");
}
private String delete(HttpServletRequest req, HttpServletResponse resp) {
System.out.println("bookDao.delete()..");
/**
* 需求:删除成功后跳转到书籍展示页面
* 缺陷:1.出现大量重复代码
* 2.不好管理(增删改重新跳到查询页面)
* 解决方案://YUXIG在一个地方统一管理重定向还是转发,并定位是跳到哪一个页面
*/
/*try {
req.getRequestDispatcher("/book.jsp").forward(req, resp);
this.list(req, resp);
resp.sendRedirect("/book.jsp");
} catch (Exception e) {
e.printStackTrace();
}*/
return "toList";
}
private void edit(HttpServletRequest req, HttpServletResponse resp) {
// TODO Auto-generated method stub
System.out.println("bookDao.edit()..");
}
private void ref(HttpServletRequest req, HttpServletResponse resp) {
// 解决参数实体类封装问题
/**
* 缺陷://YUXIG对于前端传参,实体类属性值封装req.getParammeter("")代码是重复的 需求:做一个book的新增功能
* 思考:将jsp参数塞进实体类的属性中,问题是每一个实体类的属性是不一样的,没办法做到代码的完全重复 不知道是哪一个实体类->泛型
* 不知道这个实体类有哪些属性,但又要给这些属性赋值->反射动态设置属性值 解决:泛型+反射 编码:模型驱动接口:ModelDriver
*
*//*
*
* Field[] ds = b.getClass().getDeclaredFields();
* b.setBname(req.getParameter("bname")); b.setPrice(req.getParameter("price"));
* b.setAuthor(req.getParameter("author"));
* b.setPublish(req.getParameter("publish"));
*/
System.out.println(b);
System.out.println("bookDao.ref()..");
}
private void load(HttpServletRequest req, HttpServletResponse resp) {
// TODO Auto-generated method stub
System.out.println("bookDao.load()..");
}
private String list(HttpServletRequest req, HttpServletResponse resp) throws Exception{
// TODO Auto-generated method stub
System.out.println("bookDao.list()..");
return "list";
}
@Override
public Book getModel() {
// //YUXIG TODO Auto-generated method stub
return b;
}
}
xml
<?xml version="1.0" encoding="UTF-8"?>
<config>
<action path="/book" type="com.zking.web.BookCupi">
<forward name="list" path="/book.jsp" redirect="false" />
<!-- //YUXIG增删改操作之后要再一次查询数据库的新的数据做展示 -->
<forward name="toList" path="/book.action?methodName=list" redirect="true" />
</action>
</config>