搭建一個JavaWeb項目流程詳解

来源:https://www.cnblogs.com/qy-blog/p/17976236
-Advertisement-
Play Games

本文致力於,讓編程者一步步明白書寫一個JavaWeb項目應該做些什麼,梳理清楚流程框架,需要的jar包,同時手寫了一個分頁工具類也在其中,讓你在編程中更加絲滑。 ...


搭建一個JavaWeb項目流程

本文致力於,讓編程者一步步明白書寫一個JavaWeb項目應該做些什麼,梳理清楚流程框架,需要的jar包,同時手寫了一個分頁工具類也在其中,讓你在編程中更加絲滑。

1.src\main\java\com\einmeer\qianyu

刪除系統預設生成的HelloServlet.java

1.1tools包

DruidTools.java

需要在lib中加入druid-1.1.22.jar

package com.einmeer.qianyu.tools;

import com.alibaba.druid.pool.DruidDataSource;
import com.alibaba.druid.pool.DruidDataSourceFactory;

import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;

/**
 * @author 芊嵛
 * @date 2024/1/18
 * JDBC操作,就是在java中操作資料庫
 * 封裝資料庫連接工具,帶連接池的
 */
public class DruidTools {
    //聲明連接池對象
    private  static DruidDataSource dataSource;
    //java的靜態代碼塊,作初始化用
    static {
        //讀取外部配置文件的內容,生成位元組流輸入流對象
        InputStream in = DruidTools.class.getResourceAsStream("/Druid.properties");
        //創建屬性集合對象
        Properties properties = new Properties();
        try {
            //載入輸入流對象
            properties.load(in);
            //生成連接池對象
            dataSource= (DruidDataSource) DruidDataSourceFactory.createDataSource(properties);
        } catch (IOException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }

    }
    //獲取接池對象,便於dbutils使用,簡化原生JDBC的增刪改查功能CURD
    public static DruidDataSource getDataSource(){
        return  dataSource;
    }
    //拿到資料庫連接對象,便於我們操作MYSQL
    public static Connection getConn() {//方便之後重覆使用
        Connection conn = null;
        try {
            //創建資料庫連接對象(資料庫地址,用戶名,密碼)
            conn = dataSource.getConnection();
            System.out.println("資料庫連接成功success!");
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return conn;
    }

    ////測試一下,建議用單元測試
    public static void main(String[] args) {
        getConn();
    }
}

Pagination.java

分頁,封裝好了方法之後直接調用就行

package com.einmeer.qianyu.tools;

import java.util.List;

/**
 * @author 芊嵛
 * @date 2024/1/19
 */
public class Pagination<T> { // 使用泛型是為了復用
    // 當前頁號
    private int currentPage;
    // 總頁號或總頁數
    private int totalPage;
    // 每頁記錄數或行數
    private int limitRows;
    // 總的記錄數或行數
    private int totalRows;
    // 每頁開始的記錄號
    private int startRecord;
    // 每頁結束的記錄號,這個沒用,只需每頁記錄行數limitRows即可
    // private int endRecord;
    // 存每頁中的記錄
    private List<T> list;

    // //初始化操作
    public void init() {
        // 1.求總頁數,通過總記錄數與每頁行數來計算,有幾種情況
        // (1)不夠一頁(2)有零頭(3)剛好是整數頁
        int tp = totalRows / limitRows;
        if (totalRows > limitRows) {
            // 判斷是是剛剛好還是一頁多一點
            totalPage = (totalRows % limitRows) == 0 ? tp : tp + 1;
        } else {
            totalPage = 1;
        }
        // 2.將當頁保留在第一頁或最後一頁
        if (currentPage > totalPage) {
            currentPage = totalPage;
        } else if (currentPage < 1) {
            currentPage = 1;
        }
        // 3.初始化開始記錄數,mysql應用的limit它不包括開始記錄,所以不要加1;
        // 還有limit傳入的是開始記錄號與查詢的條數,此處是每頁可顯示數limitRows,
        // 如果查到最後沒有limitRows限制的行數,則顯示剩餘部分
        this.startRecord = (currentPage - 1) * limitRows;
    }

    // 無參構造,便於使用
    public Pagination() {
    }

    // 當前頁號,總記錄數,每頁行數;這些屬性需要傳入後初始化,其它的可以set設置
    public Pagination(int currentPage, int totalRows, int limitRows) {
        this.currentPage = currentPage;
        this.totalRows = totalRows;
        this.limitRows = limitRows;
    }

    // get與set方法
    public int getCurrentPage() {
        return currentPage;
    }

