主頁 > 後端開發 > 刷leetcode必備演算法,動態規劃詳解

刷leetcode必備演算法,動態規劃詳解

2022-04-05 07:08:23 後端開發

我們刷leetcode的時候,經常會遇到動態規劃型別題目,動態規劃問題非常非常經典,也很有技巧性,一般大廠都非常喜歡問,今天跟大家一起來學習動態規劃的套路,文章如果有不正確的地方,歡迎大家指出哈,感謝感謝~

  • 什么是動態規劃?
  • 動態規劃的核心思想
  • 一個例子走進動態規劃
  • 動態規劃的解題套路
  • leetcode案例分析

公眾號:撿田螺的小男孩

什么是動態規劃?

動態規劃(英語:Dynamic programming,簡稱 DP),是一種在數學、管理科學、計算機科學、經濟學和生物資訊學中使用的,通過把原問題分解為相對簡單的子問題的方式求解復雜問題的方法,動態規劃常常適用于有重疊子問題和最優子結構性質的問題,

dynamic programming is a method for solving a complex problem by breaking it down into a collection of simpler subproblems.

以上定義來自維基百科,看定義感徑訓是有點抽象,簡單來說,動態規劃其實就是,給定一個問題,我們把它拆成一個個子問題,直到子問題可以直接解決,然后呢,把子問題答案保存起來,以減少重復計算,再根據子問題答案反推,得出原問題解的一種方法,

一般這些子問題很相似,可以通過函式關系式遞推出來,然后呢,動態規劃就致力于解決每個子問題一次,減少重復計算,比如斐波那契數列就可以看做入門級的經典動態規劃問題,

動態規劃核心思想

動態規劃最核心的思想,就在于拆分子問題,記住過往,減少重復計算,

動態規劃在于記住過往

我們來看下,網上比較流行的一個例子:

  • A : "1+1+1+1+1+1+1+1 =?"
  • A : "上面等式的值是多少"
  • B : 計算 "8"
  • A : 在上面等式的左邊寫上 "1+" 呢?
  • A : "此時等式的值為多少"
  • B : 很快得出答案 "9"
  • A : "你怎么這么快就知道答案了"
  • A : "只要在8的基礎上加1就行了"
  • A : "所以你不用重新計算,因為你記住了第一個等式的值為8!動態規劃演算法也可以說是 '記住求過的解來節省時間'"

一個例子帶你走進動態規劃 -- 青蛙跳階問題

暴力遞回

leetcode原題:一只青蛙一次可以跳上1級臺階,也可以跳上2級臺階,求該青蛙跳上一個 10 級的臺階總共有多少種跳法,

有些小伙伴第一次見這個題的時候,可能會有點蒙圈,不知道怎么解決,其實可以試想:

  • 要想跳到第10級臺階,要么是先跳到第9級,然后再跳1級臺階上去;要么是先跳到第8級,然后一次邁2級臺階上去,
  • 同理,要想跳到第9級臺階,要么是先跳到第8級,然后再跳1級臺階上去;要么是先跳到第7級,然后一次邁2級臺階上去,
  • 要想跳到第8級臺階,要么是先跳到第7級,然后再跳1級臺階上去;要么是先跳到第6級,然后一次邁2級臺階上去,

假設跳到第n級臺階的跳數我們定義為f(n),很顯然就可以得出以下公式:

f(10) = f(9)+f(8)
f (9)  = f(8) + f(7)
f (8)  = f(7) + f(6)
...
f(3) = f(2) + f(1)

即通用公式為: f(n) = f(n-1) + f(n-2)

那f(2) 或者 f(1) 等于多少呢?

  • 當只有2級臺階時,有兩種跳法,第一種是直接跳兩級,第二種是先跳一級,然后再跳一級,即f(2) = 2;
  • 當只有1級臺階時,只有一種跳法,即f(1)= 1;

因此可以用遞回去解決這個問題:

class Solution {
    public int numWays(int n) {
    if(n == 1){
        return 1;
    }
     if(n == 2){
        return 2;
    }
    return numWays(n-1) + numWays(n-2);
    }
}

去leetcode提交一下,發現有問題,超出時間限制了

為什么超時了呢?遞回耗時在哪里呢?先畫出遞回樹看看:

  • 要計算原問題 f(10),就需要先計算出子問題 f(9) 和 f(8)
  • 然后要計算 f(9),又要先算出子問題 f(8) 和 f(7),以此類推,
  • 一直到 f(2) 和 f(1),遞回樹才終止,

我們先來看看這個遞回的時間復雜度吧:

遞回時間復雜度 = 解決一個子問題時間*子問題個數
  • 一個子問題時間 = f(n-1)+f(n-2),也就是一個加法的操作,所以復雜度是 O(1);
  • 問題個數 = 遞回樹節點的總數,遞回樹的總節點 = 2^n-1,所以是復雜度O(2^n),

