主頁 > 後端開發 > 精盡Spring Boot原始碼分析 - 支持外部 Tomcat 容器的實作

精盡Spring Boot原始碼分析 - 支持外部 Tomcat 容器的實作

2021-07-03 06:09:02 後端開發

該系列文章是筆者在學習 Spring Boot 程序中總結下來的,里面涉及到相關原始碼,可能對讀者不太友好,請結合我的原始碼注釋 Spring Boot 原始碼分析 GitHub 地址 進行閱讀

Spring Boot 版本:2.2.x

最好對 Spring 原始碼有一定的了解,可以先查看我的 《死磕 Spring 之 IoC 篇 - 文章導讀》 系列文章

如果該篇內容對您有幫助,麻煩點擊一下“推薦”,也可以關注博主,感激不盡~

該系列其他文章請查看:《精盡 Spring Boot 原始碼分析 - 文章導讀》

概述

我們知道 Spring Boot 應用能夠被打成 war 包,放入外部 Tomcat 容器中運行,你是否知道 Spring Boot 是如何整合 Spring MVC 的呢?

在上一篇 《Spring Boot 內嵌 Tomcat 容器的實作》 文章中分析了 Spring Boot 白打成 jar 包后是如何創建 Tomcat 容器并啟動的,那么這篇文章主要告訴你 Spring Boot 應用被打成 war 包后放入外部 Tomcat 容器是如何運行的,

如何使用

在我們的 Spring Boot 專案中通常會引入 spring-boot-starter-web 這個依賴,該模塊提供全堆疊的 WEB 開發特性,包括 Spring MVC 依賴和 Tomcat 容器,我們將內部 Tomcat 的 Starter 模塊排除掉,如下:

<packaging>war</packaging>

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
    <exclusions>
        <exclusion>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-tomcat</artifactId>
        </exclusion>
    </exclusions>
</dependency>
<dependency>
    <groupId>javax.servlet</groupId>
    <artifactId>javax.servlet-api</artifactId>
    <version>4.0.1</version>
    <scope>provided</scope>
</dependency>

然后啟動類這樣寫:

@SpringBootApplication
public class Application extends SpringBootServletInitializer {

    // 可不寫
    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);
    }

    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder builder) {
        return builder.sources(Application.class);
    }
}

這樣你打成 war 包就可以放入外部的 Servlet 容器中運行了,

實作原理

原理在分析 Spring MVC 原始碼的時候講過,參考我的 《精盡Spring MVC原始碼分析 - 尋找遺失的 web.xml》 這篇文章

借助于 Servlet 3.0 的一個新特性,新增的一個 javax.servlet.ServletContainerInitializer 介面,在 Servlet 容器啟動時會通過 Java 的 SPI 機制從 META-INF/services/javax.servlet.ServletContainerInitializer 檔案中找到這個介面的實作類,然后呼叫它的 onStartup(..) 方法,

在 Spring 的 spring-web 模塊中該檔案是這么配置的:

org.springframework.web.SpringServletContainerInitializer

一起來看看這個類:

@HandlesTypes(WebApplicationInitializer.class)
public class SpringServletContainerInitializer implements ServletContainerInitializer {

	@Override
	public void onStartup(@Nullable Set<Class<?>> webAppInitializerClasses, ServletContext servletContext)
			throws ServletException {

		List<WebApplicationInitializer> initializers = new LinkedList<>();

		if (webAppInitializerClasses != null) {
			for (Class<?> waiClass : webAppInitializerClasses) {
				// Be defensive: Some servlet containers provide us with invalid classes,
				// no matter what @HandlesTypes says...
				if (!waiClass.isInterface() && !Modifier.isAbstract(waiClass.getModifiers()) &&
						WebApplicationInitializer.class.isAssignableFrom(waiClass)) {
					try {
						initializers.add((WebApplicationInitializer)
								ReflectionUtils.accessibleConstructor(waiClass).newInstance());
					} catch (Throwable ex) {
						throw new ServletException("Failed to instantiate WebApplicationInitializer class", ex);
					}
				}
			}
		}

		if (initializers.isEmpty()) {
			servletContext.log("No Spring WebApplicationInitializer types detected on classpath");
			return;
		}

		servletContext.log(initializers.size() + " Spring WebApplicationInitializers detected on classpath");
		AnnotationAwareOrderComparator.sort(initializers);
		for (WebApplicationInitializer initializer : initializers) {
			initializer.onStartup(servletContext);
		}
	}
}

