Spring MVC 中的攔截器的使用“攔截器基本配置” 和 “攔截器高級配置”

来源:https://www.cnblogs.com/TheMagicalRainbowSea/p/18303437
-Advertisement-
Play Games

1. Spring MVC 中的攔截器的使用“攔截器基本配置” 和 “攔截器高級配置” @目錄1. Spring MVC 中的攔截器的使用“攔截器基本配置” 和 “攔截器高級配置”2. 攔截器3. Spring MVC 中的攔截器的創建和基本配置3.1 定義攔截3.2 攔截器基本配置3.3 攔截器的 ...


1. Spring MVC 中的攔截器的使用“攔截器基本配置” 和 “攔截器高級配置”

@

目錄


2. 攔截器

攔截器(Interceptor) 類似於過濾器(Filter)

Spring MVC 的攔截器作用是在請求到達控制器之前或之後進行攔截,可以對請求和響應進行一些特定的處理。

攔截器可以用於很多場景下:

  1. 登錄驗證:對於需要登錄才能訪問的地址,使用攔截器可以判斷用戶是否已登錄,如果未登錄,則跳轉到登錄頁面。
  2. 許可權校驗:根據用戶許可權對部分網址進行訪問控制,拒絕未經授權的用戶訪問。
  3. 請求日誌:記錄請求信息,例如:請求地址,請求參數,請求時間等,用於排查問題和性能優化。
  4. 更改響應:可以對響應的內容進行修改,例如:添加頭信息,調整響應內容格式等。

攔截器和過濾器的區別在於它們的作用層面不同:

  • 過濾器更註重在請求和響應的流程中進行處理,可以修改請求和響應的內容,例如:設置編碼和字元集,請求頭,狀態碼等。
  • 攔截器則更加側重於對控制器進行前置或後置處理,在請求到達控制器之前或之後進行特定的操作,例如:列印日誌,許可權驗證等。

Filter、Servlet、Interceptor、Controller的執行順序:

在這裡插入圖片描述

3. Spring MVC 中的攔截器的創建和基本配置

3.1 定義攔截

實現org.springframework.web.servlet.HandlerInterceptor 介面,共有三個方法可以進行選擇性的實現:

在這裡插入圖片描述

  • preHandle( ):處理器方法調用之前執行。只有該方法有返回值,返回值是布爾類型,true 表示放行,false 表示攔截
  • postHandle( ):處理器方法調用之後執行。
  • afterCompletion( ):渲染完成後執行。

3.2 攔截器基本配置

第一步:編寫攔截器,該攔截器要實現org.springframework.web.servlet.HandlerInterceptor 介面

在這裡插入圖片描述

package com.rainbowsea.springmvc.interceptors;

import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;



public class Interceptor1 implements HandlerInterceptor {

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("Interceptor1's preHandle!");
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("Interceptor1's postHandle!");
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("Interceptor1's afterCompletion!");
    }
}

在 Spring MVC 中攔截器的基本配置有兩種方式:

  • 第一種方式是:通過 xml 進行配置
  • 第二種方式是:通過 @Component 註解 + xml 文件進行配置

第一種方式:通過 xml 進行配置

需要註意的是:這個基本配置,預設情況下是攔截所有請求的。

在 springmvc.xml 文件中進行如下配置:

<mvc:interceptors>
    <bean class="com.powernode.springmvc.interceptors.Interceptor1"/>
</mvc:interceptors>

