首页 > 其他分享 >【luogu题解】P9749 [CSP-J 2023] 公路

【luogu题解】P9749 [CSP-J 2023] 公路

时间:2023-11-22 16:22:05浏览次数:49  
标签:题解 复杂度 P9749 len 查询 luogu 区间 ST dp

\(Meaning\)

\(Solution\)

这道题我来讲一个不一样的解法:\(dp\)

在写 \(dp\) 之前,我们需要明确以下几个东西:状态的表示,状态转移方程,边界条件和答案的表示。

状态的表示

\(dp[i]\) 表示到达第 \(i\) 个站点所需要的最少钱数, \(w[i]\) 表示在使用最少钱数到达第 \(i\) 个站点时多余的路程。

状态转移方程

\[ dp[i]=dp[i-1]+\bigg\lceil\frac{v[i-1]-w[i-1]}{d}\bigg\rceil\times pre\_min(i-1) \]

\[ w[i]=\bigg\lceil\frac{v[i-1]-w[i-1]}{d}\bigg\rceil-v[i-1]+w[i-1] \]

其中 \(pre\_min(i)\) 表示前 \(i\) 个站点中最小的油价。

边界条件

\[ dp[i]=0,w[i]=0 \]

答案的表示

\[ dp[n] \]

问题

在状态转移方程中,怎样在 \(O(1)\) 的时间复杂度下完成 \(pre\_min\) 函数呢?

这就涉及到了一个算法:

\(ST\) 表

在算法和数据结构中,ST表(Sparse Table)是一种用于解决区间查询问题的数据结构。它可以有效地回答各种形式的查询,例如最小值、最大值、区间和等。

简介

ST表的主要思想是通过预处理来加速区间查询。它使用倍增 DP 的思想将一个数组分割成多个子区间,并在每个子区间上计算出某种操作的结果。然后,根据这些预先计算好的结果,我们可以根据需要合并区间来回答各种查询。

具体的实现过程如下:

  1. 初始化ST表,ST表是一个二维数组。
  2. 将输入的原始数组填充到ST表的第一行。
  3. 使用递推关系填充ST表的其他行,直到得到完整的ST表。
  4. 根据查询的起始位置和区间长度,在ST表中找到对应区间的值,结合适当的操作得出最终结果。

查询操作

对于任何查询操作,我们可以使用以下步骤来回答:

  1. 计算出查询区间的长度len。

  2. 找到大于等于len的最大值j,使得2^j <= len。

  3. 使用预处理的结果和递推关系,在ST表中找到对应的值,并结合适当的操作得到查询结果。

这种方法的时间复杂度为O(1),因为我们只需进行几次常数级别的操作即可回答查询。

应用场景

ST表在解决各种区间查询问题时非常有用。以下是一些常见的应用场景:

  • 查询最小值/最大值:通过选择适当的查询操作,在O(1)的时间复杂度内回答每个查询。
  • 区间和查询:可以通过使用累积和来实现区间和查询。
  • 区间gcd查询:可以通过预处理和递推关系计算区间内的最大公约数。

总结

ST表是一种高效解决区间查询问题的数据结构。通过预先计算和递推关系,我们可以在O(1)的时间复杂度内回答各种形式的查询。它的实现相对简单且灵活,适用于多种应用场景。

模板

初始化(时间复杂度 \(O(\log_2n)\) )

for(int i=1;i<=n;i++) {
   	st[i][0]=a[i];
}
for(int j=1;(1<<j)<=n;j++) {
	for(int i=1;i+(1<<j)-1<=n;i++) {
		st[i][j]=min(st[i][j-1],st[i+(1<<(j-1))][j-1]);
	}
}

查询(时间复杂度 \(O(1)\) )

l=1,r=i-1,len=log2(r-l+1);
pm=min(st[l][len],st[r-(1<<len)+1][len]);

解决问题

有了ST表,我们就可以在O(1)的时间复杂度中查询最值了,那我们程序的最终问题:TLE也解决了。程序整体时间复杂度为O(n),可以通过此题。

AC代码如下。

\(Accept\ Code\)

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll N=1e5+5;
ll v[N],a[N],w[N],dp[N],st[N][20];
ll n,d,l,r,len,pm;
int main() {
    cin>>n>>d;
    for(int i=1;i<n;i++) {
        cin>>v[i];
    }
    for(int i=1;i<=n;i++) {
        cin>>a[i];
    }
    for(int i=1;i<=n;i++) {
    	st[i][0]=a[i];
	}
	for(int j=1;(1<<j)<=n;j++) {
		for(int i=1;i+(1<<j)-1<=n;i++) {
			st[i][j]=min(st[i][j-1],st[i+(1<<(j-1))][j-1]);
		}
	}
    for(int i=2;i<=n;i++) {
		l=1,r=i-1,len=log2(r-l+1);
		pm=min(st[l][len],st[r-(1<<len)+1][len]);
		dp[i]=dp[i-1]+ceil(1.0*(v[i-1]-w[i-1])/d)*pm;
		w[i]=ceil(1.0*(v[i-1]-w[i-1])/d)*d-(v[i-1]-w[i-1]);
	}
	cout<<dp[n];
    return 0;
}

