主頁 > 後端開發 > 異步編程Ajax的詳解,并對其進行封裝整理

異步編程Ajax的詳解,并對其進行封裝整理

2020-10-13 10:26:30 後端開發

對于Ajax,肯定很多小伙伴都聽過甚至用過了,那么沒聽過的也不用著急,本文會對Ajax進行講解,其次,一定還有一些人只用過JQuery封裝好了的Ajax卻對原生的Ajax并不了解,那么也不用著急,本文從最基本的Ajax開始講起,然后最后會盡可能得模仿JQuery對其進行封裝,讓我剛才提到的兩類人能對Ajax有進一步的了解,

公眾號:Lpyexplore的編程小屋
關注我,不定時更新前端面試題
關注還有更多電子書前端面試題資料結構與演算法代碼等你來拿

異步編程——Ajax

  • 一、什么是Ajax
  • 二、Ajax的優缺點
    • (1)優點
    • (2)缺點
  • 三、Ajax的使用
    • (1)狀態碼
    • (2)xhr的基本使用
    • (3)發送get請求
    • (4)發送post請求
  • 四、封裝Ajax
    • (1)JQuery中的Ajax
    • (2)封裝準備作業
    • (3)封裝$.get方法
    • (4)封裝$.post方法
    • (5)封裝$.ajax方法
  • 五、Ajax的約束
  • 六、結束語

一、什么是Ajax

Ajax(Asynchronous JavaScript And XML)是2005年新出現的技術,它的出現是為了解決這樣一個場景:整個頁面中,只有一小部分的資料需要進行更新,按照傳統的前后端互動,我們需要向服務器請求該網頁的所有資料,然后再在客戶端重新渲染,這無疑是非常低效的操作,因此,Ajax就可以做到只向服務器請求我們想要的那一小部分資料,而不用請求全部資料,進而在重繪整個頁面的前提下更新那部分的資料,

舉個例子,我們去飯店吃飯,然后點了一桌子菜,后來發現其中有一道菜太咸了,因此我們只需要讓服務員端回去給廚師重新做這一道菜再拿回來就行了,

在這個例子中的人、物對比Ajax的關系如下表:

吃飯事件資料更新
我們客戶端
菜品頁面所有的資料
服務員ajax物件
廚師服務器

當我們發現有一道菜太咸了,不需要讓廚師把所有的菜重新做一遍,只要讓服務員拿這一道菜回去給廚師重做這一操作就相當于讓ajax物件向后端請求那一小部分資料再拿回來更新頁面而無需重繪整個頁面,

二、Ajax的優缺點

了解了Ajax的作用和定義,我們再來看看它的優缺點

(1)優點

  1. 瀏覽器默認支持(一般瀏覽器都是支持JavaScript的)
  2. 提高用戶體驗(不需要重繪整個頁面,而只需要區域重繪)
  3. 提高頁面的性能(只需要請求部分資料,所以資料量就明顯下降了)

(2)缺點

  1. 破壞了瀏覽器的前進和后退功能(Ajax不會改變網頁URL,因此不會在瀏覽器記錄前后頁面)
  2. 對搜索引擎的支持較弱(搜索引擎無法監測到JS引起的資料變化)

三、Ajax的使用

Ajax的基本流程:創建XHR物件 => 發送資料 => 接收資料

(1)狀態碼

既然Ajax涉及到前后端的資料互動,那么我們就先來簡單的看一下幾種型別的狀態碼,如下表:

狀態碼含義
100 ~ 199連接繼續
200 ~ 299各種成功的請求
300 ~ 399重定向
400 ~ 499客戶端錯誤
500 ~ 599服務端錯誤

(2)xhr的基本使用

在使用xhr之前,我們要創建一個xhr的實體物件

let xhr = new XMLHttpRequest()

然后再呼叫xhr物件上的 open() 方法,表示創建一個請求,

open() 方法接收三個引數:

  • 第一個引數: 請求的型別(例如get 、post)
  • 第二個引數: 請求的URL
  • 第三個引數: 是否異步發送請求(默認為true)
// 創建了一個Ajax請求
xhr.open('get', 'example.php', 'true')

光呼叫了 open() 方法還不夠,它只是創建了一個請求,但還沒有發送請求,因此我們還要呼叫xhr物件上的另一個方法,即 send() 方法,表示將請求發送給目標URL

send() 方法接收一個引數:

  • 第一個引數: 作為請求主體發送的資料(例如post請求攜帶的資料)
