bzoj2817[ZJOI2012]波浪
题目链接:
http://www.lydsy.com/JudgeOnline/problem.php?id=2817
波浪
【问题描述】
阿米巴和小强是好朋友。 阿米巴和小强在大海旁边看海水的波涛。小强第一次面对如此汹涌的海潮, 他兴奋地叫个不停。而阿米巴则很淡定,他回想起曾经的那些日子,事业的起伏, 情感的挫折……总之今天的风浪和曾经经历的那些风雨比起来,简直什么都不 算。 于是,这对好朋友不可避免地产生了分歧。为了论证自己的观点,小强建立 了一个模型。他海面抽象成一个 1 到N的排列P[1…N]。定义波动强度等于 相邻 两项的差的绝对值的和 ,即: L = | P2 – P1 | + | P3 – P2 | + … + | PN – PN-1 | 给你一个N和M,问:随机一个 1…N的排列,它的波动强度 不小于 M的概率 有多大? 答案请保留 小数点后 K 位输出,四舍五入 。
【输入格式】
输入文件 wavel.in 的第一行包含三个整数 N, M 和 K,分别表示排列的长度, 波动强度,输出位数。
【输出格式】
输出文件 wavel.out 包含一个小数点后 K 位的实数。
【样例输入】
3 3 3
【样例输出】
0.667
【样例说明】
N = 3 的排列有 6 个:123,132,213,231,312,321;他们的波动强度分 别为 2,3,3,3,3,2。所以,波动强度不小于 3 的概率是 4/6,即 0.667。 你也可以通过下面的代码来验证这个概率: int a[3]={0,1,2},s=0,n=3; for (int i=0;i<1000000;i++){ random_shuffle(a,a+n); int t=0; for (int j=0;j<n-1;j++) t+=abs(a[j+1]-a[j]);
if (t>=3) s++; } printf("%.3f\n",s/1000000.0);
【数据规模】
对于 30%的数据,N ≤ 10。 对于另外 30%的数据,K ≤ 3。 对于另外 30%的数据,K ≤ 8。 对于另外 10%的数据,N ≤ 50。 对于 100%的数据,N ≤ 100,K ≤ 30,0 ≤ M ≤ 2147483647。
题解
啊。。
我从未写过如此**的题目啊T^T
这种题估计也就fhq出的出来。。
从此我对小强和阿米巴产生了心理阴影。。
网上好像没多少题解啊。。那我再发个良心题解吧。。
好的现在开始正文。。
首先考虑从小到大在序列中插入每个数字。
比如。。
涂黑的格子表示已经放了数字的位置,然后。。我们现在要再放一个数字。
显然易见。。涂黑的格子的数字都比当前数字小
然后有这几种情况
红色格子表示当前填的数字
先看第一张图,新添的格子产生了新的一段!
而且,这一段两端的数还没填。。说明,当前位置贡献的权值要-2*i(假设i为当前填的数字),因为两端的数都比这个大,又因为你有(k+1)段可以插入新的一段,所以转移系数为(k+1)。。
再看第二张图,i合并了原来的两段,说明i两端的数字都填好了,这代表着当前位置贡献的权值为2*i,因为两端的数字都比这个小,又因为你有(k-1)段可以合并两个段,多已转移系数为(k-1)。。
发现了没?我们可以DP了!
设f[i][j][k][t]表示当前插入到第i个数字,权值和为j,有k段,t。。t表示整个序列两端的情况。比如t=0表示这个序列两端都没有放数字,1表示放了一个,2表示放了两个。
然后,我们可以对第一张图列出方程:f[i][j-2*i][k+1][0]+=f[i-1][j][k][0]*(k+1)
同理,可以对第二段建立转移方程:f[i][j+2*i][k-1][0]+=f[i-1][j][k][0]*(k-1)
第三张图就是f[i][j][k][0]+=f[i-1][j][k][0]*2*k
然后第四张图,出现了边界的情况,也依然可以DP。
1、当前放在边界上的数字和别的段合成了新的一段,那么i的贡献就是i,由于有两个端点,所以转移系数是2,即方程为:f[i][j+i][k][1]+=f[i-1][j][k][0]*2
2、当前放在边界上的数字创造了新的一段,那么i的贡献就是-i,由于有两个端点,所以转移系数是2,即方程为:f[i][j-i][k+1][1]+=f[i-1][j][k][0]*2
-----------------------------------------------------------------------------以上是t=0的情况。。t=1的情况也类似,只不过。。由于你能放的位置会减少。。所以转移系数也会改变。。具体看代码。----------------------------------------------------
对了,当t=2时,就不用考虑边界情况了!
好的大体内容讲完了!
但是
但是
但是
精度30位会炸啊!!!!!
这时神器来了:__float128
优点:精度高
缺点:速度慢速度慢速度慢速度慢速度慢速度慢速度慢速度慢速度慢空间大空间大空间大空间大空间大空间大空间大空间大
又由于出题人****卡内存卡时限。。
你会发现这个代码T了。。
#include <bits/stdc++.h>
#define MAX 4500
#define N 110
using namespace std;
typedef long long ll;
double f[][MAX*+][N][];
int i,j,k,n,m,x,y,t,q,la,no;
void print2(__float128 x,int k){
for (i=;i<=n;i++)x=x/(__float128)i;
ll a=x;x-=a;int s[],l,j=;s[]=;
while (a)s[++s[]]=a%,a/=;if (!s[])s[++s[]]=;l=s[];
for (i=;i<=k+;i++){x*=;a=(int)x;x-=a;s[++s[]]=a;}
if (s[s[]]>=)j=;int p=s[];
while (j&&p){s[p-]++;j=s[p-]/;s[p-]%=;p--;}
if (j){for (i=s[];i>;i--)s[i]=s[i-];s[]=;}
putchar(''+s[]);if (k)putchar('.');for (i=;i<=k+;i++)putchar(s[i]+'');
return;
}
inline void solve(int n,int m,int q){
f[][MAX-*][][]=;f[][MAX-][][]=;f[][MAX][][]=;no=;la=;
for (i=;i<=n;i++){
no=-no,la=-la;
memset(f[no],,sizeof f[no]);
for (j=;j<=MAX*;j++)
for (k=;k<=n-;k++){
if (f[la][j][k][]){
if (j>=*i)f[no][j-*i][k+][]+=f[la][j][k][]*(k+);
if (j+*i<=MAX*)f[no][j+*i][k-][]+=f[la][j][k][]*(k-);
f[no][j][k][]+=f[la][j][k][]*k*;
if (j+i<=MAX*)f[no][j+i][k][]+=f[la][j][k][]*;
if (j>=i)f[no][j-i][k+][]+=f[la][j][k][]*;
}
if (f[la][j][k][]){
if (j>=*i)f[no][j-*i][k+][]+=f[la][j][k][]*k;
if (j+*i<=MAX*)f[no][j+*i][k-][]+=f[la][j][k][]*(k-);
f[no][j][k][]+=f[la][j][k][]*(*k-);
if (j+i<=MAX*)f[no][j+i][k][]+=f[la][j][k][];
if (j>=i)f[no][j-i][k+][]+=f[la][j][k][];
}
if (f[la][j][k][]){
if (j+i*<=MAX*)f[no][j+i*][k-][]+=f[la][j][k][]*(k-);
if (j>=*i)f[no][j-i*][k+][]+=f[la][j][k][]*(k-);
f[no][j][k][]+=*f[la][j][k][]*(k-);
}
}
}
__float128 ans=;
for (i=m+MAX;i<=MAX*;i++)ans+=(__float128)(f[no][i][][]*1.000000000000);
print2(ans,q);
}
int main(){
scanf("%d%d%d",&n,&m,&q);
solve(n,m,q);
return ;
}
TLE啊。。
555~
本机测试22s。。其中两个点9s。。
说明
bzoj太慢了!!!
bzoj太慢了!!!
bzoj太慢了!!!
bzoj太慢了!!!
bzoj太慢了!!!
bzoj太慢了!!!
bzoj太慢了!!!
bzoj太慢了!!!
这时,膜了一下别人的代码,发现好妙啊
原来可以将k<=8的用double做,k>8的用__float128做!
写个namespace。。。inline什么的。。就过了!!跑得飞快啊!
AC代码
#include <bits/stdc++.h>
#define N 110
#define MAX 4500
using namespace std;
typedef long long ll;
int i,j,k,n,m,x,y,t,q,la,no;
namespace dob {typedef double db;db f[][][][];}
namespace fl {typedef __float128 db;db f[][][][];}
template <class T> inline
void print2(T x,int k){
for (i=;i<=n;i++)x=x/(T)i;
ll a=x;x-=a;int s[],l,j=;s[]=;
while (a)s[++s[]]=a%,a/=;if (!s[])s[++s[]]=;l=s[];
for (i=;i<=k+;i++){x*=;a=(int)x;x-=a;s[++s[]]=a;}
if (s[s[]]>=)j=;int p=s[];
while (j&&p){s[p-]++;j=s[p-]/;s[p-]%=;p--;}
if (j){for (i=s[];i>;i--)s[i]=s[i-];s[]=;}
putchar(''+s[]);if (k)putchar('.');for (i=;i<=k+;i++)putchar(s[i]+'');
return;
}
template <class T> inline
void solve(T f[][][][]){
f[][MAX-*][][]=;f[][MAX-][][]=;f[][MAX][][]=;no=;la=;
for (int i=;i<=n;i++){
no=-no,la=-la;
memset(f[no],,sizeof f[no]);
for (int j=;j<=MAX*;j++)
for (int k=;k<=n-;k++){
if (f[la][j][k][]){
if (j>=*i)f[no][j-*i][k+][]+=f[la][j][k][]*(k+);
if (j+*i<=MAX*)f[no][j+*i][k-][]+=f[la][j][k][]*(k-);
f[no][j][k][]+=f[la][j][k][]*k*;
if (j+i<=MAX*)f[no][j+i][k][]+=f[la][j][k][]*;
if (j>=i)f[no][j-i][k+][]+=f[la][j][k][]*;
}
if (f[la][j][k][]){
if (j>=*i)f[no][j-*i][k+][]+=f[la][j][k][]*k;
if (j+*i<=MAX*)f[no][j+*i][k-][]+=f[la][j][k][]*(k-);
f[no][j][k][]+=f[la][j][k][]*(*k-);
if (j+i<=MAX*)f[no][j+i][k][]+=f[la][j][k][];
if (j>=i)f[no][j-i][k+][]+=f[la][j][k][];
}
if (f[la][j][k][]){
if (j+i*<=MAX*)f[no][j+i*][k-][]+=f[la][j][k][]*(k-);
if (j>=*i)f[no][j-i*][k+][]+=f[la][j][k][]*(k-);
f[no][j][k][]+=*f[la][j][k][]*(k-);
}
}
}
T ans=;
for (int i=m+MAX;i<=MAX*;i++)ans+=(T)f[no][i][][];
print2(ans,q);
}
int main(){
scanf("%d%d%d",&n,&m,&q);if (q<=)solve(dob::f);else solve(fl::f);
return ;
}
AC啦~
本机测试5s!!
飞起~
然后就是几个注意点。。
1:用滚动数组。。
2:要输出优化。。我曾分类讨论了K为1~30的所有情况。。。
3、做好卡评测的准备。。
4、为bzoj买一台好点的机子
完结?撒花!
bzoj2817[ZJOI2012]波浪的更多相关文章
- 洛谷2612&&bzoj2817 [ZJOI2012]波浪
洛谷2612&&bzoj2817 [ZJOI2012]波浪 原题链接 题解 因为有abs不太好搞,考虑拆掉abs. 生成排列的方法之一:n个空位,从1到n一次插入一个空位. 这样搞的话 ...
- 【BZOJ2817】[ZJOI2012]波浪(动态规划)
[BZOJ2817][ZJOI2012]波浪(动态规划) 题面 BZOJ 洛谷 题解 首先这个差值最大也就是\(n^2\)级别的. 那么这样子就可以压进状态啦. 我们把这个操作看成一个个加数的操作,按 ...
- [ZJOI2012]波浪
Description: L = | P2 – P1 | + | P3 – P2 | + - + | PN – PN-1 | 给你一个N和M,问:随机一个1-N的排列,它的波动强度(L)不小于M的概率 ...
- Luogu2612 ZJOI2012 波浪 DP
传送门 花掉了自己用来搞学科的时间做了这道题-- 一道类似的题:Here 考虑拆开绝对值计算贡献.那么我们对于\(1\)到\(N\)的排列,从小到大地将插入它们插入排列中. 假设我们现在计算到了数\( ...
- [ZJOI2012]波浪弱化版(带技巧的DP)
题面 \(solution:\) 这道确实挺难的,情况特别多,而且考场上都没想到如何设置状态.感觉怎么设状态不能很好的表示当前情况并转移,考后发现是对全排列的构造方式不熟而导致的,而这一题的状态也是根 ...
- Luogu P2612 [ZJOI2012]波浪
题目 我们考虑从\(1\)到\(n\)把每个数放到序列里面去,以消掉绝对值. 在最后的序列中,如果\(i\)的某一边是序列的边界,那么\(i\)会产生\(0\)的贡献.如果\(i\)的某一边是一个比\ ...
- 题解 洛谷 P2612 【[ZJOI2012]波浪】DP+高精
题目描述 题目传送门 分析 因为有绝对值不好处理,所以我们强制从小到大填数 设 \(f[i][j][p][o]\) 为当前填到了第 \(i\) 个数,波动强度为 \(j\),有 \(p\) 个连续段并 ...
- Noip前的大抱佛脚----赛前任务
赛前任务 tags:任务清单 前言 现在xzy太弱了,而且他最近越来越弱了,天天被爆踩,天天被爆踩 题单不会在作业部落发布,所以可(yi)能(ding)会不及时更新 省选前的练习莫名其妙地成为了Noi ...
- bzoj AC倒序
Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...
随机推荐
- 【甘道夫】CDH5.2的Maven依赖
之前一直结合Maven开发Hadoop2.2.0的程序.环境换成CDH5.2后报错,发现是Maven依赖库的问题. 之前一直使用 http://mvnrepository.com/ 查找maven依赖 ...
- Docker:一个装应用的容器
一:简介:你是否经历过“我本地运行没问题啊!““哪个哥们有写死循环了““完了,服务器撑不住了“等等问题,docker就是这么帮你解决问题的工具,它可以帮你把web应用自动化打包和发布,在服务型环境下进 ...
- 【LeetCode234】Palindrome Linked List★
题目描述: 解题思路: 判断一个单向链表是否是回文链表,并且要求O(n)的时间复杂度和O(1)的空间复杂度. 方法有以下几种: 1.遍历整个链表,将链表每个节点的值记录在数组中,再判断数组是不是一个回 ...
- mssql sqlserver 保留小数位指定位数的2种方法分享
摘要: 下文讲述将"sql数值型"类型数值转换为指定小数位的数据 方法1:采用 cast 方式转换数值类型至指定小数位: ,) set @a = 18.893 ,) 方法2:采用 ...
- Python3入门(六)——函数式编程
一.高阶函数 1.可以通过变量指向函数,达到类似别名的效果: >>> f = abs >>> f(-10) 10 2.函数的参数可以是函数,也就是函数可以作为一个入 ...
- WPF的Style的TargetType不同写法的异同
原文:WPF的Style的TargetType不同写法的异同 <Style TargetType="TextBlock"> <Setter Property=&q ...
- Redis学习之路(四)之Redis集群
[toc] #Redis集群 1.Redis Cluster简介 Redis Cluster为Redis官方提供的一种分布式集群解决方案.它支持在线节点增加和减少. 集群中的节点角色可能是主,也可能是 ...
- C++学习之从C到C++
头文件的包含 包含头文件可以不加.h结尾,如iostream,一些常用的头文件在引用时可以不加.h后缀,并在开头增加c,如: #include <cstdio> #include < ...
- Kafka API: TopicMetadata
Jusfr 原创,转载请注明来自博客园 TopicMetadataRequest/TopicMetadataResponse 前文简单说过"Kafka是自描述的",是指其broke ...
- Linux内核分析——构造一个简单的Linux系统MenuOS
马悦+原创作品转载请注明出处+<Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.Linux内核源代码简 ...