zoukankan      html  css  js  c++  java
  • TRI 解题报告

    题目大意:

    在一个平面上有N(N <= 1000)个点,其中任意三点不共线,求这些点组成的三角形的面积和每和三角形内部含的点数的个数和。

    数据范围:

    20%的数据 N <= 50, 30% N <= 100, 100% N <= 1000。

    算法讨论

    算法1:

    看到这题还是有部分分的,那么我们首先映入脑袋中的就是O(N^4)的算法,暴力枚举三个点叉积算面积,然后再枚举剩下的点判断是否在当前的三角形内。

    如何判断一个点在三角形内部,有个不错的教程:http://www.yalewoo.com/in_triangle_test.html

    考场上打这个暴力还是30分妥妥的。

    算法2:

    O(N^2logn)。

    首先对于第一问:

    我们考虑,对于一条边来说,边外的点都能和其组成一个三角形,而在枚举边的过程中,可能会对一个三角形多次计算,所以为了避免这种重复计算,我们就要保证一定的顺序,做到不重不漏。于是,对于这样的散点图,我们经常用到的排序方法有两种,第一个是以x为第一关键字,以y为第二关键字进行排序,另一个就是极角排序。(至于不知道什么是极角的,自行百度)。

    我们枚举每一个点让其做为一个边的起点,然后以这个点为基准进行极角排序(两种方法,一个是让其它点的坐标都减去这个点的坐标,然后atan2,另一个就是用这个点与其它点的斜率),下面给出这样一个过程。

    我们看这样一张图,如果计算S(AOB) + S(AOC) + S(AOD),那么这个面积和就等于 OA叉OB + OA叉OC + OA叉OD

    就等于  - xb * ya + xa * yb - xc * ya + xa * yc - xd * ya + xa * yd = -(xb + xc + xd) * ya + xa * (yc + yd + yb),为了保证这个结合是成立的,我们必须保证上面计算的顺序,也要保证都是向左旋,这样叉积才是正的。所以我们对于每一个点为基准,然后枚举那个‘A',就可以得到以每个点为端点的每一个三角形的面积。然后想办法减少计算的重复,就需要极角排序一下。

    对于第二问,我们考虑,当我们每选定一个点的时候,那么它最多在C(2,n-1)个三角形中,我们采用补集的思想,当且仅当三个点在这个点同侧的时候,这个点不在三个点围成的三角形中,那么,假设我们当前找的基准点是A,那么已经枚举的那个点是A',我们只要统计AA'一侧的点的数量就可以了,然后组合计数 C(2,...),计算得出答案。同样,为了避免重复, 我们要按照一定的顺序就OK了。

    Codes:

     1 #include <cstdio>
     2 #include <iostream>
     3 #include <cstring>
     4 #include <cstdlib>
     5 #include <algorithm>
     6 #include <cmath>
     7 using namespace std;
     8 typedef long long ll;
     9 
    10 int n;
    11 struct Point{
    12     double x, y, ji;
    13     
    14     Point(double _x = 0, double _y = 0): x(_x), y(_y) {}
    15     bool operator < (const Point &a) const{
    16         return ji < a.ji;
    17     }
    18 }p[1005 * 2], T[1005];
    19 
    20 int C(int x, int y){
    21     int ret = 1;
    22     for(int i = y; i > y-x; -- i){
    23         ret = ret * i;
    24     }
    25     for(int i = 1; i <= x; ++ i)
    26         ret = ret / i;
    27     return ret;
    28 }
    29 
    30 double Cross(Point a, Point b){
    31     return a.x * b.y - a.y * b.x;
    32 }
    33 
    34 void Solve(){
    35     ll ans1 = 0, ans2 = 0;
    36     for(int i = 1; i <= n; ++ i){
    37         int cnt = 0;
    38         for(int j = 1; j <= n; ++ j){
    39             if(j != i){
    40                 ++ cnt;
    41                 p[cnt].x = T[j].x - T[i].x; p[cnt].y = T[j].y - T[i].y;
    42                 p[cnt].ji = atan2(p[cnt].y, p[cnt].x); 
    43             }
    44         }
    45         sort(p + 1, p + cnt + 1);
    46         for(int j = 1; j <= cnt; ++ j)
    47             p[cnt + j] = p[j];
    48         ans2 += C(3, cnt);
    49         for(int j = 1, k = 2; j <= cnt; ++ j){
    50             if(j == k) k ++;
    51             while(Cross(p[j], p[k]) > 0) k ++;
    52             Point tp(T[i].x + p[j].x, T[i].y + p[j].y);
    53             ans1 += (long long)Cross(T[i], tp) * (k - j - 1);
    54             ans2 -= (long long)(k - j - 1) * (k - j - 2) / 2;
    55         }
    56     }
    57     printf("%lf %lf
    ", (double) ans1 / 2 / C(3, n), (double) ans2 / C(3, n));
    58 }
    59 #define ONLINE_JUDGE
    60 int main(){
    61 #ifndef ONLINE_JUDGE
    62     freopen("tri.in", "r", stdin);
    63     freopen("tri.out", "w", stdout);
    64 #endif
    65 
    66     scanf("%d", &n);
    67     for(int i = 1; i <= n; ++ i){
    68         scanf("%lf%lf", &T[i].x, &T[i].y);
    69     }
    70     
    71     Solve();
    72 
    73 #ifndef ONLINE_JUDGE
    74     fclose(stdin); fclose(stdout);
    75 #endif
    76     return 0;
    77 }
    TRI
  • 相关阅读:
    netty源码学习
    大话数据结构读书笔记
    java编程思想读书笔记
    spring依赖注入源码分析和mongodb自带连接本地mongodb服务逻辑分析
    spring的xml的property和constructor-arg的解析
    junit测试用例加载spring配置文件
    (大二下)软件需求分析阅读笔记01
    课后作业——需求分析
    2018年春季个人阅读计划
    软件需求与分析——大二下需会知识点
  • 原文地址:https://www.cnblogs.com/sxprovence/p/5114113.html
Copyright © 2011-2022 走看看