zoukankan      html  css  js  c++  java
  • JZOJ 6439. 【GDOI2020模拟01.17】小 ω 数排列(DP)

    JZOJ 6439. 【GDOI2020模拟01.17】小 ω 数排列

    题目

    Description

    在这里插入图片描述

    Input

    在这里插入图片描述

    Output

    在这里插入图片描述

    Sample Input

    Sample Input1
    4 10
    3 6 2 9

    Sample Input2
    8 35
    3 7 1 5 10 2 11 6

    Sample Output

    Sample Output1
    6
    【样例 1 解释】
    共有 6 个排列符合条件,它们是 (1, 3, 2, 4),(2, 4, 1, 3),(3, 1, 2, 4),(3, 1, 4, 2),(4, 2, 1, 3),(4, 2, 3, 1)。

    Sample Output2
    31384

    Data Constraint

    在这里插入图片描述

    题解

    • 第一眼就是DP,先考虑暴力做法,
    • f [ i ] [ j ] [ k ] f[i][j][k] f[i][j][k]表示选出的数状压后为 i i i,当前最后一个数为 j j j,绝对值之和为 k k k的方案数,转移显然。
    • 正解也是DP,方法比较巧妙。
    • 因为式子中的绝对值不是很好处理,所以想办法去掉它,最直接的办法就是先排序后再按从小到大的顺序依次加入
    • 但这样并不知道要将当前数加入到什么位置,那怎么样才能保证能不重不漏地得出各种符合条件的排列呢?
    • 可以这样想,加入时不一定和前面已经加入的连在一起,而是后面才不断加入使前面的分散的段合并起来。
    • f [ i ] [ j ] [ k ] f[i][j][k] f[i][j][k]表示加入了 i i i个数,目前有 j j j个分散的段,答案为 k k k的方案数,
    • 至于答案 k k k如何更新,其实与 a [ i ] a[i] a[i]加入在哪个数旁边并没有关系,区别在于它是否自成一段,是否在边界,是否连接两段
    • 因为边界情况特殊,又要保证只有两个边界,所以再设多一维 l l l表示当前已经确定的边界的个数,
    • 转移情况有五种:
    • 1、加入 a [ i ] a[i] a[i]自成一段且不在边界;
    • 2、加入 a [ i ] a[i] a[i]在某一段的一端且不在边界;
    • 3、加入 a [ i ] a[i] a[i]连接在不相连的两段中间;
    • 4、加入 a [ i ] a[i] a[i]自成一段放在边界;
    • 5、加入 a [ i ] a[i] a[i]在某一段的一端且成为边界。
    • 考虑每种转移对 k k k的贡献(注意符号有正有负):
    • 1、 a [ i ] a[i] a[i]会被后来加在它左右两边的给减去,所以贡献是 − 2 ∗ a [ i ] -2*a[i] 2a[i]
    • 2、 a [ i ] a[i] a[i]需要加上一个用来减去它旁边的,也会被后来加在旁边的给减去,所以贡献是 0 0 0
    • 3、 a [ i ] a[i] a[i]需要加上两个用来减去它左右两边的,所以贡献是 2 ∗ a [ i ] 2*a[i] 2a[i]
    • 4、 a [ i ] a[i] a[i]会被后来加在它一边的给减去,所以贡献是 − a [ i ] -a[i] a[i]
    • 5、 a [ i ] a[i] a[i]需要加上一个用来减去它旁边的,且另一边没有数了,所以贡献是 a [ i ] a[i] a[i]
    • 其他对于 i , j , l i,j,l i,j,l的变化显然。
    • 但这样一来 k k k值的变化范围特别大,还是不能过,所以想办法控制 k k k的范围。
    • 首先把每个 a [ i ] a[i] a[i]减去 a [ i ] m a x a[i]_{max} a[i]max,但这还不够,
    • 发现每一次填入 a [ i ] a[i] a[i]时,对所有 k k k的贡献都可以看作是 ( a [ i ] − a [ i − 1 ] ) ∗ ( 2 ∗ j − l ) (a[i]-a[i-1])*(2*j-l) (a[i]a[i1])(2jl)
    • 那么这样就全是正的了,如果大于 L L L即可退出,
    • 于是 k k k的范围被控制在了 [ 0 , L ] [0,L] [0,L],可以愉快地通过此题。

    代码

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    #define N 1001
    #define md 1000000007
    #define ll long long
    int a[N],t[N];
    int f[2][110][N][3];
    int main()
    {
       int n,L,i,j,k,l;
       scanf("%d%d",&n,&L);
       if(n==1) 
       {
       	printf("1");
       	return 0;
       }
       for(i=1;i<=n;i++) scanf("%d",&a[i]);
       if(n==2)
       {
       	if(a[1]-a[2]<=L&&a[2]-a[1]<=L) printf("2"); else printf("0");
       	return 0;
       }
       sort(a+1,a+n+1);
       if(a[n]-a[1]>L)
       {
       	printf("0");
       	return 0;
       }
       for(i=n;i;i--) a[i]-=a[1];
       for(i=1;i<=n;i++) t[i]=a[i]-a[i-1];
       f[0][0][0][0]=1;
       for(i=0;i<n;i++)
       {
       	memset(f[1-i%2],0,sizeof(f[1-i%2]));
       	for(j=0;j<=n;j++) 
       	{
       		for(k=0;k+t[i+1]*(2*j-2)<=L&&k<=L;k++)
       		{
       			for(l=0;l<3;l++) if(f[i%2][j][k][l])
       			{
       				int x=a[i+1],k1=k+t[i+1]*(2*j-l);
       				if(k1>L) continue;
       				
       				if(j<n) f[1-i%2][j+1][k1][l]=(f[1-i%2][j+1][k1][l]+f[i%2][j][k][l])%md;
    
       				ll s=2*j-l;
       				f[1-i%2][j][k1][l]=(f[1-i%2][j][k1][l]+f[i%2][j][k][l]*s)%md;
       				
       				s=(j-l)*(j-1);
       				if(i+1==n) s++;
       				if(2*j-l>=2&&j>=2) f[1-i%2][j-1][k1][l]=(f[1-i%2][j-1][k1][l]+f[i%2][j][k][l]*s)%md;
       				
       				s=2-l;
       				if(j<n&&l<2) f[1-i%2][j+1][k1][l+1]=(f[1-i%2][j+1][k1][l+1]+f[i%2][j][k][l]*s)%md;
       				
       				s=j*2-j*l-l;
       				if(i+1==n) s++;
       				if(l<2&&j) f[1-i%2][j][k1][l+1]=(f[1-i%2][j][k1][l+1]+f[i%2][j][k][l]*s)%md;
       			}
       		}
       	}
       }
    
       int ans=0;
       for(k=0;k<=L;k++) ans=(ans+f[n%2][1][k][2])%md;
       printf("%d",ans);
       return 0;
    }
    
    哈哈哈哈哈哈哈哈哈哈
  • 相关阅读:
    通过抓取pintpoint2的页面信息把数据存入数据库python3
    jenkins 2.282+Publish over ssh 1.22版本发布日志不能实时显示
    ELKF搭建
    pinpoint2.0.2 定制开发--增加钉钉群通知、@到具体个人,解决手机号码无法输入BUG、删除客户端需要认证、查看数据日期范围扩大等。
    通过命令行通知RANCHER重新部署程序。
    Jenkins rancher cli 配置
    centos7 outlook mailx配置
    创业公司使用的敏捷GIT FLOW管理方式(多BUG、多项目多环境,多任务并发开发,支持需求多变)
    Dubbo Admin 2.6.2 版本连接zk 集群的连接字符串配置
    jenkins典型配置-多分支选择
  • 原文地址:https://www.cnblogs.com/LZA119/p/13910046.html
Copyright © 2011-2022 走看看