首页 > 其他 > 详细

HDU5738

时间:2016-07-23 14:53:01      阅读:610      评论:0      收藏:0      [点我收藏+]

题意就是要找一条直线上的点,注意有重点;

技术分享

注意运用第一个公式,因为第i个点必定要用到,其他的点用法有两种,只用i的重点和不止用i的重点

只用i的重点相当于从1开始取组合数

不止用i的重点则重点可以从0开始取组合数,非重点则从1开始取

#include <stdio.h>
#include<algorithm>
#include<string.h>
#include<iostream>
#include<map>
using namespace std;
typedef long long LL;
map<pair<LL,LL>,int>vis;
const int mod=1e9+7;
int t,n;
struct st{
LL x,y;
}num[1005];
bool cmp(st a,st b){
if(a.x==b.x)
    return a.y<b.y;
else
    return a.x<b.x;
}
LL gcd(LL a,LL b){
return b?gcd(b,a%b):a;
}

LL qpow(LL a,LL b){
    if(b<0)
        return 0;
    a%=mod;
    LL ans = 1;
    while(b) {
        if(b&1) ans = (ans * a) % mod;
        b >>= 1;
        a = (a * a) % mod;
    }
    return ans;
  }


int main()
{
    freopen("in.txt","r",stdin);
 cin>>t;
 while(t--){

    scanf("%d",&n);
    int i,j;
    for( i=1; i<=n; i++)
        scanf("%lld%lld",&num[i].x,&num[i].y);
    sort(num+1,num+1+n,cmp);
    LL ans=0;
    for( i=1; i<=n; i++){
        vis.clear();
        int res=1;
        for( j=i+1; j<=n; j++){
        if(num[j].x==num[i].x&&num[i].y==num[j].y)
            res++;
        else{
        LL dx=num[i].x-num[j].x;
        LL dy=num[i].y-num[j].y;
        LL gg=gcd(dx,dy);
        if(gg!=0){
            dx/=gg;
            dy/=gg;
        }
    
        vis[make_pair(dx,dy)]++;
        }
        }
        if(res>1){
        ans+=(qpow(2,res-1)-1)%mod;//第一种只用这个点的情况
        ans%=mod;
        }
     // cout<<ans<<" "<<i<<endl;


            for(map<pair<LL,LL>,int>::iterator it = vis.begin();it!=vis.end();it++)
            {
                LL cnt=(it->second);
                ans = (ans+((qpow(2,cnt)-1)*(qpow(2,res-1)))%mod)%mod;//用重点和其他点的情况
            }


    }
    cout<<ans<<endl;
 }
}

 

HDU5738

原文:http://www.cnblogs.com/shimu/p/5698541.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!