\[\texttt{Preface}
\]

赛时,把 " 任意时刻 " 理解成 " 整数时刻 " 了,看起来一脸不可做的亚子,还各种推式子。

话说我为什么觉得 E 比 F 还难。

\[\texttt{Description}
\]

一个坐标轴 \(OX\) 上有 \(n\) 个点,第 \(i\) 个点位于整数点 \(x_i\) ,速度为 \(v_i\) 。

所有点以恒定的速度移动,在时刻 \(t\) ( \(t\) 可以是非整数),第 \(i\) 个点的坐标可以被计算为 \(x_i + t \times v_i\) 。

定义 \(d(i,j)\) 为:任何可能的时刻,第 \(i\) 个点和第 \(j\) 个点的最小可能距离,即

\[\min\limits_{t\in[0,∞)} \{ |(x_i+t \times v_i)-(x_j+t \times v_j)| \}
\]

求 \(\sum\limits_{1 \leq i < j \leq n}d(i,j)\) 。

\[\texttt{Solution}
\]

  • 考虑任意两个点 \(i,j\) ,不妨设 \(x_i \leq x_j\) :
  1. 当 \(v_i \leq v_j\) 时:

    说明点 \(i\) 的速度不比点 \(j\) 快,意味着每过一秒,点 \(i,j\) 的距离会增大 \(v_j-v_i\) ,由于 \(t\times(v_j-v_i) \geq 0\) ,所以点 \(i,j\) 的距离永远不降,故时刻 \(0\) 时,\(d(i,j)\) 最小,此时 \(d(i,j)=x_j-x_i\) 。

  2. 当 \(v_i>v_j\) 时:

    说明点 \(i\) 的速度比点 \(j\) 快,意味着每过一秒,点 \(i,j\) 的距离会减少 \(v_i-v_j\) ,突然想到小学数学的 追及问题 ,显然可得,在时刻 \(\frac{x_j-x_i}{v_i-v_j}\) 时,点 \(i\) 与点 \(j\) 重合,故 \(d(i,j)=0\) 。

  • 我们发现情况 \(2\) 对答案没有贡献,实际上只是求情况 \(1\) 的贡献,也就是 \(\sum\limits_{x_i \leq x_j \ \& \ v_i \leq v_j}(x_j-x_i)\) 。

  • 然后我们发现这是一个经典的偏序问题,具体的:

  • 我们将所有点按 \(x\) 这一维从小到大排序,就可以直接去掉 \(x_i \leq x_j\) 的偏序关系了。

  • 将 \(v\) 这一维离散化,建立两个 BIT(当然线段树也行),一个用于维护当前扫描到的点中, \(v\) 值在区间内的点的个数,一个用于维护 \(v\) 值在区间内的点的 \(x\) 值和。

  • 接下来我们扫描每个点 \(i\) ,计算将 \(i\) 当成位置靠后的那个点时所产生的贡献,当固定住 \(i\) 时,此时 \(x_i\) 是不变的,我们记 \(v\) 值小于等于 \(v_i\) 的点的数量为 \(c_1\) , \(v\) 值小于等于 \(v_i\) 点的 \(x\) 值和为 \(c_2\) ,贡献为 \(x_i \times c_1-c_2\) ,最后再将点 \(i\) 插入 BIT ,扫描完即可求出答案。

  • \(\mathcal{O(n \log n)}\) 。

\[\texttt{Code}
\]

#include<cstdio>
#include<algorithm> #define RI register int using namespace std; namespace IO
{
static char buf[1<<20],*fs,*ft;
inline char gc()
{
if(fs==ft)
{
ft=(fs=buf)+fread(buf,1,1<<20,stdin);
if(fs==ft)return EOF;
}
return *fs++;
}
#define gc() getchar()
inline int read()
{
int x=0,f=1;char s=gc();
while(s<'0'||s>'9'){if(s=='-')f=-f;s=gc();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=gc();}
return x*f;
}
}using IO::read; const int N=200100; int n; struct Node{
int x;
int v;
}a[N]; bool cmp(Node a,Node b)
{
return a.x<b.x;
} int len,mapval[N]; void discrete() // 离散化
{
sort(mapval+1,mapval+1+len);
len=unique(mapval+1,mapval+1+len)-mapval-1;
} int Real(int x)
{
return lower_bound(mapval+1,mapval+1+len,x)-mapval;
} long long c[N][2]; // 0 维是数量 , 1 维是 x 值和 void add(int x,int k,int val)
{
for(;x<=len;x+=x&-x)c[x][k]+=val;
} long long ask(int x,int k)
{
long long ans=0;
for(;x;x-=x&-x)ans+=c[x][k];
return ans;
} long long ans; int main()
{
n=read(); for(RI i=1;i<=n;i++)
a[i].x=read(); for(RI i=1;i<=n;i++)
mapval[++len]=a[i].v=read(); sort(a+1,a+1+n,cmp); // 排序 discrete(); for(RI i=1;i<=n;i++)
a[i].v=Real(a[i].v); for(RI i=1;i<=n;i++)
{
ans+=a[i].x*ask(a[i].v,0)-ask(a[i].v,1); // 计算
add(a[i].v,0,1),add(a[i].v,1,a[i].x);
} printf("%lld\n",ans); return 0;
}

