IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> ST表初识(C++) -> 正文阅读

[C++知识库]ST表初识(C++)

ST表:Sparse Table,稀疏表,一种数据结构主要用来解决静态的区间最大/最小值问题。

主要思想:倍增思想。

在看ST表之前,先看一个问题:

24153

在这个序列中找出区间【1,3】、【3,5】、【1,5】

max【1,3】= 4?

max【3,5】= 5

max【1,5】= 5

大家可以发现,在区间【1,3】中最大值是4,区间【3,5】中最大值是5,在整个区间【1,5】中最大值是5,max(a,b) = max(max(a,c), max(c,b)); (a<=c<=b)

这个性质我们称之为:可重复性贡献

如果我们单单通过这个倍增思想的话,每次询问的时间复杂度仍认为O(logn),显然这并没有得到优化。

但是基于以上性质再结合我们倍增的思想(每次前进2^i),使用两个预处理过的区间来覆盖询问区间,时间复杂度就被降至O(1)。

首先:

令f(i,j)表示从a[i] 开始连续2^j个数的最大值,区间表示为[ i,2 ^ j - 1 ];

显然,f[ i , 0 ] = a[ i ];

第二维则表示:倍增时跳的步数(2 ^ j - 1),跨越的长度。

通过以上的分析,我们得到一个状态转化方程:

max[ l , r ]?= max( f [ l ] ,[r - 1 ]?, f [ l + ( 1 << ( r - 1 ) ] [ r - 1 ] );

?以上为预处理部分,获得各区间的最大值。

接下来我们考虑询问部分,根据转化方程,我们将询问区间[ l , r ] 分成两部分。

f[l, l+2^s-1] 和 f[r-(2^s)+1,r];?

s:前进的步数??

我们希望前一个子区间的右端点尽可能接近r。当l + 2 ^ s -1 =? r??时,有 s = log2( r - l + 1);

但因为s是整数,所以我们向下取整? ,根据上面“可重复贡献”的性质,重叠并不会对区间最大值产生影响。同时两个区间的并完全覆盖[ l , r ]?。

由于输入输出数据一般很多,防止因为I/O被卡,这里提供一个快速读入的函数:

inline int read() { 
  char c = getchar();
  int x = 0, f = 1;
  while (c < '0' || c > '9') {
    if (c == '-') f = -1;
    c = getchar();
  }
  while (c >= '0' && c <= '9') {
    x = x * 10 + c - '0';
    c = getchar();
  }
  return x * f;
}

?由于s 也就是log函数每次都需要计算,这里我们直接通过递推将log函数预处理出来

//由于log[1] = 0;若想通过下面的循环直接处理,需要将log[N] 初始化为 -1;
log[N] = {-1};

for(int i =1;i<=n;i++) 
	{
		log[i] = log[i/2] + 1;
	}

洛谷模板题:?https://www.luogu.com.cn/problem/P3865

#include <bits/stdc++.h>
using namespace std;
const int MAX = 2000010;
int f[MAX][50], logn[MAX]= {-1};

inline int read() { 
  char c = getchar();
  int x = 0, f = 1;
  while (c < '0' || c > '9') {
    if (c == '-') f = -1;
    c = getchar();
  }
  while (c >= '0' && c <= '9') {
    x = x * 10 + c - '0';
    c = getchar();
  }
  return x * f;
}

int main() {
  int n = read(), m = read();
  for (int i = 1; i <= n; i++)
  {
      f[i][0] = read();
      logn[i] = logn[i/2] + 1; //预处理log函数
  } 
  
  //预处理最大值
  for (int j = 1; j <= logn[n]; j++)
    for (int i = 1; i + (1 << j) - 1 <= n; i++)
      f[i][j] = max(f[i][j - 1], f[i + (1 << (j - 1))][j - 1]); 


  while(m--) 
 {
    int x = read(), y = read();
    int s = logn[y - x + 1];
    printf("%d\n", max(f[x][s], f[y - (1 << s) + 1][s]));
  }
  return 0;
}

?代码中f [ MAX ][ 50 ],这里的50,也就是第二维,指的就是“跳跃”的步数,大小根据数据范围而定,不小于log[MAX];

最后我们来看一下ST表的时间复杂度:

预处理:O(nlogn);

询问:O(1);

时间复杂度:O(nlogn + m);

?ST表优点:时间复杂度较低 、代码量较少;

ST表缺点:维护的信息有限,只支持静态操作(不支持修改操作);

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2021-08-16 11:33:49  更:2021-08-16 11:35:48 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年12日历 -2024/12/26 16:33:54-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码
数据统计