zoukankan      html  css  js  c++  java
  • Geometry 离散化

    【问题描述】
    山山在数学课上学到了平面上两点间直线距离公式:
    山山对新的知识非常感兴趣,于是他上网查询了一下,发现他所学的叫做“欧几里得距
    离(简称欧氏距离)


    ,还有一种距离叫“曼哈顿距离”,公式为:


    山山被这两个东西绕晕了。他开始尝试画一些点然后计算它们的这两种距离。发现有一
    些点的欧氏距离和曼哈顿距离是相等的,他认为这个现象特别有趣。为了发现一些规律和性
    质,他给出了 n 个点,想知道这些点中有多少对点的欧氏距离与曼哈顿距离相等。
    由于山山还只是一名小学生,他的计算能力不是很强,于是他希望你能帮他算出这样的
    点对数。 注意(A, B) 和(B, A) 被看作是相同的点对,只计算一次 。


    【输入格式】
    输入文件名为 geo.in。
    第一行为一个正整数 n。
    接下来 n 行每行两个整数 xi yi,表示每个点的坐标。


    【输出格式】
    输出文件名为 geo.out。
    输出仅一行一个整数 ans 表示答案。 答案可能会很大,请使用 64 位整数。


    【样例输入与输出】


    【数据范围】
    对于 20%的数据:0 < n <= 1000
    对于 70%的数据:0 < n <= 100000,0 <= xi , yi <= 10 6 ,保证数据是随机生成的
    对于 100%的数据:0 < n <= 100000,|xi |, |yi | <= 10 9


    一开始我用的map做的这题,但是判重的时候用了离散化,结果细节没处理好,只拿了60分。

    其实我们可以直接离散化,没必要用map。

    代码:

    #include<iostream>
    #include<cstdio>
    #include<cstdlib>
    #include<cstring>
    #include<algorithm>
    
    #define ll long long
    #define il inline
    #define db double
    
    using namespace std;
    
    il int gi()
    {
    	int x=0,y=1;
    	char ch=getchar();
    	while(ch<'0'||ch>'9')
    		{
    			if(ch=='-')
    				y=-1;
    			ch=getchar();
    		}
    	while(ch>='0'&&ch<='9')
    		{
    			x=x*10+ch-'0';
    			ch=getchar();
    		}
    	return x*y;
    }
    
    struct node
    {
    	int x,y;
    }c[200045];
    
    bool cmp1(node a,node b)
    {
    	return a.x<b.x;
    }
    
    bool cmp2(node a,node b)
    {
    	if(a.y==b.y)
    		return a.x<b.x;
    	else
    		return a.y<b.y;
    }
    
    int main()
    {
    	freopen("geo.in","r",stdin);
    	freopen("geo.out","w",stdout);
    
    	int n=gi();
    
    	for(int i=1;i<=n;i++)
    		c[i].x=gi(),c[i].y=gi();
    
    	sort(c+1,c+1+n,cmp1);
    
    	ll ans=0;
    
    	int now=c[1].x;
    	ll p=0;
    	for(int i=1;i<=n;i++)
    		{
    			if(c[i].x==now)
    				p++;
    			else
    				{
    					ans+=p*(p-1)/2;
    					now=c[i].x;
    					p=1;
    				}
    		}
    
    	if(p!=1)
    		ans+=p*(p-1)/2;
    
    	sort(c+1,c+1+n,cmp2);
    
    	now=c[1].y,p=0;
    	for(int i=1;i<=n;i++)
    		{
    			if(c[i].y==now)
    				p++;
    			else
    				{
    					ans+=p*(p-1)/2;
    					now=c[i].y;
    					p=1;
    				}
    		}
    
    	if(p!=1)
    		ans+=p*(p-1)/2;
    
    	int now1=c[1].x,now2=c[1].y;
    	p=0;
    	for(int i=1;i<=n;i++)
    		{
    			if(now1==c[i].x&&now2==c[i].y)
    				p++;
    			else
    				{
    					ans-=p*(p-1)/2;
    					now1=c[i].x,now2=c[i].y;
    					p=1;
    				}
    		}
    
    	if(p!=1)
    		ans-=p*(p-1)/2;
    
    	printf("%lld
    ",ans);
    
    	return 0;
    }
    
  • 相关阅读:
    获取文件的MD5码(C#)
    存储过程查询表的记录数
    通用分页
    解析SQLServer事务日志压缩与删除的方法
    性能测试之测试工具(一)
    测试框架学习HttpRunner之功能扩展(十一)
    测试框架学习HttpRunner之Locust性能测试(十)
    测试框架学习之HttpRunner Validate & Prettify(九)
    测试框架学习之HttpRunner测试用例之用例分层设计(八)
    测试框架学习之HttpRunner用例执行 运行测试以及测试报告(七)
  • 原文地址:https://www.cnblogs.com/gshdyjz/p/7681804.html
Copyright © 2011-2022 走看看