[bzoj4282]慎二的随机数列_动态规划_贪心

慎二的随机数列 bzoj-4282

题目大意:一个序列,序列上有一些数是给定的,而有一些位置上的数可以任意选择。问最长上升子序列。

注释:$1\le n\le 10^5$。



想法:结论:逢N必选。N是可以任意选择的位置。

具体的,我们将所有N踢出序列,将给定的权值-=前面N的个数。再在当前序列上求最长上升子序列。

正确性的话如果当前序列中的数:

如果前面的数小于后面的数,显然中间的N我也可以加上。

如果前面的数大于后面的数:

  如果前面的数在原序列中的权值大于后面的数在原序列中的权值,那么这两个数无论如何都不能同时选择。

  而如果前面的数在原序列中的数小于后面的数在原序列中的权值,那么我们选择抛弃后面的数转而选择中间的所有N,显然更优。

最后,附上丑陋的代码... ...

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N 100010
using namespace std;
int dp[N],sum,a[N],cnt;
int q[N];
int maxn=0;
int main()
{
    int n; cin >> n ;
    char opt[10];
    for(int i=1;i<=n;i++)
    {
        scanf("%s",opt+1);
        if(opt[1]==‘K‘)
        {
            int x; scanf("%d",&x);
            x-=sum;
            a[++cnt]=x;
        }
        else sum++;
    }
    int ans=0;
    for(int i=1;i<=cnt;i++)
    {
        int l=0,r=ans;
        while(l!=r)
        {
            int mid=(l+r+1)>>1;
            if(a[q[mid]]<a[i]) l=mid;
            else r=mid-1;
        }
        l++;
        ans=max(ans,l);
        q[l]=i;
    }
    printf("%d\n",ans+sum);
}

小结:这题...不禁让我想到了Claris的CDQ分治+扫描线+树状数组...

证明对于计算机竞赛的用处,就是可以简化一个复杂的算法(个人理解)。

原文地址:https://www.cnblogs.com/ShuraK/p/9537849.html

时间: 2024-10-30 18:39:50

[bzoj4282]慎二的随机数列_动态规划_贪心的相关文章

BZOJ4282 : 慎二的随机数列

首先在开头加上-inf,结尾加上inf,最后答案减2即可. 设s[i]为i之前未知的个数,f[i]为以i结尾的LIS,且a[i]已知,那么: f[i]=max(f[j]+min(s[i]-s[j],a[i]-a[j]-1))+1,其中j<i,a[j]<a[i]且a[j]已知 将min分类讨论后可转化为三维偏序,CDQ分治+扫描线+树状数组即可,时间复杂度$O(n\log^2n)$. #include<cstdio> #include<algorithm> using s

【bzoj4282】慎二的随机数列

扯几句题外的,最近在看Fate/StayNight,对此人毫无好感-- 每次减一下当前可辨认数,然后随意dp一个LIS,最后记得加回去就好. 1 #include<bits/stdc++.h> 2 #define N 100010 3 using namespace std; 4 int q[N],s[N],dp[N],sum,n,top,t,x; 5 int main(){ 6 scanf("%d",&n); 7 for(int i=1;i<=n;i++){

算法导论_动态规划_最长公共子序列

一.动态规划的概念 动态规划(Dynamic Programming)是通过组合子问题的解而解决整个问题的.分治是指将问题划分成一些独立的子问题,递归地求解各子问题,然后合并子问题的解而得到原始问题的解,与此不同,动态规划适用于子问题不是独立的情况,也就是各个子问题包含公共的子问题.在这种情况下,采用分治法会做许多不必要的工作,即重复地求解公共地子问题.动态规划算法对每个子问题只求解一次,将其结果保存在一张表中,从而避免每次遇到各个子问题时重新计算答案. 动态规划通常应用于最优化问题.此类问题可

算法导论_动态规划_最长回文子序列

一.问题的描述 回文序列(Palindromic sequence, Palindrome)是指正向遍历和反向遍历完全相同的序列,例如字符串"AAAAA"显然是一个回文序列,又如字符串"[email protected]"也是一个回文序列.现在,我们要在一个(字符)序列中找出最长回文子序列的长度.例如字符序列"BBABCBCAB",最长回文子序列是"BACBCAB"(可能不唯一),它的长度是7:子序列"BBBBB&q

[bzoj2748][HAOI2012]音量调节_动态规划_背包dp

音量调节 bzoj-2748 HAOI-2012 题目大意:有一个初值,给你n个$\delta$值,求最后不超过给定的限制的情况下的改变的最大值.每个$\delta$值可以+也可以-. 注释:$1\le n\le 50$,$1\le 限制\le 1000$. 想法:正负背包.在背包的之后更新用两行更新,保证每一个$\delta$值都被计入了答案. 最后,附上丑陋的代码... ... #include <iostream> #include <cstdio> #include <

[bzoj1084][SCOI2005]最大子矩阵_动态规划_伪&#183;轮廓线dp

最大子矩阵 bzoj-1084 SCOI-2005 题目大意:给定一个n*m的矩阵,请你选出k个互不重叠的子矩阵使得它们的权值和最大. 注释:$1\le n \le 100$,$1\le m\le 2$,$1\le k\le 10$. 想法:不会...看了数据范围..卧槽?m<=2?????我们就可以进行一个简单的轮廓线dp. 首先,先分m==1和m==2分类讨论,m==1不说了 m==2 令f[k][i][j]是第一列到了i,第二列到了j,已经选取了k个矩形的最大权值. 转移:有3种转移方式:

[bzoj1879][Sdoi2009]Bill的挑战_动态规划_状压dp

Bill的挑战 bzoj-1879 Sdoi-2009 题目大意: 注释:$1\le t \le 5$,$1\le m \le 15$,$1\le length \le 50$. 想法: 又是一个看数据范围想做法的题,我们想到状压dp. 看了题解... ...网上给的状态是f[len][s]表示长度为len满足状态s的字符串个数. 光看状态... ...可能算重啊?! 其实... ... 状态:dp[len][s]表示长度为len,能且只能满足状态为s的字符串个数. 转移:我们先预处理出g[i]

[bzoj3622]已经没有什么好害怕的了_动态规划_容斥原理

bzoj-3622 已经没有什么好害怕的了 题目大意: 数据范围:$1\le n \le 2000$ , $0\le k\le n$. 想法: 首先,不难求出药片比糖果小的组数. 紧接着,我开始的想法是 $f_{(i,j)}$表示前$i$个糖果中,满足糖果比药片大的组数是$j$的方案数. 进而发现需要将两个数组排序. 到这里一切都很正常,但是我们发现了一个问题:就是我在转移的时候,分两种情况讨论.第一种是当前糖果配对的药片比自己大,第二种是比自己小. 这样的话我需要乘上两个组合数. 但是我们仔细

HDU 4876 ZCC loves cards _(:зゝ∠)_ 随机输出保平安

GG,,,g艹 #include <cstdio> #include <iostream> #include <algorithm> #include <string.h> #include <vector> #include <queue> #include <math.h> using namespace std; vector<int>G[21][7];//G[i][j] 表示n=i k=j的情况下 二进