标签:题解,复杂度,P9749,len,查询,luogu,区间,ST,dp
From: https://www.cnblogs.com/daiyulong/p/tijie-luogu-p9749.html

相关文章

  • 【模板】最小度限制生成树 题解
    其他的题解感觉都好高级,分享一种好想且好实现的方法。我们可以先把点\(s\)和与其相连的边都删除,我们发现剩下的部分变成了一些连通块。我们不难发现,当要求与\(s\)点相连的边的个数为\(k\)时,我们的连通块个数显然是\(k\)的。接下来这个问题就转化成了:\(n-1\)个点中生......
  • [IOI2015] Teams 题解
    妙妙题。不难发现,我们对于每个\(k\)取出的人都是满足\(a_i\leqk\leqb_i\)的。经典的,我们直接将\((a_i,b_i)\)转化到二维平面上,将它转化成一个二维数点问题。我们对于每一个询问,都使\(k\)有序,从小到大贪心的选择,也就相当于\(x\)轴限制不断向右,\(y\)轴限制不断......
  • zookeeper3.5.5以上8080端口占用问题解决
    zookeeper3.5.5启动默认会把AdminService服务启动,这个服务默认是8080端口,是一个通过jetty启动的管理控制台,一般不会用到,网上的复制粘贴就是来自同一个办法如下:方法一、删除jetty方法二、修改端口。修改方法的方法有两种:在启动脚本中增加-Dzookeeper.admin.serverPort=你的端......
  • P9868 题解
    blog。NOIP2023T1。可以对字符串随意交换,即可以重排每个单词。对于询问\(i\),最优方案显然是将\(\forallj\nei\)的\(w_j\)重排至字典序最大,将\(w_i\)重排至字典序最小。这件事情本质是将\(w_i\)与\(\min\limits_{j\nei}w_j\)比较。在开始时将全部串重排至字典序......
  • 【AGC】鸿蒙应用软件包上传问题解析
    ​【问题背景】近期收到了一些反馈,一些鸿蒙元服务开发者在发布应用市场的过程中,上传.app包时遇到了不同的报错,导致上传失败,下面来看一下这些报错的具体原因,如何正确打包上传。 【问题描述1】HarmonyOS元服务软件包上传后,提示“软件包解析失败,请重新上传”,错误详情(5)​​​【......
  • T401305 平面划分(easy) 题解
    LinkT401305平面划分(easy)Solution平面上\(n\)条直线所划分处的区域最大个数\(L_n\)是多少我们考虑假设已经有\(n-1\)条直线,我们需要画一条直线,这条直线最多和\(n-1\)条直线相交产生\(n\)个新的区域所以我们得到了\[\begin{align*} &L_0=1\\ &L_n=L_{n-1}......
  • jsmpeg视频播放器使用方法和常见问题解决方案
    JSMpeg是一个使用JavaScript编写的视频播放器,它可以在浏览器中播放MPEG1视频和MP2音频流。JSMpeg的特点是它能够通过WebSockets实时传输视频流,并且可以在不支持HTML5视频播放器的浏览器上运行。以下是JSMpeg的基本使用方法和一些常见问题的解决方案:主要用来解决移移动端视频播放问......
  • [ABC328D] Take ABC 题解
    链接如果只是扫一遍肯定是不行的,所以我们使用一个栈,遇到C就判断栈顶的两个元素是不是分别为B和A。这样就能做出来这道题了。代码#include<bits/stdc++.h>usingnamespacestd;strings;charstk[200010];intmain(){ cin>>s; intn=s.size(),p=0;//字符串长度和......
  • 【树链剖分】P3401 洛谷树 题解
    P3401考虑先将路径权值进行转化,因为很难对路径直接进行统计。考虑如何表示出这条路径的权值。记\(s_i=\oplus_{j\in\text{path}(1,i)}w_j\),其中\(\text{path}(i,j)\)表示\(i\)到\(j\)的路径上的边集。则\(u\tov\)的路径的权值为\(s_u\opluss_v\)。现在转变为......
  • P9620 歌姬 题解
    感觉题解做法都好神秘。来一个容易理解,通俗易懂的树剖解法。思路容易发现原问题等价于维护一个虚树。每一次询问虚树的根的所有儿子的最大值。要求链修。容易发现仅仅动态维护根是好做的。我们用一个\(\text{set}\)。每次维护\(\text{dfs}\)的最小值和最大值。对于这......