使用樹狀數組(離線查詢),將範圍L到R中大於K的元素數量計算出來
在電腦科學領域,我們必須處理大型資料集,其中包括查詢選擇和更新操作。以較低的時間複雜度即時執行這些操作對於開發人員來說是一項具有挑戰性的任務。
使用 Fenwick 樹是解決這些基於範圍的查詢問題的有效方法。
Fenwick Tree 是一種資料結構,可以有效地更新元素並計算表中數字的前綴和。它也稱為二元索引樹。
在本文中,我們將討論如何使用 Fenwick 樹來找出 L 到 R 範圍內大於 K 的元素數量。
輸入輸出場景
假設您有一個包含 N 個元素的數組,並且您想要在數組中找到 L 到 R 範圍內大於 K 的元素數量。
Input: arr = {1, 15, 12, 13, 6, 18, 14, 10, 23, 41, 16, 9} N = 11, K = 17, L = 1, R = 8 Output: 2
什麼是離線查詢?
離線查詢是對預先決定的資料集執行的查詢操作。換句話說,這些操作僅對那些在處理查詢時沒有進一步修改的資料集執行。
使用芬威克樹
在這裡,我們將使用芬威克樹,它的每個節點都有向量,其中按順序包含數組的所有元素。然後,我們使用二分搜尋來回答每個查詢並計算元素的數量。
建立兩個函數,updateTree() 和total(),分別用於更新和檢索BIT中的前綴和。
接下來,建立另一個函數,該函數將計算 L 到 R 範圍內大於「K」的元素。函數接受輸入值「arr」、「N」、「L」、「R」和「K 」。
計數是用最大範圍N的累加和減去K的累加和來計算的。
為了排除超出範圍的元素,減去L-1的累加和(如果L大於1)。
範例
#include <iostream> #include <vector> using namespace std; // Updating the Fenwick Tree void updateTree(vector<int>& BIT, int index, int value) { while (index < BIT.size()) { BIT[index] += value; index += index & -index; } } int total(vector<int>& BIT, int index) { int result = 0; while (index > 0) { result += BIT[index]; index -= index & -index; } return result; } // Counting the number of elements int elementsGreaterThanK(vector<int>& arr, int N, int K, int L, int R) { vector<int> BIT(N+1, 0); for (int i = L; i <= R; i++) { updateTree(BIT, arr[i], 1); } int count = total(BIT, N) - total(BIT, K); if (L > 1) { count -= total(BIT, L-1); } return count; } int main() { vector<int> arr = {0, 5, 2, 3, 6, 8, 7, 1, 8, 4, 6, 9}; int N = 11; // Total range of values int K = 4; // Highest value int L = 1; // Start index of the range int R = 8; // End index of the range int count = elementsGreaterThanK(arr, N, K, L, R); cout << "Number of elements greater than " << K << " in the range [" << L << ", " << R << "]: " << count << endl; return 0; }
輸出
Number of elements greater than 4 in the range [1, 8]: 5
替代方法
在這裡,我們將建立一個單獨的向量來儲存查詢。每個查詢由兩個變數組成,index 和 val。 index 用於儲存陣列中的位置,而val用於儲存該索引處元素的值。這種方法使開發人員能夠執行各種查詢操作。此外,我們使用 BIT 計算每個查詢大於 K 的元素數量。
範例
#include <iostream> #include <vector> using namespace std; struct Query { int index; int val; }; void updateTree(vector<int>& BIT, int index, int value) { while (index < BIT.size()) { BIT[index] += value; index += index & -index; } } int total(vector<int>& BIT, int index) { int result = 0; while (index > 0) { result += BIT[index]; index -= index & -index; } return result; } vector<int> elementsGreaterThanK(vector<int>& arr, vector<Query>& queries, int K) { int N = arr.size(); vector<int> BIT(N+1, 0); vector<int> result; for (int i = 0; i < N; i++) { updateTree(BIT, i+1, (arr[i] > K) ? 1 : 0); } for (auto query : queries) { if (query.val > K) { result.push_back(total(BIT, query.index)); } else { result.push_back(0); } } return result; } int main() { vector<int> arr = {3, 14, 32, 7, 11, 5, 8, 6, 16, 2, 15}; vector<Query> queries = {{2, 4}, {5, 3}, {3, 6}, {0, 3}}; int K = 2; vector<int> counts = elementsGreaterThanK(arr, queries, K); for (int count : counts) { cout << "Number of elements greater than " << K << ": " << count << endl; } return 0; }
輸出
Number of elements greater than 2: 2 Number of elements greater than 2: 5 Number of elements greater than 2: 3 Number of elements greater than 2: 0
結論
我們可以簡單地從索引 L 到 R 迭代數組,並在每次數組元素大於 K 時加 1,以便找到每個查詢的答案。然而,為了降低時間複雜度,我們使用Fenwick樹來進行此類查詢操作。我們也可以使用線段樹和稀疏表來取代Fenwick樹來進行此類查詢操作。
以上是使用樹狀數組(離線查詢),將範圍L到R中大於K的元素數量計算出來的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

熱門話題

C語言數據結構:樹和圖的數據表示與操作樹是一個層次結構的數據結構由節點組成,每個節點包含一個數據元素和指向其子節點的指針二叉樹是一種特殊類型的樹,其中每個節點最多有兩個子節點數據表示structTreeNode{intdata;structTreeNode*left;structTreeNode*right;};操作創建樹遍歷樹(先序、中序、後序)搜索樹插入節點刪除節點圖是一個集合的數據結構,其中的元素是頂點,它們通過邊連接在一起邊可以是帶權或無權的數據表示鄰

文件操作難題的真相:文件打開失敗:權限不足、路徑錯誤、文件被佔用。數據寫入失敗:緩衝區已滿、文件不可寫、磁盤空間不足。其他常見問題:文件遍歷緩慢、文本文件編碼不正確、二進製文件讀取錯誤。

文章討論了在C中有效使用RVALUE參考,以進行移動語義,完美的轉發和資源管理,重點介紹最佳實踐和性能改進。(159個字符)

C 20範圍通過表現力,合成性和效率增強數據操作。它們簡化了複雜的轉換並集成到現有代碼庫中,以提高性能和可維護性。

C35 的計算本質上是組合數學,代表從 5 個元素中選擇 3 個的組合數,其計算公式為 C53 = 5! / (3! * 2!),可通過循環避免直接計算階乘以提高效率和避免溢出。另外,理解組合的本質和掌握高效的計算方法對於解決概率統計、密碼學、算法設計等領域的許多問題至關重要。

本文討論了C中的動態調度,其性能成本和優化策略。它突出了動態調度會影響性能並將其與靜態調度進行比較的場景,強調性能和之間的權衡

C語言函數是代碼模塊化和程序搭建的基礎。它們由聲明(函數頭)和定義(函數體)組成。 C語言默認使用值傳遞參數,但也可使用地址傳遞修改外部變量。函數可以有返回值或無返回值,返回值類型必須與聲明一致。函數命名應清晰易懂,使用駝峰或下劃線命名法。遵循單一職責原則,保持函數簡潔性,以提高可維護性和可讀性。

本文討論了使用C中的移動語義來通過避免不必要的複制來提高性能。它涵蓋了使用std :: Move的實施移動構造函數和任務運算符,並確定了關鍵方案和陷阱以有效
