Educational Codeforces Round 68 E. Count The Rectangles

传送门

题意:

给出不超过\(n,n\leq 5000\)条直线,问共形成多少个矩形。

思路:

考虑到\(n\)的范围不大,所以可以暴力枚举两条平行的直线,接下来处理的就是与其垂直的直线的数量。

满足相交成矩形有两个条件,假如我们枚举的直线是垂直于\(x\)轴的,那么两个条件即为\(low\leq y_i\leq high,x_{i,0}\leq left,right\leq x_{i,1}\)。

所以我们可以考虑扫描线的思想,根据左右端点选择插入/删除直线,插入的时候利用权值树状数组维护其\(y_i\)值。之后对于竖直的两条直线,查询范围在\(low\)~\(high\)之间的水平直线个数即可。设其为\(x\),那么对答案的贡献就为\(C_{x}^2\)。

注意一下\(low\)可能大于\(high\)的情况。

代码如下:

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 50005;
int n;
int x[N][2], y[N][2];
int X[N], Y[N];
struct seg1{
int x, y1, y2;
bool operator < (const seg1 &A) const {
return x < A.x;
}
}s1[N];
struct seg2{
int y, x1, x2;
}s2[N];
vector <int> v1[N], v2[N], v3[N];
int c[N];
int lowbit(int x) {
return x & (-x);
}
void add(int x, int v) {
for(int i = x; i < N; i += lowbit(i)) c[i] += v;
}
int query(int p) {
int ans = 0;
for(int i = p; i; i -= lowbit(i)) ans += c[i];
return ans;
}
int main() {
ios::sync_with_stdio(false); cin.tie(0);
cin >> n;
int D1 = 0, D2 = 0;
for(int i = 1; i <= n; i++) {
cin >> x[i][0] >> y[i][0] >> x[i][1] >> y[i][1];
X[++D1] = x[i][0]; X[++D1] = x[i][1];
Y[++D2] = y[i][0]; Y[++D2] = y[i][1];
}
sort(X + 1, X + D1 + 1); sort(Y + 1, Y + D2 + 1);
D1 = unique(X + 1, X + D1 + 1) - X - 1;
D2 = unique(Y + 1, Y + D2 + 1) - Y - 1;
int t1 = 0, t2 = 0;
for(int i = 1; i <= n; i++) {
x[i][0] = lower_bound(X + 1, X + D1 + 1, x[i][0]) - X;
x[i][1] = lower_bound(X + 1, X + D1 + 1, x[i][1]) - X;
y[i][0] = lower_bound(Y + 1, Y + D2 + 1, y[i][0]) - Y;
y[i][1] = lower_bound(Y + 1, Y + D2 + 1, y[i][1]) - Y;
if(x[i][0] > x[i][1]) swap(x[i][0], x[i][1]);
if(y[i][0] > y[i][1]) swap(y[i][0], y[i][1]);
if(x[i][0] == x[i][1]) s1[++t1] = seg1{x[i][0], y[i][0], y[i][1]};
if(y[i][0] == y[i][1]) s2[++t2] = seg2{y[i][0], x[i][0], x[i][1]};
if(y[i][0] == y[i][1]) v1[x[i][0]].push_back(t2);
}
sort(s1 + 1, s1 + t1 + 1);
for(int i = 1; i <= t1; i++) v3[s1[i].x].push_back(i);
ll res = 0;
int bound = D1;
for(int x = 1; x <= bound; x++) {
for(auto it : v2[x]) {
add(s2[it].y, -1);
}
for(auto it : v1[x]) {
add(s2[it].y, 1);
v2[s2[it].x2].push_back(it);
}
for(auto i : v3[x]) {
for(int j = x + 1; j <= bound; j++) {
for(auto k : v3[j]) {
int R = min(s1[i].y2, s1[k].y2), L = max(s1[i].y1, s1[k].y1);
if(L > R) continue;
int tmp = query(R) - query(L - 1);
res += 1ll * tmp * (tmp - 1) / 2;
}
for(auto it : v2[j]) add(s2[it].y, -1);
}
for(int j = x + 1; j <= bound; j++) {
for(auto it : v2[j]) add(s2[it].y, 1);
}
}
}
cout << res;
return 0;
}

