zoukankan      html  css  js  c++  java
  • 【LOJ#10180】烽火传递 单调队列+dp

    题目大意:给定一个 N 个非负整数数组成的序列,每个点有一个贡献值,现选出其中若干数,使得每连续的 K 个数中至少有一个数被选,要求选出的数贡献值最小。

    题解:设 (dp[i]) 表示考虑了序列前 i 个数的情况,且第 i 个数被选上时的最小贡献值,因此状态转移方程为:(dp[i]=min{dp[j],jin[i-k,i-1] }+val[i]​)
    这种状态转移方程满足:决策取值范围上下界均单调变化,每个决策在候选集合中插入和删除至多一次。因此可以用单调队列在决策候选集合上进行优化。

    代码如下

    #include <bits/stdc++.h>
    using namespace std;
    const int maxn=2e5+10;
    
    inline int read(){
    	int x=0,f=1;char ch;
    	do{ch=getchar();if(ch=='-')f=-1;}while(!isdigit(ch));
    	do{x=x*10+ch-'0';ch=getchar();}while(isdigit(ch));
    	return f*x;
    }
    
    int n,k,val[maxn],q[maxn<<1],l,r,dp[maxn];
    
    void read_and_parse(){
    	n=read(),k=read();
    	for(int i=1;i<=n;i++)val[i]=read();
    }
    
    void solve(){
    	l=1,r=0;q[++r]=0;
    	for(int i=1;i<=n;i++){
    		while(l<=r&&q[l]<i-k)l++;
    		while(l<=r&&dp[i-1]<dp[q[r]])r--;
    		q[++r]=i-1;
    		dp[i]=dp[q[l]]+val[i];
    	}
    	int ans=0x3f3f3f3f;
    	for(int i=n-k+1;i<=n;i++)if(dp[i]<ans)ans=dp[i];
    	printf("%d
    ",ans);
    }
    
    int main(){
    	read_and_parse();
    	solve();
    	return 0;
    } 
    
  • 相关阅读:
    IO模型详解
    Java中的CAS实现原理
    深入理解幂等性
    区块链基本原理入门
    通俗易懂讲解IO模型
    java高级特性(4)--枚举
    static、final和finalize详解
    锁(3)-- DB锁
    浅析项目中的并发
    分布式(1)-- 分布式锁
  • 原文地址:https://www.cnblogs.com/wzj-xhjbk/p/9957850.html
Copyright © 2011-2022 走看看