码迷,mamicode.com
首页 > 其他好文 > 详细

ZJOI2012 波浪))))))))))))))))))))))))))))))))))))))

时间:2018-01-28 11:26:03      阅读:158      评论:0      收藏:0      [点我收藏+]

标签:http   none   rand   opened   hid   ret   lan   can   abs   

题链

一道很良心的省选题。

我们先考虑如何优雅的骗分。

有30%的程序精度要求K<=3,这个时候我们只要把题目给你的那个程序拿下来改一改就好了。

RP不差的话有30分。

我们再观察一波数据,发现N<=10时我们可以暴力枚举所有可能,除以N!求概率。

来一发50分代码(为了稳健,没播种子)

技术分享图片
// luogu-judger-enable-o2
#include<bits/stdc++.h>
#define SIZ 5000000
#define abs(t) (t)>0?(t):-(t)
using namespace std;
int a[107],gg[11],p[11],usd[11],tot,ot;
int n,s,t,m,GG,k,anss,ans;
void dfs(int x){
    if (x==n+1) {if (anss>=m) ans++; return;}
    for (int i=1;i<=n;i++)
     if (!usd[i]) {
         usd[i]=1; p[x]=i; anss+=abs(i-p[x-1]); dfs(x+1); 
        anss-=abs(i-p[x-1]); usd[i]=0;
     }
}
void out(long double xx,int g){
    if (g==-1) {
        printf("%d",(int)xx); return;
    }
    printf("%d.",(int)xx); xx=xx*10;
    while (g--) {
       printf("%d",(int)xx); xx=xx*10-((int)xx)*10;
    }
    xx=xx*10; GG=(int)xx; if (GG%10<4) GG=GG/10; else GG=GG/10+1;
    printf("%d",GG); 
}
void Out(int x,int y,int g){
    if (g==-1) {
        if (x*2>=y) printf("1\n");
        else printf("0\n"); return;
    }
    printf("%d.",x/y); x=x%y*10;
    while (g--) {
        printf("%d",x/y); x=x%y*10;
    }
    GG=x/y; ot=x%y; if (ot*2>=y) GG++;
    printf("%d",GG);
}
int main () {
    scanf("%d%d%d",&n,&m,&k);
    if (n<=10){
        for (int i=1;i<=n;i++) {
            anss=0; p[1]=i; usd[i]=1; dfs(2); usd[i]=0;
        }
         tot=1;
        for (int i=1;i<=n;i++) tot=tot*i;
        Out(ans,tot,k-1); 
        return 0;
    }
    for (int i=1;i<n;i++) a[i]=i;
    s=0;
    for (int i=0; i<SIZ; i++) {
        random_shuffle(a,a+n); t=0;
        for (int j=0; j<n-1; j++) t+=abs(a[j+1]-a[j]);
        if (t>=m) s++;
    }
    long double XX=s/(1.0*SIZ);
    out(XX,k-1);
}
View Code

其实第二个暴力给我们了一种接近标算的思路。

我们只要快速求出比M大的有几种。

那么我们考虑DP:

绝对值不好搞,那么我们按大小从小到大依次插入。

那么插入一个数有很多种情况。

1.插入以后它两边都没有数。

2.插入以后它两边都有数。

3.插入以后它的一边有数。

4.插入在边界上,且它旁边没有数。

5.插入在边界上,且它旁边有数。

55种情况对应55个不同的转移,于是我们可以设一个状态来表示这些转移。

f[i][j][k][l表示插入i个数,当前序列波动值为j,序列现在被分为k个连续的段,序列边界的状态为ll为0表示边界没有数,为1表示边界有一个数,为2表示边界有两个数。

那么我们可以分几种情况转移,就好了。

__float128可以替代高精度。

#include<bits/stdc++.h>
#define M 4500
using namespace std;
int n,m,k;
namespace db{ double f[2][9001][101][3]; }
namespace f128{ __float128 f[2][9001][101][3]; }
template<class T>
inline void pout(T ans,int k){
    printf("%d.",(int)ans);
    while (k--) {
        ans=(ans-(int)ans)*10;
        if(!k) ans+=0.5;
        printf("%d",(int)ans);
    }
}
template<class T>
void solve(T f[][9001][101][3]){ T A;
    f[1][M-2][1][0]=1;f[1][M-1][1][1]=2;f[0][M][1][2]=1;
    for (int i=2,cur=0,pre=1;i<=n;i++,cur=pre,pre^=1) {
        memset(f[cur],0,sizeof f[cur]);
        for (int j=2*M;~j;j--) 
         for (int k=n-1;k;k--)
          for (int l=2;~l;l--) {
              if (!(A=f[pre][j][k][l])) continue;
              if (j+2*i<=2*M) f[cur][j+2*i][k-1][l]+=A*(k-1);
              if (j>=2*i) f[cur][j-2*i][k+1][l]+=A*(k+1-l);
              f[cur][j][k][l]+=A*(k*2-l);
              if (l<2) {
                  if (j+i<=2*M) f[cur][j+i][k][l+1]+=(2-l)*A;
                  if (j>=i) f[cur][j-i][k+1][l+1]+=A*(2-l);
              }
          }
    }
    T ans=0;for (int i=M+m;i<=2*M;i++) ans+=f[n&1][i][1][2];
    for (int i=2;i<=n;i++) ans/=i; 
    pout(ans,k);
}
int main () {
    scanf("%d %d %d",&n,&m,&k);
    if (k<=8) solve(db::f);
    else solve(f128::f);
}

 

ZJOI2012 波浪))))))))))))))))))))))))))))))))))))))

标签:http   none   rand   opened   hid   ret   lan   can   abs   

原文地址:https://www.cnblogs.com/rrsb/p/8370521.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!