\[\texttt{Thanks} \ \texttt{for} \ \texttt{watching}
\]

题解【CF1311F Moving Points】的更多相关文章

  1. [CF1311F] Moving Points - 树状数组

    Solution 按 \(x\) 关键字升序排序,依次枚举每个点 考虑对任意 \(x_j < x_i\),那么当 \(v_j \leq v_i\) 时,它们不会相交,且 \(dis\) 就是它们 ...

  2. F. Moving Points 解析(思維、離散化、BIT、前綴和)

    Codeforce 1311 F. Moving Points 解析(思維.離散化.BIT.前綴和) 今天我們來看看CF1311F 題目連結 題目 略,請直接看原題. 前言 最近寫1900的題目更容易 ...

  3. HDOJ 4717 The Moving Points

    The Moving Points Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  4. HDU 4717The Moving Points warmup2 1002题(三分)

    The Moving Points Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  5. The Moving Points hdu4717

    The Moving Points Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  6. HDU 4717 The Moving Points (三分)

    The Moving Points Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  7. HDUOJ---The Moving Points

    The Moving Points Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  8. 题解 CF576C 【Points on Plane】

    题解 CF576C [Points on Plane] 一道很好的思维题. 传送门 我们看这个曼哈顿距离,显然如果有一边是按顺序排列的,显然是最优的,那另一边怎么办呢? 假如你正在\(ioi\)赛场上 ...

  9. HDU-4717 The Moving Points(凸函数求极值)

    The Moving Points Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

随机推荐

  1. Java常见问题汇总

    1.String,StringBuffer,StringBulider的区别及应用场景 2.Servlet生命周期 3.向上转型与向下转型 4.Java的多态性 5.重写和重载的区别 6.深拷贝和浅拷 ...

  2. springIOC源码接口分析(二):ConfigurableBeanFactory

    一 继承功能 1 SingletonBeanRegistry接口 此接口是针对Spring中的单例Bean设计的.提供了统一访问单例Bean的功能,类中定义了以下方法: 2 HierarchicalB ...

  3. divide and conquer - 最大连续子序列 - py

    以HDU1231为例,代码之没法交如下: inf = 0x3f3f3f3f a = [0 for i in range(10005)] ans, L, R = -inf, 0, 0 def divid ...

  4. CentOS 7 上CNVnator安装

    1.到github上下载最新版本 https://github.com/abyzovlab/CNVnator/releases 2.先看INSTALL文件,要求以下依赖,我的机器上已经安装了前两个,所 ...

  5. 实验6:路由器IOS升级

    路由器IOS升级 Cisco路由器IOS映像恢复及升级方法 一.Cisco 1000,1600,2500,4000系列 1.IOS映像恢复的方法及步骤 1) 连接PC的COM1口与路由器的consol ...

  6. void * 和 void 在函数返回值中的区别

    一个很容易糊涂的问题. 在函数的返回值中, void 是没有任何返回值, 而 void * 是返回任意类型的值的指针. 还是看代码吧: #include <stdlib.h> #inclu ...

  7. html作业记录

    <html> <head> <title>Hello World</title> </head> <body> <!-- ...

  8. 简单处理IP XML数据

    ///* 编译环境: visual c++ */ //#include <stdio.h> //#include <winsock2.h> //#pragma comment( ...

  9. python学习--quote()函数

    屏蔽特殊的字符.比如如果url里面的空格!url里面是不允许出现空格的. 在 Python2.x 中的用法是:urllib.quote(text)Python3.x 中是urllib.parse.qu ...

  10. 如何为wordpress 的文章添加分页

    原文参考:http://www.wpdaxue.com/add-next-page-button-wordpress-post-editor.html 1.在编辑文章时切到text 模式,然后加上&l ...