因此,青蛙跳階,遞回解法的時間復雜度 = O(1) * O(2^n) = O(2^n),就是指數級別的,爆炸增長的,如果n比較大的話,超時很正常的了,

回過頭來,你仔細觀察這顆遞回樹,你會發現存在大量重復計算,比如f(8)被計算了兩次,f(7)被重復計算了3次...所以這個遞回演算法低效的原因,就是存在大量的重復計算!

既然存在大量重復計算,那么我們可以先把計算好的答案存下來,即造一個備忘錄,等到下次需要的話,先去備忘錄查一下,如果有,就直接取就好了,備忘錄沒有才開始計算,那就可以省去重新重復計算的耗時啦!這就是帶備忘錄的解法,

帶備忘錄的遞回解法(自頂向下)

一般使用一個陣列或者一個哈希map充當這個備忘錄,

  • 第一步,f(10)= f(9) + f(8),f(9) 和f(8)都需要計算出來,然后再加到備忘錄中,如下:

  • 第二步, f(9) = f(8)+ f(7),f(8)= f(7)+ f(6), 因為 f(8) 已經在備忘錄中啦,所以可以省掉,f(7),f(6)都需要計算出來,加到備忘錄中~

第三步, f(8) = f(7)+ f(6),發現f(8),f(7),f(6)全部都在備忘錄上了,所以都可以剪掉,

所以呢,用了備忘錄遞回演算法,遞回樹變成光禿禿的樹干咯,如下:

帶備忘錄的遞回演算法,子問題個數=樹節點數=n,解決一個子問題還是O(1),所以帶備忘錄的遞回演算法的時間復雜度是O(n),接下來呢,我們用帶備忘錄的遞回演算法去擼代碼,解決這個青蛙跳階問題的超時問題咯~,代碼如下:

public class Solution {
    //使用哈希map,充當備忘錄的作用
    Map<Integer, Integer> tempMap = new HashMap();
    public int numWays(int n) {
        // n = 0 也算1種
        if (n == 0) {
            return 1;
        }
        if (n <= 2) {
            return n;
        }
        //先判斷有沒計算過,即看看備忘錄有沒有
        if (tempMap.containsKey(n)) {
            //備忘錄有,即計算過,直接回傳
            return tempMap.get(n);
        } else {
            // 備忘錄沒有,即沒有計算過,執行遞回計算,并且把結果保存到備忘錄map中,對1000000007取余(這個是leetcode題目規定的)
            tempMap.put(n, (numWays(n - 1) + numWays(n - 2)) % 1000000007);
            return tempMap.get(n);
        }
    }
}

去leetcode提交一下,如圖,穩了:

其實,還可以用動態規劃解決這道題,

自底向上的動態規劃

動態規劃跟帶備忘錄的遞回解法基本思想是一致的,都是減少重復計算,時間復雜度也都是差不多,但是呢:

  • 帶備忘錄的遞回,是從f(10)往f(1)方向延伸求解的,所以也稱為自頂向下的解法,
  • 動態規劃從較小問題的解,由交疊性質,逐步決策出較大問題的解,它是從f(1)往f(10)方向,往上推求解,所以稱為自底向上的解法,

動態規劃有幾個典型特征,最優子結構、狀態轉移方程、邊界、重疊子問題,在青蛙跳階問題中:

  • f(n-1)和f(n-2) 稱為 f(n) 的最優子結構
  • f(n)= f(n-1)+f(n-2)就稱為狀態轉移方程
  • f(1) = 1, f(2) = 2 就是邊界啦
  • 比如f(10)= f(9)+f(8),f(9) = f(8) + f(7) ,f(8)就是重疊子問題,

我們來看下自底向上的解法,從f(1)往f(10)方向,想想是不是直接一個for回圈就可以解決啦,如下:

帶備忘錄的遞回解法,空間復雜度是O(n),但是呢,仔細觀察上圖,可以發現,f(n)只依賴前面兩個數,所以只需要兩個變數a和b來存盤,就可以滿足需求了,因此空間復雜度是O(1)就可以啦

動態規劃實作代碼如下:

public class Solution {
    public int numWays(int n) {
        if (n<= 1) {
            return 1;
        }
        if (n == 2) {
            return 2;
        }
        int a = 1;
        int b = 2;
        int temp = 0;
        for (int i = 3; i <= n; i++) {
            temp = (a + b)% 1000000007;
            a = b;
            b = temp;
        }
        return temp;
    }
    }

動態規劃的解題套路

什么樣的問題可以考慮使用動態規劃解決呢?

如果一個問題,可以把所有可能的答案窮舉出來,并且窮舉出來后,發現存在重疊子問題,就可以考慮使用動態規劃,

