HDU 4506

腾讯编程马拉松第二场比赛的第二题。

真坑爹!昨天不知道怎么搞的就是wa,今天早上重新敲了一下代码还是wa。

最后,就觉得是不是格式出问题了,就把全部的long long改成了_int64结果还真他妈的过了。

但是,我朋友用long long写的代码却过了。真心坑爹,不知道是怎么回事。

 

总结一下 主要的“难点”,就是那个分治法的吧。(今早才知道,原来这个叫分治法)

就是求k的t次。  分成 k的t/2次*k的t/2次,当然要判断t是不是偶数,然后判断是否需要再乘上本身。分治下去最后t一定会变成1.(证明略)

然后其余的特殊情况,自己考虑一下。输出的时候变化的话,周期就是n。然后,就简单了。

下面是代码:

#include<iostream>
#include<cstdio>
using namespace std;

#define mm 1000000007
_int64 cal(int t, long long k)
{
    _int64 x = 1;
    if(!t) return 1;
    if(!k) return 0;
    if(t == 1) return k;
    while(t)
    {
        if(t & 1)
            x = x * k % mm;
        t >>= 1;
        k = k * k % mm;
    }
    return x;
}

int main()
{
    int T;
    int n, t;
    _int64 k, a[10005];
    scanf("%d", &T);
    while(T--)
    {
        scanf("%d%d%I64d", &n, &t, &k);
        for(int i = 0; i < n; ++i)
            scanf("%I64d", a + i);
        k = cal(t, k);
        t %= n;
        int y, i;
        for(i = 0; i < n - 1; ++i)
        {
            if(i < t)
                y = i + n - t;
            else
                y = i - t;
            printf("%I64d ", a[y] * k % mm);
        }
        if(i < t) y = i + n - t;
        else y = i - t;
        printf("%I64d\n", a[y] * k % mm);
    }
    return 0;
}


 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值