题目描述

维护一个向量集合,在线支持以下操作:
"A x y (|x|,|y| < =10^8)":加入向量(x,y);
"Q x y l r (|x|,|y| < =10^8,1 < =L < =R < =T,其中T为已经加入的向量个数)":询问第L个到第R个加入的向量与向量(x,y)的点积的最大值。
集合初始时为空。

输入

输入的第一行包含整数N和字符s,分别表示操作数和数据类别;
接下来N行,每行一个操作,格式如上所述。
请注意s≠'E'时,输入中的所有整数都经过了加密。你可以使用以下程序得到原始输入:
inline int decode (int x long long lastans) {
     return x ^ (lastans & Ox7fffffff);
}
其中x为程序读入的数,lastans为之前最后一次询问的答案。在第一次询问之前,lastans=0。

注:向量(x,y)和(z,w)的点积定义为xz+yw。

输出

对每个Q操作,输出一个整数表示答案。

样例输入

6 A
A 3 2
Q 1 5 1 1
A 15 14
A 12 9
Q 12 8 12 15
Q 21 18 19 18

样例输出

13
17
17


题解

线段树+STL-vector维护凸包

本质是什么呢:给出 $x_0$ 和 $y_0$ ,求满足条件的 $x,y$ 使得 $x_0x+y_0y$ 最大。

我们不妨设这个值是 $c$ ,则要最大化 $c$ 。

当 $y>0$ 时:

给这个式子 $c=x_0x+y_0y$ 变形,得:$y=-\frac {x_0}{y_0}x+\frac c{y_0}$ 。想要 $c$ 最大,即要 $\frac c{y_0}$ 最大。

相当于是一条斜率为 $-\frac{x_0}{y_0}$ 的直线,要使纵截距最大,答案一定在上凸壳上,且所选点与答案形成单峰函数。

因此维护上凸壳,在凸壳上二分即可。

$y<0$ 的情况同理,维护下凸壳并二分即可。$y=0$ 的情况只要求 $x$ 最大/小,在两个凸壳上二分都可以。

考虑使用线段树维护一段区间的凸壳。

由于强制在线,因此不能提前求凸壳。但我们可以发现:如果线段树的一个节点没有被全部加入(即没加全),那么这个点一定不会出现在询问区间中。因此我们在线段树上插入,当且仅当 $tot==r$ 时,对该节点建立上凸壳即可。

时间复杂度 $O(n\log^2 n)$