    public void setCurrentPage(int currentPage) {
        this.currentPage = currentPage;
    }

    public int getTotalPage() {
        return totalPage;
    }

    public void setTotalPage(int totalPage) {
        this.totalPage = totalPage;
    }

    public int getLimitRows() {
        return limitRows;
    }

    public void setLimitRows(int limitRows) {
        this.limitRows = limitRows;
    }

    public int getTotalRows() {
        return totalRows;
    }

    public void setTotalRows(int totalRows) {
        this.totalRows = totalRows;
    }

    public int getStartRecord() {
        return startRecord;
    }

    public void setStartRecord(int startRecord) {
        this.startRecord = startRecord;
    }

    public List<T> getList() {
        return list;
    }

    public void setList(List<T> list) {
        this.list = list;
    }
}

1.2entity包

Entity層,實體層,放入實體類

詳解Lombok中的@Builder用法

// Lombok註解
@Builder	// 一步步創建一個對象,它對用戶屏蔽了裡面構建的細節,但卻可以精細地控制對象的構造過程,不寫@Builder,@AllArgsConstructor會報紅
@Data	// 提供了get、set、equals、toString方法
@NoArgsConstructor	// 生成一個無產構造函數
@AllArgsConstructor // 生成一個包含所有變數的有參構造函數

實現序列化

Serializable介面就是Java提供用來進行高效率的異地共用實例對象的機制,實現這個介面即可。

參考java實體類為什麼要實現Serializable介面

implements Serializable

1.3dao包

DAO數據訪問層,把訪問資料庫的代碼封裝起來,不涉及業務邏輯

// 介面:寫需要的細分功能的名字,能用sql語句表示出來,如格局用戶id查詢用戶全部信息
User selectUserById(Long userId);
// 實現類
//1.建議先定義一個全局結果集返回對象,應為每次運行sql語句都會返回結果
QueryRunner qr = new QueryRunner(DruidTools.getDataSource());

// 2.每個介面第一句都把返回值類型置為空(查詢語句置為null)或者0(增刪改置為0)
List<User> list = null;
int n = 0;

// 3.每個介面第二句寫sql語句(我這裡舉例簡單寫寫*號實際一定不要寫*)
String sql = "select * from user where username like ? and userGender = ? limit ?,?";

// 4.創建一個對象類數組(裡面放?號代替的東西,註意位置不要錯)
// 如果就一個參數或者不需要參數不用寫這個,多個參數再寫
Object[] param = {"%"+username+"%",userGender,start,number};

// 5.處理異常,執行sql
try {
    // 查詢
    // 返回list結果集BeanListHandler<>()
    // 返回對象BeanHandler<>()
    // 返回單個數據ScalarHandler<>()
    // query(String sql, ResultSetHandler<T> rsh)
    // query(String sql, Object param, ResultSetHandler<T> rsh)
    // query(String sql, Object[] params, ResultSetHandler<T> rsh)
    // 更多可以看看源碼
    list = qr.query(sql, new BeanListHandler<User>(User.class));
    user = qr.query(sql, param, new BeanHandler<>(User.class));
    // 這個看其他資料說是返回Object類型需要轉,但是我測試並不需要,返回的就是我要的類型
    n = qr.query(sql,parm,new ScalarHandler<>());
    
    // 更新/刪除/插入
    // update(String sql, Object param)
    // update(String sql, Object... params)
    n = qr.update(sql, username);
    n = qr.update(sql, param);
} catch (SQLException e) {
    throw new RuntimeException(e);
}

// 6.return 第一步的名字;
return list;
return n;

1.4service包

Service業務邏輯層,處理邏輯上的業務,而不去考慮具體的實現。

通過調用數據訪問層,實現邏輯上的業務,一個介面的實現可能需要多個dao層的介面

// 介面:定義好方法,之後servlet直接調用,與servlet方法數量相同
// 名字最好不要跟dao層一樣
// dao:select
// service:query
// 實現類
// 1.全局調一下dao層,方便下麵調用
private UserDao userDao = new UserDaoImpl;
// 2.自由發揮

1.5servlet包

Servlet(Server Applet)是Java Servlet的簡稱,是為小服務程式或服務連接器,用Java編寫的伺服器端程式,主要功能在於互動式地瀏覽和修改數據,生成動態Web內容。

// 1.註解,
@WebServlet("/user")

// 2.繼承HttpServlet
public class UserServlet extends HttpServlet{}

// 3.創建服務類對象
private UserService userService = new UserServiceImpl();

// 4.重寫doGet()方法,如果調用它就會調用doPost()方法
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    doPost(req, resp);
}

