MVC架构模式学习笔记(动力节点老杜2022)
GitHub代码笔记:laodu-mvc: 动力节点学习javaweb中的mvc笔记。
文章目录
1.视频链接
2.不使用MVC架构模式程序存在的缺陷
3.MVC架构模式理论基础
4.JavaEE设计模式-DAO模式
5.pojo & bean & domain
6.业务层抽取以及业务类实现
7.控制层
8.MVC架构模式与三层架构的关系
9.手撕ThreadLocal源码
10.ThreadLocal的使用
11.不同功能的类放在不同的包下
12.层与层之间应该使用接口进行衔接以及当前项目存在的两大问题
1 视频链接
JavaWeb课程导读_哔哩哔哩_bilibili
2 不使用MVC架构模式程序存在的缺陷
- 在不使用MVC架构模式的前提下,完成银行账户转账。
- 分析以下AccountTransferServlet他都负责了什么?
- 1> 负责了数据接收
- 2> 负责了核心的业务处理
- 3> 负责了数据库表中数据的CRUD操作(Create【增】 Retrieve【查】 Update【改】 Delete【删】)
- 4> 负责了页面的数据展示
- 分析这个程序存在哪些问题?
- 缺点1> 代码的复用性太差。(代码的重用性太差)
- 导致缺点1的原因?
- 因为没有进行“职能分工”,没有独立组件的概念,所以没有办法进行代码复用。代码和代码之间的耦合度太高,扩展力太差。
- 缺点2> 耦合度高,导致了代码很难扩展。
- 缺点3> 操作数据库的代码和业务逻辑混杂在一起,很容易出错。编写代码的时候很容易出错,无法专注业务逻辑的编写。
- 缺点1> 代码的复用性太差。(代码的重用性太差)
3 MVC架构模式理论基础