#include <cstdio>
#include <vector>
#include <algorithm>
#define lson l , mid , x << 1
#define rson mid + 1 , r , x << 1 | 1
using namespace std;
typedef long long ll;
struct point
{
ll x , y;
point() {}
point(ll a , ll b) {x = a , y = b;}
ll operator*(const point &a)const {return x * a.x + y * a.y;}
ll operator^(const point &a)const {return x * a.y - y * a.x;}
point operator-(const point &a)const {return point(x - a.x , y - a.y);}
bool operator<(const point &a)const {return x == a.x ? y < a.y : x < a.x;}
};
vector<point> t[1600010] , v1[1600010] , v2[1600010];
char opt[5] , str[5];
void insert(int p , point o , int l , int r , int x)
{
t[x].push_back(o);
if(p == r)
{
sort(t[x].begin() , t[x].end());
int i;
for(i = 0 ; i <= r - l ; i ++ )
{
while(v1[x].size() > 1 && ((t[x][i] - v1[x][v1[x].size() - 1]) ^ (v1[x][v1[x].size() - 2] - v1[x][v1[x].size() - 1])) >= 0) v1[x].pop_back();
v1[x].push_back(t[x][i]);
while(v2[x].size() > 1 && ((t[x][i] - v2[x][v2[x].size() - 1]) ^ (v2[x][v2[x].size() - 2] - v2[x][v2[x].size() - 1])) <= 0) v2[x].pop_back();
v2[x].push_back(t[x][i]);
}
}
if(l == r) return;
int mid = (l + r) >> 1;
if(p <= mid) insert(p , o , lson);
else insert(p , o , rson);
}
ll query1(int b , int e , point o , int l , int r , int x)
{
if(b <= l && r <= e)
{
int L = 1 , R = v1[x].size() - 1 , Mid , Ans = 0;
while(L <= R)
{
Mid = (L + R) >> 1;
if(o * v1[x][Mid] > o * v1[x][Mid - 1]) Ans = Mid , L = Mid + 1;
else R = Mid - 1;
}
return o * v1[x][Ans];
}
int mid = (l + r) >> 1;
ll ans = -1ll << 62;
if(b <= mid) ans = max(ans , query1(b , e , o , lson));
if(e > mid) ans = max(ans , query1(b , e , o , rson));
return ans;
}
ll query2(int b , int e , point o , int l , int r , int x)
{
if(b <= l && r <= e)
{
int L = 1 , R = v2[x].size() - 1 , Mid , Ans = 0;
while(L <= R)
{
Mid = (L + R) >> 1;
if(o * v2[x][Mid] > o * v2[x][Mid - 1]) Ans = Mid , L = Mid + 1;
else R = Mid - 1;
}
return o * v2[x][Ans];
}
int mid = (l + r) >> 1;
ll ans = -1ll << 62;
if(b <= mid) ans = max(ans , query2(b , e , o , lson));
if(e > mid) ans = max(ans , query2(b , e , o , rson));
return ans;
}
int main()
{
int n , tot = 0 , i , l , r;
ll last = 0;
point o;
scanf("%d%s" , &n , opt);
for(i = 1 ; i <= n ; i ++ )
{
scanf("%s%lld%lld" , str , &o.x , &o.y) , o.x ^= last , o.y ^= last;
if(str[0] == 'A') insert(++tot , o , 1 , n , 1);
else
{
scanf("%d%d" , &l , &r) , l ^= last , r ^= last;
if(o.y > 0) last = query1(l , r , o , 1 , n , 1);
else last = query2(l , r , o , 1 , n , 1);
printf("%lld\n" , last) , last &= 0x7fffffff;
if(opt[0] == 'E') last = 0;
}
}
return 0;
}

