【链接】点击打开链接


【题意】


询问n个点的完全k叉树,所有子树节点个数的异或总和为多少。


【题解】


考虑如下的一棵k=3叉树,假设这棵树恰好有n个节点.

因为满的k叉树,第i层的节点个数为k^(i-1);

则我们找到最大的d;

使得

k^0+k^1+..+k^(d-1) <=n

自此,我们会发现,整棵树可以分成若干个树的深度大小d和d-1满k叉树

如上图,d=3,两个圈起来的,就是一个深度为3的满k叉树和一个深度为2的满k叉树

当然,还可能会有一个子树,它不是满的k叉树

如上图中没有被圈出来的部分.

深度大小为d的满k叉树的个数,可以用rest =
n-(k^0+k^1+..+k^(d-1))算出来;

它的个数就是rest/num[d],(其中num[d]表示第d层的满k叉树的节点个数.)

就是每num[d]个最下边的那些点可以将一个深度为d-1的满k叉树组成深度为d的满k叉树

然后看看rest%num[d]是不是为0.

如果不为0的话,就会有一棵子树不是满k叉树;

则肯定还有k-rest/num[d]-1棵是深度为d-1的满k叉树

然后思考那个不是满k叉树的

会发现.那棵子树的问题。仍然是我们最开始面临的问题

可以递归解决的!

获取那个子树的大小,然后递归解决就好!

(深度为d和深度为d-1的满k叉树的子树异或和,单独写一个dfs来做就好)

(不要忘记有多个子树,所以对应的答案要异或和子树的个数对应次数)

k=1的情况要单独处理。找规律就好。就是1^2^3..^n的值。。

(要注意根节点的儿子不够k个的情况。。那个时候就不用递归了,直接处理答案)


【错的次数】


0

【反思】



编程比较难。思路不难想。

【代码】

#include <bits/stdc++.h>
using namespace std;
#define rep1(i,a,b) for (int i = a;i <= b;i++)
#define rep2(i,a,b) for (int i = a;i >= b;i--)
#define ri(x) scanf("%d",&x)
#define rl(x) scanf("%lld",&x)
#define oi(x) printf("%d",x)
#define ol(x) printf("%lld",x)
#define os(x) printf(x)
#define LL long long LL k,num[100],tot[100],ans; LL f(LL x,LL y){
    if (y&1)
        return x;
    else
        return 0;
} LL dfs(int d){
    LL now = 0,temp = 0;
    rep2(i,d,1){
        temp = temp*k + 1;
        now ^= f(temp,num[i]);
    }
    return now;
} void solve(LL n){
    num[1] = 1,tot[1] = 1;
    int d = 1;
    while (1){
        d++;
        num[d] = num[d-1]*k;
        tot[d] = tot[d-1]+num[d];
        if (tot[d]>n) break;
    }
    d--;
    LL numdp1 = 0,numd = 0,rest = n-tot[d];
    if (d == 1 && rest <= k){
        ans ^= f(1,rest);
        ans ^= n;
        return;
    }
    numdp1 = rest/num[d],numd = k - numdp1 - 1;
    LL trest = rest%num[d];     ans ^= f(dfs(d),numdp1);     LL temp1 = dfs(d-1);
    ans ^= f(temp1,numd);     if (trest==0)
        ans ^= temp1;
     else
        solve(trest+tot[d-1]);     ans ^= n;
} int main(){
    //freopen("D:\\rush.txt","r",stdin);
    int T;
    ri(T);
    while (T--){
        LL n;
        rl(n),rl(k);
        ans = 0;
        if (k==1){
            n--;
            if (n%4==0) ans = 1;
            if (n%4==1) ans = n + 2;
            if (n%4==2) ans = 0;
            if (n%4==3) ans = n+1;
        }else
            solve(n);
        ol(ans);puts("");
    }
    return 0;
}

