看到这题讨论版里有说用公式的有说用循环节的,但是个人觉得这两种方法都不靠谱,比赛场上做这种题能直接推出公式需要很强数学功底,而循环节的方法如果循环节比较大就不太好发现了。这种已知通项公式的题还是用矩阵快速幂比较通用,但同是矩阵快速幂,对于这题,也有很大的差异;

注:memmove这个函数可能不太常见,但还是比较好用的,可以用较低的时间复杂度完成数组的拷贝

  • 方法一

    Time Limit Exceeded 2802 1000MS 1348K 1734 B G++
    #include "bits/stdc++.h"
    using namespace std;
    const int MOD = ;
    /*
    {f(n - 1), f(n - 2), n ^ 3, (n - 1) ^ 3, n ^ 2, n, 1}
    乘MAT得到
    {f(n), f(n - 1), (n + 1) ^ 3, n ^ 3, (n + 1) ^ 2, n + 1, 1}
    */
    const int MAT[][] = {
    {, , , -, , , },
    {, , , , , , },
    {, , , , , , },
    {, , , , , , },
    {, , , , , , },
    {, , , , , , },
    {, , , , , , }
    };
    const int TABLE[] = {, , , , , , };
    struct Mat {
    int mat[][];
    Mat() {
    memset(mat, , sizeof(mat));
    }
    friend Mat operator * (Mat n, Mat m) {
    Mat res;
    for (int k = ; k < ; k++)
    for (int i = ; i < ; i++)
    for (int j = ; j < ; j++) {
    res.mat[i][j] += n.mat[i][k] * m.mat[k][j];
    // 因为当n >=0 时 (n + 1) ^ 3 一定大于 n ^ 3,所以要保证结果是正数;
    res.mat[i][j] = (res.mat[i][j] % MOD + MOD) % MOD;
    }
    return res;
    }
    } m;
    Mat mat_pow(Mat n, int k) {
    Mat res;
    for (int i = ; i < ; i++) {
    res.mat[i][i] = ;
    }
    while (k) {
    if (k & ) {
    res = res * n;
    }
    n = n * n;
    k >>= ;
    }
    return res;
    }
    int main() {
    int n;
    while (scanf("%d", &n) && n) {
    if (n == ) {
    puts("");
    continue;
    }
    if (n == ) {
    puts("");
    continue;
    }
    memmove(m.mat, MAT, sizeof(m.mat));
    m = mat_pow(m, n - );
    int res = ;
    for (int i = ; i < ; i++) {
    res = (res + m.mat[][i] * TABLE[i]) % MOD;
    }
    printf("%d\n", res);
    }
    return ;
    }

    这是拿到题直接把递推式拿来套的解法,7层矩阵,超时。于是想到n ^ 3  - (n - 1) ^ 3 = (n - 1) ^ 3 + 3 * (n - 1) ^ 2 + 3 * (n - 1) + 1 - (n - 1) ^ 3;可以抵消(n  - 1) ^ 3;

  • 方法二

    Accepted 2802 889MS 1384K 1462 B G++
    #include "bits/stdc++.h"
    using namespace std;
    const int MOD = ;
    /*
    {f(n - 1), f(n - 2), n ^ 2, n, 1}
    乘MAT得到
    {f(n), f(n - 1), (n + 1) ^ 2, n + 1, 1}
    */
    const int MAT[][] = {
    {, , , , },
    {, , , , },
    {, , , , },
    {, , , , },
    {, , , , }
    };
    const int TABLE[] = {, , , , };
    struct Mat {
    int mat[][];
    Mat() {
    memset(mat, , sizeof(mat));
    }
    friend Mat operator * (Mat n, Mat m) {
    Mat res;
    for (int k = ; k < ; k++)
    for (int i = ; i < ; i++)
    for (int j = ; j < ; j++)
    res.mat[i][j] = (res.mat[i][j] + n.mat[i][k] * m.mat[k][j]) % MOD;
    return res;
    }
    } m;
    Mat mat_pow(Mat n, int k) {
    Mat res;
    for (int i = ; i < ; i++) {
    res.mat[i][i] = ;
    }
    while (k) {
    if (k & ) {
    res = res * n;
    }
    n = n * n;
    k >>= ;
    }
    return res;
    }
    int main() {
    int n;
    while (scanf("%d", &n) && n) {
    if (n == ) {
    puts("");
    continue;
    }
    if (n == ) {
    puts("");
    continue;
    }
    memmove(m.mat, MAT, sizeof(m.mat));
    m = mat_pow(m, n - );
    int res = ;
    for (int i = ; i < ; i++) {
    res = (res + m.mat[][i] * TABLE[i]) % MOD;
    }
    printf("%d\n", res);
    }
    return ;
    }

    降到5层矩阵之后能AC掉这题了,但是这个矩阵还不是最好的。这个代码的效率还是不高。

  • 方法三
    Accepted 2802 483MS 1392K 1551 B G++
    #include "bits/stdc++.h"
    using namespace std;
    const int MOD = ;
    /*
    {f(n - 2), (n + 1) ^ 2, (n + 1), 1}
    乘MAT得到
    {f(n), (n + 3) ^ 2, n + 3, 1}
    */
    const int MAT[][] = {
    {, , , },
    {, , , },
    {, , , },
    {, , , }
    };
    const int TABLE1[] = {, , , };
    const int TABLE2[] = {, , , };
    struct Mat {
    int mat[][];
    Mat() {
    memset(mat, , sizeof(mat));
    }
    friend Mat operator * (Mat n, Mat m) {
    Mat res;
    for (int k = ; k < ; k++)
    for (int i = ; i < ; i++)
    for (int j = ; j < ; j++)
    res.mat[i][j] = (res.mat[i][j] + n.mat[i][k] * m.mat[k][j]) % MOD;
    return res;
    }
    } m;
    Mat mat_pow(Mat n, int k) {
    Mat res;
    for (int i = ; i < ; i++) {
    res.mat[i][i] = ;
    }
    while (k) {
    if (k & ) {
    res = res * n;
    }
    n = n * n;
    k >>= ;
    }
    return res;
    }
    int main() {
    int n;
    while (scanf("%d", &n) && n) {
    if (n == ) {
    puts("");
    continue;
    }
    if (n == ) {
    puts("");
    continue;
    }
    memmove(m.mat, MAT, sizeof(m.mat));
    m = mat_pow(m, n - >> );
    int res = ;
    /*
    假设重定义两个函数,a(n) = f(2 * n - 1), b(n) = f(2 * n);
    if (n & 1) 求得a((n - 1) >> 1)即为 f(n)
    else 求得b((n - 1) >> 1) 即为f(n)
    */
    if (n & ) {
    for (int i = ; i < ; i++) {
    res = (res + m.mat[][i] * TABLE1[i]) % MOD;
    }
    } else {
    for (int i = ; i < ; i++) {
    res = (res + m.mat[][i] * TABLE2[i]) % MOD;
    }
    }
    printf("%d\n", res);
    }
    return ;
    }

    递推式是从f(n)直接到f(n + 2)的,所以相当于把f拆成两个函数分开讨论,只用4层矩阵就够了。对于矩阵快速幂,4层应该是最少的了。如果还要再快,那就只能采取循环节或公式的方式了。

  • 方法四(来自讨论版,代码省略)
    这题的循环节是4018(正好是MOD的两倍,不知道是不是巧合)。
  • 方法五(来自讨论版,代码省略)
    n为奇数 (n+1)(n+1)(2n-1)/4;
    n为偶数 n*n*(2n+3)/4