// 我們上面創建的是get請求,因此send()方法無需傳參
xhr.send()

請求發送出去后,客戶端需要接收服務器回應回來的資料,xhr物件中有一些屬性,它們存盤著服務端回傳來的一些資料資訊,如下表所示

屬性名含義
responseText服務端回傳的文本資訊
responseXML服務端回傳的XML DOM檔案
statusHTTP狀態碼
statusTextHTTP狀態碼說明
readyStatexhr物件的請求回應階段

既然我們要獲取服務端回傳的資料,我們就要知道服務端是何時回傳資料的,這就可以通過上面表格中的 readyState 屬性來判斷了

readyState 屬性一共有5個值,分別表示不同的請求回應階段:

  • 0: 還未創建請求,即未呼叫 open() 方法
  • 1: 已呼叫 open() 方法,但未發送 send() 方法
  • 2: 已呼叫 send() 方法,但未接收到回應
  • 3: 已接收到部分回應
  • 4: 已接收到全部的回應

同時,xhr物件可以系結一個 readystatechange 事件,每當 readyState 屬性發生改變,都會觸發該事件,因此,該事件在一次請求中會被多次觸發

xhr.onreadystatechange = function() {
	console.log('readyState屬性發生改變了')
}

所以,我們可以在 readystatechange 事件中判斷一下 readyState 屬性是否為 4,即是否已經接收所有的回應,然后還可以再繼續判斷一下 status 屬性,看看狀態碼是否為 200,當上述都成立了,我們再去 responseText 屬性 或 responseXML 屬性中獲取回應資料

xhr.onreadystatechange = function() {
	// 判斷是否已接收所有回應
	if(xhr.readyState === 4) {
		// 判斷狀態碼是否為200
		if(xhr.status === 200) {
			console.log(xhr.responseText)
		}
	}
}

(3)發送get請求

上面也講解了Ajax請求的簡單應用,同時也是拿 get 請求來舉得例子,因此這里我就不多做說明,唯一要講的就是,get請求所攜帶的資料是明文的,大小只有4k左右,而且它是寫在URL的 ? 后面的,例如這樣 example.php?query=4&em=0,所以若是我們要在發送get請求時攜帶資料,只需要在呼叫 open() 方法時,將資料寫在第二個引數的URL的 ? 后面即可

直接來寫一次完整的 get 請求,代碼如下:

let xhr = new XMLHttpRequest()
xhr.open('get', 'example.php?query=4&em=0')
xhr.send()
xhr.onreadystatechange = function() {
    if(xhr.readyState === 4) {
        if(xhr.status === 200){
            console.log(xhr.responseText);
        }
    }
}

(4)發送post請求

發送post請求的程序幾乎和get請求一樣,唯一不一樣的是資料的傳遞,大家都知道post請求的資料是放在請求體中的,因此我們需要呼叫xhr物件上的 setRequestHeader() 方法來模仿表單提交時的內容型別

該方法傳入的引數比較固定,代碼如下

xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded')

然后我們上面也說過,send() 方法接收的一個引數是請求主體發送的資料,所以我們的post請求要發送的資料就要作為該方法的引數,代碼如下:

xhr.send('query=4&em=0')

那我們來看一次完整的post請求是怎么樣的吧,代碼如下:

let xhr = new XMLHttpRequest()
xhr.open('post', 'example.php')
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded')
xhr.send('query=4&em=0')
xhr.onreadystatechange = function() {
    if(xhr.readyState === 4) {
        if(xhr.status === 200){
            console.log(xhr.responseText);
        }
    }
}

四、封裝Ajax

文章開頭提到,JQuery早已對Ajax請求進行了成熟的封裝,所以我們可以借鑒它,甚至盡可能地去模仿它進行封裝,在這之前,我們得先了解JQuery中Ajax的使用

(1)JQuery中的Ajax

這里我找來了幾段使用JQuery發送Ajax請求的代碼,如下所示:

  • 發送get請求
$.get('example.php', {query: 4, em: 0}, function(data, status, xhr) {
	console.log(`
		回傳的資料為${data}
		回傳的狀態為${status}
		回傳xhr物件為${xhr}
	`)
}, 'json')

這段代碼發送了一個 get 請求,攜帶的引數有 query 值為 4em 值為 0,規定回傳的資料型別為 json,同時設定了一個回呼函式用于接收請求回傳的資料、狀態和xhr物件

  • 發送post請求
