「JSOI2014」矩形并

传送门

我们首先考虑怎么算这个期望比较好。

我们不难发现每一个矩形要和 \(n - 1\) 个矩形去交,而总共又有 \(n\) 个矩形,所以我们把矩形两两之间的交全部加起来再除以 \(n(n - 1)\) 就是答案。

至于算矩形之间的交我们可以考虑把每个矩形都视为在这个矩形范围内区间加上 \(1\) ,那么我们只需要查询一个矩形内的和 - 该矩形自身的贡献就可以算出一个矩形与其他矩形的交。

所以现在相当于我们只需要实现二维的区间加/查询。

但是数据范围很大我们不可能用二维树状数组搞 不然这题就被爆艹了 。

所以我们考虑扫描线 + 一维树状数组来搞。

我们把纵坐标用树状数组维护起来,横坐标用扫描线代替。

对于每个矩形弄四条扫描线,两条修改两条查询。

其中一条修改在左边界加,另一条在右边界右边一个位置减。

其中一个查询在左边界左边一个位置算负的贡献,另一个在右边界算正的贡献。

那么有一个问题,为什么我们要对于左边界左边搞一条扫描线算负贡献呢?

我们考虑差分时的效果,我们对于一条加的修改扫描线,我们是把这条扫描线以左的一整块矩形区域都加了 \(1\)

那么我们如果在加之前,把一段区间的贡献减掉,实际上是减掉了若干个两个不同矩形左边界之间的值,那么我们这时如果在右边碰到一条查询正贡献的扫描线,我们会发现这两条扫描线的贡献一结合刚好就得到了一块矩形交的贡献,也就是说我们这样算就可以很方便地算矩形之间的交了。

具体细节可以手画一下图,那么我们接下来就用二维树状数组的实现技巧来搞。

还有一些小问题:运算过程中可能会爆 long long 所以我们用 long double 来存;然后就是树状数组不能以零为下标,所以我们再输入时把所有矩形的横纵坐标都往各正方向平移 \(2\) 个单位。

参考代码:

  1. #include <algorithm>
  2. #include <cstdio>
  3. #define rg register
  4. #define file(x) freopen(x".in", "r", stdin), freopen(x".out", "w", stdout)
  5. using namespace std;
  6. template < class T > inline void read(T& s) {
  7. s = 0; int f = 0; char c = getchar();
  8. while ('0' > c || c > '9') f |= c == '-', c = getchar();
  9. while ('0' <= c && c <= '9') s = s * 10 + c - 48, c = getchar();
  10. s = f ? -s : s;
  11. }
  12. typedef long double LL;
  13. const int _ = 2e5 + 5, __ = 2e6 + 5;
  14. int n, num = 0; LL ans = 0;
  15. struct node { int opt, x, l, r, v; } t[_ << 2];
  16. inline bool cmp(const node& a, const node& b)
  17. { return a.x != b.x ? a.x < b.x : (a.opt != b.opt ? a.opt < b.opt : a.v < b.v); }
  18. struct BIT {
  19. LL tr[__];
  20. inline void update(int x, LL v) { for (rg int i = x; i < __; i += i & -i) tr[i] += v; }
  21. inline LL query(int x) { LL res = 0; for (rg int i = x; i >= 1; i -= i & -i) res += tr[i]; return res; }
  22. } tr1, tr2, tr3, tr4;
  23. inline void Update(int x, int y, int v) {
  24. tr1.update(y, v), tr2.update(y, x * v), tr3.update(y, y * v), tr4.update(y, (LL) x * y * v);
  25. }
  26. inline LL Query(int x, int y) {
  27. return tr1.query(y) * (x + 1) * (y + 1) - tr2.query(y) * (y + 1) - tr3.query(y) * (x + 1) + tr4.query(y);
  28. }
  29. int main() {
  30. #ifndef ONLINE_JUDGE
  31. file("cpp");
  32. #endif
  33. read(n);
  34. for (rg int x, y, a, b, i = 1; i <= n; ++i) {
  35. read(x), read(y), read(a), read(b);
  36. x += 2, y += 2;
  37. t[++num] = (node) { 0, x, y, y + b - 1, 1 };
  38. t[++num] = (node) { 0, x + a, y, y + b - 1, -1 };
  39. t[++num] = (node) { 1, x - 1, y, y + b - 1, -1 };
  40. t[++num] = (node) { 1, x + a - 1, y, y + b - 1, 1 };
  41. ans -= 1ll * a * b;
  42. }
  43. sort(t + 1, t + num + 1, cmp);
  44. for (rg int i = 1; i <= num; ++i) {
  45. if (t[i].opt == 0)
  46. Update(t[i].x, t[i].l, t[i].v), Update(t[i].x, t[i].r + 1, -t[i].v);
  47. else
  48. ans += t[i].v * (Query(t[i].x, t[i].r) - Query(t[i].x, t[i].l - 1));
  49. }
  50. printf("%.9Lf\n", 1.0 * ans / n / (n - 1));
  51. return 0;
  52. }