比如一些求最值的場景,如最長遞增子序列、最小編輯距離、背包問題、湊零錢問題等等,都是動態規劃的經典應用場景,

動態規劃的解題思路

動態規劃的核心思想就是拆分子問題,記住過往,減少重復計算, 并且動態規劃一般都是自底向上的,因此到這里,基于青蛙跳階問題,我總結了一下我做動態規劃的思路:

  • 窮舉分析
  • 確定邊界
  • 找出規律,確定最優子結構
  • 寫出狀態轉移方程

1. 窮舉分析

  • 當臺階數是1的時候,有一種跳法,f(1) =1
  • 當只有2級臺階時,有兩種跳法,第一種是直接跳兩級,第二種是先跳一級,然后再跳一級,即f(2) = 2;
  • 當臺階是3級時,想跳到第3級臺階,要么是先跳到第2級,然后再跳1級臺階上去,要么是先跳到第 1級,然后一次邁 2 級臺階上去,所以f(3) = f(2) + f(1) =3
  • 當臺階是4級時,想跳到第3級臺階,要么是先跳到第3級,然后再跳1級臺階上去,要么是先跳到第 2級,然后一次邁 2 級臺階上去,所以f(4) = f(3) + f(2) =5
  • 當臺階是5級時......

自底向上的動態規劃

2. 確定邊界

通過窮舉分析,我們發現,當臺階數是1的時候或者2的時候,可以明確知道青蛙跳法,f(1) =1,f(2) = 2,當臺階n>=3時,已經呈現出規律f(3) = f(2) + f(1) =3,因此f(1) =1,f(2) = 2就是青蛙跳階的邊界,

3. 找規律,確定最優子結構

n>=3時,已經呈現出規律 f(n) = f(n-1) + f(n-2) ,因此,f(n-1)和f(n-2) 稱為 f(n) 的最優子結構,什么是最優子結構?有這么一個解釋:

一道動態規劃問題,其實就是一個遞推問題,假設當前決策結果是f(n),則最優子結構就是要讓 f(n-k) 最優,最優子結構性質就是能讓轉移到n的狀態是最優的,并且與后面的決策沒有關系,即讓后面的決策安心地使用前面的區域最優解的一種性質

4, 寫出狀態轉移方程

通過前面3步,窮舉分析,確定邊界,最優子結構,我們就可以得出狀態轉移方程啦:

5. 代碼實作

我們實作代碼的時候,一般注意從底往上遍歷哈,然后關注下邊界情況,空間復雜度,也就差不多啦,動態規劃有個框架的,大家實作的時候,可以考慮適當參考一下:

dp[0][0][...] = 邊界值
for(狀態1 :所有狀態1的值){
    for(狀態2 :所有狀態2的值){
        for(...){
          //狀態轉移方程
          dp[狀態1][狀態2][...] = 求最值
        }
    }
}

leetcode案例分析

我們一起來分析一道經典leetcode題目吧

給你一個整數陣列 nums ,找到其中最長嚴格遞增子序列的長度,

示例 1:

輸入:nums = [10,9,2,5,3,7,101,18]
輸出:4
解釋:最長遞增子序列是 [2,3,7,101],因此長度為 4 ,

示例 2:

輸入:nums = [0,1,0,3,2,3]
輸出:4

我們按照以上動態規劃的解題思路,

  • 窮舉分析
  • 確定邊界
  • 找規律,確定最優子結構
  • 狀態轉移方程

1.窮舉分析

因為動態規劃,核心思想包括拆分子問題,記住過往,減少重復計算, 所以我們在思考原問題:陣列num[i]的最長遞增子序列長度時,可以思考下相關子問題,比如原問題是否跟子問題num[i-1]的最長遞增子序列長度有關呢?

自頂向上的窮舉

這里觀察規律,顯然是有關系的,我們還是遵循動態規劃自底向上的原則,基于示例1的資料,從陣列只有一個元素開始分析,

  • 當nums只有一個元素10時,最長遞增子序列是[10],長度是1.
  • 當nums需要加入一個元素9時,最長遞增子序列是[10]或者[9],長度是1,
  • 當nums再加入一個元素2時,最長遞增子序列是[10]或者[9]或者[2],長度是1,
  • 當nums再加入一個元素5時,最長遞增子序列是[2,5],長度是2,
  • 當nums再加入一個元素3時,最長遞增子序列是[2,5]或者[2,3],長度是2,
  • 當nums再加入一個元素7時,,最長遞增子序列是[2,5,7]或者[2,3,7],長度是3,
  • 當nums再加入一個元素101時,最長遞增子序列是[2,5,7,101]或者[2,3,7,101],長度是4,
  • 當nums再加入一個元素18時,最長遞增子序列是[2,5,7,101]或者[2,3,7,101]或者[2,5,7,18]或者[2,3,7,18],長度是4,
  • 當nums再加入一個元素7時,最長遞增子序列是[2,5,7,101]或者[2,3,7,101]或者[2,5,7,18]或者[2,3,7,18],長度是4.
