poj 2991 起重机
地址 http://poj.org/problem?id=2991




题解
本来以为这是一个简单的线段树模板 不料始终不太明白线段树如何记录转动角度后的各个线段端的XY值
学习了网络上的一些博客题解 感觉似是而非 谈到复数 角度 向量等,有点不太好理解
现在这里将自己的理解记录如下
如图

1 预备知识
使用线段树记录的内容如下 指示某段线段的组合 以第一条线段为垂直 最后的线段的端点的X Y值
途中1~2 线段 和3~5线段 就是线段树节点1~5的子节点 那么线段树节点1~5 就记录1~5结合后的X Y 值以及两个子节点结合的角度值
由于3~5线段的XY 是以自己的第一条线段为垂直起点为0 0 计算出来的X Y
那么在于1~2线段合并的时候 并不是简单的将两子节点的X Y相加即可得到1~5线段的XY 而是要加入旋转了相对角度 该角度由记录1~5线段的线段树节点记录
1~2线段部分的X Y值 旋转相对角度的公式推导如下
https://blog.csdn.net/hjq376247328/article/details/45113563
其实也就是
xNew = x * cosB - y * sinB
yNew = x * sinB + y * cosB
再来和 1~2线段的X Y相加即可得到1~5线段的X Y,并将该两子节点的相对角度记录在父节点中
预备知识讲完
2 解答步骤如下
一 建造线段树 build(1,1,n); 由于每条线段都是垂直连接 所以X 均为0 相对角度全部为0
void build(int k,int l,int r) {
angT[k]=x[k]=0.0;
if(r==l) y[k]=L[l];
else {
int lson=k*, rson=k*+;
int m=(l+r)/;
build(lson,l,m);
build(rson,m+,r);
y[k]=y[lson]+y[rson];
}
}
二 某段线段转动角度后
题意输入的角度是si和si+1逆时针角度而不是旋转的角度,而是需要转到的结果角度, 所以我们需要进行转换 。所以使用了pre数组记录每段线段与相邻线段的逆时针间隔角度,这样接收到题意输入角度a后
a-pre[s] 就是实际要转动的角度 而且需要更新pre[s] 以便下次计算
change(s,ang-pre[s],,,n);
pre[s]=ang; // 要求改变为a度 考虑之前已改变过
chang()代码就是批量更新需要转换的角度和X Y
只有旋转的起点线段在当前线段树节点的左子节点 我们才更新当前线段树节点的角度记录
如图

