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 ...
随机推荐
- Feeling_2018_5_22
“我打你,你会走吗?” “不会!!” “我骂你,你会走吗?” “不会!!” “那我不爱你了,你会走吗?” “会.”
- Python面向对象之异常捕获(一)-----抛出一个异常
大部分的异常都继承自Exception这个类(而这个类有继承自BaseException这个类) 常见的异常 ValueError TypeError IndexError 抛出一个异常 下面这个类的 ...
- 大数据入门第十七天——storm上游数据源 之kafka详解(一)入门与集群安装
一.概述 1.kafka是什么 根据标题可以有个概念:kafka是storm的上游数据源之一,也是一对经典的组合,就像郭德纲和于谦 根据官网:http://kafka.apache.org/intro ...
- 20155202《网络对抗》Exp9 web安全基础实践
20155202<网络对抗>Exp9 web安全基础实践 实验前回答问题 (1)SQL注入攻击原理,如何防御 SQL注入产生的原因,和栈溢出.XSS等很多其他的攻击方法类似,就是未经检查或 ...
- Compensating-Transaction模式
在应用中,会将一系列相关的操作定义为一个连续的操作,当其中一个或者多个步骤失败的时候,Compensating-Transaction模式会重置(回滚)这个连续的操作.在云应用中,这些需要保证一致性的 ...
- [LOJ#6066]. 「2017 山东一轮集训 Day3」第二题[二分+括号序列+hash]
题意 题目链接 分析 首先二分,假设二分的答案为 \(mid\),然后考虑利用括号序列来表示树的形态. 点 \(u\) 的 \(k-\) 子树的括号序列表示实际上是刨去了 \(u\) 子树内若干个与 ...
- zooland 新开源的RPC项目,希望大家在开发的微服务的时候多一种选择,让微服务开发简单,并且容易上手。
zooland 我叫它动物园地,一个构思很长时间的一个项目.起初只是觉得各种通信框架都封装的很好了,但是就是差些兼容,防错,高可用.同时在使用上,不希望有多余的代码,像普通接口一样使用就可以了. 基于 ...
- Js_网站右下角悬浮视窗可关闭广告
站右下角悬浮视窗可关闭广告代码,可收缩.展开,关闭,内容区可自定义html,兼容IE8+.FireFox.Chrome.Safari等主流浏览器.广告图片尺寸300x250. 使用方法:在head区域 ...
- kubernetes 集群新增node 节点并将应用分配到新增节点
第一章 1.重新安装一台kubernetes node节点,新增节点:192.168.1.192 网址:https://www.cnblogs.com/zoulixiang/p/9504324.htm ...
- 分分钟让你理解HTTPS
一.HTTP存在的问题 1.1 可能被窃听 HTTP 本身不具备加密的功能,HTTP 报文使用明文方式发送 由于互联网是由联通世界各个地方的网络设施组成,所有发送和接收经过某些设备的数据都可能被截获或 ...