zoukankan      html  css  js  c++  java
  • 【27.22%】【poj2991】Crane

    Time Limit: 2000MS   Memory Limit: 65536K
    Total Submissions: 5772   Accepted: 1571   Special Judge

    Description

    ACM has bought a new crane (crane -- jeřáb) . The crane consists of n segments of various lengths, connected by flexible joints. The end of the i-th segment is joined to the beginning of the i + 1-th one, for 1 ≤ i < n. The beginning of the first segment is fixed at point with coordinates (0, 0) and its end at point with coordinates (0, w), where w is the length of the first segment. All of the segments lie always in one plane, and the joints allow arbitrary rotation in that plane. After series of unpleasant accidents, it was decided that software that controls the crane must contain a piece of code that constantly checks the position of the end of crane, and stops the crane if a collision should happen. 

    Your task is to write a part of this software that determines the position of the end of the n-th segment after each command. The state of the crane is determined by the angles between consecutive segments. Initially, all of the angles are straight, i.e., 180o. The operator issues commands that change the angle in exactly one joint. 

    Input

    The input consists of several instances, separated by single empty lines. 

    The first line of each instance consists of two integers 1 ≤ n ≤10 000 and c 0 separated by a single space -- the number of segments of the crane and the number of commands. The second line consists of n integers l1,..., ln (1 li 100) separated by single spaces. The length of the i-th segment of the crane is li. The following c lines specify the commands of the operator. Each line describing the command consists of two integers s and a (1 ≤ s < n, 0 ≤ a ≤ 359) separated by a single space -- the order to change the angle between the s-th and the s + 1-th segment to a degrees (the angle is measured counterclockwise from the s-th to the s + 1-th segment).

    Output

    The output for each instance consists of c lines. The i-th of the lines consists of two rational numbers x and y separated by a single space -- the coordinates of the end of the n-th segment after the i-th command, rounded to two digits after the decimal point. 

    The outputs for each two consecutive instances must be separated by a single empty line.

    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

    Source

    【题解】

    题意就是说把n条线段从下往上竖直堆积。然后有时候会给你几个操作。要求把第s个的上面那个棍子(s+1)转到一定的角度。(以第s个棍子作为衡量角度的参照物);

    这题我们不要管它转到多少度。而只需要知道它转的这个角度和前一个角度的差是多少就可以了。

    每次遇到一个角度就减去180度。

    然后用减去180度的角和之前它的角相减。就能得到角度差了。为什么减去180度等下会讲到。

    然后我们把这n条线段看成一个个向量。那么从头加到尾。就能够得到最上面那条线段的顶点坐标了。(向量加法);

    这样看来。我们只要维护这n条线段的向量和就可以了。

    然后其中一个向量m它的角度转了x度,则m+1,m+2..n它们的向量也都会转x度。

    而一个向量砖了x度之后。我们是能够求出它新的向量坐标形式的。

    x' = cosa*x-sina*y;

    y' = sina*x+cosa*y;

    (a是旋转的角度);

    推导过程看下面.

    推导的基础就是一条线段它转了之后,长度是不会变的!



    然后注意这是逆时针转的情况。

    所以我们一开始减去180度。

    比如样例90,减去180就变成-90了

    然后套用这个公式就变成逆时针旋转(-90)也就是顺时针转90度了。

    而这正是我们所需的。(可以带进样例试试。)

    注意减去180度之后,要重新记录k+1的角度,方便下次对k+1操作。(输入的是k,但是我们实际在对k+1进行操作!);

    然后我们在做线段树的时候对k+1,k+2...n进行相同的旋转操作即可。

    然后维护一下区间的向量和。

    //然后一开始的区间向量和的x坐标都是0,y坐标的和就用前缀和的方法获得。

    具体的过程看代码;

    ps:突然发现如果define里面比如define MAXN 10000+10 ,交上去后会runtime error。。。就是define里面貌似不能写表达式。。以后都用const吧。const 即不会有事。。

    acos(-1.0)==pi

    里面的点号不能漏,不然它会以为你输入的是整形然后会Compile error

    【代码】

    #include <cstdio>
    #include <cmath>
    #define lson begin,m,rt<<1
    #define rson m+1,end,rt<<1|1
    
    const int MAXN = 10000 + 100;
    const double pi = acos(-1.0);
    
    int n, c;
    int a[MAXN];
    double sum[MAXN * 4][2];
    int lazy_tag[MAXN * 4];
    int pre[MAXN];
    
    void build(int begin, int end, int rt)//建树
    {
    	lazy_tag[rt] = 0;
    	sum[rt][1] = a[end] - a[begin - 1];//一开始建树的时候就能处理区间和了
    	sum[rt][0] = 0;
    	if (begin == end)
    		return;
    	int m = (begin + end) >> 1;
    	build(lson);
    	build(rson);
    }
    
    void input_data()
    {
    	a[0] = 0;
    	for (int i = 1; i <= n; i++)
    	{
    		int len;
    		scanf("%d", &len);
    		a[i] = a[i - 1] + len;//获取前缀和,用来处理一开始的初始区间和
    		pre[i] = 0;//这是第i个棍子的初始角度置0就可以。
    	}
    	build(1, n, 1);
    }
    
    double get_rad(int k) //获取角度k的弧度制。
    {
    	double temp = (k * pi)/180;
    	return temp;
    }
    
    void rotation(int rt, double rad) //根据坐标和选择角度更改这个向量
    {
    	//y = sina*x+cosa*y
    	//x = cosa*x-sina*y
    	double dx = sum[rt][0], dy = sum[rt][1];
    	sum[rt][0] = cos(rad)*dx - sin(rad)*dy;
    	sum[rt][1] = sin(rad)*dx + cos(rad)*dy;
    }
    
    void push_up(int rt) 
    {
    	sum[rt][0] = sum[rt << 1][0] + sum[rt << 1 | 1][0];
    	sum[rt][1] = sum[rt << 1][1] + sum[rt << 1 | 1][1];
    }
    
    void push_down(int rt)//传递懒惰标记
    {
    	if (lazy_tag[rt] != 0)
    	{
    		double temp = get_rad(lazy_tag[rt]);
    		rotation(rt << 1, temp);
    		rotation(rt << 1 | 1, temp); //把儿子们旋转
    		lazy_tag[rt << 1] += lazy_tag[rt];
    		lazy_tag[rt << 1 | 1] += lazy_tag[rt];
    		lazy_tag[rt] = 0;
    	}
    }
    
    void updata(int k, int change, int begin, int end, int rt)
    {
    	if (begin == end) 
    	{
    		double hudu = get_rad(change);
    		rotation(begin,hudu);//只有一个点 那就是要旋转的了。
    		return;
    	}
    	push_down(rt);
    	int m = (begin + end) >> 1;
    	if (k <= m + 1) //这个意思是说右儿子全是大于等于k的节点。那么就要全部旋转
    	{
    		double hudu = get_rad(change);
    		if (k <= m)//左区间可能还有部分需要旋转
    			updata(k, change, lson);
    		rotation(rt << 1 | 1, hudu);//因为要全部旋转所以就用懒惰标记。
    		lazy_tag[rt << 1 | 1] += change;
    	}
    	else
    		if (k > m + 1)//k>m+1的话左边就全是小于k的节点了(不用管)。只要处理右边就好
    			updata(k, change, rson);
    	push_up(rt);
    }
    
    void get_ans()
    {
    	for (int i = 1; i <= c; i++)
    	{
    		int number, degree;
    		scanf("%d%d", &number, °ree);
    		number++;//要处理的是number+1
    		int d = degree - 180;
    		int change = d - pre[number];
    		pre[number] = d;
    		updata(number, change,1, n, 1);
    		printf("%.2lf %.2lf
    ", sum[1][0], sum[1][1]);
    	}
    }
    
    int main()
    {
    	//freopen("F:\rush.txt", "r", stdin);
    	//freopen("F:\rush_out.txt", "w", stdout);
    	bool flag = false;
    	while (scanf("%d%d", &n, &c) == 2)
    	{
    		if (flag)
    			printf("
    ");
    			input_data();
    		get_ans();
    		flag = true;
    	}
    	return 0;
    }


  • 相关阅读:
    第9课
    FreeRTOS 定时器组
    FMC—扩展外部 SDRAM
    FreeRTOS 事件标志组
    第8课
    FreeRTOS 系统时钟节拍和时间管理
    第七课 线性表的顺序存储结构
    手把手教你调试Linux C++ 代码(一步到位包含静态库和动态库调试)
    Windows GUI代码与Windows消息问题调试利器
    谈谈数据挖掘和机器学习
  • 原文地址:https://www.cnblogs.com/AWCXV/p/7632266.html
Copyright © 2011-2022 走看看