题意:

给你一个C,再给你n组a,b,让你求x取什么值的时候,$ \sum_{i=1}^n |a_i*x+b_i| =C $,要求求出解的个数,并用最简分数从小到大表示,如果有无穷多解,输出-1.

题解:

其实这些方程就是在平面上的一组曲线,都是V形的,最低点都在x轴上,求出所有的零点,以这个零点从左到右排序。

容易看出,这些函数之和也是一条曲线y=f(i),这条曲线最多有n个转折点,就是刚才那些零点,那么就在这n个转折点分出的n+1个区间内,和这n个点上,用比例公式找和y=C的交点即可。无穷多解的情况是存在一条与y=C重合的线段。

首先预处理出f(i)上所有转折点的值,注意n的范围是1e5,因此不可能让你$O(n^2)$求每一点的值,其实,只需维护a与b的前缀和和后缀和,要求某点$x_k$时,将零点在此点左边的函数取正,零点在此点右边的的函数取反。

$(\sum_{i=1}^{k-1}a_i) *x_k+\sum_{i=1}^{k-1}b_i-(\sum_{i=k+1}^{n}a_i) *x_k-\sum_{i=k+1}^{n}b_i$

注意判断零点重合情况。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<string>
#include<stack>
#include<algorithm>
using namespace std;
typedef long long LL;
typedef long long ll;
const int M = 1e5 + ;
const double eps = 1e-;
const LL mod = ;
const LL lINF = 0x3f3f3f3f3f3f3f3f;
struct node {
int a, b;
}tr[M];
int t;
int n, c;
int fenzi[M], fenmu[M];
int ans;
int gcd(int a, int b)
{
if (!b)
return a;
else
return gcd(b, a % b);
}
double lst;
bool cmp(node x, node y)
{
return x.a * y.b - x.b * y.a < ;
}
bool cmp2(node x, node y)
{
return (double)x.b / -x.a < (double)y.b / -y.a;
}
bool cmp1(node x, node y)
{
return (double)x.b / -x.a <= (double)y.b / -y.a;
}
int suma[M], sumb[M];
int flag;
double nw;
double nx;
int main()
{
scanf("%d", &t);
while (t--)
{
scanf("%d%d", &n, &c);
for (int i = ; i <= n; i++)
{
scanf("%d%d", &tr[i].a, &tr[i].b);
}
sort(tr + , tr + n + , cmp);
suma[] = sumb[] = ;
for (int i = ; i <= n; i++)
{
suma[i] = suma[i - ] + tr[i].a;
sumb[i] = sumb[i - ] + tr[i].b;
}
flag = ans = ;
lst = -10000.0;
for (int i = ; i <= n; i++)
{
int tmpa = -suma[n];
int tmpb = -sumb[n];
tmpa += * suma[i];
tmpb += * sumb[i];
nw = (double)sumb[i] / suma[i];
nx = (double)sumb[i + ] / suma[i + ];
if (fabs(nw - nx) < eps)
continue;
if (!tmpa && tmpb == c)
{
flag = ;
break;
}
if (!i)
{
node tmpc;
tmpc.a = tmpa, tmpc.b = tmpb - c;
if (cmp1(tmpc, tr[]))
{
fenzi[ans] = -tmpc.b;
fenmu[ans] = tmpa;
int d = gcd(fenzi[ans], fenmu[ans]);
fenzi[ans] /= d;
fenmu[ans] /= d;
if (fenmu[ans] < )
{
fenzi[ans] = -fenzi[ans], fenmu[ans] = -fenmu[ans];
}
ans++;
}
}
else if (i == n)
{
node tmpc;
tmpc.a = tmpa, tmpc.b = tmpb - c;
if (cmp2(tr[n], tmpc))
{
fenzi[ans] = -tmpc.b;
fenmu[ans] = tmpa;
int d = gcd(fenzi[ans], fenmu[ans]);
fenzi[ans] /= d;
fenmu[ans] /= d;
if (fenmu[ans] < )
{
fenzi[ans] = -fenzi[ans], fenmu[ans] = -fenmu[ans];
}
ans++;
}
}
else
{
node tmpc;
tmpc.a = tmpa, tmpc.b = tmpb - c;
if (cmp2(tr[i], tmpc) && cmp1(tmpc, tr[i + ]))
{
fenzi[ans] = -tmpc.b;
fenmu[ans] = tmpa;
int d = gcd(fenzi[ans], fenmu[ans]);
fenzi[ans] /= d;
fenmu[ans] /= d;
if (fenmu[ans] < )
{
fenzi[ans] = -fenzi[ans], fenmu[ans] = -fenmu[ans];
}
ans++;
}
}
lst = (double)(tmpb - c) / tmpa;
}
if (flag)
{
printf("-1\n");
}
else
{
printf("%d", ans);
for (int i = ; i < ans; i++)
{
printf(" %d/%d", fenzi[i], fenmu[i]);
}
puts("");
}
}
}

