矩陣快速冪小結

来源:https://www.cnblogs.com/zwfymqz/archive/2018/09/13/9642901.html
-Advertisement-
Play Games

矩陣 並不想扯什麼高端線代的內容因為我也不會 定義 由$n \times m$個數$a_{ij}$排成的$n$行$m$列的數表稱為$n$行$m$列的矩陣,簡稱$n \times m$矩陣。 $$A =\begin{bmatrix}a_{11} & a_{12} & \dots a_{1m} \\ a ...


矩陣

並不想扯什麼高端線代的內容因為我也不會

定義

由$n \times m$個數$a_{ij}$排成的$n$行$m$列的數表稱為$n$行$m$列的矩陣,簡稱$n \times m$矩陣。

$$
A =
\begin{bmatrix}
a_{11} & a_{12} & \dots a_{1m} \\
a_{21}, & \dots & \dots \\
a_{31}, & \dots & \dots \\
a_{41} & \dots & a_{nm}
\end{bmatrix}
$$

運算

這裡只講加法減法和乘法,其他的例如矩陣求逆等與本文內容出入較大,有興趣的可以自己學習

加法

註意,只有行列均相同的矩陣才有加法!

運算也比較簡單,把對應位置的數相加得到一個新的矩陣,即為答案

例如

$$
\begin{bmatrix} 1 & 1 & 2 \\ 1 & 0 & 1 \end{bmatrix}
+
\begin{bmatrix} 2 & 3 & 3 \\ 3 & 3 & 2 \end{bmatrix}
=
\begin{bmatrix} 3 & 4 & 5 \\ 4 & 3 & 3 \end{bmatrix}
$$

加法滿足以下運算律

$A + B = B + A$

$(A + B) + C = A + (B + C)$

減法

與加法同理。

乘法

這才是重點!!

兩個矩陣能進行乘法的前提條件是:一個矩陣的行數等於另一個矩陣的列數

形式化的來說,若$A$是$i \times k$的矩陣,那麼$B$必須是$k \times j$的矩陣!

他們相乘得到的$C$是$i \times j$的矩陣

其中$C_{ij} = \sum_{i = 1}^n A_{ik} * B_{kj}$

比如

$$
\begin{bmatrix} 1 & 2\\ 2 & 3 \end{bmatrix}
\times
\begin{bmatrix} 2 & 4 & 5 \\ 3 & 4 & 3 \end{bmatrix}
=
\begin{bmatrix} 8 & 12 & 11 \\ 13 & 20 & 19 \end{bmatrix}
$$

乘法滿足結合律,左分配律,右分配律,即

$(A \times B) \times C = A \times (B \times C)$

$(A + B) \times C = A \times C + B \times C$

$C(A + B) = C \times A + C \times B$

千萬註意!矩陣乘法不滿足交換律!(很多情況下交換之後都不能相乘)

 

矩陣快速冪

因為矩陣有結合律,因此我們可以把整數的快速冪推廣的矩陣上面

題目鏈接

同樣是利用二進位倍增的思想,不難得到以下代碼

其中的base,代表的是單位矩陣,也就是除了對角線全為$1$,其他位置都為$0$的矩陣,可以證明任意矩陣乘單位矩陣都等於自身

顯然矩陣快速冪的複雜度為$O(n^3 log k)$

#include<cstdio>
#define LL long long 
using namespace std;
const int mod = 1e9 + 7;
int N;
LL K;
struct Matrix {
    int m[101][101];
    Matrix operator * (const Matrix &rhs) const {
        Matrix ans = {};
        for(int k = 1; k <= N; k++) 
            for(int i = 1; i <= N; i++) 
                for(int j = 1; j <= N; j++) 
                    (ans.m[i][j] += 1ll * m[i][k] * rhs.m[k][j] % mod) %= mod;
        return ans;
    }
};
Matrix fastpow(Matrix a, LL p) {
    Matrix base;
    for(int i = 1; i <= N; i++) base.m[i][i] = 1;//構造單位矩陣
    while(p) {
        if(p & 1) base = base * a;
        a = a * a; p >>= 1;
    }
    return base;
}
int main() {
    scanf("%d %lld", &N, &K);
    Matrix a; 
    for(int i = 1; i <= N; i++)    
        for(int j = 1; j <= N; j++)
            scanf("%d", &a.m[i][j]);
    a = fastpow(a, K);
    for(int i = 1; i <= N; i++, puts("")) 
        for(int j = 1; j <= N; j++)
            printf("%d ", a.m[i][j]);
    
    return 0;
}

 

應用

矩陣快速冪最常見的應用就是優化遞推啦

還是從最常見的斐波那契數列說起吧。

眾周所知,斐波那契數列的遞推公式為$$f_{n} = f_{n - 1} + f_{n - 2}, f_1 = 1, f_2 = 1$$

一般來說,這種看起來長得很簡單,只與自身的函數值有關(可能帶幾個常數)的式子,一般都可以用矩陣快速冪來加速。

當然,如果你想找刺激,可以學一下這玩意兒

矩陣快速冪具體是怎麼加速遞推的呢?

首先我們把斐波那契數列寫成矩陣的形式,因為$f_n$的取值與$f_{n - 1}, f_{n - 2}$這兩項有關,因此我們需要同時保留這兩項的值,我們不難得到一個$2 \times 1$的矩陣

