zoukankan      html  css  js  c++  java
  • 【BZOJ4927】第一题 双指针+DP(容斥?)

    【BZOJ4927】第一题

    Description

    给定n根直的木棍,要从中选出6根木棍,满足:能用这6根木棍拼
    出一个正方形。注意木棍不能弯折。问方案数。
    正方形:四条边都相等、四个角都是直角的四边形。

    Input

    第一行一个整数n。
    第二行包含n个整数ai,代表每根木棍的长度。
    n ≤ 5000, 1 ≤ ai ≤ 10^7

    Output

    一行一个整数,代表方案数。

    Sample Input

    8
    4 5 1 5 1 9 4 5

    Sample Output

    3

    题解:这。。。这不是沈阳集训的原题吗?(xqz说是山东集训的原题)

    由于题目让你拼的是正方形,那么这个正方形的组成显然只有两种情况:3+1+1+1或2+2+1+1(这里指的是正方形的四条边),那么我们分类讨论这两种情况。

    如果是2+2+1+1,那么我们可以枚举最长的那2根木棍(也就是两个1的长度)。显然要先排序,并将长度相同的木棍合在一起。然后我们已知正方形的边长,那么问题就变成了如何选出4根木棍a,b,c,d使得a+b=c+d=这个边长。这里我采用的是双指针法。两个指针从两段向中间移动,就可以顺便统计出有多少对木棍符合条件。于是ans+=C(最长的木棍的条数,2)*C(符合条件的对数,2),当然,别忘了去重!

    如果是3+1+1+1,我们依旧是枚举最长的那3根木棍。然后问题就变成了如何在一堆木棍中选出3根使得长度之和为一个定值。这个显然是O(n3)的背包啊,而我们要的是O(n2)的复杂度,怎么办?

    看来我们枚举最长棍的做法不太可行,那么我们可以换个角度,枚举3条短棍中最长的那一条。那么我们可以用s[i]表示在之前的木棍中,选出两根使得长度之和为i的方案数。这样,我们在枚举到i的时候,先枚举i后面的所有木棍j,判断一下j能否成为最长的木棍,也就是判断s[j的长度-i的长度]是否为0。如果是,则更新答案;枚举完j后,我们再用i来更新s数组,这就变成了一个背包问题。

    然而考试的时候大佬们都是用容斥来处理的3+1+1+1,感觉容斥学的不好,没太听懂~

    #include <cstdio>
    #include <cstring>
    #include <iostream>
    #include <algorithm>
    using namespace std;
    typedef long long ll;
    int n,m;
    ll ans,sum,cnt;
    int v[5010],val[5010],num[5010],bel[5010];
    int s[10000010];
    int main()
    {
    	//freopen("yist.in","r",stdin);
    	//freopen("yist.out","w",stdout);
    	scanf("%d",&n);
    	int i,j,l,r;
    	for(i=1;i<=n;i++)	scanf("%d",&v[i]);
    	sort(v+1,v+n+1);
    	for(i=1;i<=n;i++)
    	{
    		if(v[i]>v[i-1])	val[++m]=v[i];
    		num[m]++,bel[i]=m;
    	}
    	for(i=1;i<=m;i++)
    	{
    		if(num[i]>=2)
    		{
    			sum=cnt=0;
    			for(l=1,r=i-1;l<=r;l++)
    			{
    				while(l<=r&&val[l]+val[r]>val[i])	r--;
    				if(val[l]+val[r]!=val[i]||l>r)	continue;
    				if(l==r)
    				{
    					if(num[l]>=4)	cnt+=(ll)num[l]*(num[l]-1)*(num[l]-2)*(num[l]-3)/2/3/4;
    					cnt+=(ll)num[l]*(num[l]-1)/2*sum;
    				}
    				else
    				{
    					if(num[l]>=2&&num[r]>=2)	cnt+=(ll)num[l]*(num[l]-1)/2*num[r]*(num[r]-1)/2;
    					cnt+=(ll)num[l]*num[r]*sum;
    					sum+=(ll)num[l]*num[r];
    				}
    			}
    			ans+=cnt*num[i]*(num[i]-1)/2;
    		}
    	}
    	for(i=1;i<=n;i++)
    	{
    		for(j=bel[i]+1;j<=m;j++)	if(num[j]>=3)	ans+=(ll)num[j]*(num[j]-1)*(num[j]-2)/2/3*(s[val[j]-v[i]]);
    		for(j=1;j<i;j++)	if(v[j]+v[i]<=v[n])	s[v[j]+v[i]]++;
    	}
    	printf("%lld",ans);
    	return 0;
    }

     

  • 相关阅读:
    javascript回调函数笔记
    JavaScript回调函数的实现
    深入理解JS执行细节(写的很精辟)
    javascript中return function与return function()的区别
    windows下dubbo-admin2.6.x之后版本的安装
    shiro经典通俗易懂javase例子
    字符串转数字练习--String to Integer (atoi)
    字符串按照Z旋转90度然后上下翻转的字形按行输出字符串--ZigZag Conversion
    SQL ----post漏洞测试注入
    nginx笔记----解决windows80端口被iis占用
  • 原文地址:https://www.cnblogs.com/CQzhangyu/p/7076081.html
Copyright © 2011-2022 走看看