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 ... 
随机推荐
- 再整理:Visual Studio Code(vscode)下的通用C语言环境搭建
			版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://www.cnblogs.com/czlhxm/p/11794743.ht ... 
- CVE-2019-0708复现之旅
			CVE-2019-0708 0x00 前言: CVE-2019-0708漏洞已经出来三四个月了对应的poc也出现了很久 ,exp 还是没动静前段时间出了个蓝屏伪exp 不过肯定那些大厂手里早已经了有了 ... 
- C++学习笔记1_ 指针.引用
			1.引用的本质struct typeA{ int &a;}struct typeB{ int *a;}int main(void){ cout<<sizeof(struct typ ... 
- 2019.NET Conf China(中国.NET开发者峰会)活动全纪录:.NET技术之崛起,已势不可挡
			一.微软走向开放与.NET Core的诞生 当今时代,气象更新,技术飞速发展. 当今时代,开发者大概是最优秀的群体.每一位开发者,无不奋勇向前,努力追寻时代的步伐,以大无畏的精神迎接挑战,紧跟大时代成 ... 
- IoTClient开发4 - ModBusTcp协议服务端模拟
			前言 上篇我们实现了ModBusTcp协议的客户端读写,可是在很多时候编写业务代码之前是没有现场环境的.总不能在客户现场去写代码,或是蒙着眼睛写然后求神拜佛不出错,又或是在办公室部署一套硬件环境.怎么 ... 
- 2018年7月份JAVA开源软件TOP3
			微信开发 Java SDK Weixin Java Tools 评分: 9.6 介绍: 信开发 Java 开发工具包(SDK),支持包括微信支付.微信开放平台.小程序.企业号/企业微信.公众号(包括服 ... 
- jquery ajax在 IE8/IE9 中无效
			你们是不是也曾经和我以为遇到过这样的情况呢,jquery ajax在 IE8/IE9 中无效获取不到数据呢,经过熬夜找到好的东西和你们分享一下就是jQuery-ajaxTransport-XDomai ... 
- Linux下编写-makefile-详细教程(跟我一起写-Makefile-Markdown整理版)
			目录 概述 关于程序的编译和链接 Makefile 介绍 Makefile的规则 一个演示例子 make是怎样工作的 makefile中使用变量 让make自己主动推导 另类风格的makefile 清 ... 
- [RAM] FPGA的学习笔记——RAM
			1.RAM——随机存取存储器, 分为SRAM和DRAM. SRAM:存和取得速度快,操作简单.然而,成本高,很难做到很大.FPGA的片内存储器,就是一种SRAM,用来存放程序,以及程序执行过程中,产生 ... 
- linux服务器MySQL数据从磁盘拷贝以及恢复
			偶有感触:遇到这个问题,经过一个下午的排查, 终于解决. 故事情节:我的阿里云服务器突然被黑客攻击了,整个系统down了. 找客服,他们排查说usr目录的文件全部丢失.让我重新初始化系统盘.初始化之前 ... 