分析找規律,拆分子問題

通過上面分析,我們可以發現一個規律:

如果新加入一個元素nums[i], 最長遞增子序列要么是以nums[i]結尾的遞增子序列,要么就是nums[i-1]的最長遞增子序列,看到這個,是不是很開心,nums[i]的最長遞增子序列已經跟子問題 nums[i-1]的最長遞增子序列有關聯了,

原問題陣列nums[i]的最長遞增子序列 = 子問題陣列nums[i-1]的最長遞增子序列/nums[i]結尾的最長遞增子序列

是不是感覺成功了一半呢?但是如何把nums[i]結尾的遞增子序列也轉化為對應的子問題呢?要是nums[i]結尾的遞增子序列也跟nums[i-1]的最長遞增子序列有關就好了,又或者nums[i]結尾的最長遞增子序列,跟前面子問題num[j](0=<j<i)結尾的最長遞增子序列有關就好了,帶著這個想法,我們又回頭看看窮舉的程序:

nums[i]的最長遞增子序列,不就是從以陣列num[i]每個元素結尾的最長子序列集合,取元素最多(也就是長度最長)那個嘛,所以原問題,我們轉化成求出以陣列nums每個元素結尾的最長子序列集合,再取最大值嘛,哈哈,想到這,我們就可以用dp[i]表示以num[i]這個數結尾的最長遞增子序列的長度啦,然后再來看看其中的規律:

其實,nums[i]結尾的自增子序列,只要找到比nums[i]小的子序列,加上nums[i] 就可以啦,顯然,可能形成多種新的子序列,我們選最長那個,就是dp[i]的值啦

  • nums[3]=5,以5結尾的最長子序列就是[2,5],因為從陣列下標0到3遍歷,只找到了子序列[2]5小,所以就是[2]+[5]啦,即dp[4]=2
  • nums[4]=3,以3結尾的最長子序列就是[2,3],因為從陣列下標0到4遍歷,只找到了子序列[2]3小,所以就是[2]+[3]啦,即dp[4]=2
  • nums[5]=7,以7結尾的最長子序列就是[2,5,7][2,3,7],因為從陣列下標0到5遍歷,找到2,5和3都比7小,所以就有[2,7],[5,7],[3,7],[2,5,7]和[2,3,7]這些子序列,最長子序列就是[2,5,7]和[2,3,7],它倆不就是以5結尾和3結尾的最長遞增子序列+[7]來的嘛!所以,dp[5]=3 =dp[3]+1=dp[4]+1

很顯然有這個規律:一個以nums[i]結尾的陣列nums

  • 如果存在j屬于區間[0,i-1],并且num[i]>num[j]的話,則有,dp(i) =max(dp(j))+1,

最簡單的邊界情況

當nums陣列只有一個元素時,最長遞增子序列的長度dp(1)=1,當nums陣列有兩個元素時,dp(2) =2或者1, 因此邊界就是dp(1)=1,

確定最優子結構

從窮舉分析,我們可以得出,以下的最優結構:

dp(i) =max(dp(j))+1,存在j屬于區間[0,i-1],并且num[i]>num[j],

max(dp(j)) 就是最優子結構,

狀態轉移方程

通過前面分析,我們就可以得出狀態轉移方程啦:

所以陣列num[i]的最長遞增子序列就是:

最長遞增子序列 =max(dp[i])

代碼實作

class Solution {
    public int lengthOfLIS(int[] nums) {
        if (nums.length == 0) {
            return 0;
        }
        int[] dp = new int[nums.length];
        //初始化就是邊界情況
        dp[0] = 1;
        int maxans = 1;
        //自底向上遍歷
        for (int i = 1; i < nums.length; i++) {
            dp[i] = 1;
            //從下標0到i遍歷
            for (int j = 0; j < i; j++) {
                //找到前面比nums[i]小的數nums[j],即有dp[i]= dp[j]+1
                if (nums[j] < nums[i]) {
                    //因為會有多個小于nums[i]的數,也就是會存在多種組合了嘛,我們就取最大放到dp[i]
                    dp[i] = Math.max(dp[i], dp[j] + 1);
                }
            }
            //求出dp[i]后,dp最大那個就是nums的最長遞增子序列啦
            maxans = Math.max(maxans, dp[i]);
        }
        return maxans;
    }
}

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

標籤:其他

上一篇:安裝pyinstaller,打包python檔案

下一篇:EbitenCookBook中文教程 第一課:安裝 Ebiten

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