$.post('example.php', {query: 4, em: 0}, function(data, status, xhr) {
	console.log(`
		回傳的資料為${data}
		回傳的狀態為${status}
		回傳xhr物件為${xhr}
	`)
}, 'json')

這段代碼發送了一個 post 請求,攜帶的引數有 query 值為 4em 值為 0,規定回傳的資料型別為 json,同時設定了一個回呼函式用于接收請求回傳的資料、狀態和xhr物件

  • 綜合方法
// 該方法既可以發送get請求又可以發送post請求
$.ajax({
	url: 'example.php', // 請求的URL
	type: 'get', //請求型別,若為post,則表示發送post請求
	data: {query: 4, em: 0},     // 請求攜帶資料
	dataType: 'json',  // 接收的資料型別
	isAsync: true     // 是否異步請求
})
.then(data => {
	console.log(`請求成功,資料為${data}`)
})
.catch(err => {
	console.log(`請求失敗,狀態為${err}`)
})

其呼叫的是一個綜合的方法,傳入的引數是一個物件,物件中傳入多個引數,這段代碼是發送了一個 get 請求,地址為 example.php,攜帶的引數有 query 值為 4em 值為 0,所接識訓傳資料的型別為 json,請求為異步請求

特別的是,該方法的回呼函式是通過 promise 實作的,即該方法回傳一個 promise 物件,在 then 函式中處理請求成功的情況,在 catch 函式中處理請求失敗的情況

若沒有了解過 promise 的小伙伴建議先花幾分鐘了解一下,因為這是異步編程最常用的一個語法,下面放上文章鏈接——深入了解Promise物件,寫出優雅的回呼代碼,告別回呼地獄

接下來我們就針對上述給出的例子,逐個封裝

(2)封裝準備作業

因為 XMLHttpRequest 物件有一定的兼容性,因此我們在封裝ajax方法之前可以先封裝一個方法用來動態創建一個兼容性稍微好點的XHR物件(其中主要是兼容IE5和IE6)

我們都知道JQuery都是將方法封裝在一個名為 $ 的物件中的,我們也這么做

let $ = {
	createXHR: function() {
		// 若瀏覽器支持,則創建XMLHttpRequest物件
		if(window.XMLHttpRequest) {
			return new XMLHttpRequest()
		} 
		// 若不支持,則創建ActiveXobject物件
		else {
			return new ActiveXObject()
		} 
	}
}

(3)封裝$.get方法

首先查閱JQuery的 get 方法可知,其接收四個引數:URLdatacallbackdataType,分別表示請求的url地址、攜帶的引數、成功回呼函式、回傳資料的型別

let $ = {
	// 動態生成XHR物件的方法
	createXHR: function() {
		if(window.XMLHttpRequest) {
			return new XMLHttpRequest()
		} else {
			return new ActiveXObject()
		} 
	},
	get: function(url, data, callback, dataType) {
		// 避免dataType大小寫的問題
		let dataType = dataType.toLowerCase()
		// 如果有傳入data,則在url后面跟上引數
		if(data) {
			url += '?'
			Object.keys(data).forEach(key => url += `${key}=${data[key]}&`)
			url = url.slice(0, -1)
		}
		// 呼叫我們封裝的方法生成XHR物件
		let xhr = this.createXHR()
		// 創建get請求
		xhr.open('get', url)
		// 發送請求
		xhr.send()
		xhr.onreadystatechange = function() {
			if(xhr.readyState === 4) {
				if(xhr.status >= 200 && xhr.status < 300 || xhr.status == 304) {
					// 若dataType為json,則將回傳的資料通過JSON.parse格式化
					let res = dataType === 'json' ? JSON.parse(xhr.responseText) : xhr.responseText
					// 呼叫回呼函式,并把引數傳進去
					callback(res, xhr.status, xhr)
				}
			}
		}
	},
}

(4)封裝$.post方法

JQuery的 post 方法傳入的引數跟 get 方法一樣,只不過其內部的實作有略微的區別,就是攜帶引數的發送不一樣,所以直接來看代碼吧