假设节点4 向3旋转90度
那么合并1 2的时候无更新
合并3 4的时候 3~4节点的角度要在原来记录上再旋转90
合并1~2 3~4为1~4的时候无需更新角度 因为 3~4 已经旋转 与1 ~2的相对角度 并没有变化
同样 5~8节点流程中无变化
但是合并1~4 5~8节点的时候 角度需要旋转90
整个流程下来 4 5~8 角度均旋转更新1次 符合题目的集合意义
最后返回1~8节点记录的X Y即可
代码如下
#include <iostream>
#include <vector>
#include <math.h>
#include <stdio.h>
using namespace std;
const double PI = acos(-1.0);
const int N = ; int n, c, L[N];
double pre[N]; double angT[N << ];
double x[N << ], y[N << ]; void build(int k, int l, int r) {
angT[k] = x[k] = 0.0;
if (r == l) y[k] = L[l];
else {
int lson = k * , rson = k * + ;
int m = (l + r) / ;
build(lson, l, m);
build(rson, m + , r);
y[k] = y[lson] + y[rson];
}
} void change(int s, double ang, int k, int l, int r) {
if (s < l || l == r) return; // 操作位置不在范围内 或 区间长度为1 不作处理
else if (s <= r) {
int lson = k * , rson = k * + ;
int m = (l + r) / ;
change(s, ang, lson, l, m);
change(s, ang, rson, m + , r); // 先处理左右子区间
if (s <= m) angT[k] += ang; // 操作位置位于区间的左子区间内 可根据左右子区间的向量更新 double sina = sin(angT[k]), cosa = cos(angT[k]);
x[k] = x[lson] + (x[rson] * cosa - y[rson] * sina);
y[k] = y[lson] + (x[rson] * sina + y[rson] * cosa);
}
} /*
Sample Input 2 1
10 5
1 90 3 2
5 5 5
1 270
2 90
Sample Output 5.00 10.00 -10.00 5.00
-5.00 10.00
*/ int main()
{
while (~scanf("%d%d", &n, &c)) {
for (int i = ; i <= n; i++) {
scanf("%d", &L[i]);
pre[i] = PI;
}
build(, , n);
while (c--) {
int s, a; scanf("%d%d", &s, &a);
double ang = (double)a / 180.0*PI;
change(s, ang - pre[s], , , n);
pre[s] = ang;
printf("%.2f %.2f\n", x[], y[]);
}
printf("\n");
} return ;
}
poj 2991 起重机的更多相关文章
- POJ 2991 Crane(线段树+计算几何)
POJ 2991 Crane 题目链接 题意:给定一个垂直的挖掘机臂.有n段,如今每次操作能够旋转一个位置,把[s, s + 1]专程a度,每次旋转后要输出第n个位置的坐标 思路:线段树.把每一段当成 ...
- AC日记——Crane poj 2991
POJ - 2991 思路: 向量旋转: 代码: #include <cmath> #include <cstdio> #include <cstring> #in ...
- [poj 2991]Crane[线段树表示向量之和,而非数量]
题意: 起重机的机械臂, 由n段组成, 对某一些连接点进行旋转, 询问每次操作后的末端坐标. 思路: 由于旋转会影响到该点之后所有线段的角度, 因此容易想到用线段树记录角度, 成段更新. (但是不是每 ...
- POJ 2991–Crane【线段树+几何】
题意: 把手臂都各自看成一个向量,则机械手的位置正好是手臂向量之和.旋转某个关节,其实就是把关节到机械手之间的手臂向量统统旋转. 由于手臂很多,要每个向量做相同的旋转操作很费时间.这时就可以想到用线段 ...
- 线段树 poj 2991
我们只要把这些向量求和,最终所指的位置就是终点,因此我们只要维护好向量的区间和就可以了.对于第二个问题,我们可以用一个数组degree[i]表示第i个向量和第i-1一个向量当前的夹角,这样就有了当前的 ...
- POJ 2991 Crane
线段树+计算几何,区间更新,区间求和,向量旋转. /* *********************************************** Author :Zhou Zhentao Ema ...
- (中等) POJ 2991 Crane , 几何+线段树。
Description ACM has bought a new crane (crane -- jeřáb) . The crane consists of n segments of variou ...
- Crane (POJ 2991)
//线段树 延迟标签 // #include <bits/stdc++.h> using namespace std; const int maxn=1e4+5; double x[max ...
- POJ 2991 Crane(线段树)
Crane Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 7687 Accepted: 2075 Special J ...
随机推荐
- 设计模式C++描述----01.单例(Singleton)模式
一.概念 单例模式:其意图是保证一个类仅有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享. class CSingleton { //公有的静态方法,来获取该实例 public: s ...
- 《Effective Java》 读书笔记(二) 在构造参数过多的时候优先考虑使用构造器
刚开始看见这个标题的时候,我想到了python可以选择初始化参数的语法,C++.C#能有默认参数. 为什么Java什么都没有~~ 好吧,我们是使用构造器来实现它. 1.当一个类的构造函数需要很多构造函 ...
- AtCoder Grand Contest 038E - Gachapon
\(\bf Description\) 一个 \(0\) 到 \(n-1\) 的随机数生成器,生成 \(i\) 的概率是 \(A_i/S\) ,其中 \(S=\sum_{i=0}^{n} A_i\) ...
- Weed:线段树
观察复杂度,是log级别以下回答询问的. O(1)?逗我kx呢? 自然而然地想到线段树. 学长讲的原题啊考场上还不会打. 线段树上的每个节点都表示一个操作区间. 线段树上维护的权值有3个:这个子区间一 ...
- svg路径蒙版动画
svg路径蒙版动画,是比较实用的一种动画效果,能够绘制如下图所示的动画. 接下来细说这样的动画是如何做成的: 1.准备工作 2.SVG路径动画 3.SVG路径蒙版动画 4.复杂图形的编辑技巧 1.准备 ...
- ios jquery css('left')无法读取属性解决的方法
ios jquery css('left')无法读取属性解决的方法 <pre>$(this).position().left因为display:none状态下是读取不了 $(this).o ...
- linux内核崩溃之kdump机制
kdump相关概念 standard(production) kernel 生产内核 ,是指我们正在使用的kernel. Crash(capture)kernel 捕 ...
- JavaScript中BOM与DOM的使用
BOM: 概念:Browser Object Model 浏览器对象模型 将浏览器的各个组成部分封装成对象. 组成: Window:窗口对象 Navigator:浏览器对象 Screen:显示器屏幕对 ...
- Junit使用方法
Junit使用方法 步骤: 定义 一个测试类(测试用例) 建议 测试类名:被测试类名+Test 包名:xxx.xxx.xxx.test 测试方法 建议: 方法名:test测试的方法名 返回值: voi ...
- 深入理解计算机系统 第八章 异常控制流 part1
本章主旨 第八章的目的是阐述清楚应用程序是如何与操作系统交互的(之前章节的学习是阐述应用程序是如何与硬件交互的) 异常控制流 异常控制流,即 ECF(exceptional contril flow) ...