HDU-2802-F(N)的更多相关文章

  1. HDU 2802 F(N)(简单题,找循环解)

    题目链接 F(N) Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Sub ...

  2. HDU - 2802 F(N) (周期)

    题目链接:HDU 2009-4 Programming Contest 分析:具有一定的周期性——4018处理下就可以A了 Sample Input Sample Output AC代码: #incl ...

  3. HDU 2802 F(N) 数论+打表

    题目大意:f[n]-n^3=f[n-2]-(n-1)^3 (n >=3),f[1]=1,f[2]=7,求f[n]. 题目思路:将n^3移到到等式右边化简的到:f[n]=f[n-2]+3n*(n- ...

  4. 数学--数论--HDU 2802 F(N) 公式推导或矩阵快速幂

    Giving the N, can you tell me the answer of F(N)? Input Each test case contains a single integer N(1 ...

  5. 【数位DP】 HDU 4734 F(x)

    原题直通车:HDU 4734 F(x) 题意:F(x) = An * 2n-1 + An-1 * 2n-2 + ... + A2 * 2 + A1 * 1, 求0.....B中F[x]<=F[A ...

  6. HDU 4734 F(x) 2013 ACM/ICPC 成都网络赛

    传送门:http://acm.hdu.edu.cn/showproblem.php?pid=4734 数位DP. 用dp[i][j][k] 表示第i位用j时f(x)=k的时候的个数,然后需要预处理下小 ...

  7. HDU 4734 - F(x) - [数位DP][memset优化]

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4734 Time Limit: 1000/500 MS (Java/Others) Memory Lim ...

  8. HDU 1937 F - Finding Seats 枚举

    F - Finding Seats Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u S ...

  9. hdu 4734 F(x)(数位dp+优化)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4734 题意:我们定义十进制数x的权值为f(x) = a(n)*2^(n-1)+a(n-1)*2(n-2 ...

  10. E - Tunnel Warfare HDU - 1540 F - Hotel G - 约会安排 HDU - 4553 区间合并

    E - Tunnel Warfare HDU - 1540 对这个题目的思考:首先我们已经意识到这个是一个线段树,要利用线段树来解决问题,但是怎么解决呢,这个摧毁和重建的操作都很简单,但是这个查询怎么 ...

随机推荐

  1. 3.react 基础 - JSX 语法

    1.最基础的 JSX 语法 普通javaScript中 引入 标签 let html = '<h1>hello</h1>'; jsx语法 let JSX_html = < ...

  2. 86.QuerySet API常用的方法详解:get方法

    get方法的查询条件只能有一条数据满足,如果匹配到多条数据都满足,就会报错:如果没有匹配到满足条件的数据,也会报错. 示例代码如下: from django.http import HttpRespo ...

  3. JavaScript sort()方法总结

    sort() 方法用于对数组的元素进行排序. 语法:arrayObject.sort(sortby):参数sortby可选.规定排序顺序.必须是函数. 注:如果调用该方法时没有使用参数,将按字母顺序对 ...

  4. PTA 自测-4 Have Fun with Numbers

    #include<iostream> #include<string> #include<cstring> #include<vector> using ...

  5. ZJNU 2235 - EnDlEsS ChAsE

    因为速度值保证各不相同 所以n只战斗人形会出现 n! 种不同情况 可以用不同id表示不同人形的速度 比如1 2 3三只人形 他们可能的排列有 1 2 3 1 3 2 2 1 3 2 3 1 3 1 2 ...

  6. Tooltips

    #include<windows.h> #include<Commctrl.h> #include"resource.h" #pragma comment( ...

  7. PHP 5.3 新特性

    1 PHP 5.3中的新特性 1.1 支持命名空间 (Namespace) 毫无疑问,命名空间是PHP5.3所带来的最重要的新特性. 在PHP5.3中,则只需要指定不同的命名空间即可,命名空间的分隔符 ...

  8. Python——气象数据分析

    将对意大利北部沿海地区的气象数据进行分析与可视化.我们在实验过程中先会运用 Python 中 matplotlib 库的对数据进行图表化处理,然后调用 scikit-learn 库当中的的 SVM 库 ...

  9. 函数(Python)

    函数是什么? 计算机的函数,是一个固定的一个程序段,或称其为一个子程序,它在可以实现固定运算功能的同时,还带有一个入口和一个出口,所谓的入口,就是函数所带的各个参数,我们可以通过这个入口,把函数的参数 ...

  10. Ansible部署rsync、nfs及sersync

    rsync nfs sersync httpd环境: 角色 外网IP(NAT) 内网IP(LAN) 主机名 Rsync服务端 eth0:10.0.1.51 eth1:172.16.1.51 backu ...