4.Java Web开发模式(javaBean+servlet+MVC)
Java Web开发模式
一、Java Web开发模式
1.javaBean简介
JavaBeans是Java中一种特殊的类,可以将多个对象封装到一个对象(bean)中。特点是可序列化,提供无参构造器,提供getter方法和setter方法访问对象的属性。名称中的“Bean”是用于Java的可重用软件组件的惯用叫法。 --from 维基百科
JavaBean是一种可重复使用、且跨平台的软件组件。JavaBean可分为两种:
- 一种是有用户界面(UI,User Interface)的JavaBean;
- 另一种是没有用户界面,==主要负责处理事务(如数据运算,操纵数据库)的JavaBean
1)特点:
- JavaBean是一种Java类,而且是一种特殊的、可重用的类。
- JavaBean必须具有无参数的构造器,所有的属性都是private的,通过提供setter和getter方法来实现对成员属性的访问。
- Javabean 是为了和 jsp 页面传数据化简交互过程而产生的。
2.开发模式分类
-
模式一(JSP+JavaBean): 简便、灵活,在小规模、业务逻辑简单的项目开发中有一定优势,开发效率高
JSP页面通过JavaBean处理数据,响应请求并返回结果
-
模式二(JSP+Servlet+JavaBean): 程序层次清晰、分工明确,可维护性、扩展性高,尤其在规模较大或是业务逻辑复杂的项目中倾向使用
1)模式一特点
- 模式 一 体系结构用于开发简单的应用程序
- 模式 一体系结构包括多个用户可与之交互的页面
- 客户端能够直接访问到服务器上的JSP页面
- 在采用模式 一 开发的Web 应用程序中混杂了大量的业务逻辑代码,HTML内容、Java代码交织在一起,使程序的维护性和扩展性较差
- 在JSP页面中可以通过链接等方式直接转向其他页面。在业务逻辑较为复杂的项目中管理页面流程较为困难。
2)模式二特点
模式 二 体系结构结合使用 JSP 页面、Servlet和 JavaBean 来开发 Web 应用程序
二、MVC模式
1.简介
MVC是英文“Model-View-Controller”的缩写,最初是在Smalltalk-80中被用来构建用户界面的。其中M代表模型Model,V代表视图View,C代表控制器Controller。
通过模型视图控制器来
采用MVC模式的目的,就是为了增加代码的重用率,减少数据表达、数据描述和提高应用操作的偶合度。同时也使得软件的可维护性、可修复性、可扩展性、灵活性以及封装性大大提高。
2.MVC模式结构
MVC,把一个应用的输入、处理、输出流程按照Model、View、Controller的方式进行分离,这样一个应用将被分成三层:模型层、视图层、控制层。
- 输入>>模型层(Model)
- 处理>>视图层(View)
- 输出>>控制层(Controller)
controller负责处理客户端发送的请求,经过service层业务处理并间接调用mapper接口层的数据库操作,后将结果返回给视图层
客户端请求>>controller>>service>>mapper
1)Model层
分为: DAO层、service层
-
DAO层:
负责访问数据库进行数据的操作,取得结果集之后将结果集中的数据取出封装到VO类对象之后返回给service层
负责数据库操作,取得结果返回给service层
public interface HrMapper {int deleteByPrimaryKey(Long id);int insert(Hr record);int insertSelective(Hr record);Hr selectByPrimaryKey(Long id);int updateByPrimaryKeySelective(Hr record);int updateByPrimaryKey(Hr record);Hr login(@Param("username") String username, @Param("password") String password);List<Hr> queryAll();List<Hr> queryByUsername(String username); }
-
service层:
主要负责一些业务处理,比如多个操作需要放在一个事务中进行管理,事务回滚,一些复杂的逻辑业务处理就放到service层
通过调用DAO层接口实现业务功能
Service层的业务实现,具体要调用到已定义的DAO层的接口。封装Service层的业务逻辑有利于通用的业务逻辑的独立性和重复利用性。
@Service("hrService") public class HrServiceImpl implements HrService {@Autowiredprivate HrMapper hrMapper;@Overridepublic Hr login(String username, String password) {return hrMapper.login(username, password);}@Overridepublic List<Hr> queryAll() {return hrMapper.queryAll();}@Overridepublic int insertSelective(Hr record) {return hrMapper.insertSelective(record);}@Overridepublic int deleteByPrimaryKey(Long id) {return hrMapper.deleteByPrimaryKey(id);}@Overridepublic Hr selectByPrimaryKey(Long id) {return hrMapper.selectByPrimaryKey(id);}@Overridepublic int updateByPrimaryKeySelective(Hr record) {return hrMapper.updateByPrimaryKeySelective(record);}@Overridepublic List<Hr>queryByUsername(String username) {return hrMapper.queryByUsername(username);} }
2)View层
负责处理用户界面的显示细节,以及如何向用户展示业务处理的结果(页面效果)
<%@ page import="com.woniuxy.hrms.entity.Hr" %>
<%@ page import="java.util.List" %>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %><%String path = request.getContextPath();String basePath = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + path + "/";
%>
<base href="<%=basePath%>"/>
<!DOCTYPE html><head><meta charset="UTF-8"><title>Title</title>
</head>
<link rel="stylesheet" type="text/css" href="static/bootstrap/css/bootstrap.min.css"><script type="text/javascript" src="static/bootstrap/js/bootstrap.min.js"></script>
<script type="text/javascript" src="static/bootstrap/js/jquery.min.js"></script>
<script type="text/javascript">function check(id) {var flag = confirm("确认要删除吗?");if (flag) {//删除确认,服务器请求跳转至delete方法处理数据库数据location.href = "hr/delete?id=" + id;}}
</script><body>
<div class="container" style="padding-top: 40px;"><div class="form-group"><div class="row"><div class="col-md-8"><form id="searchForm" action="hr/queryByUsername" method="post"><input type="text" class="form-control" id="selectUserName" name="username"placeholder="请输入用户名"/><button type="submit" class="btn btn-danger search" id="searchBtn">搜索</button></form></div><div class="col-md-3"><a class="btn btn-round btn-square btn-default" href="manage/hr/add.jsp">添加<iclass="mdi mdi-eye"></i></a><!-- <button class="btn btn-default add" data-toggle="modal" data-target="#addModel">增加</button>--></div></div></div><%List<Hr> list = (List<Hr>) request.getAttribute("list");%><table class="table table-bordered text-center"><tr><td>编号</td><td>姓名</td><td>用户名</td><td>电话</td><td>地址</td><td>操作</td></tr><%-- 迭代集合--%><%for (Hr hr : list) {%><tr><td><%=hr.getId()%></td><td><%=hr.getRealName()%></td><td><%=hr.getUsername()%></td><td><%=hr.getPhone()%></td><td><%=hr.getAddress()%></td><td><%-- 点击编辑链接,传递当前对象id,并向服务器发送请求--%><%-- href="hrms/queryById?id=<%=hr.getId()% 查询字符串以?字符开始 --%><%-- <%=hr.getId()%>jsp脚本表达式 --%><a class="btn btn-round btn-square btn-info" href="hr/queryById?id=<%=hr.getId()%>">编辑<iclass="mdi mdi-eye"></i></a><a class="btn btn-round btn-square btn-warning" href="javascript:check(<%=hr.getId()%>)">删除<iclass="mdi mdi-eye"></i></a></td></tr><%}%></table>
</div>
</body>
</html>
3)Cotroller层
处理客户端请求
叫做控制层,主要的功能是处理用户发送的请求。
负责协调视图与模型,在两者之间处于桥梁和纽带的位置
public class BaseServlet extends HttpServlet {//HttpServlet生命周期: 初始化init(),服务运行service(),销毁destroy()//1.init()仅加载一次//2.service()随时响应客户端请求,每次获取请求对象调用此方法//3.destroy()仅调用一次, 释放servlet所占用的资源。如关闭文件输入输出流,关闭与数据库的连接。//多次调用,每次服务器获取请求对象调用service方法,随时响应客户端请求@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//获取请求路径String requestURI = req.getRequestURI();//检索请求路径最后的方法名String methodName = requestURI.substring(requestURI.lastIndexOf("/") + 1);Method declaredMethod;try {//动态调用对象方法declaredMethod = this.getClass().getDeclaredMethod(methodName, HttpServletRequest.class, HttpServletResponse.class);declaredMethod.invoke(this, req, resp);} catch (NoSuchMethodException | InvocationTargetException | IllegalAccessException e) {throw new RuntimeException(e);}}
}
@WebServlet(value = "/hr/*", loadOnStartup = 1)
public class HrServlet extends BaseServlet {HrService hrService;@Overridepublic void init(ServletConfig config) throws ServletException {ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-config.xml");//config.getServletContext()创建ServletContext对象,用来存储applicationContext对象,整个Web应用范围内访问它config.getServletContext().setAttribute("applicationContext", applicationContext);hrService = applicationContext.getBean("hrService", HrService.class);}protected void queryByUsername(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {String username = req.getParameter("username");List<Hr> list = hrService.queryByUsername(username);list.forEach(System.out::println);req.setAttribute("list", list);req.getRequestDispatcher("../manage/hr/show.jsp").forward(req, resp);}protected void queryAll(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {List<Hr> list = hrService.queryAll();req.setAttribute("list", list);req.getRequestDispatcher("../manage/hr/show.jsp").forward(req, resp);}protected void add(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// req.setCharacterEncoding("utf-8");Hr hr = new Hr();hr.setRealName(req.getParameter("realName"));hr.setUsername(req.getParameter("username"));hr.setAddress(req.getParameter("address"));hr.setPhone(req.getParameter("phone"));System.out.println(hr);int i = hrService.insertSelective(hr);//数据库添加数据,并重新调用queryAll走页面展示流程queryAll(req, resp);}protected void queryById(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {String id = req.getParameter("id");Hr hr = hrService.selectByPrimaryKey(Long.parseLong(id));req.setAttribute("hr", hr);
// 跳转页面,带着对象forward(),地址栏不变req.getRequestDispatcher("../manage/hr/update.jsp").forward(req, resp);
// 单纯跳转页面,并不能携带对象,地址栏更新
// resp.sendRedirect("/hr/update.jsp");}protected void update(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// req.setCharacterEncoding("utf-8");Hr hr = new Hr();//更新必须有id,负责更新所有内容,依据id更新// where dept_id = #{deptId,jdbcType=INTEGER}//获取页面update.jsp请求对象传递的参数hr.setId(Integer.parseInt(req.getParameter("id")));hr.setRealName(req.getParameter("realName"));hr.setUsername(req.getParameter("username"));hr.setAddress(req.getParameter("address"));hr.setPhone(req.getParameter("phone"));int update = hrService.updateByPrimaryKeySelective(hr);//更新完显示全部数据(查询全部数据反馈给show.jsp)queryAll(req, resp);}protected void delete(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {String id = req.getParameter("id");hrService.deleteByPrimaryKey(Long.parseLong(id));queryAll(req, resp);}protected void login(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//负责处理login.jsp提交//获取提交对象的username和passwordString username = req.getParameter("username");String password = req.getParameter("password");Hr hr = hrService.login(username, password);//HttpSession session = req.getSession();session.setAttribute("hr", hr);if (hr != null) {//本地存储,在浏览器存储数据Cookie[] cookies = req.getCookies();boolean flag = false;for (Cookie cookie : cookies) {String name = cookie.getName();if (name.equals("username")) {flag = true;}}if (!flag) {Cookie username1 = new Cookie("username", username);Cookie password1 = new Cookie("password", password);//设置cookie存储路径username1.setPath("/hrms");password1.setPath("/hrms");username1.setMaxAge(60 * 60 * 24);password1.setMaxAge(60 * 60 * 24);resp.addCookie(username1);resp.addCookie(password1);}//路径可以: ../或/hrms根路径下的页面resp.sendRedirect("/hrms/admin.jsp");} else {resp.sendRedirect("../login.jsp");}}
}
3.MVC的优点:
-
1、耦合性低
视图层和业务层分离,这样就允许更改视图层代码而不用重新编译模型和控制器代码,同样,一个应用的业务流程或者业务规则的改变只需要改动MVC的模型层即可。因为模型与控制器和视图相分离,所以很容易改变应用程序的数据层和业务规则。
-
2、重用性高
MVC模式允许使用各种不同样式的视图来访问同一个服务器端的代码,因为多个视图能共享一个模型,它包括任何WEB(HTTP)浏览器或者无线浏览器(wap),比如,用户可以通过电脑也可通过手机来订购某样产品,虽然订购的方式不一样,但处理订购产品的方式是一样的。由于模型返回的数据没有进行格式化,所以同样的构件能被不同的界面使用。
-
3、部署快,生命周期成本低
MVC使开发和维护用户接口的技术含量降低。使用MVC模式使开发时间得到相当大的缩减,它使程序员(Java开发人员)集中精力于业务逻辑,界面程序员(HTML和JSP开发人员)集中精力于表现形式上。
-
4、可维护性高
分离视图层和业务逻辑层也使得WEB应用更易于维护和修改。
4.MVC框架
- Struts2框架:Struts2是基于MVC的轻量级的web应用框架
规则。
-
2、重用性高
MVC模式允许使用各种不同样式的视图来访问同一个服务器端的代码,因为多个视图能共享一个模型,它包括任何WEB(HTTP)浏览器或者无线浏览器(wap),比如,用户可以通过电脑也可通过手机来订购某样产品,虽然订购的方式不一样,但处理订购产品的方式是一样的。由于模型返回的数据没有进行格式化,所以同样的构件能被不同的界面使用。
-
3、部署快,生命周期成本低
MVC使开发和维护用户接口的技术含量降低。使用MVC模式使开发时间得到相当大的缩减,它使程序员(Java开发人员)集中精力于业务逻辑,界面程序员(HTML和JSP开发人员)集中精力于表现形式上。
-
4、可维护性高
分离视图层和业务逻辑层也使得WEB应用更易于维护和修改。
4.MVC框架
-
Struts2框架:Struts2是基于MVC的轻量级的web应用框架
-
SpringMVC:Spring家族产品,我们后期重点使用的框架
相关文章:

4.Java Web开发模式(javaBean+servlet+MVC)
Java Web开发模式 一、Java Web开发模式 1.javaBean简介 JavaBeans是Java中一种特殊的类,可以将多个对象封装到一个对象(bean)中。特点是可序列化,提供无参构造器,提供getter方法和setter方法访问对象的属性。名称中…...

centos7 mysql 基本测试(6)主从简单测试
centos7 xtrabackup mysql 基本测试(6)主从简单测试 mysql -u etc -p 1234aA~1 参考: centos7 时区设置 时间同步 https://blog.csdn.net/wowocpp/article/details/135931129 Mysql数据库:主从复制与读写分离 https://blog.csd…...

信息安全工程师题
防火墙安全策略有两种类型:白名单策略、黑名单策略白名单策略:只允许符合安全规则的包通过防火墙,其他通信包禁止黑名单策略:禁止与安全规则相冲突的包通过防火墙,其他通信包允许实现网络地址转换的方式主要有静态NAT、…...

springcloud rocketmq 新增的消费者组从哪里开始消费
如果新建一个新的消费者组,是否会消费历史消息,导致重复消费? 直接在 console 界面新增消费者组,但是没有办法绑定订阅关系,没有找到入口,在 控制台项目源码 rocketmq-externals 也没有找到可以确定订阅关系…...

Redis-缓存
什么是缓存? 缓存就像自行车和越野车的避震器,降低硬着陆造成的损害 缓存就是系统的避震器,,防止过高的数据访问猛冲系统,导致其操作线程无法及时处理信息而瘫痪 缓存(Cache),就是数据交换的缓冲区,俗称的缓存就是缓冲区内的数据,一般从数…...

MySQL练习05
题目 步骤 触发器 use mydb16_trigger; #使用数据库create table goods( gid char(8) primary key, name varchar(10), price decimal(8,2), num int);create table orders( oid int primary key auto_increment, gid char(10) not null, name varchar(10), price decima…...

[C++][STL源码剖析] 详解AVL树的实现
目录 1.概念 2.实现 2.1 初始化 2.2 插入 2.2.1 旋转(重点) 左单旋 右单旋 双旋 2.❗ 双旋后,对平衡因子的处理 2.3 判断测试 完整代码: 拓展:删除 1.概念 二叉搜索树虽可以缩短查找的效率,但…...

Kubernetes存储 - Node本地存储卷
官方文档 Kubernetes管理的Node本地存储目前有三种,分别是EmptyDir,HostPath,Local,EmptyDir是一种与Pod同生命周期的Node临时存储;HostPath是Node的目录;Local是基于持久卷(PV)管理的Node目录。接下来详细说明这几种类型如何以存…...

Cocos Creator2D游戏开发-(2)Cocos 常见名词
场景(Scene): 它一个容器,容纳游戏中的各个元素,如精灵,标签,节点对象。它负责着游戏的运行逻辑,以帧为单位渲染这些内容。就是你理解到的那个场景; 个人理解就是一个画面, 一个游戏不同的关卡,会有不同的…...

【不同设备间的数据库连接】被连接设备如何开权限给申请连接的设备
为了方便叙述,简称申请连接数据库的设备为a,被连接的为b 1.确保在同一局域网下,检查a的ip 如果你设置的动态ip,那么每重启一次这个ip都会变。两种选择,每次都给b同步一下你的最新ip,或者a设置成静态ip。具…...

Whisper离线部署问题处理
Whisper是OpenAI开发一款开源语音识别模型,可以帮我们低成本的拥有语音识别的能力。具体的安装部署方法,我在这里就不详细说了,网上有很多相关文章: 使用OpenAI的Whisper 模型进行语音识别 (baidu.com) 我这里主要想说的是&…...

【Hive SQL】数据探查-数据抽样
文章目录 数据随机抽样1、随机数排序抽样(rand())2、数据块抽样(tablesample())3、分桶抽样 数据随机抽样 在大规模数据量的数据分析及建模任务中,往往针对全量数据进行挖掘分析时会十分耗时和占用集群资源,…...

微信答题小程序产品研发-需求分析与原型设计
欲知应候何时节,六月初迎大暑风。 我前面说过,我决意仿一款答题小程序,所以我做了大量的调研。 题库软件产品开发不仅仅是写代码这一环,它包含从需求调研、分析与构思、设计到开发、测试再到部署上线一系列复杂过程。 需求分析…...

基础模板Mybatis-plus+Springboot+Mysql开发配置文件
1.pom.xml <dependencies><dependency><groupId>org.mybatis.spring.boot</groupId><artifactId>mybatis-spring-boot-starter</artifactId><version>2.2.2</version></dependency>// mybatisplus功能<dependency&g…...

java-poi实现excel自定义注解生成数据并导出
因为项目很多地方需要使用导出数据excel的功能,所以开发了一个简易的统一生成导出方法。 依赖 <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-ooxml</artifactId> <version>4.0.1</version…...

LeetCode707 设计链表
前言 题目: 707. 设计链表 文档: 代码随想录——设计链表 编程语言: C 解题状态: 代码功底不够,只能写个大概 思路 主要考察对链表结构的熟悉程度,对链表的增删改查,比较考验代码功底以及对链表…...

[Mysql-DDL数据操作语句]
目录 DDL语句操作数据库 库: 查看:show 创建:creat 删除:drop 使用(切换):use 表: 查看:desc show 创建:create 表结构修改 rename as add drop modify change rename as …...

google 浏览器插件开发简单学习案例:TodoList;打包成crx离线包
参考: google插件支持: https://blog.csdn.net/weixin_42357472/article/details/140412993 这里是把前面做的TodoList做成google插件,具体网页可以参考下面链接 TodoList网页: https://blog.csdn.net/weixin_42357472/article/de…...

如何学习Doris:糙快猛的大数据之路(从入门到专家)
引言:大数据世界的新玩家 还记得我第一次听说"Doris"这个名字时的情景吗?那是在一个炎热的夏日午后,我正在办公室里为接下来的大数据项目发愁。作为一个刚刚跨行到大数据领域的新手,我感觉自己就像是被丢进了深海的小鱼—周围全是陌生的概念和技术。 就在这时,我的…...

梯度下降算法,gradient descent algorithm
定义:是一个优化算法,也成最速下降算法,主要的部的士通过迭代找到目标函数的最小值,或者收敛到最小值。 说人话就是求一个函数的极值点,极大值或者极小值 算法过程中有几个超参数: 学习率n,又称…...

Spring boot 2.0 升级到 3.3.1 的相关问题 (六)
文章目录 Spring boot 2.0 升级到 3.3.1 的相关问题 (六)spring-data-redis 和 Spring AOP 警告的问题问题描述问题调研结论解决方案方案1-将冲突的Bean 提升为InfrastructureBean方案2 其他相关资料 Spring boot 2.0 升级到 3.3.1 的相关问题 ÿ…...

C++模版基础知识与STL基本介绍
目录 一. 泛型编程 二. 函数模板 1. 概念 2. 函数模版格式 3. 函数模版的原理 4. 模版函数的实例化 (1). 隐式实例化 (2.) 显式实例化 5. 模版参数的匹配原则 三. 类模板 1. 类模板的定义格式 2. 类模板的实例化 四. STL的介绍 1. 什么是STL? 2. STL的版…...

Android 防止重复点击
1.第一种方式: // 两次点击按钮之间的点击间隔不能少于1000毫秒 private static final int MIN_CLICK_DELAY_TIME 700; private static long lastClickTime; /** * 是否是快速点击 * return */ public static boolean isFastClick() { …...

使用阿里云云主机通过nginx搭建文件服务器
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、准备基础环境二、安装配置nginx三、阿里云安全组配置安全组配置 
微信Android一面凉经(2024)
微信Android一面凉经(2024) 笔者作为一名双非二本毕业7年老Android, 最近面试了不少公司, 目前已告一段落, 整理一下各家的面试问题, 打算陆续发布出来, 供有缘人参考。今天给大家带来的是《微信Android一面凉经(2024)》。 面试职位: 微信-客户端开发工程师-基础功能(广州) And…...

VMware、Docker - 让虚拟机走主机代理,解决镜像封禁问题
文章目录 虚拟机全局代理配置找到 VMnet8 的 IPv4 地址代理相关配置虚拟机代理配置 Docker 代理配置修改镜像修改 Docker 代理配置 虚拟机全局代理配置 找到 VMnet8 的 IPv4 地址 a)打开此电脑,输入 “控制面板”,然后回车. b)之…...

版本管理|为什么不推荐使用Git Rebase
文章目录 什么是 Git Rebase?如何使用 Git Rebase?基本语法示例更多选项 注意事项何时使用何时避免其他注意事项 为什么需要谨慎使用 Git Rebase?面试中的常见问题问题 1: Git Rebase 和 Git Merge 有何不同?问题 2: 为什么有时应…...

Https post 请求时绕过证书验证方案
解决异常:Caused by: java.security.cert.CertificateException: No subject alternative names matching IP address xxx.xx.xx.xx found // Https POST 请求private cn.hutool.json.JSON PostGsData(String url, String appKey, String token, Map<String, Ob…...

C# 数组常用遍历方式
// 假设数组Point[] points new Point[2];// 第一种遍历 forfor (int i 0; i < points.Length; i){Point p points[i];Console.WriteLine($"X{p.X},y{p.Y}");}// 第二种遍历 foreachforeach (Point p in points){Console.WriteLine($"X{p.X},y{p.Y}"…...

【JavaScript】详解Day.js:轻量级日期处理库的全面指南
文章目录 一、Day.js简介1. 什么是Day.js?2. 安装Day.js 二、Day.js的基本用法1. 创建日期对象2. 格式化日期3. 解析日期字符串4. 操作日期5. 比较日期 三、Day.js的高级功能1. 插件机制2. 国际化支持 四、实际应用案例1. 事件倒计时2. 日历应用 在JavaScript开发中…...