「JSOI2014」矩形并的更多相关文章

  1. 「JSOI2014」打兔子

    「JSOI2014」打兔子 传送门 首先要特判 \(k \ge \lceil \frac{n}{2} \rceil\) 的情况,因为此时显然可以消灭所有的兔子,也就是再环上隔一个点打一枪. 但是我们又 ...

  2. 「JSOI2014」电信网络

    「JSOI2014」电信网络 传送门 一个点选了就必须选若干个点,最大化点权之和,显然最大权闭合子图问题. 一个点向它范围内所有点连边,直接跑最大权闭合子图即可. 参考代码: #include < ...

  3. 「JSOI2014」学生选课

    「JSOI2014」学生选课 传送门 看到这题首先可以二分. 考虑对于当前的 \(mid\) 如何 \(\text{check}\) 我们用 \(f_{i,j}\) 来表示 \(i\) 对 \(j\) ...

  4. 「JSOI2014」歌剧表演

    「JSOI2014」歌剧表演 传送门 没想到吧我半夜切的 这道题应该算是 \(\text{JSOI2014}\) 里面比较简单的吧... 考虑用集合关系来表示分辨关系,具体地说就是我们把所有演员分成若 ...

  5. 「JSOI2014」支线剧情2

    「JSOI2014」支线剧情2 传送门 不难发现原图是一个以 \(1\) 为根的有根树,所以我们考虑树形 \(\text{DP}\). 设 \(f_i\) 表示暴力地走完以 \(i\) 为根的子树的最 ...

  6. 「JSOI2014」强连通图

    「JSOI2014」强连通图 传送门 第一问很显然就是最大的强连通分量的大小. 对于第二问,我们先把原图进行缩点,得到 \(\text{DAG}\) 后,统计出入度为零的点的个数和出度为零的点的个数, ...

  7. 「JSOI2014」序列维护

    「JSOI2014」序列维护 传送门 其实这题就是luogu的模板线段树2,之所以要发题解就是因为学到了一种比较NB的 \(\text{update}\) 的方式.(参见这题) 我们可以把修改操作统一 ...

  8. 「AHOI2014/JSOI2014」拼图

    「AHOI2014/JSOI2014」拼图 传送门 看到 \(n \times m \le 10^5\) ,考虑根号分治. 对于 \(n < m\) 的情况,我们可以枚举最终矩形的上下边界 \( ...

  9. 「AHOI2014/JSOI2014」宅男计划

    「AHOI2014/JSOI2014」宅男计划 传送门 我们首先要发现一个性质:存货天数随买食物的次数的变化类似于单峰函数. 具体证明不会啊,好像是二分加三分来证明?但是没有找到明确的严格证明. 感性 ...

随机推荐

  1. IntelliJ IDEA 2017.3尚硅谷-----设置字体大小行间距

  2. IntelliJ IDEA 2017.3尚硅谷-----自动导包

  3. Windows10_64位下upload-labs靶场搭建+phpstudy_v8.1安装教程

     之前介绍了Windows10_64位搭建WampServer的教程,这一次再来水一篇phpstudy的搭建教程.哈哈哈.     顺便安装一下upload-labs,搭着玩玩~         操作 ...

  4. Codeforces Round #599 (Div. 2) B1. Character Swap (Easy Version)

    This problem is different from the hard version. In this version Ujan makes exactly one exchange. Yo ...

  5. linux用字符模式修改权限

    -rw-r--r--  tt.htm 从第二个字符起rw是说用户apple有读.写权,没有运行权,接着的r--表示用户组users只有读权限,没有运行权,最后的r--指其他人 (others)只有读权 ...

  6. 前端——语言——Core JS——《The good part》读书笔记——第一章节(Good Parts)

    本章是引言,有四个小节,具体内容如下: 第一小节 第一小节介绍作者的观点,作者编写本书的目的. 原文:I discovered that I could be a better programmer ...

  7. asp.net 大文件上传配置

    <system.web> <httpRuntime requestValidationMode=" ></httpRuntime> <!--单位:K ...

  8. Dart语言学习(七)Dart Map类型

    映射(Maps)是无序的键值对: 常用属性: keys 获取所有的key值 values 获取所有的value值 isEmpty 是否为空 isNotEmpty 是否不为空 常用方法: remove( ...

  9. 排序算法之希尔排序的python实现

    希尔排序(Shell’s Sort)是插入排序的一种,是直接插入排序算法的一种更高版本的改进版本. 希尔排序的工作原理 如下: (1)把记录按步长gap分组,对每组记录采用直接插入排序方法进行排序: ...

  10. C语言随笔2: rom ram 及其运行的过程

    内存的分配方式 五个区 :堆区.栈区.全局区.文字常量区.程序代码区 这五个区是ram中的.或者说,堆栈段.数据段.代码段 而rom中,没有堆区和栈区,只有数据段和代码段. ram是在程序运行是的内存 ...