// 5.重寫doPost()方法
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    // 首局一定要寫這句,把用戶傳來的數據如果有漢字轉為漢字,不然亂碼
    req.setCharacterEncoding("utf8");
    // 這裡我前端頁面隱藏了一個表單,區分頁面傳入的是哪個方法
    // <input type="hidden" name="method" value="pn-prid">
    String method = req.getParameter("method");
    // 根據method判斷需要調用什麼方法
}

// 6.具體方法,舉一個例子
public void queryAll(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    // 前端傳來的參數
    String proCode = req.getParameter("queryProCode");
    String proName = req.getParameter("queryProName");
    Object current = req.getParameter("current");
    // 實現分頁(當時劃分不清楚,這塊感覺可以寫進service進行封裝)
    int temp1 = 0;
    if (current != null) {
        temp1 = Integer.parseInt(req.getParameter("current"));
    }
    long temp = ps.queryCount(proCode,proName);
    int totalRows = (int) temp;
    Pagination<Provider> pg = new Pagination<>(temp1, totalRows, 10);
    pg.init();
    List<Provider> providers = ps.queryLimit(proCode,proName,pg.getStartRecord(), 10);
    pg.setList(providers);

	// 攜帶數據請求轉發
    RequestDispatcher rd = req.getRequestDispatcher("WEB-INF/jsp/providerlist.jsp");
    // 攜帶數據
    req.setAttribute("p1",proCode);
    req.setAttribute("p2",proName);
    req.setAttribute("providers", pg.getList());
    req.setAttribute("pg", pg);
    rd.forward(req, resp);
}

2.src\main\resources

druid.properties

# 如果mysql是5版本的去掉cj
driverClassName=com.mysql.cj.jdbc.Driver
# 埠號預設3306如果修改了記得把此處進行修改
# 記得修改資料庫的名字
url=jdbc:mysql://localhost:3306/資料庫名字?rewriteBatchedStatements=true&useSSL=false&characterEncoding=utf8&serverTimezone=Asia/Shanghai
# 資料庫賬號,根據實際填寫
username=root
# 資料庫密碼,根據實際填寫
password=quanyu9988.gmail.com
# 一般不需要更改,初始化資料庫連接池
initialSize=10
# 一般不需要更改,連接池的最大資料庫連接數
maxActive=20
# 一般不需要更改,超時等待時間一毫秒為單位
maxWait=1000
# 一般不需要更改,連接池的最小空閑連接數,如果空閑的連接數大於該值,則關閉多餘連接,反之創建更多連接滿足最小連接數的要求
minIdle=5

3.src\main\webapp

3.1WEB-INF\lib

image

image

// 不導入,連接池用不了
druid-1.1.22.jar

// 導入可以使用QueryRunner類+ResultSetHandler類,更方便的完成curd
commons-dbutils-1.7.jar

// 資料庫驅動,我這裡用的8
mysql-connector-java-8.0.25.jar

//使用JSTL核心標簽庫,需要導入下麵兩個依賴
standard.jar
jstl.jar

3.2WEB-INF\jsp

存放無法直接在地址欄訪問的界面

3.3WEB-INF\web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0">
    <!--歡迎頁,也就是一啟動就能看見的頁面-->
    <welcome-file-list>
        <welcome-file>login.jsp</welcome-file>
    </welcome-file-list>
</web-app>

3.4暴露在外面的內容

例如:
css
js
images

以及能直接在地址欄訪問的界面
login.jsp
register.jsp

下麵jsp頁面第八行必須寫上,才能證明他是jsp頁面

<%--
    Created by IntelliJ IDEA.
    User: Qy
        Date: 2024/1/18
            Time: 20:42
                To change this template use File | Settings | File Templates.
                --%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%--如果用到了核心標簽記得加上核心標簽庫--%>
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<html>
    <head>
        <title>Title</title>
        <%--所有外面的靜態資源一定要用<%=request.getContextPath()%>代替..不然找不到數據--%>
        <link type="text/css" rel="stylesheet" href="<%=request.getContextPath()%>/css/style.css"/>
    </head>
    <body>

    </body>
</html>

4.pom.xml