在這裡插入圖片描述

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:mvc="http://www.springframework.org/schema/mvc"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/cache http://www.springframework.org/schema/cache/spring-cache.xsd http://www.springframework.org/schema/mvc https://www.springframework.org/schema/mvc/spring-mvc.xsd">

    <!--    組件掃描-->
    <context:component-scan
            base-package="com.rainbowsea.springmvc.controller,com.rainbowsea.springmvc.interceptors"></context:component-scan>

    <!--    視圖解析器-->
    <bean id="thymeleafViewResolver" class="org.thymeleaf.spring6.view.ThymeleafViewResolver">
        <!--作用於視圖渲染的過程中,可以設置視圖渲染後輸出時採用的編碼字元集-->
        <property name="characterEncoding" value="UTF-8"/>
        <!--如果配置多個視圖解析器,它來決定優先使用哪個視圖解析器,它的值越小優先順序越高-->
        <property name="order" value="1"/>
        <!--當 ThymeleafViewResolver 渲染模板時,會使用該模板引擎來解析、編譯和渲染模板-->
        <property name="templateEngine">
            <bean class="org.thymeleaf.spring6.SpringTemplateEngine">
                <!--用於指定 Thymeleaf 模板引擎使用的模板解析器。模板解析器負責根據模板位置、模板資源名稱、文件編碼等信息,載入模板並對其進行解析-->
                <property name="templateResolver">
                    <bean class="org.thymeleaf.spring6.templateresolver.SpringResourceTemplateResolver">
                        <!--設置模板文件的位置(首碼)-->
                        <property name="prefix" value="/WEB-INF/templates/"/>
                        <!--設置模板文件尾碼(尾碼),Thymeleaf文件擴展名不一定是html,也可以是其他,例如txt,大部分都是html-->
                        <property name="suffix" value=".html"/>
                        <!--設置模板類型,例如:HTML,TEXT,JAVASCRIPT,CSS等-->
                        <property name="templateMode" value="HTML"/>
                        <!--用於模板文件在讀取和解析過程中採用的編碼字元集-->
                        <property name="characterEncoding" value="UTF-8"/>
                    </bean>
                </property>
            </bean>
        </property>
    </bean>

    <!--    配置攔截器-->
        <mvc:interceptors>
    <!--        基本配置,第一種方式
            註意:基本配置,預設情況下是攔截所有請求的-->
            <bean class="com.rainbowsea.springmvc.interceptors.Interceptor1"></bean>
        </mvc:interceptors>
</beans>

編寫對應的 Controller 控制器進行測試:

在這裡插入圖片描述

package com.rainbowsea.springmvc.controller;


import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

@Controller  // 交給 Spring IOC 容器管理
public class IndexController {


    @RequestMapping("/index")
    public String toIndex() {
        System.out.println("IndexController#toIndex()  ---> 處理器方法執行了");
        return "index";
    }


    @RequestMapping("ok")
    public String toOK() {
        System.out.println("IndexController#OK() ---> 處理器方法執行了");
        return "ok";
    }
}

運行測試:

在這裡插入圖片描述

在這裡插入圖片描述

第二種方式是:通過 @Component 註解 + xml 文件進行配置

註意:同樣的,對於這種基本配置來說,攔截器是攔截所有請求的。

第二種方式的前提:

  1. 前提1:包掃描,在 spring mvc 中配置組件掃描

在這裡插入圖片描述

  1. 前提2:使用 @Component 註解進行對 編寫的攔截器類進行標註即可。
    在這裡插入圖片描述
  1. 兩個前提都搞定了,就可以在 spring mvc.xml 文件中進行配置了。
  2. 在這裡插入圖片描述
<mvc:interceptors>
    <ref bean="interceptor1"/>
</mvc:interceptors>

運行測試:

在這裡插入圖片描述

3.3 攔截器的高級配置

採用以上基本配置方式,攔截器是攔截所有請求路徑的。如果要針對某些路徑進行攔截,某些路徑不攔截,某些路徑攔截,可以採用高級配置:在 spring mvc.xml 文件當中進行配置

在這裡插入圖片描述