【bzoj3533】[Sdoi2014]向量集 线段树+STL-vector维护凸包的更多相关文章

  1. BZOJ3533:[SDOI2014]向量集(线段树,三分,凸包)

    Description 维护一个向量集合,在线支持以下操作: "A x y (|x|,|y| < =10^8)":加入向量(x,y); " Q x y l r (| ...

  2. BZOJ 3533: [Sdoi2014]向量集( 线段树 + 三分 )

    答案一定是在凸壳上的(y>0上凸壳, y<0下凸壳). 线段树维护, 至多N次询问, 每次询问影响O(logN)数量级的线段树结点, 每个结点O(logN)暴力建凸壳, 然后O(logN) ...

  3. bzoj 3533: [Sdoi2014]向量集 线段树维护凸包

    题目大意: http://www.lydsy.com/JudgeOnline/problem.php?id=3533 题解: 首先我们把这些向量都平移到原点.这样我们就发现: 对于每次询问所得到的an ...

  4. bzoj 3533 [Sdoi2014]向量集 线段树+凸包+三分(+动态开数组) 好题

    题目大意 维护一个向量集合,在线支持以下操作: "A x y (|x|,|y| < =10^8)":加入向量(x,y); "Q x y l r (|x|,|y| & ...

  5. BZOJ3533 [Sdoi2014]向量集 【线段树 + 凸包 + 三分】

    题目链接 BZOJ3533 题解 我们设询问的向量为\((x_0,y_0)\),参与乘积的向量为\((x,y)\) 则有 \[ \begin{aligned} ans &= x_0x + y_ ...

  6. [SDOI2014][BZOJ3533] 向量集 [线段树+凸包]

    题面 BZOJ传送门 思路 首先当然是推式子 对于一个询问点$(x_0,y_0$和给定向量$(x_1,y_1)$来说,点积这么表达: $A=x_0x_1+y_0y_1$ 首先肯定是考虑大小关系:$x_ ...

  7. bzoj3533: [Sdoi2014]向量集

    Description 维护一个向量集合,在线支持以下操作:"A x y (|x|,|y| < =10^8)":加入向量(x,y);" Q x y l r (|x| ...

  8. 【BZOJ4311】向量(线段树分治,斜率优化)

    [BZOJ4311]向量(线段树分治,斜率优化) 题面 BZOJ 题解 先考虑对于给定的向量集,如何求解和当前向量的最大内积. 设当前向量\((x,y)\),有两个不同的向量\((u1,v1),(u2 ...

  9. UVA1455 - Kingdom(并查集 + 线段树)

    UVA1455 - Kingdom(并查集 + 线段树) 题目链接 题目大意:一个平面内,给你n个整数点,两种类型的操作:road x y 把city x 和city y连接起来,line fnum ...

随机推荐

  1. 修改cmd为utf-8编码:

    1.组合键WIN+R键,组合键后就会弹出窗口,然后输入CMD,回车: 2.要修改成UTF8编码,输入命令CHCP 65001(设置为65001): 3.鼠标放在命令窗口的标题部分右键,在弹出的右键菜单 ...

  2. 可以用来求急的开源项目es5-shim

    最近发现一个JS库,无法在IE8中运行.比较纠结,又不想另找个库代替它,还好发现了这玩意.         它的作用:就给傻逼浏览器做兼容性,使得傻逼浏览器可以支持一些 es5 的 api       ...

  3. struts2框架实例

    一,Struts2框架介绍 它是一个View框架,对Servle进行了封装,使用核心过滤器对servlet进行了解耦,可以自动封装数据 核心是结果视图导航 二,程序实例 1.导入框架依赖包 2.注册框 ...

  4. 无旋treap的区间操作实现

    最近真的不爽...一道维修数列就做了我1上午+下午1h+1晚上+晚上1h+上午2h... 一道不错的自虐题... 由于这一片主要讲思想,代码我放这里了 不会无旋treap的童鞋可以进这里 呵呵... ...

  5. 单元测试或main方法 进行单元测试时 idea检查其他类的语法是否正确的去除方法

    在进行单元测试或者main方法时,在 运行/调试 设置中设置想要使用的测试单位的 before launch 即可

  6. R的数据库访问-MySQL

    目录 1 RMySQL 2 环境与安装 3 建立通信 1 RMySQL R作为一款数据分析的工具,,而MySQL是一款常用的开源关系型数据库软件,非常适用于中小型的数据存储,当二者相互结合时才能爆发出 ...

  7. Java EE JSP内置对象及表达式语言

    一.JSP内置对象 JSP根据Servlet API规范提供了一些内置对象,开发者不用事先声明就可使用标准变量来访问这些对象. JSP提供了9种内置对象: (一).request 简述: JSP编程中 ...

  8. Halcon如何保存仿射变换矩阵

    这里我们通过序列化来实现的,如下图,写到硬盘的HomMat2D_1内容和从硬盘里HomMat2D_2读出的内容一致,源代码在图片下方. Halcon源代码: hom_mat2d_identity (H ...

  9. eFPGA与FPGA SoC,谁将引领下一代可编程硬件之潮流?|半导体行业观察

    eFPGA:冉冉升起的新星 eFPGA即嵌入式FPGA(embedded FPGA),是近期兴起的新型电路IP. 随着摩尔定律越来越接近瓶颈,制造ASIC芯片的成本越来越高.因此,设计者会希望ASIC ...

  10. php命名空间学习笔记。

    为什么要用命名空间? 在PHP中,命名空间用来解决在编写类库或应用程序时创建可重用的代码如类或函数时碰到的两类问题: 用户编写的代码 与  PHP内部的类/函数/常量或第三方类/函数/常量之间的名字冲 ...