hdu多校第五场1004 (hdu6627) equation 1 计算几何的更多相关文章

  1. hdu 6088 Rikka with Rock-paper-scissors (2017 多校第五场 1004) 【组合数学 + 数论 + 模意义下的FFT】

    题目链接 首先利用组合数学知识,枚举两人的总胜场数容易得到 这还不是卷积的形式,直接搞的话复杂度大概是O(n^2)的,肯定会TLE.但似乎和卷积有点像?想半天没想出来..多谢Q巨提醒,才知道可以用下面 ...

  2. hdu多校第五场1005 (hdu6628) permutation 1 排列/康托展开/暴力

    题意: 定义一个排列的差分为后一项减前一项之差构成的数列,求对于n个数的排列,差分的字典序第k小的那个,n<=20,k<=1e4. 题解: 暴力打表找一遍规律,会发现,对于n个数的排列,如 ...

  3. 2014 HDU多校弟五场J题 【矩阵乘积】

    题意很简单,就是两个大矩阵相乘,然后求乘积. 用 Strassen算法 的话,当N的规模达到100左右就会StackOverFlow了 况且输入的数据范围可达到800,如果变量还不用全局变量的话连内存 ...

  4. 2014 HDU多校弟五场A题 【归并排序求逆序对】

    这题是2Y,第一次WA贡献给了没有long long 的答案QAQ 题意不难理解,解题方法不难. 先用归并排序求出原串中逆序对的个数然后拿来减去k即可,如果答案小于0,则取0 学习了归并排序求逆序对的 ...

  5. HDU 多校对抗赛第二场 1004 Game

    Game Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  6. hdu多校第五场1002 (hdu6625) three arrays 字典树/dfs

    题意: 给你两个序列a,b,序列c的某位是由序列a,b的此位异或得来,让你重排序列ab,找出字典序最小的序列c. 题解: 如果能找到a,b序列中完全一样的值当然最好,要是找不到,那也尽量让低位不一样. ...

  7. hdu多校第五场1006 (hdu6629) string matching Ex-KMP

    题意: 给你一个暴力匹配字符串公共前缀后缀的程序,为你对于某个字符串,暴力匹配的次数是多少. 题解: 使用扩展kmp构造extend数组,在扩展kmp中,设原串S和模式串T. extend[i]表示T ...

  8. hdu多校第五场1007 (hdu6630) permutation 2 dp

    题意: 给你n个数,求如下限制条件下的排列数:1,第一位必须是x,2,最后一位必须是y,3,相邻两位之差小于等于2 题解: 如果x<y,那么考虑把整个数列翻转过来,减少讨论分支. 设dp[n]为 ...

  9. 2018 HDU多校第四场赛后补题

    2018 HDU多校第四场赛后补题 自己学校出的毒瘤场..吃枣药丸 hdu中的题号是6332 - 6343. K. Expression in Memories 题意: 判断一个简化版的算术表达式是否 ...

随机推荐

  1. jsp引擎是什么

    1.JSP引擎 执行JSP代码需要在服务器上安装JSP引擎,比较常见的引擎有webLogic和Tomcat.把这些支持JSP的web服务器配置好后,就可以在客户端通过浏览器来访问JSP页面了. 2.J ...

  2. 管理mysql

    要管理MySQL,可以使用可视化图形界面MySQL Workbench. MySQL Workbench可以用可视化的方式查询.创建和修改数据库表,但是,归根到底,MySQL Workbench是一个 ...

  3. (转)Linux C 多线程编程----互斥锁与条件变量

    转:http://blog.csdn.net/xing_hao/article/details/6626223 一.互斥锁 互斥量从本质上说就是一把锁, 提供对共享资源的保护访问. 1. 初始化: 在 ...

  4. 微软引入了两种新的网络过滤系统,WFP和NDISfilter

    Windows 8是微软公司推出的最新的客户端OS,内部名称Windows NT 80.相对于Windows NT 5.x,其网络结构变化非常大,原有的TDI,NDIS系统挂接方法不再适用.在Wind ...

  5. Maven 打包jar清单

    <project> ... <build> <plugins> <plugin> <groupId>org.apache.maven.plu ...

  6. 【node】---token的原理及使用---【alley】

    一.登陆的验证流程 当用户请求登录的时候,如果没有问题,我们在服务端生成一条记录,这个记录里可以说明一下登录的用户是谁,然后把这条记录的 ID 号发送给客户端,客户端收到以后把这个 ID 号存储在 C ...

  7. Pandas异常值处理

    import pandas as pd #生成异常数据 df=pd.DataFrame({'col1':[1,120,3,5,2,12,13], 'col2':[12,17,31,53,22,32,4 ...

  8. 【洛谷】P1247取火柴游戏

    题目链接:https://www.luogu.org/problemnew/show/P1247 题意:nim取石子的题意,多了一个判断先手赢的话,输出先手第一把怎么拿,以及拿完之后每堆还剩多少. 题 ...

  9. Apache版hadoop编译

    前言  做为大数据入门的基础,hadoop是每个大数据开发人员几乎不可避免的基础,目前hadoop已经发展到3.x.x版本,但当前企业使用的主流还是2.x.x版本,hadoop官网提供了编译后的had ...

  10. Unity中动态绘制圆柱体

    问题背景 上次写了动态绘制立方体,这最近又来了新功能,绘制圆柱(风筒),要求是给了很多节点,根据节点去动态绘制风筒,风筒就是圆柱连接而成的,可以理解为管道,还有就是拐角处注意倒角,圆润过度过来. 实现 ...