【模拟】CSU 1807 最长上升子序列~ (2016湖南省第十二届大学生计算机程序设计竞赛)
题目链接:
http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1807
题目大意:
给你一个长度为N(N<=105)的数列,数列中的0可以被其他数字替换,最终形成一个1~N的排列,求这个排列的最长上升子序列长度为N-1的方案数。
题目思路:
【模拟】
这道题需要分类讨论。
首先可以肯定,一个长度为n的序列最长上升子序列长度为n-1(最长下降子序列长度为2),那么这个序列的样子是1~n从小到大排列后其中一个数字挪到其余数字中间(错位)
一个长度为L的0区间,考虑第i个数字,如果放在第i位上,则f[i]=f[i-1]
如果放在第i-1位上,则第i位有i-1种放法(前面i-1个数都可以放在第i位,此时错位的为第i位上的数)
如果放在其他位置,则这个数就是错位的数,那这个数可以放的位置有i-2个(其他的数必须按从小到大排列才能保证错位数为1个)
现在考虑给定的数字,如果某个数字a[i]与i的偏移量超过1,那么意味着它前面必然有至少2个比它大的(a[i]-i>1),或它后面有至少2个比它小的(a[i]-i<-1),那么当前的a[i]必然是唯一的最终答案里偏移的数,或者无解。
如果给定的数字没有偏移,那么那个偏移的数字一定在只含0的区间中的一个,那么ans=Σ F[l] (F[l]即为长度为l的0区间偏移量为1的方案数)
如果区间[L,R]里给定的数字都往前偏移了1,那么代表有一个小于L的数在R的后面(如果区间[L,R]里给定的数字都往后偏移了1,代表一个大于R的数在L前面),并且根据前面判断得出这个数字还未被填(偏移量超过1),左端小于L的能填的数=L左端到第一个a[i]=i之间的0数(l[L]),能够填的位置就是R右端到第一个a[i]=i之间的0数(r[R]),所以ans=l[L]*r[R]。(另一种情况同理)
还有一种情况是某两个数字交换位置,这个我一开始考虑漏了。两位数字交换只有在这两个数字相邻的时候才可行且只有唯一解,并且要求其他数字必须要从小到大排列不能再有错位。
//
//by coolxxx
//#include<bits/stdc++.h>
#include<iostream>
#include<algorithm>
#include<string>
#include<iomanip>
#include<map>
#include<stack>
#include<queue>
#include<set>
#include<bitset>
#include<memory.h>
#include<time.h>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
//#include<stdbool.h>
#include<math.h>
#define min(a,b) ((a)<(b)?(a):(b))
#define max(a,b) ((a)>(b)?(a):(b))
#define abs(a) ((a)>0?(a):(-(a)))
#define lowbit(a) (a&(-a))
#define sqr(a) ((a)*(a))
#define swap(a,b) ((a)^=(b),(b)^=(a),(a)^=(b))
#define mem(a,b) memset(a,b,sizeof(a))
#define eps (1e-10)
#define J 10000
#define mod 1000000007
#define MAX 0x7f7f7f7f
#define PI 3.14159265358979323
#pragma comment(linker,"/STACK:1024000000,1024000000")
#define N 100004
using namespace std;
typedef long long LL;
double anss;
LL aans;
int cas,cass;
int n,m,lll,ans;
int a[N];
LL f[N];
void init()
{
int i;
f[]=f[]=;
for(i=;i<N;i++)f[i]=f[i-]+i-+i-;
}
void work1(int x)
{
int i,t=;
for(i=;i<=n;i++)
{
if(i==x)continue;
if(t==a[x])t++;
if(t==a[i] || !a[i])t++;
else break;
}
if(i<=n)puts("");
else puts("");
}
void work2(int l,int r)
{
int i,x,ll=,rr=;
for(i=l;i<=r;i++)
{
if(!a[i])continue;
if(a[i]-i!=a[l]-l)break;
}
if(i<=r){puts("");return;}
for(i=l-;i;i--)
{
if(!a[i])ll++;
else if(a[i]-i!=a[l]-l)break;
}
for(i=r+;i<=n;i++)
{
if(!a[i])rr++;
else if(a[i]-i!=a[l]-l)break;
}
if(a[l]-l==)ll++;
else rr++;
printf("%lld\n",1LL*ll*rr);
}
void work3(int x,int y,int l,int r)
{
int i;
if(x!=y || l!=r || x!=l-){puts("");return;}
puts("");
}
void work4()
{
int i,sz=;
for(i=;i<=n;i++)
{
if(!a[i])sz++;
else aans+=f[sz],sz=;
}
if(sz)aans+=f[sz];
printf("%lld\n",aans);
}
int main()
{
#ifndef ONLINE_JUDGE
// freopen("1.txt","r",stdin);
// freopen("2.txt","w",stdout);
#endif
int i,j,k;
int x,y,l,r;
init();
// for(scanf("%d",&cass);cass;cass--)
// for(scanf("%d",&cas),cass=1;cass<=cas;cass++)
// while(~scanf("%s",s))
while(~scanf("%d",&n))
{
aans=;
x=l=n+,y=r=;
for(i=;i<=n;i++)scanf("%d",a+i);
for(i=;i<=n;i++)
{
if(!a[i])continue;
if(abs(a[i]-i)>){work1(i);break;}
if(a[i]-i==)x=min(x,i),y=max(y,i);
if(a[i]-i==-)l=min(l,i),r=max(y,i);
}
if(i<=n)continue;
if(x<=n && y> && !r)work2(x,y);
else if(l<=n && r> && !y)work2(l,r);
else if(y && r)work3(x,y,l,r);
else work4();
}
return ;
}
/*
// //
*/
【模拟】CSU 1807 最长上升子序列~ (2016湖南省第十二届大学生计算机程序设计竞赛)的更多相关文章
- 【模拟】【数学】CSU 1803 2016 (2016湖南省第十二届大学生计算机程序设计竞赛)
题目链接: http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1803 题目大意: 给定n,m(n,m<=109)1<=i<=n,1& ...
- 【最短路】【数学】CSU 1806 Toll (2016湖南省第十二届大学生计算机程序设计竞赛)
题目链接: http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1806 题目大意: N个点M条有向边,给一个时间T(2≤n≤10,1≤m≤n(n-1), ...
- 【树状数组】CSU 1811 Tree Intersection (2016湖南省第十二届大学生计算机程序设计竞赛)
题目链接: http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1811 题目大意: 一棵树,N(2<=N<=105)个节点,每个节点有一种颜 ...
- 【数学】CSU 1810 Reverse (2016湖南省第十二届大学生计算机程序设计竞赛)
题目链接: http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1810 题目大意: 一个长度为N的十进制数,R(i,j)表示将第i位到第j位翻转过来后的 ...
- 【拓扑】【宽搜】CSU 1084 有向无环图 (2016湖南省第十二届大学生计算机程序设计竞赛)
题目链接: http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1804 题目大意: 一个有向无环图(DAG),有N个点M条有向边(N,M<=105 ...
- 【最短路】【STL】CSU 1808 地铁 (2016湖南省第十二届大学生计算机程序设计竞赛)
题目链接: http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1808 题目大意: N个点M条无向边(N,M<=105),每条边属于某一条地铁Ci ...
- 【贪心】CSU 1809 Parenthesis (2016湖南省第十二届大学生计算机程序设计竞赛)
题目链接: http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1809 题目大意: 给一个长度为N(N<=105)的合法括号序列.Q(Q<= ...
- 2016年湖南省第十二届大学生计算机程序设计竞赛---Parenthesis(线段树求区间最值)
原题链接 http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1809 Description Bobo has a balanced parenthes ...
- 2016年湖南省第十二届大学生计算机程序设计竞赛Problem A 2016 找规律归类
Problem A: 2016 Time Limit: 5 Sec Memory Limit: 128 MB Description 给出正整数 n 和 m,统计满足以下条件的正整数对 (a,b) ...
随机推荐
- (转)smarty实现多级分类的方法
--http://www.aspku.com/kaifa/php/44679.html 这篇文章主要介绍了smarty实现多级分类的方法,涉及循环读取的技巧,非常具有实用价值,需要的朋友可以参考下 ...
- ToString格式.
C 货币 2.5.ToString("C") ¥2.50 D 十进制数 25.ToString("D5") 00025 E 科学型 25000.ToString ...
- Linux运行C#程序
首先需要安装mono 安装教程http://www.cnblogs.com/aixunsoft/p/3422099.html 然后 用终端执行C#程序就可以了,mono 程序文件名 可以直接执行win ...
- Eclipse从数据库逆向生成Hibernate带注解的实体类
http://www.2cto.com/database/201501/372023.html
- JavaScript学习笔记之原型对象
本文是学习<JavaScript高级程序设计>第六章的笔记. JS中,便于批量创建对象的三种模式: 1.工厂模式:用一个函数封装创建对象的细节,传入必要的参数,在函数内部new一个对象并返 ...
- 狗狗40题~(Volume B)
H - Sorting Slides 应该是个二分匹配的模板题的,但我还不会写 = = 其实数据规模很小,就用贪心的方法就水过了(没加vis判冲突wa了几发,从此开始艰难的没有1A 的生活) #inc ...
- GET和POST详解
GET和POST 表单提交方式 http的get提交方法把表单数据编码到url中,可以在浏览器地址栏中看到, post提交把表单数据编码到http请求包的正文部分,在url中啊可能不到数据
- CSS实现DIV三角形
本文内容收集来自网络 #triangle-up { width:; height:; border-left: 50px solid transparent; border-right: 50px s ...
- js以json形式提交数据,后台接受
$("#savename").click(function(){ var fananname=$("#editname").val(); var jsonLis ...
- python读写Excel文件的函数--使用xlrd/xlwt
python中读取Excel的模块或者说工具有很多,如以下几种: Packages 文档下载 说明 openpyxl Download | Documentation | Bitbucket The ...