【2017 Multi-University Training Contest - Team 7 && hdu 6121】Build a tree的更多相关文章

  1. 【 2017 Multi-University Training Contest - Team 9 && hdu 6162】Ch’s gift

    [链接]h在这里写链接 [题意] 给你一棵树,每个节点上都有一个权值. 然后给你m个询问,每个询问(x,y,a,b); 表示询问x->y这条路径上权值在[a,b]范围内的节点的权值和. [题解] ...

  2. 2017 Multi-University Training Contest - Team 2 &&hdu 6050 Funny Function

    Funny Function Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  3. 2017 Multi-University Training Contest - Team 2 &&hdu 6053 TrickGCD

    TrickGCD Time Limit: 5000/2500 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others)Total ...

  4. 2017 Multi-University Training Contest - Team 2&&hdu 6047 Maximum Sequence

    Maximum Sequence Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  5. 2017 Multi-University Training Contest - Team 2 &hdu 6055 Regular polygon

    Regular polygon Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)T ...

  6. 【2017中国大学生程序设计竞赛 - 网络选拔赛 && hdu 6154】CaoHaha's staff

    [链接]点击打开链接 [题意] 给你一个面积,让你求围成这个面积最少需要几条边,其中边的连线只能是在坐标轴上边长为1的的线或者是两个边长为1 的线的对角线. [题解] 找规律题 考虑s[i]表示i条边 ...

  7. 【2017中国大学生程序设计竞赛 - 网络选拔赛 hdu 6150】Vertex Cover

    [链接]点击打开链接 [题意] 有人写了一个最小点覆盖的贪心算法,然后,让你去hack它. 并且,要求这个算法得到的错误答案,是正确答案的三倍. 让你任意输出hack数据,点数<=500 [题解 ...

  8. 2017 Multi-University Training Contest - Team 1 1006&&HDU 6038 Function【DFS+数论】

    Function Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)Total ...

  9. 2017 Multi-University Training Contest - Team 9 1005&&HDU 6165 FFF at Valentine【强联通缩点+拓扑排序】

    FFF at Valentine Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) ...

随机推荐

  1. 127.0.0.1和localhost和本机IP三者的区别!

    1, 先来说下回送地址(Loopback Address): 回送地址是主机用于向自身发送通信的一个特殊地址(也就是一个特殊的目的地址).可以这么说:同一台主机上的两项服务若使用回送地址而非分配的主机 ...

  2. 实现人脸识别性别之路---matplotlib

    Np.linspace(start,stop,num,endpoint,dtype)函数 1.参数:范围值,在范围值中取到的数值总数.是否包含范围值.类型 2.返回值:返回一维数据 3.在指定的范围内 ...

  3. Linux下java/bin目录下的命令集合

    Linux下JAVA命令(1.7.0_79) 命令 详解 参数列表 示例 重要程度 资料 appletviewer Java applet 浏览器.appletviewer 命令可在脱离万维网浏览器环 ...

  4. vs平台 error link:2019

    需要XXX.lib或XXX.dll库.手动添加,项目->属性->配置属性->链接器->输入 然后在附件依赖项添加XXX.lib,再生成第一个无法解析的外部符号错误消失了.

  5. [Python] Python Libs

    The Python Standard Library has a lot of modules! To help you get familiar with what's available, he ...

  6. ognl.OgnlException: target is null for setProperty(null,&quot;XXXX&quot;...)

    今天遇到了这个奇葩问题,最后来回比对了一下前辈写过的一段完整代码后才发现问题. Error大概描写叙述为: 警告: Error setting expression 'XXX' with value ...

  7. OpenCASCADE中散乱Edge生成Wire

    OpenCASCADE中散乱Edge生成Wire eryar@163.com Abstract. In OpenCASCADE a wire can be built from any number ...

  8. 解决wget下载文件名乱码的一些方法

    在下载用apache或者nginx做的索引目录时,遇到文件名乱码问题.搜索了不少资料,尝试了好几种方案,大家可以结合使用. 一般情况下加上–restrict-file-names=nocontrol参 ...

  9. vim基础学习之自动补全功能

    本章我们学习自动补全功能1.自动补全优先从当前的编辑区获得补全列表例如:我们写下如下内容 aaaaa aabbb aaab 当我们再次输入aa,然后我们按下Tab的时候,会弹出一个包含 aaaaa a ...

  10. Entity Framework的原理及使用方式

    ADO.NET Entity Framework操作数据库的过程对用户是透明的(当然我们可以通过一些工具或方法了解发送到数据库的SQL语句等).我们唯一能做的是操作EDM,EDM会将这个操作请求发往数 ...