通過 @HandlesTypes 注解指定只處理 WebApplicationInitializer 型別的類

這個程序很簡單,實體化所有 WebApplicationInitializer 型別的物件,然后依次呼叫它們的 onStartup(ServletContext) 方法

通過打斷點你會發現,有一個 DemoApplication 就是我們的啟動類

這也就是為什么如果你的 Spring Boot 應用需要打成 war 包放入外部 Tomcat 容器運行的時候,你的啟動類需要繼承 SpringBootServletInitializer 這個抽象類,因為這個抽象類實作類 WebApplicationInitializer 介面,我們只需要繼承它即可

SpringBootServletInitializer

org.springframework.boot.web.servlet.support.SpringBootServletInitializer 抽象類,實作了 WebApplicationInitializer 介面,目的就是支持你將 Spring Boot 應用打包成 war 包放入外部的 Servlet 容器中運行

public abstract class SpringBootServletInitializer implements WebApplicationInitializer {

	protected Log logger; // Don't initialize early

	private boolean registerErrorPageFilter = true;

	protected final void setRegisterErrorPageFilter(boolean registerErrorPageFilter) {
		this.registerErrorPageFilter = registerErrorPageFilter;
	}

	@Override
	public void onStartup(ServletContext servletContext) throws ServletException {
		// Logger initialization is deferred in case an ordered
		// LogServletContextInitializer is being used
		this.logger = LogFactory.getLog(getClass());
		// <1> 創建一個 WebApplicationContext 作為 Root Spring 應用背景關系
		WebApplicationContext rootAppContext = createRootApplicationContext(servletContext);
		if (rootAppContext != null) {
			// <2> 添加一個 ContextLoaderListener 監聽器,會監聽到 ServletContext 的啟動事件
			// 因為 Spring 應用背景關系在上面第 `1` 步已經準備好了,所以這里什么都不用做
			servletContext.addListener(new ContextLoaderListener(rootAppContext) {
				@Override
				public void contextInitialized(ServletContextEvent event) {
					// no-op because the application context is already initialized
				}
			});
		} else {
			this.logger.debug("No ContextLoaderListener registered, as createRootApplicationContext() did not "
					+ "return an application context");
		}
	}
}

onStartup(ServletContext) 方法中就兩步:

  1. 呼叫 createRootApplicationContext(ServletContext) 方法,創建一個 WebApplicationContext 作為 Root Spring 應用背景關系
  2. 添加一個 ContextLoaderListener 監聽器,會監聽到 ServletContext 的啟動事件,因為 Spring 應用背景關系在上面第 1 步已經準備好了,所以這里什么都不用做

1 步是不是和 Spring MVC 類似,同樣創建一個 Root WebApplicationContext 作為 Spring 應用背景關系的父物件

createRootApplicationContext 方法

createRootApplicationContext(ServletContext) 方法,創建一個 Root WebApplicationContext 物件,如下:

