题意

给出一个长度为n的,所有元素大小在[1,n]的整数数列,要求选出一个尽量长的区间使得区间内所有元素组成一个1到区间长度k的排列,输出k的最大值

n<=1e5

分析

不会做,好菜啊.jpg

学习了西方那一套理论,里面别人的题解写得挺吼的但是没多少人点赞23333.

外来的和尚好念经

一个合法排列必然包括一个1.那么我们可以枚举这个1的位置,既然包含了这个1就不能包含其他的1,所以我们可以找出这个1左边的第一个1的位置L(如果没有就是0)和右边的第一个1的位置R(如果没有就是n+1).包括这个1的排列左端不能超过L+1,右端不超过R-1.

于是我们把问题转化成若干个子问题,每个子问题处理的区间中只有一个1.所有子问题的区间总长之和是O(n)的,因为原序列中的每个位置只会对最多两个子问题贡献1的区间长度.

现在考虑怎样解决一个子问题.

一个长度为L的序列如果合法,那么序列中的最大值等于L且序列中没有重复元素.

在一个子问题中,区间内只有一个1,那么选出的序列必然包含这个1,而选出的序列中的最大值可以在这个1的左侧,也可以在这个1的右侧.由对称性,只考虑最大值在左侧的情况.在右侧的情况反着跑一遍就行.

如果最大值在左侧,我们可以枚举选出的序列的左端点l,那么从1的位置一直到l的最大值就是1左侧的最大值,因为我们假设最大值在1的左侧,所以现在我们也认为这个最大值是整个选出的序列的最大值.那么选出的序列的长度也就确定了,序列的右端点r也就确定了,我们只需判断[l,r]是不是一个合法的区间.也就是判一下[l,r]的最大值是不是在1的位置的左侧,判一下[l,r]中是否有重复元素.如果是合法的,就用r-l+1更新答案.

求最大值只需要预处理Max[i]表示从1所在的位置一直到i的最大值.

判断[l,r]是否有重复元素只需预处理lsame[i]表示i左侧第一个数值和a[i]相等的位置,如果l<=i<=r的所有i中存在一个i使得lsame[i]>=l,那么有重复元素,否则没有.

#include <cstdio>
#include <algorithm>
using namespace std;
const int maxn=300005;
int a[maxn];
int lsame[maxn],rsame[maxn];
int pos[maxn];
int ans=0;
int Max[maxn],MaxLsame[maxn];
void work(int s,int L,int R){
if(ans==0)ans=1;
Max[s]=a[s];
for(int i=s-1;i>=L;--i)Max[i]=max(Max[i+1],a[i]);
for(int i=s+1;i<=R;++i)Max[i]=max(Max[i-1],a[i]);
MaxLsame[s]=lsame[s];
for(int i=s-1;i>=L;--i)MaxLsame[i]=max(MaxLsame[i+1],lsame[i]);
for(int i=s+1;i<=R;++i)MaxLsame[i]=max(MaxLsame[i-1],lsame[i]);
for(int i=s-1;i>=L;--i){//maximum number left of s
if(MaxLsame[i]>=i)continue;
int length=Max[i];
int r=i+length-1;
if(r>R)continue;
if(MaxLsame[r]>=i||Max[r]>=Max[i])continue;
if(length>ans)ans=length;
}
for(int i=s+1;i<=R;++i){//maximum number right of s
if(MaxLsame[i]>=s)continue;
int length=Max[i];
int l=i-length+1;
if(l<L)continue;
if(MaxLsame[l]>=l||MaxLsame[i]>=l||Max[l]>=Max[i])continue;
if(length>ans)ans=length;
}
}
int main(){
int n;scanf("%d",&n);
for(int i=1;i<=n;++i)scanf("%d",a+i);
for(int i=1;i<=n;++i){
lsame[i]=pos[a[i]];pos[a[i]]=i;
}
for(int i=1;i<=n;++i)pos[i]=n+1;
for(int i=n;i>=1;--i){
rsame[i]=pos[a[i]];pos[a[i]]=i;
}
for(int i=1;i<=n;++i){
if(a[i]==1){
work(i,lsame[i]+1,rsame[i]-1);
}
}
printf("%d\n",ans);
return 0;
}

