DAO设计模式

简介: DAO设计模式 1.认识DAO 2.DAO各部分的详解 3.DAO设计模式流程

 

DAO设计模式

1.认识DAO

DAO(Data Acess Object 数据库访问对象)的主要功能是操作数据库,所以DAO在标准开发架构中数据数据层,以下是标准开发的架构

    1. 客户层:目前使用B/S开发架构居多,客户可以通过浏览器访问
    2. 显示层:使用Vue框架、 React框架、 JSP/Servlet等进行页面展示
    3. 业务层:负责将DAO层的操作进行组合,形成一个完整的业务逻辑
    4. 数据层:提供原子性操作,比如增删查改

    2.DAO各部分的详解

    DAO的设计流程包括六个部分,如下:

    1.DataBaseConnection

    设计一个专门负责打开连接数据库和关闭数据库操作的类

    命名规则: xxx.dbc.DataBaseConnection

    2.VO

    设计VO(值对象),其主要由属性,setter和getter组成与数据库中的字段进行对应。

    命名规则:xxx.vo.ttt ; 其中ttt要和数据库中的表的名字一致

    3.DAO

    定义一系列原子性操作,比如增删查改,和实现业务的接口

    命名规则:xxx.dao.I.xxx.DAO

    4.Impl

    设计DAO接口真正的实现类,完成具体的操作,但是不负责数据库的开关

    命名规则:xxx.dao.imp.xxxDAOImpI

    5.Proxy

    Proxy代理类的实现,主要将以上四个部分组合起来,完成整个操作过程

    命名规则:xxx.dao.Proxy.xxx.Proxy

    6.Factory

    Factory类主要用于获得DAO类的实例对象

    命名规则:xxx.factory.DAOFactory


    3.DAO设计模式流程

    通过一个案例讲解DAO设计模式的流程

    现在需要实现一个能够注册和查询工作者的案例(JDBC与DAO结合)

    1.首先需要实现数据库的创建与表单的创建

    /*======================= 删除数据库 =======================*/
    DROP DATABASE IF EXISTS smile ;
    /*======================= 创建数据库 =======================*/
    CREATE DATABASE smile ;
    /*======================= 使用数据库 =======================*/
    USE smile ;
    /*======================= 删除数据表 =======================*/
    DROP TABLE IF EXISTS worker ;
    /*======================= 创建数据表 =======================*/
    CREATE TABLE worker(
       empno      INT(4)      PRIMARY KEY,
       ename      VARCHAR(10),
       job        VARCHAR(9),
       hiredate     DATE,
       sal        FLOAT(7,2)
    ) ;
    /*======================= 插入测试数据 =======================*/
    INSERT INTO worker(empno,ename,job,hiredate,sal) VALUES (7369,'董鸣楠','销售','2003-10-09',1500.90) ;
    INSERT INTO worker(empno,ename,job,hiredate,sal) VALUES (8964,'李祺','分析员','2003-10-01',3000) ;
    INSERT INTO worker(empno,ename,job,hiredate,sal) VALUES (7698,'张惠','销售','2005-03-12',800) ;
    INSERT INTO worker(empno,ename,job,hiredate,sal) VALUES (7782,'杨军','分析员','2005-01-12',2500) ;
    INSERT INTO worker(empno,ename,job,hiredate,sal) VALUES (7762,'刘明','销售','2005-03-09',1000) ;
    INSERT INTO worker(empno,ename,job,hiredate,sal) VALUES (7839,'王月','经理','2006-09-01',2500) ;

    image.gif


    2.定义VO类

    package com.JavaWebDAO.vo;
    import java.util.Date ;
    //定义工人类
    public class Worker {
    //    设置工人属性
        private int empno ;
        private String ename ;
        private String job ;
        private Date hiredate ;
        private float sal ;
    //    配置setter函数
        public void setEmpno(int empno){
            this.empno = empno ;
        }
        public void setEname(String ename){
            this.ename = ename ;
        }
        public void setJob(String job){
            this.job = job ;
        }
        public void setHiredate(Date hiredate){
            this.hiredate = hiredate ;
        }
        public void setSal(float sal){
            this.sal = sal ;
        }
    //    配置getter函数
        public int getEmpno(){
            return this.empno ;
        }
        public String getEname(){
            return this.ename ;
        }
        public String getJob(){
            return this.job ;
        }
        public Date getHiredate(){
            return this.hiredate ;
        }
        public float getSal(){
            return this.sal ;
        }
    }

    image.gif

    3.定义数据库连接类

    package com.JavaWebDAO.db;
    import java.sql.Connection ;
    import java.sql.DriverManager ;
    public class DatabaseConnection {
    //    配置相关信息
        private static final String DBDRIVER = "com.mysql.jdbc.Driver" ;
        private static final String DBURL = "jdbc:mysql://localhost:3306/smile" ;
        private static final String DBUSER = "root" ;
        private static final String DBPASSWORD = "357703" ;
        private Connection conn ;
    //    连接函数
        public DatabaseConnection() throws Exception {
    //        加载类库
            Class.forName(DBDRIVER) ;
    //        建立连接
            this.conn = DriverManager.getConnection(DBURL,DBUSER,DBPASSWORD) ;
        }
    //    返回连接对象
        public Connection getConnection(){
            return this.conn ;
        }
    //    断开连接函数
        public void close() throws Exception {
            if(this.conn != null){
                try{
                    this.conn.close() ;
                }catch(Exception e){
                    throw e ;
                }
            }
        }
    }

    image.gif


    4.新建DAO接口

    package com.JavaWebDAO.dao;
    import java.util.* ;
    import com.JavaWebDAO.vo.Worker;
    public interface IWorkerDAO {
    //    业务需求--注册员工
        public boolean doCreate(Worker work) throws Exception ;
    //    业务需求--查询员工
        public List<Worker> findAll(String keyWord) throws Exception ;
    //    业务需求--通过id查询员工
        public Worker findById(int empno) throws Exception ;
    }

    image.gif


    5.定义完DAO接口后就需要写好具体的实现类,实现类分为两种。

    一种是真实实现类,一种是代理操作类。

    真实实现类

    package com.JavaWebDAO.dao.impI;
    import com.JavaWebDAO.dao.IWorkerDAO;
    import com.JavaWebDAO.vo.Worker;
    import java.sql.Connection;
    import java.sql.PreparedStatement;
    import java.sql.ResultSet;
    import java.util.ArrayList;
    import java.util.List;
    //真实实现类,实现IWorker中所有业务需求
    public class WorkerDAOImpl implements IWorkerDAO {
        private Connection conn = null ;
        private PreparedStatement pstmt = null ;
    //    通过构造函数获取连接对象
        public WorkerDAOImpl(Connection conn){
            this.conn = conn ;
        }
    //    实现注册员工功能
        public boolean doCreate(Worker work) throws Exception{
            boolean flag = false ;
            String sql = "INSERT INTO worker(empno,ename,job,hiredate,sal) VALUES (?,?,?,?,?)" ;
            this.pstmt = this.conn.prepareStatement(sql) ;
            this.pstmt.setInt(1,work.getEmpno()) ;
            this.pstmt.setString(2,work.getEname()) ;
            this.pstmt.setString(3,work.getJob()) ;
            this.pstmt.setDate(4,new java.sql.Date(work.getHiredate().getTime())) ;
            this.pstmt.setFloat(5,work.getSal()) ;
            if(this.pstmt.executeUpdate() > 0){
                flag = true ;
            }
            this.pstmt.close() ;
            return flag ;
        }
    //    实现查询员工的功能
        public List<Worker> findAll(String keyWord) throws Exception{
            List<Worker> all = new ArrayList<Worker>() ;
            String sql = "SELECT empno,ename,job,hiredate,sal FROM worker WHERE ename LIKE ? OR job LIKE ?" ;
            this.pstmt = this.conn.prepareStatement(sql) ;
            this.pstmt.setString(1,"%"+keyWord+"%") ;
            this.pstmt.setString(2,"%"+keyWord+"%") ;
            ResultSet rs = this.pstmt.executeQuery() ;
            Worker work = null ;
            while(rs.next()){
                work = new Worker() ;
                work.setEmpno(rs.getInt(1)) ;
                work.setEname(((ResultSet) rs).getString(2)) ;
                work.setJob(rs.getString(3)) ;
                work.setHiredate(rs.getDate(4)) ;
                work.setSal(rs.getFloat(5)) ;
                all.add(work) ;
            }
            this.pstmt.close() ;
            return all ;
        }
    //    实现通过id查询员工的功能
        public Worker findById(int empno) throws Exception{
            Worker emp = null ;
            String sql = "SELECT empno,ename,job,hiredate,sal FROM worker WHERE empno=?" ;
            this.pstmt = this.conn.prepareStatement(sql) ;
            this.pstmt.setInt(1,empno) ;
            ResultSet rs = this.pstmt.executeQuery() ;
            if(rs.next()){
                emp = new Worker() ;
                emp.setEmpno(rs.getInt(1)) ;
                emp.setEname(rs.getString(2)) ;
                emp.setJob(rs.getString(3)) ;
                emp.setHiredate(rs.getDate(4)) ;
                emp.setSal(rs.getFloat(5)) ;
            }
            this.pstmt.close() ;
            return emp ;
        }
    }

    image.gif

    代理操作类

    package com.JavaWebDAO.dao.proxy;
    import com.JavaWebDAO.dao.IWorkerDAO;
    import com.JavaWebDAO.dao.impI.WorkerDAOImpl;
    import com.JavaWebDAO.db.DatabaseConnection;
    import com.JavaWebDAO.vo.Worker;
    import java.util.List;
    //代理操作IWorkerDAo中的业务需求
    public class WorkerDAOProxy implements IWorkerDAO {
        private DatabaseConnection db = null ;
        private IWorkerDAO dao = null ;
    //    代理建立连接数据库
        public WorkerDAOProxy() throws Exception {
            this.db = new DatabaseConnection() ;
            this.dao = new WorkerDAOImpl(this.db.getConnection()) ;
        }
    //    代理实现注册功能
        public boolean doCreate(Worker work) throws Exception{
            boolean flag = false ;
            try{
                if(this.dao.findById(work.getEmpno()) == null){
                    flag = this.dao.doCreate(work) ;
                }
            }catch(Exception e){
                throw e ;
            }finally{
                this.db.close() ;
            }
            return flag ;
        }
    //    代理实现查询用户功能
        public List<Worker> findAll(String keyWord) throws Exception{
            List<Worker> all = null ;
            try{
                all = this.dao.findAll(keyWord) ;
            }catch(Exception e){
                throw e ;
            }finally{
                this.db.close() ;
            }
            return all ;
        }
    //    代理通过id查询用户的功能
        public Worker findById(int empno) throws Exception {
            Worker emp = null;
            try {
                emp = this.dao.findById(empno);
            } catch (Exception e) {
                throw e;
            } finally {
                this.db.close();
            }
            return emp;
        }
    }

    image.gif

    代理类只是调用了真实类中的方法,但是代理类可以让代码开发的结构更加清晰


    6.定义工厂类

    package com.JavaWebDAO.factory;
    import com.JavaWebDAO.dao.IWorkerDAO;
    import com.JavaWebDAO.dao.proxy.WorkerDAOProxy;
    public class DAOFactory {
    //    通过工厂获取代理类的实例
        public static IWorkerDAO getIWorkerDAOInstance() throws Exception{
            return new WorkerDAOProxy() ;
        }
    }

    image.gif


    设计完所有类和接口后,为了保证定义后功能可用,则需要做一个测试类,来测试所有功能是否能够正常使用

    package com.lzl.dao.test ;
    import com.lzl.factory.DAOFactory ;
    import com.lzl.vo.* ;
    public class TestdoCreate{
      public static void main(String args[]) throws Exception{
        Worker work = null ;
          work = new Worker() ;
          work.setEmpno(1000) ;
          work.setEname("SMILE") ;
          work.setJob("程序员 " ) ;
          work.setHiredate(new java.util.Date()) ;
          work.setSal(10000) ;
          DAOFactory.getIWorkerDAOInstance().doCreate(work) ;
          }
      }

    image.gif

    package com.lzl.dao.test ;
    import java.util.* ;
    import com.lzl.factory.DAOFactory ;
    import com.lzl.vo.* ;
    public class TestfindAll{
      public static void main(String args[]) throws Exception{
        List<Worker> all = DAOFactory.getIWorkerDAOInstance().findAll("") ;
        Iterator<Worker> iter = all.iterator() ;
        while(iter.hasNext()){
          Worker work = iter.next() ;
          System.out.println(work.getEmpno() + "、" + work.getEname() + " 、 " + work.getJob()+ "、"+work.getHiredate() + " 、 "+work.getSal()) ;
        }
      }
    }

    image.gif


    整体项目结构

    image.gif编辑

    这样DAO层就开发完毕了


    目录
    相关文章
    |
    8月前
    |
    设计模式 存储 前端开发
    18:JavaBean简介及其在表单处理与DAO设计模式中的应用-Java Web
    18:JavaBean简介及其在表单处理与DAO设计模式中的应用-Java Web
    145 4
    |
    SQL Java 数据库连接
    DAO 设计模式
    介绍 DAO功能是数据操作.客户发送数据到显示层,显示层发送数据到业务层,业务发送数据到数据层,数据层进行持久化.即.保存进入databases 一些简称 BO 业务对象的简称 DAO 一个数据访问对象,增删查改 PO数据库一条记录,映射成javaBean对象,拥有getter/setter方法 ...
    8290 0
    |
    SQL
    DAO设计模式
    本文完全根据我对DAO设计模式的理解程度完成,在未来不断地学习中,我也会不断更新这篇文章,如果你对文章内容有建议欢迎评论! DAO(Data Access Object,数据访问接口)设计模...
    1462 0
    |
    关系型数据库 Java 数据库
    DAO设计模式---实现一个简单的注册(上)
    DAO设计模式是属于Java Web中对数据层的具体操作,使用DAO设计模式可以简化大量代码的编写和增加程序的可移植性,可以很方便在日后对代码的修改。 如果在JSP中使用JDBC操作数据库,那样在JSP中就会嵌入大量的Java代码,显示和逻辑功能的代码混在一起,可读性差,难以维护。
    1389 0
    |
    SQL 设计模式
    DAO设计模式---实现一个简单的注册(中)
    4、DAO接口实现类: package com.song.Impl; import java.sql.*; import com.song.DAO.RegisterDAO; import com.
    949 0
    |
    Java 设计模式
    DAO设计模式---实现一个简单的注册(下)
    6、DAO工厂类: package com.song.DAOFactory; import com.song.DAO.RegisterDAO; import com.song.Impl.RegisterImpl; public class RegisterFactory{ publ...
    968 0
    |
    11天前
    |
    设计模式 前端开发 搜索推荐
    前端必须掌握的设计模式——模板模式
    模板模式(Template Pattern)是一种行为型设计模式,父类定义固定流程和步骤顺序,子类通过继承并重写特定方法实现具体步骤。适用于具有固定结构或流程的场景,如组装汽车、包装礼物等。举例来说,公司年会节目征集时,蜘蛛侠定义了歌曲的四个步骤:前奏、主歌、副歌、结尾。金刚狼和绿巨人根据此模板设计各自的表演内容。通过抽象类定义通用逻辑,子类实现个性化行为,从而减少重复代码。模板模式还支持钩子方法,允许跳过某些步骤,增加灵活性。
    |
    2月前
    |
    设计模式 安全 Java
    Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
    Kotlin教程笔记(51) - 改良设计模式 - 构建者模式