protected WebApplicationContext createRootApplicationContext(ServletContext servletContext) {
    // <1> 創建一個 SpringApplication 構造器
    SpringApplicationBuilder builder = createSpringApplicationBuilder();
    // <2> 設定 `mainApplicationClass`,主要用于列印日志
    builder.main(getClass());
    // <3> 從 ServletContext 背景關系中獲取最頂部的 Root ApplicationContext 應用背景關系
    ApplicationContext parent = getExistingRootWebApplicationContext(servletContext);
    // <4> 如果已存在 Root ApplicationContext,則先置空,因為這里會創建一個 ApplicationContext 作為 Root
    if (parent != null) {
        this.logger.info("Root context already created (using as parent).");
        servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, null);
        // <4.1> 添加一個 ApplicationContextInitializer 初始器,
        // 用于設定現在要創建的 Root ApplicationContext 應用背景關系的父容器為 `parent`
        builder.initializers(new ParentContextApplicationContextInitializer(parent));
    }
    /**
     * <5> 添加一個 ApplicationContextInitializer 初始器
     * 目的是往 ServletContext 背景關系中設定 Root ApplicationContext 為現在要創建的 Root ApplicationContext 應用背景關系
     * 并將這個 ServletContext 保存至 ApplicationContext 中,參考 {@link ServletWebServerApplicationContext#createWebServer()} 方法,
     * 如果獲取到了 ServletContext 那么直接呼叫其 {@link ServletWebServerApplicationContext#selfInitialize} 方法來注冊各個 Servlet、Filter
     * 例如 {@link DispatcherServlet}
     */
    builder.initializers(new ServletContextApplicationContextInitializer(servletContext));
    // <6> 設定要創建的 Root ApplicationContext 應用背景關系的型別(Servlet)
    builder.contextClass(AnnotationConfigServletWebServerApplicationContext.class);
    // <7> 對 SpringApplicationBuilder 進行擴展
    builder = configure(builder);
    // <8> 添加一個 ApplicationListener 監聽器
    // 用于將 ServletContext 中的相關屬性關聯到 Environment 環境中
    builder.listeners(new WebEnvironmentPropertySourceInitializer(servletContext));
    // <9> 構建一個 SpringApplication 物件,用于啟動 Spring 應用
    SpringApplication application = builder.build();
    // <10> 如果沒有設定 `source` 源物件,那么這里嘗試設定為當前 Class 物件,需要有 `@Configuration` 注解
    if (application.getAllSources().isEmpty()
            && MergedAnnotations.from(getClass(), SearchStrategy.TYPE_HIERARCHY).isPresent(Configuration.class)) {
        application.addPrimarySources(Collections.singleton(getClass()));
    }
    // <11> 因為 SpringApplication 在創建 ApplicationContext 應用背景關系的程序中需要優先注冊 `source` 源物件,如果為空則拋出例外
    Assert.state(!application.getAllSources().isEmpty(),
            "No SpringApplication sources have been defined. Either override the "
                    + "configure method or add an @Configuration annotation");
    // Ensure error pages are registered
    if (this.registerErrorPageFilter) {
        // <12> 添加一個錯誤頁面 Filter 作為 `sources`
        application.addPrimarySources(Collections.singleton(ErrorPageFilterConfiguration.class));
    }
    // <13> 呼叫 `application` 的 `run` 方法啟動整個 Spring Boot 應用
    return run(application);
}