$$
\begin{bmatrix}
f_{n} \\
f_{n - 1}
\end{bmatrix}
$$

現在我們要進行遞推,也就是得到這樣一個矩陣

$$
\begin{bmatrix}
f_{n + 1} \\
f_{n}
\end{bmatrix}
$$

展開

$$
\begin{bmatrix}
f_{n} + f_{n - 1} \\
f_{n}
\end{bmatrix}
$$

觀察一下,上面的一項需要用到$f_{n}$和$f_{n - 1}$,下麵的一項只需要用到$f_n$

同時結合上面的矩陣乘法的定義,我們不難得到一個轉移矩陣

$$
\begin{bmatrix} 1 & 1 \\ 1 & 0 \\ \end{bmatrix}
\begin{bmatrix} f_{n} \\ f_{n - 1}\\ \end{bmatrix}
=
\begin{bmatrix} f_{n} + f_{n - 1} \\ f_{n}\\ \end{bmatrix}
$$

這樣我們乘一次即可遞推到下一項。

但是這樣好像並沒有什麼卵用啊,複雜度上還多了個矩陣相乘

嘿嘿

不要忘了,我們前面可以講過矩陣有結合律的!

這樣的話我們只需要把轉移矩陣自乘$n - 1$次,然後再與初始矩陣相乘,就能得到答案矩陣啦!

題目鏈接

// luogu-judger-enable-o2
#include<cstdio>
#include<cstring>
#define LL long long 
using namespace std;
const int mod = 1e9 + 7;
LL K;
struct Matrix {
    int m[101][101], N;
    Matrix() {
        memset(m, 0, sizeof(m));
        N = 2;
    }
    Matrix operator * (const Matrix &rhs) const {
        Matrix ans;
        for(int k = 1; k <= N; k++) 
            for(int i = 1; i <= N; i++) 
                for(int j = 1; j <= N; j++) 
                    (ans.m[i][j] += 1ll * m[i][k] * rhs.m[k][j] % mod) %= mod;
        return ans;
    }
};
Matrix fastpow(Matrix a, LL p) {
    Matrix base; 
    for(int i = 1; i <= base.N; i++) base.m[i][i] = 1;//鏋勯€犲崟浣嶇煩闃?
    while(p) {
        if(p & 1) base = base * a;
        a = a * a; p >>= 1;
    }
    return base;
}
int main() {
    scanf("%lld", &K);
    Matrix a;
    a.m[1][1] = 1; a.m[1][2] = 1;
    a.m[2][1] = 1; a.m[2][2] = 0;
    a = fastpow(a, K - 1);
    printf("%d", a.m[1][1]);
    return 0;
}
代碼

 


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

-Advertisement-
Play Games
更多相關文章
  • c/c++ 標準順序容器 容器的訪問,刪除 操作 pop_front:vector,string不支持 pop_back:forward_list不支持 知識點 1,front, back, at 成員函數的使用,對應代碼里的test1 2,刪除最後一個元素pop_back, 刪除第一個元素pop_ ...
  • 明天老王要給我們講JVM的知識,提前發了一個小Demo給我們看,代碼如下: 運行上述代碼,結果毫無疑問,電腦瞬間開始狂躁起來,過了十幾秒,然後G了 基於JDK1.8運行的,估計老版本會崩的更快。。。 如果不計算記憶體,這個HashMap一共要插入4000*4000*4個對象,但是其實只有4個是不重覆的 ...
  • 據說Visual Studio Code(VS Code)的諸多好處,瞭解了一下果然很喜歡,我喜歡它的原因主要有3個,一是VS Code開源且跨平臺,二是因為其界面非常酷,三是可以我的大所屬代碼需求,除此之外當然還有強大的好奇心。使用VScode編寫第一個Python程式“one.py”,並將其打包... ...
  • 本文主要記錄如何使用aop切麵的方式來實現日誌記錄功能。 主要記錄的信息有: 操作人,方法名,參數,運行時間,操作類型(增刪改查),詳細描述,返回值。 ...
  • 最近工作上的事不忙了,開始忙著找房子了 我打算在天津再買一套房子,一個是投資一個是給我爸換個環境 但是二手的房子真的很少有非常完美的 現在天天守著以鏈家為主的APP看各種房子,晚上睡覺滿腦袋飛戶型圖 再有一個愁人的就是因為我在北京有房貸,所以在天津也算二套 二套要60%的首付,我手裡的錢不夠,還要湊 ...
  • c/c++ 標準順序容器 之 push_back,push_front,insert,emplace 操作 關鍵概念:向容器添加元素時,添加的是元素的拷貝,而不是對象本身。隨後對容器中元素的任何改變都不會影響到原始對象,反之亦然。 關鍵警告:因為vector,deque,string的記憶體存儲都是在 ...
  • 前面的菜單、部門、職位與管理員管理功能完成後,接下來要處理的是將它們關聯起來,根據職位管理中選定的許可權控制菜單顯示以及頁面數據的訪問和操作。 那麼要怎麼改造呢?我們可以通過用戶的操作步驟來一步步進行處理,具體思路如下: 1.用戶在管理端登錄時,通過用戶記錄所綁定的職位信息,來確定用戶所擁有的許可權。我 ...
  • /*介面List分為LinkedList和ArrayList;泛型<String>規定該類new出的對象或聲明的引用只能存放String類的對象 eg:List<String> map = new LinkedList<String>(); List<Integer> map = new Linke ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...