HDU 4611 Balls Rearrangement 数学 循环节优化

7 篇文章 0 订阅

题目链接点这儿

就是给一个n,a,b求这明显是会很蛋疼的式子。。。

首先。。。肯定是试直接模拟。。。不意外TLE。。。

然后,可以发现它是由循环节的,那就是当lcm(a,b)<n时,有循环节。兴冲冲的写了一个交,然后仍然TLE。。。想起来当n很大但是lcm更大的情况还是没有优化。。。当时都要心灰意冷了。。。

纸上模拟了一个n=13,a=3,b=5的情况。。。

发现了一个东西。。。

球上标的数字012345678910111213
初始位置1201201201201
新位置01234012340123
差值绝对值00033211141122
发现了一个东西。。。就是值有一个一个的重复小段。。。然后想起来。。。在遇到下一个0之前,两个数列都是公差为1的等差数列,它们同一个项的差值当然是恒定的。。。所以。。。在循环节内还能继续优化。。。

然后就有了下面的代码

#include <vector>
#include <list>
#include <map>
#include <set>
#include <deque>
#include <queue>
#include <stack>
#include <bitset>
#include <algorithm>
#include <functional>
#include <numeric>
#include <utility>
#include <sstream>
#include <iostream>
#include <iomanip>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <cctype>
#include <string>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <ctime>
#include <climits>
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)>(b)?(b):(a))
#define rep(i,initial_n,end_n) for(ll (i)=(initial_n);(i)<(end_n);i++)
#define repp(i,initial_n,end_n) for(ll (i)=(initial_n);(i)<=(end_n);(i)++)
#define reep(i,initial_n,end_n) for((i)=(initial_n);(i)<(end_n);i++)
#define reepp(i,initial_n,end_n) for((i)=(initial_n);(i)<=(end_n);(i)++)
#define eps 1.0e-9
#define MAX_N 1010

using namespace std;
typedef pair<int, int> pii;
typedef pair<double, double> pdd;
typedef __int64 ll;
typedef unsigned __int64 ull;

ll gcd(ll a, ll b) {
    return b == 0 ? a : gcd(b, a%b);
}

ll val_loop(ll n, ll a, ll b);

int main() {
    ll nn;
    scanf("%I64d", &nn);
    while(nn--) {
        ll n, a, b;
        scanf("%I64d%I64d%I64d", &n, &a, &b);
        if(a < b) swap(a,b);
        ll g = gcd(a,b);
        ll lcm = (ll)(a*b/g);
        ll quo = n / lcm, mod = n % lcm;
        if(a == b) { puts("0"); continue; }
        if(n <= lcm) printf("%I64d\n", val_loop(n, a, b));
        else
            printf("%I64d\n", quo * val_loop(lcm, a, b) + val_loop(mod, a, b));
            //printf("1 = %lld 2 = %lld\n", quo * val_loop(lcm, a, b) , val_loop(mod, a, b));
    }
    return 0;
}

ll val_loop(ll n, ll a, ll b) {
    ll i = 0, ta = 0, tb = 0;
    ll ans = 0, p = 0;
    while(i < n) {
        //printf("ta %d tb %d i %d p %lld\n", ta, tb, i, p);
       // printf("ta + a = %d tb + b = %d\n", ta + a, tb + b);
        if(ta + a >= n && tb + b >= n) { ans += p * (n-i); i = n; break; }
        else if(ta + a < tb + b) {
            ans += p * (ta + a - i);
            ta += a;
            i = ta;
            p = i - tb;
        }
        else if(ta + a > ta + b) {
            ans += p * (tb + b - i);
            tb += b;
            i = tb;
            p = i - ta;
        }
        else {
            ans += p * (ta + a - i);
            i = ta;
            ta += a, tb += b;
            p = 0;
        }
    }
    return ans;
}




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值