程序如下:

  1. 創建一個 SpringApplication 構造器,目的就是啟動 Spring 應用咯

    protected SpringApplicationBuilder createSpringApplicationBuilder() {
        return new SpringApplicationBuilder();
    }
    
  2. 設定 mainApplicationClass,也就是你的啟動類,主要用于列印日志

  3. 從 ServletContext 背景關系中獲取最頂部的 Root ApplicationContext 應用背景關系 parent,通常這里沒有父物件,所以為空

  4. 如果 parent 不為空,則先 ServletContext 中的該屬性置空,因為這里會創建一個 ApplicationContext 作為 Root

    1. 添加一個 ApplicationContextInitializer 初始器,用于設定現在要創建的 Root ApplicationContext 應用背景關系的父容器為 parent
  5. 添加一個 ApplicationContextInitializer 初始器,目的是往 ServletContext 背景關系中設定 Root ApplicationContext 為現在要創建的 Root ApplicationContext 應用背景關系,并將這個 ServletContext 保存至 ApplicationContext 中

    注意,這個物件很關鍵,會將當前 ServletContext 背景關系物件設定到 ApplicationContext 物件里面,那么后續就不會再創建 Spring Boot 內嵌的 Tomcat 了

  6. 設定要創建的 Root ApplicationContext 應用背景關系的型別(Servlet)

  7. 對 SpringApplicationBuilder 進行擴展,呼叫 configure(SpringApplicationBuilder) 方法,這也就是為什么我們的啟動類可以重寫該方法,通常不用做什么

  8. 添加一個 ApplicationListener 監聽器,用于將 ServletContext 中的相關屬性關聯到 Environment 環境中

  9. 構建一個 SpringApplication 物件 application,用于啟動 Spring 應用

  10. 如果沒有設定 source 源物件,那么這里嘗試設定為當前 Class 物件,需要有 @Configuration 注解

  11. 因為 SpringApplication 在創建 ApplicationContext 應用背景關系的程序中需要優先注冊 source 源物件,如果為空則拋出例外

  12. 添加一個錯誤頁面 Filter 作為 sources

  13. 呼叫 applicationrun 方法啟動整個 Spring Boot 應用

整個程序不復雜,SpringApplication 相關的內容在前面的 《SpringApplication 啟動類的啟動程序》文章中已經分析過,這里的關鍵在于第 5

添加的 ServletContextApplicationContextInitializer 會將當前 ServletContext 背景關系物件設定到 ApplicationContext 物件里面

ServletContextApplicationContextInitializer

public class ServletContextApplicationContextInitializer
		implements ApplicationContextInitializer<ConfigurableWebApplicationContext>, Ordered {

	private int order = Ordered.HIGHEST_PRECEDENCE;

	private final ServletContext servletContext;

	private final boolean addApplicationContextAttribute;

	public ServletContextApplicationContextInitializer(ServletContext servletContext) {
		this(servletContext, false);
	}

	public ServletContextApplicationContextInitializer(ServletContext servletContext,
			boolean addApplicationContextAttribute) {
		this.servletContext = servletContext;
		this.addApplicationContextAttribute = addApplicationContextAttribute;
	}

	public void setOrder(int order) {
		this.order = order;
	}

	@Override
	public int getOrder() {
		return this.order;
	}

	@Override
	public void initialize(ConfigurableWebApplicationContext applicationContext) {
		// 將這個 ServletContext 背景關系物件設定到 ApplicationContext 中
		applicationContext.setServletContext(this.servletContext);
		if (this.addApplicationContextAttribute) {
			this.servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE,
					applicationContext);
		}
	}
}

可以看到會將這個 ServletContext 背景關系物件設定到 ApplicationContext 中

那么我們回顧到上一篇 《Spring Boot 內嵌 Tomcat 容器的實作》 文章的 1. onRefresh 方法小節呼叫的 createWebServer() 方法,如下:

