今天要解决的三个问题:
一、解决子控制器初始化问题?
二、解决参数实体类封装问题
三、解决结果集代码重复问题
?
?
一、解决子控制器初始化问题?
思路:
- ?参考DBAccess的数据源配置文件config.properties
? ? ? ? ? ? ? 1.减少代码改动的风险性 ?? ? ? ? ? ? ?2.减少代码的编译次数(对于已经部署到服务器后)
?? ?解决步骤 ?? ? ? ? ? ? ?1.必须有配置文件config.xml ?? ? ? ? ? ? ?2.配置文件config.xml中要包含处理业务的子控制器 ?? ? ? ? ? ? ?3.读取到配置文件config.xml中对应的处理浏览器请求的子控制器
private ConfigModel configModel=null;
@Override
public void init() throws ServletException {
try {//建模
configModel = ConfigModelFactory.build();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
二、解决参数实体类封装问题
思路: ?? ??? ? 1.不知道是哪一个实体类--》泛型 ? ? ? ? ?2.不知道这个实体类有哪些属性,但是又要给这些属性赋值-》反射动态设置属性值
解决:泛型+反射
编码用模型驱动接口:modelDriver
package com.zking.gwq.framework;
public interface ModelDriver<T> {
/**
* 是中央控制器来做实体类封装的:提供对象
* @return
*/
T getModel();
}
在子控制器实现它
package com.zking.gwq.framework;
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;
public class BookAction extends ActionSupport implements ModelDriver<Book>{
private Book book = new Book();
private String delete(HttpServletRequest req, HttpServletResponse resp) {
System.out.println("BookAction调用bookDao.delete()...");
/**
* 需求:删除数据后跳转到书籍展示页面
* 缺陷:1.出现大量重复代码 2.不好管理(增删改重新跳到查询页面)
* 解决方案:在一个地方统一管理重定向还是转发,并且是定位跳转到哪一个页面
* 预测结果
* BookAction调用bookDao.delete()...
* BookAction调用bookDao.list()...
*/
/* try {
req.getRequestDispatcher("/booklist.jsp").forward(req, resp);
// resp.sendRedirect("/booklist.jsp");
this.list(req, resp);
} catch (Exception e) {
e.printStackTrace();
}*/
return "toList";
}
private void add(HttpServletRequest req, HttpServletResponse resp) {
System.out.println("BookAction调用bookDao.add...");
}
private String list(HttpServletRequest req, HttpServletResponse resp) {
System.out.println("BookAction调用bookDao.list...");
/*try {
req.getRequestDispatcher("/booklist.jsp").forward(req, resp);
} catch (Exception e) {
e.printStackTrace();
}*/
return "list";
}
private String ref(HttpServletRequest req, HttpServletResponse resp) {
//解决参数实体类封装问题
/**
* 缺陷:对于前端传参,实体类属性值封装req.getParammeter("")代码 是重复的
* 需求:做一个book的新增功能
* 思考:将jsp参数塞进实体类的属性中,问题是每个实体类的属性是不一样的,没办法做到代码的完全重复
* 思路:
* 不知道是哪一个实体类--》泛型
* 不知道这个实体类有哪些属性,但是又要给这些属性赋值-》反射动态设置属性值
* 解决:泛型+反射
* 编码:
* 模型驱动接口:modelDriver
*/
/*Book book = new Book();*/
// book.setBid(req.getParameter("bid"));
// book.setBname(req.getParameter("bname"));
// book.setPrice(req.getParameter("price"));
// book.setAuthor(req.getParameter("author"));
// book.setPulish(req.getParameter("publish"));
System.out.println(book);
System.out.println("BookAction调用bookDao.ref(book)...");
/*req.getRequestDispatcher("index.jsp")*/
return "toList";
}
@Override
public Book getModel() {
return book;
}
}
去中央控制器接收
package com.zking.gwq.framework;
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.gwq.Books.GoodsAction;
/**
* 目标:
* 根据自定义mvc框架的原理图 完成 框架的研发
* @author T440s
* 中央控制器
*寻找子控制器
*/
@WebServlet("*.action")
public class DispathServlet extends HttpServlet {
//存放子控制器的容器
// private Map<String, ActionSupport> actions = new HashMap<String, ActionSupport>();
private ConfigModel configModel=null;
//初始化子控制器(集合),经过初始化,actions容器内部就有了子控制器
//init,service,destroy
/**
* 需求:
* 再增加一个商品类的增删查改
* 步骤:
* 改动init中代码
* 思考:
* 能不能不改动代码完成这个需求
* 参考DBAccess的数据源配置文件config.properties
* 1.减少代码改动的风险性
* 2.减少代码的编译次数(对于已经部署到服务器后)
*
* 解决方案:
* 改成子控制器可配置
* 解决步骤
* 1.必须有配置文件config.xml
* 2.配置文件config.xml中要包含处理业务的子控制器
* 3.读取到配置文件config.xml中对应的处理浏览器请求的子控制器
*
* 编码
* ...
*/
@Override
public void init() throws ServletException {
try {//建模
configModel = ConfigModelFactory.build();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
// actions.put("/book", new BookAction());
// actions.put("/goods", new GoodsAction());
// actions.put("/order", new BookAction());
// actions.put("/orderItem", new BookAction());
}
@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/t266_mvc/book.action?methodName=add
//目标:bookaction.add();
/**
* 思路:
* 1.从浏览器URL中获取到“/book”字符串
* 2.在子控制器容器中拿到BookAction
* 3.BookAction.add()
*/
String uri = req.getRequestURI();
uri = uri.substring(uri.lastIndexOf("/"), uri.lastIndexOf("."));
//action=BookAction
// ActionSupport action = config.get(uri);
// ActionSupport action = actions.get(uri);
ActionModel actionModel = configModel.pop(uri);
String type = actionModel.getType();
try {
ActionSupport action = (ActionSupport) Class.forName(type).newInstance();
if(action instanceof ModelDriver) {
ModelDriver m = (ModelDriver) action;
Object obj = m.getModel();
//有对象
//接收所有的前端jsp传递到后台的参数
Map<String, String[]> parameterMap = req.getParameterMap();
//对对象赋值
// PropertyUtils.getProperty(obj, "")
BeanUtils.populate(obj, parameterMap);
}
//execute--delete
String res = action.execute(req, resp);
/**
* 思路:
* 1.方法执行后必须有一个返回值
* 2.通过返回值决定是否重定向,还是转发
* 3.通过返回值决定跳转那一个页面
*/
ForwardModel forwardModel = actionModel.pop(res);
if(forwardModel.isRedirect()) {//代表是重定向
resp.sendRedirect(req.getContextPath()+forwardModel.getPath());
}else {//代表转发
req.getRequestDispatcher(forwardModel.getPath()).forward(req, resp);
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
?三、解决结果集代码重复问题
解决方案:在一个地方统一管理重定向还是转发,并且是定位跳转到哪一个页面
?思路: ? ? ? ? ? ? ?1.方法执行后必须有一个返回值
package com.zking.gwq.framework;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 1.运行/处理前台的浏览器请求
* 2.运行JSP传递到后台的方法字符串所代表的方法
* @author T440s
*
*/
public interface Action {
public String execute(HttpServletRequest req, HttpServletResponse resp);
}
? ? ? ? ? ? 2.通过返回值决定是否重定向,还是转发
private String delete(HttpServletRequest req, HttpServletResponse resp) {
System.out.println("BookAction调用bookDao.delete()...");
/**
* 需求:删除数据后跳转到书籍展示页面
* 缺陷:1.出现大量重复代码 2.不好管理(增删改重新跳到查询页面)
* 解决方案:在一个地方统一管理重定向还是转发,并且是定位跳转到哪一个页面
* 预测结果
* BookAction调用bookDao.delete()...
* BookAction调用bookDao.list()...
*/
/* try {
req.getRequestDispatcher("/booklist.jsp").forward(req, resp);
// resp.sendRedirect("/booklist.jsp");
this.list(req, resp);
} catch (Exception e) {
e.printStackTrace();
}*/
return "toList";
}
? ? ? ? ? ? 3.通过返回值决定跳转那一个页面
package com.zking.gwq.framework;
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.gwq.Books.GoodsAction;
/**
* 目标:
* 根据自定义mvc框架的原理图 完成 框架的研发
* @author T440s
* 中央控制器
*寻找子控制器
*/
@WebServlet("*.action")
public class DispathServlet extends HttpServlet {
//存放子控制器的容器
// private Map<String, ActionSupport> actions = new HashMap<String, ActionSupport>();
private ConfigModel configModel=null;
//初始化子控制器(集合),经过初始化,actions容器内部就有了子控制器
//init,service,destroy
/**
* 需求:
* 再增加一个商品类的增删查改
* 步骤:
* 改动init中代码
* 思考:
* 能不能不改动代码完成这个需求
* 参考DBAccess的数据源配置文件config.properties
* 1.减少代码改动的风险性
* 2.减少代码的编译次数(对于已经部署到服务器后)
*
* 解决方案:
* 改成子控制器可配置
* 解决步骤
* 1.必须有配置文件config.xml
* 2.配置文件config.xml中要包含处理业务的子控制器
* 3.读取到配置文件config.xml中对应的处理浏览器请求的子控制器
*
* 编码
* ...
*/
@Override
public void init() throws ServletException {
try {//建模
configModel = ConfigModelFactory.build();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
// actions.put("/book", new BookAction());
// actions.put("/goods", new GoodsAction());
// actions.put("/order", new BookAction());
// actions.put("/orderItem", new BookAction());
}
@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/t266_mvc/book.action?methodName=add
//目标:bookaction.add();
/**
* 思路:
* 1.从浏览器URL中获取到“/book”字符串
* 2.在子控制器容器中拿到BookAction
* 3.BookAction.add()
*/
String uri = req.getRequestURI();
uri = uri.substring(uri.lastIndexOf("/"), uri.lastIndexOf("."));
//action=BookAction
// ActionSupport action = config.get(uri);
// ActionSupport action = actions.get(uri);
ActionModel actionModel = configModel.pop(uri);
String type = actionModel.getType();
try {
ActionSupport action = (ActionSupport) Class.forName(type).newInstance();
if(action instanceof ModelDriver) {
ModelDriver m = (ModelDriver) action;
Object obj = m.getModel();
//有对象
//接收所有的前端jsp传递到后台的参数
Map<String, String[]> parameterMap = req.getParameterMap();
//对对象赋值
// PropertyUtils.getProperty(obj, "")
BeanUtils.populate(obj, parameterMap);
}
//execute--delete
String res = action.execute(req, resp);
/**
* 思路:
* 1.方法执行后必须有一个返回值
* 2.通过返回值决定是否重定向,还是转发
* 3.通过返回值决定跳转那一个页面
*/
ForwardModel forwardModel = actionModel.pop(res);
if(forwardModel.isRedirect()) {//代表是重定向
resp.sendRedirect(req.getContextPath()+forwardModel.getPath());
}else {//代表转发
req.getRequestDispatcher(forwardModel.getPath()).forward(req, resp);
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
?
|