4 JavaEE设计模式-DAO模式
- AccountDao是负责Account数据的增删改查的。
- 什么是DAO?
- Data Access Object(数据访问对象)
- DAO实际上是一种设计模式,属于JavaEE的设计模式之一。(不是23种设计模式。)
- DAO只负责数据库表的CRUD,没有任何业务逻辑在里面。
- 没有任何业务逻辑,只负责表中数据增删改查的对象,有一个特殊的称谓:DAO对象。
- 为什么叫做AccountDao呢?
- 这是因为这个DAO是专门处理t_act这张表的。
- 如果处理t_user表的话,可以叫做:UserDao
- 如果处理t_student表的话,可以叫做:StudentDao
- 一般情况下:一张表会对应一个DAO对象(pojo / bean / domain)。
- DAO中的方法名很固定了,一般都是:
- insert
- deleteByXxx
- update
- selectByXxx
- selectAll
5 pojo & bean & domain
- 账户实体类:封装账户信息的。
- 一般是一张表对应一个。
- 这种普通简单的对象被成为pojo对象。
- 有的人也会把这种专门封装数据的对象,称为bean对象。(javabean:咖啡豆)
- 有的人也会把这种专门封装数据的对象,称为领域模型对象。domain对象。
- 不同的程序员有不同的习惯。
public class Account { // 这种普通简单的对象被成为pojo对象。/*** 主键*/// 一般这种属性不建议设计为基本数据类型,建议使用包装类。防止null带来的问题。//private long id;private Long id;/*** 账号*/private String actno;/*** 余额*///private double balance;private Double balance;@Overridepublic String toString() {return "Account{" +"id=" + id +", actno='" + actno + '\'' +", balance=" + balance +'}';}public Long getId() {return id;}public void setId(Long id) {this.id = id;}public String getActno() {return actno;}public void setActno(String actno) {this.actno = actno;}public Double getBalance() {return balance;}public void setBalance(Double balance) {this.balance = balance;}public Account(Long id, String actno, Double balance) {this.id = id;this.actno = actno;this.balance = balance;}public Account() {}
}
6 业务层抽取以及业务类实现
- service翻译为:业务。
- AccountService:专门处理Account业务的一个类。
- 在该类中应该编写纯业务代码。(只专注业务。不写别的。不和其他代码混合在一块。)
- 只希望专注业务,能够将业务完美实现,少量bug。
- 业务类一般起名:XxxService、XxxBiz…
public class AccountService {// 为什么定义到这里?因为在每一个业务方法中都可以需要连接数据库。private AccountDao accountDao = new AccountDao();// 这里的方法起名,一定要体现出,你要处理的是什么业务。// 我们要提供一个能够实现转账的业务方法(一个业务对应一个方法。)/*** 完成转账的业务逻辑* @param fromActno 转出账号* @param toActno 转入账号* @param money 转账金额*/public void transfer(String fromActno, String toActno, double money) throws MoneyNotEnoughException, AppException {// 查询余额是否充足Account fromAct = accountDao.selectByActno(fromActno);if (fromAct.getBalance() < money) {throw new MoneyNotEnoughException("对不起,余额不足");}// 程序到这里说明余额充足Account toAct = accountDao.selectByActno(toActno);// 修改余额(只是修改了内存中java对象的余额)fromAct.setBalance(fromAct.getBalance() - money);toAct.setBalance(toAct.getBalance() + money);// 更新数据库中的余额int count = accountDao.update(fromAct);// 模拟异常String s = null;s.toString();count += accountDao.update(toAct);if (count != 2) {throw new AppException("账户转账异常!!!");}}}
7 控制层
- 账户小程序
- AccountServlet是一个司令官。他负责调度其他组件来完成任务。
@WebServlet("/transfer")
public class AccountServlet extends HttpServlet { // AccountServlet作为Controllerprivate AccountService accountService = new AccountService();@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {// 接收数据String fromActno = request.getParameter("fromActno");String toActno = request.getParameter("toActno");double money = Double.parseDouble(request.getParameter("money"));try {// 调用业务方法处理业务(调度Model处理业务)accountService.transfer(fromActno, toActno, money);// 执行到这里了,说明成功了。// 展示处理结果(调度View做页面展示)response.sendRedirect(request.getContextPath() + "/success.jsp");} catch(MoneyNotEnoughException e) {// 执行到这里了,说明失败了。(余额不足)// 展示处理结果(调度View做页面展示)response.sendRedirect(request.getContextPath() + "/moneynotenough.jsp");} catch(Exception e){// 执行到这里了,说明失败了。response.sendRedirect(request.getContextPath() + "/error.jsp");}}
}
8 MVC架构模式与三层架构的关系

9 手撕ThreadLocal源码
public class Connection {
}
public class DBUtil {// 静态变量特点:类加载时执行,并且只执行一次。// 全局的大Map集合private static MyThreadLocal<Connection> local = new MyThreadLocal<>();/*** 每一次都调用这个方法来获取Connection对象* @return*/public static Connection getConnection(){Connection connection = local.get();if (connection == null) {// 第一次调用:getConnection()方法的时候,connection一定是空的。// 空的就new一次。connection = new Connection();// 将new的Connection对象绑定到大Map集合中。local.set(connection);}return connection;}}
package com.ycy.threadlocal;import java.util.HashMap;
import java.util.Map;/*** 自定义一个ThreadLocal类*/
public class MyThreadLocal<T> {/*** 所有需要和当前线程绑定的数据要放到这个容器当中*/private Map<Thread, T> map = new HashMap<>();/*** 向ThreadLocal中绑定数据*/public void set(T obj){map.put(Thread.currentThread(), obj);}/*** 从ThreadLocal中获取数据* @return*/public T get(){return map.get(Thread.currentThread());}/*** 移除ThreadLocal当中的数据*/public void remove(){map.remove(Thread.currentThread());}
}
package com.ycy.threadlocal;
// 张三发送请求,对应一个线程t1
// 李四发送请求,对应一个线程t2
public class Test {public static void main(String[] args) {Thread thread = Thread.currentThread();System.out.println(thread);// 调用serviceUserService userService = new UserService();userService.save();}
}
package com.ycy.threadlocal;public class UserService {private UserDao userDao = new UserDao();public void save(){Thread thread = Thread.currentThread();System.out.println(thread);Connection connection = DBUtil.getConnection();System.out.println(connection);userDao.insert();}}
package com.ycy.threadlocal;public class UserDao {public void insert(){Thread thread = Thread.currentThread();System.out.println(thread);Connection connection = DBUtil.getConnection();System.out.println(connection);System.out.println("User DAO insert");}
}
10 ThreadLocal的使用
JDK有ThreadLocal类的实现
public class DBUtil {private static ResourceBundle bundle = ResourceBundle.getBundle("resources/jdbc");private static String driver = bundle.getString("driver");private static String url = bundle.getString("url");private static String user = bundle.getString("user");private static String password = bundle.getString("password");// 不让创建对象,因为工具类中的方法都是静态的。不需要创建对象。// 为了防止创建对象,故将构造方法私有化。private DBUtil(){}// DBUtil类加载时注册驱动static {try {Class.forName(driver);} catch (ClassNotFoundException e) {e.printStackTrace();}}// 这个对象实际上在服务器中只有一个。private static ThreadLocal<Connection> local = new ThreadLocal<>();/*** 这里没有使用数据库连接池,直接创建连接对象。* @return 连接对象* @throws SQLException*/public static Connection getConnection() throws SQLException {Connection conn = local.get();if (conn == null) {conn = DriverManager.getConnection(url, user, password);local.set(conn);}return conn;}/*** 关闭资源* @param conn 连接对象* @param stmt 数据库操作对象* @param rs 结果集对象*/public static void close(Connection conn, Statement stmt, ResultSet rs){if (rs != null) {try {rs.close();} catch (SQLException e) {throw new RuntimeException(e);}}if (stmt != null) {try {stmt.close();} catch (SQLException e) {throw new RuntimeException(e);}}if (conn != null) {try {conn.close();// 思考一下:为什么conn关闭之后,这里要从大Map中移除呢?// 根本原因是:Tomcat服务器是支持线程池的。也就是说一个人用过了t1线程,t1线程还有可能被其他用户使用。local.remove();} catch (SQLException e) {throw new RuntimeException(e);}}}}
public class AccountService {// 为什么定义到这里?因为在每一个业务方法中都可以需要连接数据库。private AccountDao accountDao = new AccountDao();// 这里的方法起名,一定要体现出,你要处理的是什么业务。// 我们要提供一个能够实现转账的业务方法(一个业务对应一个方法。)/*** 完成转账的业务逻辑* @param fromActno 转出账号* @param toActno 转入账号* @param money 转账金额*/public void transfer(String fromActno, String toActno, double money) throws MoneyNotEnoughException, AppException {// service层控制事务try (Connection connection = DBUtil.getConnection()){System.out.println(connection);// 开启事务(需要使用Connection对象)connection.setAutoCommit(false);// 查询余额是否充足Account fromAct = accountDao.selectByActno(fromActno);if (fromAct.getBalance() < money) {throw new MoneyNotEnoughException("对不起,余额不足");}// 程序到这里说明余额充足Account toAct = accountDao.selectByActno(toActno);// 修改余额(只是修改了内存中java对象的余额)fromAct.setBalance(fromAct.getBalance() - money);toAct.setBalance(toAct.getBalance() + money);// 更新数据库中的余额int count = accountDao.update(fromAct);// 模拟异常String s = null;s.toString();count += accountDao.update(toAct);if (count != 2) {throw new AppException("账户转账异常!!!");}// 提交事务connection.commit();} catch (SQLException e) {throw new AppException("账户转账异常!!!");}}}
public class AccountDao {/*** 插入账户信息* @param act 账户信息* @return 1表示插入成功*/public int insert(Account act) {PreparedStatement ps = null;int count = 0;try {Connection conn = DBUtil.getConnection();String sql = "insert into t_act(actno, balance) values(?,?)";ps = conn.prepareStatement(sql);ps.setString(1, act.getActno());ps.setDouble(2, act.getBalance());count = ps.executeUpdate();} catch (SQLException e) {throw new RuntimeException(e);} finally {DBUtil.close(null, ps, null);}return count;}/*** 根据主键删除账户* @param id 主键* @return*/public int deleteById(Long id){PreparedStatement ps = null;int count = 0;try {Connection conn = DBUtil.getConnection();String sql = "delete from t_act where id = ?";ps = conn.prepareStatement(sql);ps.setLong(1, id);count = ps.executeUpdate();} catch (SQLException e) {throw new RuntimeException(e);} finally {DBUtil.close(null, ps, null);}return count;}/*** 更新账户* @param act* @return*/public int update(Account act) {PreparedStatement ps = null;int count = 0;try {Connection conn = DBUtil.getConnection();System.out.println(conn);String sql = "update t_act set balance = ? , actno = ? where id = ?";ps = conn.prepareStatement(sql);ps.setDouble(1, act.getBalance());ps.setString(2, act.getActno());ps.setLong(3, act.getId());count = ps.executeUpdate();} catch (SQLException e) {throw new RuntimeException(e);} finally {DBUtil.close(null, ps, null);}return count;}/*** 根据账号查询账户* @param actno* @return*/public Account selectByActno(String actno){PreparedStatement ps = null;ResultSet rs = null;Account act = null;try {Connection conn = DBUtil.getConnection();System.out.println(conn);String sql = "select id,balance from t_act where actno = ?";ps = conn.prepareStatement(sql);ps.setString(1, actno);rs = ps.executeQuery();if (rs.next()) {Long id = rs.getLong("id");Double balance = rs.getDouble("balance");// 将结果集封装成java对象act = new Account();act.setId(id);act.setActno(actno);act.setBalance(balance);}} catch (SQLException e) {throw new RuntimeException(e);} finally {DBUtil.close(null, ps, rs);}return act;}/*** 获取所有的账户* @return*/public List<Account> selectAll() {PreparedStatement ps = null;ResultSet rs = null;List<Account> list = new ArrayList<>();try {Connection conn = DBUtil.getConnection();String sql = "select id,actno,balance from t_act";ps = conn.prepareStatement(sql);rs = ps.executeQuery();while (rs.next()) {// 取出数据Long id = rs.getLong("id");String actno = rs.getString("actno");Double balance = rs.getDouble("balance");// 封装对象/*Account account = new Account();account.setId(id);account.setActno(actno);account.setBalance(balance);*/Account account = new Account(id, actno, balance);// 加到List集合list.add(account);}} catch (SQLException e) {throw new RuntimeException(e);} finally {DBUtil.close(null, ps, rs);}return list;}}
11 不同功能的类放在不同的包下
ssm里面一般都把dao包起名为mapper,把web包起名为controller
12 层与层之间应该使用接口进行衔接以及当前项目存在的两大问题
相关文章:

MVC架构模式学习笔记(动力节点老杜2022)
GitHub代码笔记:laodu-mvc: 动力节点学习javaweb中的mvc笔记。 文章目录 1.视频链接 2.不使用MVC架构模式程序存在的缺陷 3.MVC架构模式理论基础 4.JavaEE设计模式-DAO模式 5.pojo & bean & domain 6.业务层抽取以及业务类实现 7.控制层 8.MVC架构模式与三…...

docker常用操作-docker私有仓库的搭建(Harbor),并将本地镜像推送至远程仓库中。
1、docker-compose安装,下载docker-compose的最新版本 第一步:创建docker-compose空白存放文件vi /usr/local/bin/docker-compose 第二步:使用curl命令在线下载,并制定写入路径 curl -L "https://github.com/docker/compos…...
什么是MVC
MVC的全名是Model View Controller,是模型(model)-视图(view)-控制器(controller)的缩写,是一种软件设计典范。它是用一种业务逻辑、数据与界面显示分离的方法来组织代码,将众多的业务逻辑聚集到一个部件里面ÿ…...

ChatGPT浪潮来袭!谁先掌握,谁将领先!
任正非在接受采访时说 今后职场上只有两种人, 一种是熟练使用AI的人, 另一种是创造AI工具的人。 虽然这个现实听起来有些夸张的残酷, 但这就是我们必须面对的事实 📆 对于我们普通人来说,我们需要努力成为能够掌握…...
Focal and Global Knowledge Distillation forDetectors
摘要 文章指出,在目标检测中,教师和学生在不同领域的特征差异很大,尤其是在前景和背景中。如果我们 平等地蒸馏它们,特征图之间的不均匀差异将对蒸馏产生负面影响。因此,我们提出了局部和全局蒸馏。局部蒸馏分离前景和…...

FX110网:1月美国零售货币资金环比上升2.61%,嘉盛环比上升1.86%
美国商品期货交易委员会(CFTC)发布的最新月度报告显示,2024年1月零售货币存款与上月相比上升2.61%。 这份报告涵盖在美国运营的注册零售货币对交易商(RFED)和经纪自营商。包括嘉信理财(CHARLES SCHWAB Futu…...
全量知识系统的核心-全量知识的一个“恰当组织”的构想及百度AI答问
全量知识系统的核心-全量知识的一个恰当组织 Q1. 以下是对 我刚刚完成的文档“全量知识系统的核心:全量知识的一个恰当组织构想”的百度AI答复。由于字数400的限制,内容被分成四段. 第一次回答:学科和科学的框架 关于技术学科、一般学科和…...
C++中using 和 typedef 的区别
C中using 和 typedef 的区别_typedef using-CSDN博客 在C中,“using”和“typedef”执行声明类型别名的相同任务。两者之间没有重大区别。C中的“Using”被认为是类型定义同义词。此方法也称为别名声明。定义这些别名声明的工作方式类似于使用“using”语句定义C中…...

LeetCode-1944题: 队列中可以看到的人数(原创)
【题目描述】 有 n 个人排成一个队列,从左到右 编号为 0 到 n - 1 。给你以一个整数数组 heights ,每个整数 互不相同,heights[i] 表示第 i 个人的高度。一个人能 看到 他右边另一个人的条件是这两人之间的所有人都比他们两人 矮 。更正式的&…...
Java基础面试题整理2024/3/13
1、可以使用switch的数据类型 Java5以前,switch(arg)表达式中,arg只能是byte、short、char、int。 Java5之后引入了枚举类型,也可以是枚举类型。 Java7开始引入了字符串类型。 2、Java中的goto有什么作用 goto是Java中的保留字,…...

MachineSink - 优化阅读笔记
注:该优化与全局子表达式消除刚好是相反的过程,具体该不该做这个优化得看代价模型算出来的结果(有采样文件指导算得会更准确) 该优化过程将指令移动到后继基本块中,以便它们不会在不需要其结果的路径上执行。 该优化过程并非旨在替代或完全…...
虾皮shopee根据ID取商品详情 API
公共参数 名称类型必须描述keyString是免费申请调用key(必须以GET方式拼接在URL中)secretString是调用密钥api_nameString是API接口名称(包括在请求地址中)[item_search,item_get,item_search_shop等]cacheString否[yes,no]默认y…...
你知道数据库有哪些约束吗?
目录 1. NULL约束 2. 唯一(UNIQUE)约束 3. 默认值(DEFAULT)约束 4. 主键约束 5. 外键约束 6. CHECK约束 数据库约束是一种用于确保数据库中数据完整性和一致性的规则或条件。这些约束可以应用于表、列或整个数据库࿰…...

QT----基于QT的人脸考勤系统(未完成)
目录 1 编译opencv库1.1 下载源代码1.2 qt编译opencv1.3 执行Cmake一直卡着data: Download: face_landmark_model.dat 2 编译SeetaFace2代码2.1 遇到报错By not providing "FindOpenCV.cmake" in CMAKE_MODULE_PATH this project has2.2遇到报错Model missing 3 测试…...

机试:成绩排名
问题描述: 代码示例: #include <bits/stdc.h> using namespace std;int main(){cout << "样例输入" << endl; int n;int m;cin >> n;int nums[n];for(int i 0; i < n; i){cin >> nums[i];}// 排序for(int i 0; i < n; i){//…...

C编程基础四十分笔记
都是一些基础的C语言 一 输入一个整数,计算这个整数有几位二 编写程序计算一个分布函数三 输入一个字符串,再随便输入一个字母,判断这个字母出现几次四 求 1到10的阶乘之和五 求一个球体体积六 写一个链表,存1,2&#…...

k8s关于pod
目录 1、POD 的创建流程 kubectl 发起创建 Pod 请求: API Server 接收请求并处理: 写入 Etcd 数据库: Kubelet 监听并创建 Pod: Pod 状态更新和汇报: 2、POD 的状态解析 1. Pending Pod 2. Running Pod 3. S…...

yum安装mysql 数据库tab自动补全
centos7上面没有mysql,它的数据库名字叫做mariadb [rootlocalhost ~]#yum install mariadb-server -y [rootlocalhost ~]#systemctl start mariadb.service [rootlocalhost ~]#systemctl stop firewalld [rootlocalhost ~]#setenforce 0 [rootlocalhost ~]#ss -na…...

MBT-Net
feature F,edge feature E-F where r related to the relative position 辅助信息 作者未提供代码...

大数据赋能,能源企业的智慧转型之路
在数字洪流中,大数据已经成为推动产业升级的新引擎。特别是在能源行业,大数据的应用正引领着一场深刻的智慧转型。今天,我们就来探讨大数据如何在能源企业中发挥其独特的魅力,助力企业提效降本,实现绿色发展。 动态监控…...
Vim 调用外部命令学习笔记
Vim 外部命令集成完全指南 文章目录 Vim 外部命令集成完全指南核心概念理解命令语法解析语法对比 常用外部命令详解文本排序与去重文本筛选与搜索高级 grep 搜索技巧文本替换与编辑字符处理高级文本处理编程语言处理其他实用命令 范围操作示例指定行范围处理复合命令示例 实用技…...

使用VSCode开发Django指南
使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架,专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用,其中包含三个使用通用基本模板的页面。在此…...
<6>-MySQL表的增删查改
目录 一,create(创建表) 二,retrieve(查询表) 1,select列 2,where条件 三,update(更新表) 四,delete(删除表…...
逻辑回归:给不确定性划界的分类大师
想象你是一名医生。面对患者的检查报告(肿瘤大小、血液指标),你需要做出一个**决定性判断**:恶性还是良性?这种“非黑即白”的抉择,正是**逻辑回归(Logistic Regression)** 的战场&a…...

【CSS position 属性】static、relative、fixed、absolute 、sticky详细介绍,多层嵌套定位示例
文章目录 ★ position 的五种类型及基本用法 ★ 一、position 属性概述 二、position 的五种类型详解(初学者版) 1. static(默认值) 2. relative(相对定位) 3. absolute(绝对定位) 4. fixed(固定定位) 5. sticky(粘性定位) 三、定位元素的层级关系(z-i…...
大学生职业发展与就业创业指导教学评价
这里是引用 作为软工2203/2204班的学生,我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要,而您认真负责的教学态度,让课程的每一部分都充满了实用价值。 尤其让我…...

Linux --进程控制
本文从以下五个方面来初步认识进程控制: 目录 进程创建 进程终止 进程等待 进程替换 模拟实现一个微型shell 进程创建 在Linux系统中我们可以在一个进程使用系统调用fork()来创建子进程,创建出来的进程就是子进程,原来的进程为父进程。…...
从面试角度回答Android中ContentProvider启动原理
Android中ContentProvider原理的面试角度解析,分为已启动和未启动两种场景: 一、ContentProvider已启动的情况 1. 核心流程 触发条件:当其他组件(如Activity、Service)通过ContentR…...

Unity中的transform.up
2025年6月8日,周日下午 在Unity中,transform.up是Transform组件的一个属性,表示游戏对象在世界空间中的“上”方向(Y轴正方向),且会随对象旋转动态变化。以下是关键点解析: 基本定义 transfor…...
[USACO23FEB] Bakery S
题目描述 Bessie 开了一家面包店! 在她的面包店里,Bessie 有一个烤箱,可以在 t C t_C tC 的时间内生产一块饼干或在 t M t_M tM 单位时间内生产一块松糕。 ( 1 ≤ t C , t M ≤ 10 9 ) (1 \le t_C,t_M \le 10^9) (1≤tC,tM≤109)。由于空间…...