zoukankan      html  css  js  c++  java
  • B3680 吊打xxx 物理???

    看到一道很有意思的题,这个题简直有毒,是一道物理题。。。好像得用模拟退火。。。但显然我太弱了不会模拟退火,只能用正交分解暴力。。。

    每次沿着力的方向走一定的距离,假如转头了,则走的步长就减小一点。

    不过这里有一个坑,就是假如每次二分,就会错。。。0.7或0.9就可以。有可能走过了回不来了吧。

    题面:

    Description
    
    gty又虐了一场比赛,被虐的蒟蒻们决定吊打gty。gty见大势不好机智的分出了n个分身,但还是被人多势众的蒟蒻抓住了。蒟蒻们将
    n个gty吊在n根绳子上,每根绳子穿过天台的一个洞。这n根绳子有一个公共的绳结x。吊好gty后蒟蒻们发现由于每个gty重力不同,绳
    结x在移动。蒟蒻wangxz脑洞大开的决定计算出x最后停留处的坐标,由于他太弱了决定向你求助。
    不计摩擦,不计能量损失,由于gty足够矮所以不会掉到地上。
    Input
    
    输入第一行为一个正整数n(1<=n<=10000),表示gty的数目。
    接下来n行,每行三个整数xi,yi,wi,表示第i个gty的横坐标,纵坐标和重力。
    对于20%的数据,gty排列成一条直线。
    对于50%的数据,1<=n<=1000。
    对于100%的数据,1<=n<=10000,-100000<=xi,yi<=100000
    Output
    
    输出1行两个浮点数(保留到小数点后3位),表示最终x的横、纵坐标。
    Sample Input
    3
    
    0 0 1
    
    0 2 1
    
    1 1 1
    
    Sample Output
    0.577 1.000

    代码:

    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<cmath>
    #include<cstring>
    using namespace std;
    #define duke(i,a,n) for(int i = a;i <= n;i++)
    #define lv(i,a,n) for(int i = a;i >= n;i--)
    #define clean(a) memset(a,0,sizeof(a))
    const int INF = 1 << 30;
    typedef long long ll;
    typedef double db;
    template <class T>
    void read(T &x)
    {
        char c;
        bool op = 0;
        while(c = getchar(), c < '0' || c > '9')
            if(c == '-') op = 1;
        x = c - '0';
        while(c = getchar(), c >= '0' && c <= '9')
            x = x * 10 + c - '0';
        if(op) x = -x;
    }
    template <class T>
    void write(T x)
    {
        if(x < 0) putchar('-'), x = -x;
        if(x >= 10) write(x / 10);
        putchar('0' + x % 10);
    }
    struct node
    {
        db x,y,power;
    }p[50000];
    int n;
    bool bx = true,by = true;
    db x,y;
    void solve(db move)
    {
        db gx = 0,len = 0,gy = 0;
        duke(i,1,n)
        {
            len = sqrt((x - p[i].x) * (x - p[i].x) + (y - p[i].y) * (y - p[i].y));
            if(len == 0)
            continue;
            gx += p[i].power * (p[i].x - x) / len;
            gy += p[i].power * (p[i].y - y) / len;
        }
        len = sqrt(gx * gx + gy * gy);
        x += move * gx / len;
        y += move * gy / len; 
    }
    int main()
    {
        read(n);
        duke(i,1,n)
        {
            scanf("%lf%lf%lf",&p[i].x,&p[i].y,&p[i].power);
        }
        db move = 5000,tx,ty;
        while(1)
        {
            tx = x;
            ty = y;
            solve(move);
            if(abs(tx - x) < 0.000001 && abs(ty - y) < 0.000001)
                break;
            if(bx != (x > tx) || by != (y > ty))
            {
                bx = !(x > tx);
                by = !(y > ty);
                move = move * 0.9;
            }
        }
        printf("%.3lf %.3lf
    ",x,y);
        return 0;
    }
  • 相关阅读:
    c语言,动态数组
    利用Word来发布博客到博客园(onenote类似)
    c语言,volatile
    c语言,变长数组
    C语言,sprintf与sscanf函数[总结]
    c语言,数组和指针
    Linux的notifier机制的应用
    Linux进程上下文切换过程context_switch详解--Linux进程的管理与调度(二十一)
    内核线程的进程描述符task_struct中的mm和active_mm
    Linux用户抢占和内核抢占详解(概念, 实现和触发时机)--Linux进程的管理与调度(二十)
  • 原文地址:https://www.cnblogs.com/DukeLv/p/9495789.html
Copyright © 2011-2022 走看看