// ServletWebServerApplicationContext.java
private void createWebServer() {
    // <1> 獲取當前 `WebServer` 容器物件,首次進來為空
    WebServer webServer = this.webServer;
    // <2> 獲取 `ServletContext` 背景關系物件
    ServletContext servletContext = getServletContext();
    // <3> 如果 WebServer 和 ServletContext 都為空,則需要創建一個
    // 使用 Spring Boot 內嵌 Tomcat 容器則會進入該分支
    if (webServer == null && servletContext == null) {
        // <3.1> 獲取 Servlet 容器工廠物件(默認為 Tomcat)`factory`
        ServletWebServerFactory factory = getWebServerFactory();
        /**
         * <3.2> 先創建一個 {@link ServletContextInitializer} Servlet 背景關系初始器,實作也就是當前類的 {@link this#selfInitialize(ServletContext)} 方法
         * 至于為什么不用 Servlet 3.0 新增的 {@link javax.servlet.ServletContainerInitializer} 這個類,我在
         * [精盡Spring MVC原始碼分析 - 尋找遺失的 web.xml](https://www.cnblogs.com/lifullmoon/p/14122704.html)有提到過
         *
         * <3.3> 從 `factory` 工廠中創建一個 WebServer 容器物件
         * 例如創建一個 {@link TomcatWebServer} 容器物件,并初始化 `ServletContext` 背景關系,創建 {@link Tomcat} 容器并啟動
         * 啟動程序異步觸發了 {@link org.springframework.boot.web.embedded.tomcat.TomcatStarter#onStartup} 方法
         * 也就會呼叫這個傳入的 {@link ServletContextInitializer} 的 {@link #selfInitialize(ServletContext)} 方法
         */
        this.webServer = factory.getWebServer(getSelfInitializer());
    }
    // <4> 否則,如果 ServletContext 不為空,說明使用了外部的 Servlet 容器(例如 Tomcat)
    else if (servletContext != null) {
        try {
            /** 那么這里主動呼叫 {@link this#selfInitialize(ServletContext)} 方法來注冊各種 Servlet、Filter */
            getSelfInitializer().onStartup(servletContext);
        }
        catch (ServletException ex) {
            throw new ApplicationContextException("Cannot initialize servlet context", ex);
        }
    }
    // <5> 將 ServletContext 的一些初始化引數關聯到當前 Spring 應用的 Environment 環境中
    initPropertySources();
}

我們看到上面第 4 步,如果從當前 Spring 應用背景關系獲取到了 ServletContext 物件,不會走上面的第 3 步,也就是不創建 Spring Boot 內嵌的 Tomcat

主動呼叫它的 getSelfInitializer() 方法來往這個 ServletContext 物件中注冊各種 Servlet、Filter 和 EventListener 物件,包括 Spring MVC 中的 DispatcherServlet 物件,該方法參考上一篇 《Spring Boot 內嵌 Tomcat 容器的實作》 文章的 2. selfInitialize 方法 小節

總結

本文分析了 Spring Boot 應用被打成 war 包后是如何支持放入外部 Tomcat 容器運行的,原理也比較簡單,借助 Spring MVC 中的 SpringServletContainerInitializer 這個類,它實作了 Servlet 3.0 新增的 javax.servlet.ServletContainerInitializer 介面

  1. 通過 Java 的 SPI 機制,在 META-INF/services/javax.servlet.ServletContainerInitializer 檔案中寫入 SpringServletContainerInitializer 這個類,那么在 Servlet 容器啟動的時候會呼叫這個類的 onStartup(..) 方法,會找到 WebApplicationInitializer 型別的物件,并呼叫他們的 onStartup(ServletContext) 方法

  2. 在我們的 Spring Boot 應用中,如果需要打成 war 包放入外部 Tomcat 容器運行,啟動類則需要繼承 SpringBootServletInitializer 抽象類,它實作了 WebApplicationInitializer 介面

  3. SpringBootServletInitializer 中會創建一個 WebApplicationContext 作為 Root Spring 應用背景關系,同時會將 ServletContext 物件設定到 Spring 應用背景關系中

  4. 這樣一來,因為已經存在 ServletContext 物件,那么不會再創建 Spring Boot 內嵌的 Tomcat 容器,而是對 ServletContext 進行一些初始化作業

好了,到這里關于 Spring Boot 啟動 Spring 應用的整個主流程,包括內嵌 Tomcat 容器的實作,以及支持運行在外部 Servlet 容器的實作都分析完了

那么接下來,我們一起來看看 @SpringBootApplication 這個注解,也就是 @EnableAutoConfiguration 自動配置注解的實作原理

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/288829.html

標籤:Java

上一篇:hive學習筆記之四:磁區表

下一篇:如何合理的設計系統容量?

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more