关于MVC模式
MVC和三层架构的区别?
三层架构是一个经典的分层思想,将开发模式分为三层,每个人专注自己擅长模块即可
MVC是一种设计模式,其目的是让html和业务逻辑分开
MVC的概念
MVC:Model View Controller,其中Model(模型层)、View(视图层)、Controller(控制层)
它是一种软件设计典范,用于业务逻辑处理、数据、界面显示分离
V(视图层)--》 JSP
C(控制层)--》 Servlet/Action
M(模型层)--》 Dao、Entity
1、实体域模型(entity层)
2、过程域模型(dao层)
MVC模式不能跨层调用,只能由上往下进行调用;View -> Controller -> Model
优点
1、低耦合性
前后端分离,更有效率。
视图层和业务层分离,这样就允许更改视图层代码而不用重新编译模型和控制器代码,同样,一个应用的业务流程或者业务规则的改变只需要改动MVC的模型层即可。因为模型与控制器和视图相分离,所以很容易改变应用程序的数据层和业务规则。
2、高重用性和可适用性
MVC模式允许你使用各种不同样式的视图来访问同一个服务器端的代码。它包括任何WEB(HTTP)浏览器或者无线浏览器(wap),比如,用户可以通过电脑也可通过手机来订购某样产品,虽然订购的方式不一样,但处理订购产品的方式是一样的。由于模型返回的数据没有进行格式化,所以同样的构件能被不同的界面使用。例如,很多数据可能用HTML来表示,但是也有可能用WAP来表示,而这些表示所需要的命令是改变视图层的实现方式,而控制层和模型层无需做任何改变。
3、较低的生命周期成本
MVC使开发和维护用户接口的技术含量降低。
4、快速的部署
使用MVC模式使开发时间得到相当大的缩减,它使程序员(Java开发人员)集中精力于业务逻辑,界面程序员(HTML和JSP开发人员)集中精力于表现形式上。
5、可维护性
分离视图层和业务逻辑层也使得WEB应用更易于维护和修改。
6、有利于软件工程化管理
由于不同的层各司其职,每一层不同的应用具有某些相同的特征,有利于通过工程化、工具化管理程序代码。
缺点:
使用mvc,会增加代码量、相应地也会增加软件开发的成文,设计的难度也会增加。一些小项目,完全可以不用。
自定义MVC工作原理
JSP --------> Servlet(中央控制器)--------->Action(子控制器)--------->Model(Dao、DB)数据库操作
解释:由jsp传入请求到主控制器,主控制器调试截取请求路径名,获取子控制器,由子控制器来处理具体业务逻辑。
案例
下面我们用基本的主控制器和子控制器的运用来实现一个简单的计算器功能
首先定义一个实体类Cal
public class Cal {
private String num1;
private String num2;
public String getNum1() {
return num1;
}
public void setNum1(String num1) {
this.num1 = num1;
}
public String getNum2() {
return num2;
}
public void setNum2(String num2) {
this.num2 = num2;
}
public Cal(String num1, String num2) {
super();
this.num1 = num1;
this.num2 = num2;
}
public Cal() {
super();
}
}
然后定义我们的主控制器
/**
* 主控制器
* @author 86135
*
*/
public class DispatcherServlet extends HttpServlet {
private static final long serialVersionUID = -7094025920085803724L;
private Map<String, Action> actionMap=new HashMap<>();
//初始化方法
public void init() {
//添加子控制器
actionMap.put("/addCal", new AddCalAction());
actionMap.put("/delCal", new DelCalAction());
actionMap.put("/chenCal", new ChenCalAction());
actionMap.put("/chuCal", new ChuCalAction());
}
@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 {
init();//初始化
String url = req.getRequestURI();//Mvc/xxx.action
url = url.substring(url.lastIndexOf("/"), url.lastIndexOf("."));
// 相当于向上转型
Action action = actionMap.get(url);//根据url获取子控制器
action.execute(req, resp);
}
}
定义子控制器接口,来处理业务逻辑
/**
* 子控制器
* 专门用来处理业务逻辑
* @author 86135
*
*/
public interface Action {
void execute(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException ;
}
然后定义加减乘除的类,实现子控制接口
加法
public class AddCalAction implements Action {
@Override
public void execute(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String num1=req.getParameter("num1");
String num2=req.getParameter("num2");
req.setAttribute("res", Integer.valueOf(num1)+Integer.valueOf(num2));
req.getRequestDispatcher("calRes.jsp").forward(req, resp);
}
}
减法
public class DelCalAction implements Action {
@Override
public void execute(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String num1=req.getParameter("num1");
String num2=req.getParameter("num2");
req.setAttribute("res", Integer.valueOf(num1)-Integer.valueOf(num2));
req.getRequestDispatcher("calRes.jsp").forward(req, resp);
}
}
乘法
public class ChenCalAction implements Action {
@Override
public void execute(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String num1=req.getParameter("num1");
String num2=req.getParameter("num2");
req.setAttribute("res", Integer.valueOf(num1)*Integer.valueOf(num2));
req.getRequestDispatcher("calRes.jsp").forward(req, resp);
}
}
除法
public class ChuCalAction implements Action {
@Override
public void execute(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String num1=req.getParameter("num1");
String num2=req.getParameter("num2");
req.setAttribute("res", Integer.valueOf(num1)/Integer.valueOf(num2));
req.getRequestDispatcher("calRes.jsp").forward(req, resp);
}
}
然后配置web
<servlet>
<servlet-name>dispatcherServlet</servlet-name>
<servlet-class>com.xy.framework.DispatcherServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>dispatcherServlet</servlet-name>
<url-pattern>*.action</url-pattern>
</servlet-mapping>
然后再jsp页面测试
<script type="text/javascript">
function doSub(value){
if(value==1){
calForm.action="${pageContext.request.contextPath}/addCal.action";
}
else if(value==2){
calForm.action="${pageContext.request.contextPath}/delCal.action";
}
else if(value==3){
calForm.action="${pageContext.request.contextPath}/chenCal.action";
}
else{
calForm.action="${pageContext.request.contextPath}/chuCal.action";
}
calForm.submit();
}
</script>
</head>
<body>
<form id="claForm" name="calForm" action="">
num1:<input type="text" name="num1" /><br/>
num2:<input type="text" name="num2" /><br/>
<input type="button" onclick="doSub(1)" value="+" />
<input type="button" onclick="doSub(2)" value="-" />
<input type="button" onclick="doSub(3)" value="x" />
<input type="button" onclick="doSub(4)" value="/" />
</form>
测试页面
我们点一下乘法