[THUPC2019]不等式/[51Nod1598]方程最小值

题目大意:

给定\(a_{1\sim n}\)和\(b_{1\sim n}\),令\(f_k(x)=\sum_{i=1}^k|a_ix+b_i|\)。对于所有\(k=1\sim n\),求\(f_k\)在\(\mathbb{R}\)中的最小值。

\(1\le n\le 5\times10^5,|a_i|,|b_i|<10^5\)

思路:

\[\sum_{i=1}^k|a_ix+b_i|=\sum_{i=1}^k|a_i||x+\frac{b_i}{a_i}|
\]

画在数轴上就是在\(-\frac{b_i}{a_i}\)(即零点)的位置有\(|a_i|\)个点。要找到一个位置\(x\),使得\(x\)到所有点的距离之和最小。

根据小学奥数的那套理论,\(x\)就是所有零点的加权中位数。我们可以用一个大根堆、一个小根堆来维护所有的零点,并求出中位数。

考虑函数加上绝对值后,\(a_i\)实际的符号。对于\(-\frac{b_i}{a_i}<x\)的函数来说,\(a_i>0\);反之\(a_i<0\)。因此我们可以在对两个堆中的元素分别维护考虑绝对值后\(a_i,b_i\)之和。即可求出最终\(f_k(x)\)的最小值。

时间复杂度\(\mathcal O(n\log n)\)。

源代码:

#include<queue>
#include<cstdio>
#include<cctype>
#include<cassert>
#include<algorithm>
inline int getint() {
register char ch;
register bool neg=false;
while(!isdigit(ch=getchar())) neg|=ch=='-';
register int x=ch^'0';
while(isdigit(ch=getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
return neg?-x:x;
}
const int N=5e5+1;
using int64=long long;
using Node=std::pair<double,int>;
std::priority_queue<Node,std::vector<Node>,std::greater<Node>> q1;//small
std::priority_queue<Node,std::vector<Node>,std::less<Node>> q2;//big
int64 s1,s2,a1,a2,b1,b2,a[N],b[N];
double o[N];
int main() {
const int n=getint();
for(register int i=1;i<=n;i++) a[i]=getint();
for(register int i=1;i<=n;i++) b[i]=getint();
for(register int i=1;i<=n;i++) {
if(a[i]!=0) {
o[i]=-1.*b[i]/a[i];
if(s1&&o[i]>q1.top().first) {
q1.push(std::make_pair(o[i],i));
if(a[i]>0) a[i]=-a[i],b[i]=-b[i];
a1+=a[i]; b1+=b[i];
s1+=std::abs(a[i]);
} else {
q2.push(std::make_pair(o[i],i));
if(a[i]<0) a[i]=-a[i],b[i]=-b[i];
a2+=a[i]; b2+=b[i];
s2+=std::abs(a[i]);
}
} else {
b1+=std::abs(b[i]);
}
while(s1>s2) {
q2.push(q1.top());
const int i=q1.top().second;
a1-=a[i]; b1-=b[i];
s1-=std::abs(a[i]);
a[i]=-a[i]; b[i]=-b[i];
a2+=a[i]; b2+=b[i];
s2+=std::abs(a[i]);
q1.pop();
}
while(s2>s1) {
q1.push(q2.top());
const int i=q2.top().second;
a2-=a[i]; b2-=b[i];
s2-=std::abs(a[i]);
a[i]=-a[i]; b[i]=-b[i];
a1+=a[i]; b1+=b[i];
s1+=std::abs(a[i]);
q2.pop();
}
const double x=s1?q1.top().first:0;
printf("%.7f\n",a1*x+b1+a2*x+b2);
}
return 0;
}

最新文章

  1. 如何彻底卸载Oracle
  2. 关于设置border的小技巧
  3. Solr调研总结
  4. 第23章 SEH结构化异常处理(2)_编译器对系统SEH机制的封装
  5. 【壁纸自动换】自动下载、更换壁纸(Bing壁纸)--XinBSBingWallPaper[2.7更新]
  6. 2-Medium下的MultipleCommandAssembly
  7. FZU-1921+线段树
  8. 解决安装包在win7,win8系统下安装后运行没有管理员权限
  9. Intellij 中的git操作 转!
  10. git 分布式版本控制了解
  11. 处理json中影响解析的多余引號
  12. 使用crontab创建 linux 系统定时任务#
  13. C#对图片进行马赛克处理,可控制模糊程度
  14. (MariaDB)MySQL内置函数大全
  15. java - day005 - 数组工具类, 数组复制,二维数组,变量,方法, 面向对象
  16. Java 日期比较大小
  17. bootstrap datetimepicker 格式化yyyymmdd时,无法读取yyyymmdd格式
  18. 数的划分(NOIP2001&水题测试2017082401)
  19. C++缓冲区溢出
  20. 〖Linux〗svn log 每个日志记录只显示一行的方法

热门文章

  1. JZOJ5833 永恒
  2. vs2017(Visual Studio Code)安装汉化
  3. 2019 央视网java面试笔试题 (含面试题解析)
  4. 2019 网易java面试笔试题 (含面试题解析)
  5. Flutter — IDE Shortcuts for Faster Development
  6. 图片Image转换为base64编码的方法
  7. Android-----Intent通过startActivityForResult(Intent intent , int 标志符)启动新的Activity
  8. tar.bz2解压异常
  9. [FreeRTOS]FreeRTOS使用
  10. MySQL/MariaDB数据库的各种日志管理