Educational Codeforces Round 68 E. Count The Rectangles的更多相关文章

  1. Educational Codeforces Round 68 差G

    Educational Codeforces Round 68 E 题意:有 n 个线段,每个都是平行 x 或者 y 轴,只有互相垂直的两线段才会相交.问形成了多少个矩形. \(n \le 5000, ...

  2. Educational Codeforces Round 68

    目录 Contest Info Solutions A.Remove a Progression B.Yet Another Crosses Problem C.From S To T D.1-2-K ...

  3. Educational Codeforces Round 68 Editorial

    题目链接:http://codeforces.com/contest/1194                                            A.Remove a Progre ...

  4. Educational Codeforces Round 68 (Rated for Div. 2)---B

    http://codeforces.com/contest/1194/problem/B /* */ # include <bits/stdc++.h> using namespace s ...

  5. Educational Codeforces Round 84 E. Count The Blocks

    传送门: 1327- E. Count The Blocks  题意:给你一个整数n,求10^n内(每个数有前导零)长度为1到n的块分别有多少个.块的含义是连续相同数字的长度. 题解:从n=1开始枚举 ...

  6. Educational Codeforces Round 68 (Rated for Div. 2)补题

    A. Remove a Progression 签到题,易知删去的为奇数,剩下的是正偶数数列. #include<iostream> using namespace std; int T; ...

  7. Educational Codeforces Round 68 (Rated for Div. 2) C. From S To T (字符串处理)

    C. From S To T time limit per test1 second memory limit per test256 megabytes inputstandard input ou ...

  8. Educational Codeforces Round 68 (Rated for Div. 2) D. 1-2-K Game (博弈, sg函数,规律)

    D. 1-2-K Game time limit per test2 seconds memory limit per test256 megabytes inputstandard input ou ...

  9. Educational Codeforces Round 68 (Rated for Div. 2)D(SG函数打表,找规律)

    #include<bits/stdc++.h>using namespace std;int sg[1007];int main(){ int t; cin>>t; while ...

随机推荐

  1. concurrent (一)concurrent

    参考文档: 跳跃表原理分析:https://blog.csdn.net/a1259109679/article/details/46442895 一.阻塞队列 ArrayBlockingQueue : ...

  2. Gamma阶段发布说明

    访问我们 公课网 Gamma新增功能和缺陷修复 Gamma新增功能 增加找回密码功能~妈妈再也不用担心我的密码忘掉了. 增加管理员功能,有权删除评论并通知用户. 增加信箱功能,评论被管理员删除后会得到 ...

  3. cad.net 在位编辑的原理猜测及找到在位编辑状态的图元

    在位编辑的时候会产生一个名为 0-RefEdit0 的图层,原因不明.... 在位编辑的本质是一个长事务,长事务是使用编辑库的对象替换源对象,相当于剪辑.不理解没关系,请看下面的聊天记录.... [贵 ...

  4. javascript的立即执行函数

    在javascript中有引入立即执行函数的概念,那么什么是立即执行函数呢?立即执行函数又是怎么写的呢?立即执行函数与普通函数有什么区别呢? 先来看看一般的函数: function a(){ var ...

  5. [转帖]油猴脚本管理器 Tampermonkey v4.8 离线CRX安装包(谷歌浏览器版)

    https://www.52pojie.cn/thread-1010604-1-1.html 油猴脚本管理器 Tampermonkey v4.8 离线CRX安装包(谷歌浏览器版) 链接:https:/ ...

  6. Stopwatch 类用于计算程序运行时间

    Stopwatch 类 命名空间:System.Diagnostics.Stopwatch 实例化:Stopwatch getTime=new Stopwatch(); 开始计时:getTime.St ...

  7. asp获取access数据库中的一条随机记录

    针对“用一条SQL得到数据库中的随机记录集”问题在网上已经有很多答案了: SQL Server 2000: SELECT TOP n * FROM tanblename ORDER BY NEWID( ...

  8. C#生成/调用动态链接库

    参考地址:https://www.cnblogs.com/qq4004229/archive/2013/01/30/2882409.html 一.需求描述 (1)用代码生成动态链接库 (2)用C#代码 ...

  9. vant-ui rem问题

    使用vant-ui时rem单位问题 vue-cli3.x项目引入vant-ui1.6.21,要求在项目中使用rem单位,但vant是px的.故,需要添加一个自动转换插件和一段修改html根元素的fon ...

  10. 像Java一样管理对象:T&形式仅仅用在参数传递

    类的对象为了关联/包含一个T类型的instance,若成员变量包括T*/ T&, 这种设计叫做“aggregation”(聚合):而若采用T 形式,则称为"composition&q ...