以上的配置表示,除 /ok 請求路徑之外,剩下的路徑全部攔截。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:mvc="http://www.springframework.org/schema/mvc"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/cache http://www.springframework.org/schema/cache/spring-cache.xsd http://www.springframework.org/schema/mvc https://www.springframework.org/schema/mvc/spring-mvc.xsd">

    <!--    組件掃描-->
    <context:component-scan
            base-package="com.rainbowsea.springmvc.controller,com.rainbowsea.springmvc.interceptors"></context:component-scan>

    <!--    視圖解析器-->
    <bean id="thymeleafViewResolver" class="org.thymeleaf.spring6.view.ThymeleafViewResolver">
        <!--作用於視圖渲染的過程中,可以設置視圖渲染後輸出時採用的編碼字元集-->
        <property name="characterEncoding" value="UTF-8"/>
        <!--如果配置多個視圖解析器,它來決定優先使用哪個視圖解析器,它的值越小優先順序越高-->
        <property name="order" value="1"/>
        <!--當 ThymeleafViewResolver 渲染模板時,會使用該模板引擎來解析、編譯和渲染模板-->
        <property name="templateEngine">
            <bean class="org.thymeleaf.spring6.SpringTemplateEngine">
                <!--用於指定 Thymeleaf 模板引擎使用的模板解析器。模板解析器負責根據模板位置、模板資源名稱、文件編碼等信息,載入模板並對其進行解析-->
                <property name="templateResolver">
                    <bean class="org.thymeleaf.spring6.templateresolver.SpringResourceTemplateResolver">
                        <!--設置模板文件的位置(首碼)-->
                        <property name="prefix" value="/WEB-INF/templates/"/>
                        <!--設置模板文件尾碼(尾碼),Thymeleaf文件擴展名不一定是html,也可以是其他,例如txt,大部分都是html-->
                        <property name="suffix" value=".html"/>
                        <!--設置模板類型,例如:HTML,TEXT,JAVASCRIPT,CSS等-->
                        <property name="templateMode" value="HTML"/>
                        <!--用於模板文件在讀取和解析過程中採用的編碼字元集-->
                        <property name="characterEncoding" value="UTF-8"/>
                    </bean>
                </property>
            </bean>
        </property>
    </bean>

    <!--    高級配置:指定一些路徑被攔截,一些路徑不攔截-->
    <mvc:interceptors>
        <mvc:interceptor>
            <!--            /** 表示攔截所有路徑-->
            <mvc:mapping path="/**"/>
            <!--            /ok 請求路徑不攔截-->
            <mvc:exclude-mapping path="/ok"/>
            <!--            /index 請求路徑攔截-->
            <!--            <mvc:mapping path="/index"/>-->
            <!--            設置對應的那個攔截器-->
            <ref bean="interceptor1"></ref>
        </mvc:interceptor>

    </mvc:interceptors>
</beans>

運行測試:

在這裡插入圖片描述

4. Spring MVC中多個攔截器的執行順序

這裡我們為了探究,多個攔截器存在的時候的執行順序,我們創建 3 個 攔截器。如下:

4.1 如果所有攔截器 preHandle( ) 方法 都返回 true時,多個攔截器的的執行順序

在這裡插入圖片描述

在這裡插入圖片描述

在這裡插入圖片描述

配置多個攔截器

在這裡插入圖片描述

    <mvc:interceptors>

        <!--        配置多個攔截器,這個是基本配置,預設是所有請求都會進行攔截處理-->
        <ref bean="interceptor1"></ref>
        <ref bean="interceptor2"></ref>
        <ref bean="interceptor3"></ref>
    </mvc:interceptors>
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:mvc="http://www.springframework.org/schema/mvc"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/cache http://www.springframework.org/schema/cache/spring-cache.xsd http://www.springframework.org/schema/mvc https://www.springframework.org/schema/mvc/spring-mvc.xsd">

    <!--    組件掃描-->
    <context:component-scan
            base-package="com.rainbowsea.springmvc.controller,com.rainbowsea.springmvc.interceptors"></context:component-scan>

    <!--    視圖解析器-->
    <bean id="thymeleafViewResolver" class="org.thymeleaf.spring6.view.ThymeleafViewResolver">
        <!--作用於視圖渲染的過程中,可以設置視圖渲染後輸出時採用的編碼字元集-->
        <property name="characterEncoding" value="UTF-8"/>
        <!--如果配置多個視圖解析器,它來決定優先使用哪個視圖解析器,它的值越小優先順序越高-->
        <property name="order" value="1"/>
        <!--當 ThymeleafViewResolver 渲染模板時,會使用該模板引擎來解析、編譯和渲染模板-->
        <property name="templateEngine">
            <bean class="org.thymeleaf.spring6.SpringTemplateEngine">
                <!--用於指定 Thymeleaf 模板引擎使用的模板解析器。模板解析器負責根據模板位置、模板資源名稱、文件編碼等信息,載入模板並對其進行解析-->
                <property name="templateResolver">
                    <bean class="org.thymeleaf.spring6.templateresolver.SpringResourceTemplateResolver">
                        <!--設置模板文件的位置(首碼)-->
                        <property name="prefix" value="/WEB-INF/templates/"/>
                        <!--設置模板文件尾碼(尾碼),Thymeleaf文件擴展名不一定是html,也可以是其他,例如txt,大部分都是html-->
                        <property name="suffix" value=".html"/>
                        <!--設置模板類型,例如:HTML,TEXT,JAVASCRIPT,CSS等-->
                        <property name="templateMode" value="HTML"/>
                        <!--用於模板文件在讀取和解析過程中採用的編碼字元集-->
                        <property name="characterEncoding" value="UTF-8"/>
                    </bean>
                </property>
            </bean>
        </property>
    </bean>

    <mvc:interceptors>

        <!--        配置多個攔截器,這個是基本配置,預設是所有請求都會進行攔截處理-->
        <ref bean="interceptor1"></ref>
        <ref bean="interceptor2"></ref>
        <ref bean="interceptor3"></ref>
    </mvc:interceptors>
</beans>

如果所有攔截器 preHandle 都返回 true

在這裡插入圖片描述

按照 springmvc.xml文件中配置的順序,自上而下調用 preHandle:

在這裡插入圖片描述

在這裡插入圖片描述

4.2 如果其中一個攔截器 preHandle ( ) 方法,返回 false,多個攔截器的的執行順序

Interceptor3 攔截器中的 preHandle()方法返回 false。其他兩個攔截器返回 true.

在這裡插入圖片描述

規則:只要有一個攔截器preHandle返回false,任何postHandle都不執行。但返回false的攔截器的前面的攔截器按照逆序執行afterCompletion

在這裡插入圖片描述

  1. 只要有一個攔截器preHandle()方法,返回false,則任何攔截器的 postHandle()方法都不執行。但返回 false 的攔截器的前面的攔截器按照逆序執行afterCompletion

  2. 返回 false 攔截器,攔截住了,則其中的 Controllor控制器不執行了,其中的 postHandle

    一個也不會執行。而對應的 afterCompletion()方法,的執行是按照配置攔截器(自上而下)的倒序執行,但其中返回 false 的攔截器中的 afterCompletion()方法不會被執行

  3. 只要有一個攔截器preHandle返回false,任何postHandle都不執行。但返回false的攔截器的前面的攔截器按照逆序執行afterCompletion。只要有一個攔截器preHandle返回false,任何postHandle都不執行。但返回false的攔截器的前面的攔截器按照逆序執行afterCompletion

5. 補充:源碼分析

5.1 方法執行順序的源碼分析

public class DispatcherServlet extends FrameworkServlet {
    protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
        // 調用所有攔截器的 preHandle 方法
        if (!mappedHandler.applyPreHandle(processedRequest, response)) {
            return;
        }
        // 調用處理器方法
        mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
        // 調用所有攔截器的 postHandle 方法
        mappedHandler.applyPostHandle(processedRequest, response, mv);
        // 處理視圖
        processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
    }

    private void processDispatchResult(HttpServletRequest request, HttpServletResponse response,
			@Nullable HandlerExecutionChain mappedHandler, @Nullable ModelAndView mv,
			@Nullable Exception exception) throws Exception {
        // 渲染頁面
        render(mv, request, response);
        // 調用所有攔截器的 afterCompletion 方法
        mappedHandler.triggerAfterCompletion(request, response, null);
    }
}

5.2 攔截與放行的源碼分析

public class DispatcherServlet extends FrameworkServlet {
    protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
        // 調用所有攔截器的 preHandle 方法
        if (!mappedHandler.applyPreHandle(processedRequest, response)) {
            // 如果 mappedHandler.applyPreHandle(processedRequest, response) 返回false,以下的return語句就會執行
            return;
        }
    }
}
public class HandlerExecutionChain {
    boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception {
		for (int i = 0; i < this.interceptorList.size(); i++) {
			HandlerInterceptor interceptor = this.interceptorList.get(i);
			if (!interceptor.preHandle(request, response, this.handler)) {
				triggerAfterCompletion(request, response, null);
                // 如果 interceptor.preHandle(request, response, this.handler) 返回 false,以下的 return false;就會執行。
				return false;
			}
			this.interceptorIndex = i;
		}
		return true;
	}
}

5.3 DispatcherServlet 和 HandlerExecutionChain 的部分源碼:

public class DispatcherServlet extends FrameworkServlet {
    protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
        // 按照順序執行所有攔截器的preHandle方法
        if (!mappedHandler.applyPreHandle(processedRequest, response)) {
            return;
        }
        // 執行處理器方法
        mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
        // 按照逆序執行所有攔截器的 postHanle 方法
        mappedHandler.applyPostHandle(processedRequest, response, mv);
        // 處理視圖
        processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
    }

    private void processDispatchResult(HttpServletRequest request, HttpServletResponse response,
			@Nullable HandlerExecutionChain mappedHandler, @Nullable ModelAndView mv,
			@Nullable Exception exception) throws Exception {
        // 渲染視圖
        render(mv, request, response);
        // 按照逆序執行所有攔截器的 afterCompletion 方法
        mappedHandler.triggerAfterCompletion(request, response, null);
    }
}
public class HandlerExecutionChain {
    // 順序執行 preHandle
    boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception {
        for (int i = 0; i < this.interceptorList.size(); i++) {
            HandlerInterceptor interceptor = this.interceptorList.get(i);
            if (!interceptor.preHandle(request, response, this.handler)) {
                // 如果其中一個攔截器preHandle返回false
                // 將該攔截器前面的攔截器按照逆序執行所有的afterCompletion
                triggerAfterCompletion(request, response, null);
                return false;
            }
            this.interceptorIndex = i;
        }
        return true;
	}
    // 逆序執行 postHanle
    void applyPostHandle(HttpServletRequest request, HttpServletResponse response, @Nullable ModelAndView mv) throws Exception {
        for (int i = this.interceptorList.size() - 1; i >= 0; i--) {
            HandlerInterceptor interceptor = this.interceptorList.get(i);
            interceptor.postHandle(request, response, this.handler, mv);
        }
	}
    // 逆序執行 afterCompletion
	void triggerAfterCompletion(HttpServletRequest request, HttpServletResponse response, @Nullable Exception ex) {
		for (int i = this.interceptorIndex; i >= 0; i--) {
			HandlerInterceptor interceptor = this.interceptorList.get(i);
			try {
				interceptor.afterCompletion(request, response, this.handler, ex);
			}
			catch (Throwable ex2) {
				logger.error("HandlerInterceptor.afterCompletion threw exception", ex2);
			}
		}
	}
}

6. 總結:

  1. 實現org.springframework.web.servlet.HandlerInterceptor 介面,共有三個方法可以進行選擇性的實現:

    • preHandle( ):處理器方法調用之前執行。只有該方法有返回值,返回值是布爾類型,true 表示放行,false 表示攔截
    • postHandle( ):處理器方法調用之後執行。
    • afterCompletion( ):渲染完成後執行。
  2. 在 Spring MVC 中攔截器的基本配置有兩種方式:

    • 第一種方式是:通過 xml 進行配置
    • 第二種方式是:通過 @Component 註解 + xml 文件進行配置
    • 對於這種基本配置來說,攔截器是攔截所有請求的。
  3. 攔截器的高級配置:採用以上基本配置方式,攔截器是攔截所有請求路徑的。如果要針對某些路徑進行攔截,某些路徑不攔截,某些路徑攔截,可以採用高級配置:在 spring mvc.xml 文件當中進行配置

  4. Spring MVC中多個攔截器的執行順序:

    1. 如果所有攔截器 preHandle( ) 方法 都返回 true時,多個攔截器的的執行順序:

      1. 按照 springmvc.xml文件中配置的順序,自上而下調用 preHandle:

        在這裡插入圖片描述

  1. 如果其中一個攔截器 preHandle ( ) 方法,返回 false,多個攔截器的的執行順序

    1. 只要有一個攔截器preHandle()方法,返回false,則任何攔截器的 postHandle()方法都不執行。但返回 false 的攔截器的前面的攔截器按照逆序執行afterCompletion
  2. 攔截器源碼分析。

7. 最後:

“在這個最後的篇章中,我要表達我對每一位讀者的感激之情。你們的關註和回覆是我創作的動力源泉,我從你們身上吸取了無盡的靈感與勇氣。我會將你們的鼓勵留在心底,繼續在其他的領域奮鬥。感謝你們,我們總會在某個時刻再次相遇。”

在這裡插入圖片描述


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

-Advertisement-
Play Games
更多相關文章
  • 摘要:本文詳細介紹Nuxt.js中的useAsyncData組合式函數,它用於在服務端渲染(SSR)過程中非同步獲取數據,確保客戶端正確水合,避免重覆請求。內容包括基本概念、參數說明(key, handler, options)、示例用法、如何監聽參數變化自動刷新數據及返回值詳解,展示了在頁面組件中使... ...
  • 定義 裝飾模式(Decorator Pattern),又稱為包裝模式,是一種結構型設計模式。它允許在不改變現有對象結構的情況下,動態地添加新的功能。通過將每個功能封裝在單獨的裝飾器類中,並且這些裝飾器類通過引用原始對象來實現功能的組合,從而提供了靈活性和可擴展性的優勢。裝飾模式避免了通過繼承方式增加 ...
  • 沒想到自己的小破站也逃不掉被攻擊的命,分分鐘就給我刷欠費了。 本來不想寫這篇文章的,但看到好多大佬(小林coding、 JavaGuide)近期cdn都被盜刷了。 還是來提醒下大家,防火防盜防cdn流量盜刷 事故時間:2024年7月5日晚8點左右 事故現場:好不容易到了周五,想著第二天就周末了,和朋 ...
  • 本文主要介紹了日誌的定義和作用,以及 Python 內置日誌處理的 logging 模塊,同時簡單說明瞭日誌等級和 logging 模塊的簡易使用方法。 ...
  • 本文主要介紹了Python中創建自定義類時鴨子類型的基本定義、特點和應用場景,同時列舉了“file-like object“的例子對鴨子類型進行了說明。 ...
  • Spring Boot註解主要用於簡化配置、自動裝配組件和實現聲明式服務。以下是詳細的介紹: 1、Springboot註解 核心註解 1. @SpringBootApplication 作用: 標註一個主程式類,表明這是一個Spring Boot應用程式的入口。 功能: 這是一個複合註解,組合了@C ...
  • 有的時候博客內容會有變動,首發博客是最新的,其他博客地址可能會未同步,認準https://blog.zysicyj.top 官方地址 Ubuntu 針對Docker客戶端版本大於 1.10.0 的用戶 您可以通過修改daemon配置文件/etc/docker/daemon.json來使用加速器 su ...
  • 在開發過程中,我們可能會遇到需要生成word,或者通過模板word替換相應內容的需求。但在文檔中插入圖片時,如果段落格式設置不對,就會導致圖片只顯示一點點或者不顯示。接下來就介紹一下java編輯word和插入圖片需怎麼處理。 1.引入依賴 首先我們在項目中引入Apache POI,用於讀取和操作wo ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...