let $ = {
	// 動態生成XHR物件的方法
	createXHR: function() {
		if(window.XMLHttpRequest) {
			return new XMLHttpRequest()
		} else {
			return new ActiveXObject()
		} 
	},
	post: function(url, data, callback, dataType) {
		// 避免dataType大小寫的問題
		let dataType = dataType.toLowerCase()
		// 呼叫我們封裝的方法動態生成XHR物件
		let xhr = this.createXHR()

		let str = ''
		// 若傳入引數,則將引數序列化
		if(data) {
			Object.keys(data).forEach(key => str += `${key}=${data[key]}&`)
			str = str.slice(0, -1)
		}
		// 設定頭部資訊
		xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded')
		// 發送請求,并攜帶引數
		xhr.send(str)
		xhr.onreadystatechange = function() {
			if(xhr.readyState === 4) {
				if(xhr.status >= 200 && xhr.status < 300 || xhr.status == 304) {
					// 若dataType為json,則將回傳的資料通過JSON.parse格式化
					let res = dataType === 'json' ? JSON.parse(xhr.responseText) : xhr.responseText
					// 呼叫回呼函式,把對應引數傳進去
					callback(res, xhr.status, xhr)
				}
			}
		}
	}
}

(5)封裝$.ajax方法

在JQuery中還有一個 ajax 方法,其既可以發送 get 請求,也可以發送 post 請求,該方法可傳入多種引數,且支持 promise 處理回呼函式

let $ = {
	createXHR: function() {
		if(window.XMLHttpRequest) {
			return new XMLHttpRequest()
		} else {
			return new ActiveXObject()
		} 
	},
	ajax: function(params) {
		// 初始化引數
		let type = params.type ? params.type.toLowerCase() : 'get'
		let isAsync = params.isAsync ? params.isAsync : 'true'
		let url = params.url
		let data = params.data ? params.data : {}
		let dataType = params.dataType.toLowerCase()
		// 用我們封裝的方法動態生成XHR物件
		let xhr = this.createXHR()
		
		let str = ''
		
		// 拼接字串
		Object.keys(data).forEach(key => str += `${key}=${data[key]}&`)
		str = str.slice(0, -1)
		// 如果是get請求就把攜帶引數拼接到url后面
		if(type === 'get') url += `?${str}`;
		// 回傳promise物件,便于外部then和catch函式呼叫
		return new Promise((resolve, reject) => {
			// 創建請求
			xhr.open(type, url, isAsync)
			
			if(type === 'post') {
				xhr.setRequestHeader('Content-Type', 'application/x-www-form-rulencoded')
				xhr.send(str)
			} else {
				xhr.send()
			}

			xhr.onreadystatechange = function() {
				if(xhr.readyState === 4) {
					if(xhr.status >= 200 && xhr.status < 300 || xhr.status == 304) {
						let res = dataType === 'json' ? JSON.parse(xhr.responseText) : xhr.responseText
						resolve(res) // 請求成功,回傳資料
					} else {
						reject(xhr.status) // 請求失敗,回傳狀態碼
					}
				}
			}
		})	
	}
}

五、Ajax的約束

默認情況下,Ajax一般只能向同源的域發送請求,這是受到了瀏覽器的同源策略的限制,關于同源策略,你們可以去看一下我以前寫過的一篇博客,里面寫了同源策略的定義以及解決方案——前端人員都懂的瀏覽器的同源策略,以及如何進行不同源間的相互訪問

了解過同源策略以后,我們來看看如何讓Ajax不受同源策略的限制而成功發送請求,CORS(跨域資源共享)要求我們在發送請求時自定義一個HTTP頭部與服務器進行溝通,我們只需要設定一個名為 Origin 的頭部,值為當前頁面的源資訊(協議、域名、埠),例如 Origin : http://example.com ;然后服務器需要設定一個名為 Access-Control-Allow-Origin 的回應頭部,其值為允許跨域訪問的源資訊,若服務器設定的 Access-Control-Allow-Origin 與我們設定的 Origin 相同,則表示服務器允許我們跨域請求其資源,或者服務器可以將 Access-Control-Allow-Origin 值設為 *,此時表示允許任何域向其發送請求并且不受同源策略的限制,

現在的大部分瀏覽器幾乎都支持了在發送Ajax請求后,自動向請求頭部添加當前的源資訊

六、結束語

建議你們好好了解JS的Ajax的使用,這樣在面試中問起來你還能說出個一二三,并且有時候面試官還會直接讓你親手寫一個簡單的Ajax請求呢,而不會讓你使用JQuery的,看了本文,想必面試官如果讓你當場封裝一個類似JQuery的Ajax請求,你也不會手足無措呢

歡迎關注公眾號:Lpyexplore的編程小屋 , 不定時更新前端面試題,與我一起學習前端,早日斬獲大廠Offer

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

標籤:java

上一篇:大資料開發實習生--------------------入職篇

下一篇:Canvas圓形時鐘

標籤雲
其他(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