這份說明沒用到maven,採取自己導入jar包,這麼的代碼沒有修改只是作為說明

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <!--    maven的基本信息-->
    <modelVersion>4.0.0</modelVersion>
    <!--    聲明遵循哪一個pom模型版本-->
    <!--    組織表示,一般是公司網站倒過來-->
    <groupId>com.einmeer</groupId>
    <!--    本項目的唯一標識ID,項目名稱-->
    <artifactId>qianyu</artifactId>
    <!--    項目當前版本號-->
    <version>1.0-SNAPSHOT</version>
    <!--    右邊maven名字-->
    <name>qianyu</name>
    <!--    打包方式-->
    <packaging>war</packaging>
    <!--    POM之間的關係-->
    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <maven.compiler.target>1.8</maven.compiler.target>
        <maven.compiler.source>1.8</maven.compiler.source>
        <junit.version>5.9.2</junit.version>
    </properties>
    <!--    依賴關係列表-->
    <dependencies>

        <!--        自己添加的lombok可以讓我們在寫實體類的時候大大減少代碼量-->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.24</version>
            <scope>provided</scope>
        </dependency>


        <dependency>
            <!--            依賴項的組織名-->
            <groupId>javax.servlet</groupId>
            <!--            依賴項的子項目名-->
            <artifactId>javax.servlet-api</artifactId>
            <!--            依賴項的版本-->
            <version>4.0.1</version>
            <!--            依賴項的適用範圍-->
            <scope>provided</scope>
        </dependency>
        <dependency>
            <groupId>org.junit.jupiter</groupId>
            <artifactId>junit-jupiter-api</artifactId>
            <version>${junit.version}</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.junit.jupiter</groupId>
            <artifactId>junit-jupiter-engine</artifactId>
            <version>${junit.version}</version>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <!--    構建設置-->
    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-war-plugin</artifactId>
                <version>3.3.2</version>
            </plugin>
        </plugins>
    </build>
</project>

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • selenium4框架學習 https://blog.csdn.net/qq_45158700/article/details/135363339 瀏覽器驅動&selenium文檔下載 Selenium with Python中文翻譯文檔:https://selenium-python-zh.rea ...
  • 1.約束(constraint)概述 1.1 為什麼需要約束 數據完整性(Data Integrity)是指數據的精確性(Accuracy)和可靠性(Reliability)。它是防止資料庫中存在不符合語義規定的數據和防止因錯誤信息的輸入輸出造成的無效操作或錯誤信息而提出的。 為了保證數據的完整性, ...
  • 個人博客:無奈何楊(wnhyang) 個人語雀:wnhyang 共用語雀:線上知識共用 Github:wnhyang - Overview 前文講了Sa-Token介紹與SpringBoot環境下使用,但是satoken最重要的登錄鑒權直接略過了,那這篇文章就開講,😂當然不是啦。看標題就知道這次要 ...
  • 二叉樹 前言 二叉樹的遍歷主要有深度優先遍歷和廣度優先遍歷,深度優先遍歷是優先訪問一個子樹上的所有節點,訪問的屬性是豎向的,而廣度優先遍歷則是優先訪問同一層的所有節點,訪問屬性是橫向的。 深度優先遍歷 深度優先遍歷主要有三種順序: 前序遍歷 —— 根左右 中序遍歷 —— 左根右 後序遍歷 —— 左右 ...
  • 在現代社會中,時間的規劃和安排對於個人和企業來說非常重要。在我們處理時間上的事務時,我們需要知道某一天是否是國家法定節假日或者法定工作日。因此,開發一個能夠查詢特定日期的法定工作日的API介面就變得非常有必要了。 一、功能說明 該API介面的主要功能是根據用戶輸入的日期,返回該日期是否是國家法定節假 ...
  • Rust 所有權和 Move 語義 所有權和生命周期是 Rust 和其它編程語言的主要區別,也是 Rust 其它知識點的基礎。 動態數組因為大小在編譯期無法確定,所以放在堆上,並且在棧上有一個包含了長度和容量的胖指針指向堆上的記憶體。 恰到好處的限制,反而會釋放無窮的創意和生產力。 Rust 所有權規 ...
  • UAC(User Account Control) 是 Windows 平臺的用戶許可權控制。它可以讓程式使用管理員許可權執行某些操作。 靜態 UAC 提權 靜態 UAC 提權讓程式一直運行在管理員許可權下,只要在項目設置里把 "UAC Execution Level" 設置為 "requireAdmin ...
  • Spring AOP 技術實現原理 在Spring框架中,AOP(面向切麵編程)是通過代理模式和反射機制來實現的。本文將詳細介紹Spring AOP的技術實現原理,包括JDK動態代理和CGLIB代理的使用,並通過實例演示其在實際項目中的應用。 1. AOP的實現原理概述 Spring AOP的實現基 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...