bzoj1318[spoj 744] Longest Permutation的更多相关文章

  1. 【bzoj1318】[Spoj744] Longest Permutation(乱搞)

    题目传送门:https://www.lydsy.com/JudgeOnline/problem.php?id=1318 这道题的大意是要求一个长度为len,并包含1~len所有数,并使len最大的子区 ...

  2. bzoj 1318: [Spoj744] Longest Permutation 智商题

    1318: [Spoj744] Longest Permutation Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 361  Solved: 215 ...

  3. 【SPOJ】Longest Common Substring II (后缀自动机)

    [SPOJ]Longest Common Substring II (后缀自动机) 题面 Vjudge 题意:求若干个串的最长公共子串 题解 对于某一个串构建\(SAM\) 每个串依次进行匹配 同时记 ...

  4. 【SPOJ】Longest Common Substring(后缀自动机)

    [SPOJ]Longest Common Substring(后缀自动机) 题面 Vjudge 题意:求两个串的最长公共子串 题解 \(SA\)的做法很简单 不再赘述 对于一个串构建\(SAM\) 另 ...

  5. 【SPOJ】Longest Common Substring II

    [SPOJ]Longest Common Substring II 多个字符串求最长公共子串 还是将一个子串建SAM,其他字符串全部跑一边,记录每个点的最大贡献 由于是所有串,要对每个点每个字符串跑完 ...

  6. 【SPOJ】Longest Common Substring

    [SPOJ]Longest Common Substring 求两个字符串的最长公共子串 对一个串建好后缀自动机然后暴力跑一下 废话 讲一下怎么跑吧 从第一个字符开始遍历,遍历不到了再沿着\(pare ...

  7. SPOJ Another Longest Increasing Subsequence Problem 三维最长链

    SPOJ Another Longest Increasing Subsequence Problem 传送门:https://www.spoj.com/problems/LIS2/en/ 题意: 给 ...

  8. SPOJ 1812 Longest Common Substring II(后缀自动机)(LCS2)

    A string is finite sequence of characters over a non-empty finite set Σ. In this problem, Σ is the s ...

  9. SPOJ LCS2 - Longest Common Substring II

    LCS2 - Longest Common Substring II A string is finite sequence of characters over a non-empty finite ...

随机推荐

  1. 【MongoDB】NoSQL Manager for MongoDB 教程(进阶篇)

    项目做完,有点时间,接着写下第二篇吧.回顾戳这里  基础篇:安装.连接mongodb.使用shell.增删改查.表复制 本文属于进阶篇,为什么叫进阶篇,仅仅是因为这些功能属于DB范畴,一般使用的不多, ...

  2. 【转载】ID3DXSPRITE接口简单使用

    原文:ID3DXSPRITE接口简单使用 前些日子一直研究DDraw,毕竟是DirectX7的东西了,所以转手用DirectD3D9,用了Surface进行绘图,可是怎么做透明色也都是不行loadfr ...

  3. Jmeter接口测试(四)传递参数

    参数设置 Jmeter 支持通过 查询字符串参数(Query String Parameters) 或者 Request body 请求体来传递参数. 1.get请求是普通键值对 get请求一般通过p ...

  4. Linux内核学习笔记(6)-- 进程优先级详解(prio、static_prio、normal_prio、rt_priority)

    Linux 中采用了两种不同的优先级范围,一种是 nice 值,一种是实时优先级.在上一篇粗略的说了一下 nice 值和实时优先级,仍有不少疑问,本文来详细说明一下进程优先级.linux 内核版本为 ...

  5. JS对字符串编码的几种方式

    函数 描述 encodeURI() 把字符串编码为 URI encodeURIComponent() 把字符串编码为 URI 组件 escape() 对字符串进行编码 上面是查询来自w3school的 ...

  6. Python数据挖掘——数据概述

    Python数据挖掘——数据概述 数据集由数据对象组成: 数据的基本统计描述 中心趋势度量 均值 中位数 众数 中列数 数据集的最大值和最小值的平均 度量数据分布 极差 最大值与最小值的差 四分位数 ...

  7. Pyhone学习之环境搭建

    一.python 环境搭建 本章节我们将向大家介绍如何在本地搭建Python开发环境.Python可应用于多平台包括 Linux 和 Mac OS X.你可以通过终端窗口输入 "python ...

  8. Scrum立会报告+燃尽图(十月十七日总第八次)

    本次作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2246 一.小组介绍 组名:杨老师粉丝群 组长:乔静玉 组员:吴奕瑶.公 ...

  9. 20135313_exp5

    课程:Java程序与设计     班级:1353 姓 名:吴子怡  学号:20135313 小组成员: 20135113肖昱 成绩:             指导教师:娄嘉鹏       实验日期:2 ...

  10. 【java】中缀表达式转后缀表达式 java实现

    算法: 中缀表达式转后缀表达式的方法:1.遇到操作数:直接输出(添加到后缀表达式中)2.栈为空时,遇到运算符,直接入栈3.遇到左括号:将其入栈4.遇到右括号:执行出栈操作,并将出栈的元素输出,直到弹出 ...