NYOJ-63 小猴子下落(二叉树及优化算法详解)

 

小猴子下落

时间限制:3000 ms  |  内存限制:65535 KB

难度:3

描述

有一颗二叉树,最大深度为D,且所有叶子的深度都相同。所有结点从左到右从上到下的编号为1,2,3,·····,2的D次方减1。在结点1处放一个小猴子,它会往下跑。每个内结点上都有一个开关,初始全部关闭,当每次有小猴子跑到一个开关上时,它的状态都会改变,当到达一个内结点时,如果开关关闭,小猴子往左走,否则往右走,直到走到叶子结点。

一些小猴子从结点1处开始往下跑,最后一个小猴儿会跑到哪里呢?

输入
输入二叉树叶子的深度D,和小猴子数目I,假设I不超过整棵树的叶子个数,D<=20.最终以 0 0 结尾
输出
输出第I个小猴子所在的叶子编号。
样例输入
4 2
3 4
0 0
样例输出
12
7

第一感觉就是这题题目思路和清晰,一开始想到直接算出答案输出即可,考虑到正在学习数据结构,因此还是选择了用树进行暴力求解。

没想到运气好居然过了。供学习树的朋友一同学习。

下面还将讲解优化算法:

树写代码如下:

 1 #include<iostream>
 2 #include<queue>
 3 #include<cmath>
 4 using namespace std;
 5 struct node
 6 {
 7     int data;
 8     int flag;
 9     node *lchild,*rchild;
10     node();
11 };
12 node::node()
13 {
14     flag=-1;
15     rchild=lchild=NULL;
16 }
17 void createTree(int d,node *&root)
18 {
19     queue<node *> q;
20     while(!q.empty())
21         q.pop();
22     root=new node;
23     static int count=0;
24     root->data=++count;
25     q.push(root);
26     node *t=root;
27     while(count!=pow(2,d)-1)
28     {
29         t=q.front();
30         q.pop();
31         t->lchild=new node;
32         t->lchild->data=++count;
33         q.push(t->lchild);
34         t->rchild=new node;
35         t->rchild->data=++count;
36         q.push(t->rchild);
37     }
38     t=NULL;
39     count=0;
40 }
41 /*
42 void LevelOrder(node *root)
43 {    //队列实现
44     queue<node *> q;
45     node *t=root;
46     if(t!=NULL) q.push(t);                //根非空,入队
47     while(!q.empty())                //队不空
48     {
49         t=q.front();
50         q.pop();                    //出队
51         cout<<t->data<<" ";
52         if(t->lchild)
53             q.push(t->lchild);  //遍历左孩子
54         if(t->rchild)
55             q.push(t->rchild); //遍历右孩子
56     }
57
58 }
59 */
60 void Go(int &t,node *&root)
61 {
62     if(root->lchild&&root->rchild){
63     if(root->flag==-1)
64     {
65         Go(t,root->lchild);
66         root->flag=1;
67     }
68     else
69     {
70         Go(t,root->rchild);
71         root->flag=-1;
72     }
73     }
74     else
75         t=root->data;
76 }
77
78 int main()
79 {
80     int d,num;
81     while(cin>>d>>num,d&&num){
82     node *root=NULL;
83     createTree(d,root);
84     int t;
85     for(int i=0;i<num;i++)
86         Go(t,root);
87     cout<<t<<endl;
88     }
89 return 0;
90 }

但是如果测试数据有N组,层数D有19层呢(D<=20),那么树将建立2^19-1个结点,时间和空间耗费都很大。那么怎么办?

下面讲一下优化算法:

               1               
        2                    3      
  4       5       6       7  
    10   11    12     13   14   15 

根据右图测试数据可知,一共有n行(3,4,5),x个猴子中每2^n出现一循环,理由就是它是满二叉树。

根据左图四层我们列出数据看看:

第1只猴子 1 2 4 8
第2只猴子 1 3 6 12
第3只猴子 1 2 5 10
第4只猴子 1 3 7 14
第5只猴子 1 2 4 9
第6只猴子 1 3 6 13
第7只猴子 1 2 5 11
第8只猴子 1 3 7 15

请读者看看四层二叉树(上左图)和上表中对比不难发现,进入第n个结点的次数i为奇数(即前面已有n-1过猴子访问过该结点),那么遍历其左子树根;

若为偶数,则遍历其右子树根。

因此,对照上表,得出规律:i为奇数,k=k*2;i=(i+1)/2;//第i个进入左子树

             i为偶数,k=k*2+1;i=i/2; //第i个进入右子树

例如

第1个猴子:则对于第一个结点来说,i=1为奇数,那么下一个要走的结点k=1*2=2;然后i=(1+1)/2=1(第一个进入左子树),继续判断其左子树i的奇偶性……

第3个猴子:则对于第一个结点来说,i=3为奇数,那么下一个要走的结点k=1*2=2;然后i=(3+1)/2=2(第二个进入左子树)……

第5个猴子:则对于第一个结点来说,i=5为奇数,那么下一个要走的结点k=1*2=2;然后i=(5+1)/2=3(第三个进入左子树)……

……

1 for (int j=0;j<d-1;j++)
2        if(i%2) {k=k*2;i=(i+1)/2;}
3        else {k=k*2+1;i /=2;}

OK接着按照输入标准写出完整算法如下:

 1
 2 #include<iostream>
 3 using namespace std;
 4
 5 int main()
 6 {
 7     int d,i,k;
 8     while(cin>>d>>i && (d+i) !=0)
 9     {
10         k=1;
11         for (int j=0;j<d-1;j++)
12             if(i%2) {k=k*2;i=(i+1)/2;}
13             else {k=k*2+1;i /=2;}
14         cout<<k<<endl;
15
16     }
17 }        

当然,你可以将/2换成位运算左移一位,效率更高。

时间: 2024-12-24 13:14:33

NYOJ-63 小猴子下落(二叉树及优化算法详解)的相关文章

nyoj 63(小猴子下落)(模拟,二叉树)

小猴子下落 时间限制:3000 ms  |  内存限制:65535 KB 难度:3 描述 有一颗二叉树,最大深度为D,且所有叶子的深度都相同.所有结点从左到右从上到下的编号为1,2,3,·····,2的D次方减1.在结点1处放一个小猴子,它会往下跑.每个内结点上都有一个开关,初始全部关闭,当每次有小猴子跑到一个开关上时,它的状态都会改变,当到达一个内结点时,如果开关关闭,小猴子往左走,否则往右走,直到走到叶子结点. 一些小猴子从结点1处开始往下跑,最后一个小猴儿会跑到哪里呢? 输入 输入二叉树叶

nyoj 63 小猴子下落

小猴子下落 时间限制:3000 ms  |  内存限制:65535 KB 难度:3 描述 有一颗二叉树,最大深度为D,且所有叶子的深度都相同.所有结点从左到右从上到下的编号为1,2,3,·····,2的D次方减1.在结点1处放一个小猴子,它会往下跑.每个内结点上都有一个开关,初始全部关闭,当每次有小猴子跑到一个开关上时,它的状态都会改变,当到达一个内结点时,如果开关关闭,小猴子往左走,否则往右走,直到走到叶子结点. 一些小猴子从结点1处开始往下跑,最后一个小猴儿会跑到哪里呢? 输入 输入二叉树叶

机器学习经典算法详解及Python实现--CART分类决策树、回归树和模型树

摘要: Classification And Regression Tree(CART)是一种很重要的机器学习算法,既可以用于创建分类树(Classification Tree),也可以用于创建回归树(Regression Tree),本文介绍了CART用于离散标签分类决策和连续特征回归时的原理.决策树创建过程分析了信息混乱度度量Gini指数.连续和离散特征的特殊处理.连续和离散特征共存时函数的特殊处理和后剪枝:用于回归时则介绍了回归树和模型树的原理.适用场景和创建过程.个人认为,回归树和模型树

最短路算法 :Bellman-ford算法 &amp; Dijkstra算法 &amp; floyd算法 &amp; SPFA算法 详解

 本人QQ :2319411771   邮箱 : [email protected] 若您发现本文有什么错误,请联系我,我会及时改正的,谢谢您的合作! 本文为原创文章,转载请注明出处 本文链接   :http://www.cnblogs.com/Yan-C/p/3916281.html . 很早就想写一下最短路的总结了,但是一直懒,就没有写,这几天又在看最短路,岁没什么长进,但还是加深了点理解. 于是就想写一个大点的总结,要写一个全的. 在本文中因为邻接表在比赛中不如前向星好写,而且前向星效率并

支持向量机(SVM)(五)-- SMO算法详解

一.我们先回顾下SVM问题. A.线性可分问题 1.SVM基本原理: SVM使用一种非线性映射,把原训练            数据映射到较高的维.在新的维上,搜索最佳分离超平面,两个类的数据总可以被超平面分开. 2.问题的提出: 3.如何选取最优的划分直线f(x)呢? 4.求解:凸二次规划 建立拉格朗日函数: 求偏导数: B.线性不可分问题 1.核函数 如下图:横轴上端点a和b之间红色部分里的所有点定为正类,两边的黑色部分里的点定为负类. 设: g(x)转化为f(y)=<a,y> g(x)=

Manacher算法详解

[转] Manacher算法详解 转载自: http://blog.csdn.net/dyx404514/article/details/42061017 Manacher算法 算法总结第三弹 manacher算法,前面讲了两个字符串相算法——kmp和拓展kmp,这次来还是来总结一个字符串算法,manacher算法,我习惯叫他 “马拉车”算法. 相对于前面介绍的两个算法,Manacher算法的应用范围要狭窄得多,但是它的思想和拓展kmp算法有很多共通支出,所以在这里介绍一下.Manacher算法

Tarjan算法详解

Tarjan算法详解 [概念] 在有向图G中,如果两个顶点间至少存在一条路径,称两个顶点强连通(strongly connected).如果有向图G的每两个顶点都强连通,称G是一个强连通图.非强连通图有向图的极大强连通子图,称为强连通分量(strongly connected components). 下图中,子图{1,2,3,4}为一个强连通分量,因为顶点1,2,3,4两两可达.{5},{6}也分别是两个强连通分量. [功能] Tarjan算法的用途之一是,求一个有向图G=(V,E)里极大强连

机器学习经典算法详解及Python实现---朴素贝叶斯分类及其在文本分类、垃圾邮件检测中的应用

摘要: 朴素贝叶斯分类是贝叶斯分类器的一种,贝叶斯分类算法是统计学的一种分类方法,利用概率统计知识进行分类,其分类原理就是利用贝叶斯公式根据某对象的先验概率计算出其后验概率(即该对象属于某一类的概率),然后选择具有最大后验概率的类作为该对象所属的类.总的来说:当样本特征个数较多或者特征之间相关性较大时,朴素贝叶斯分类效率比不上决策树模型:当各特征相关性较小时,朴素贝叶斯分类性能最为良好.另外朴素贝叶斯的计算过程类条件概率等计算彼此是独立的,因此特别适于分布式计算.本文详述了朴素贝叶斯分类的统计学

安全体系(二)——RSA算法详解

本文主要讲述RSA算法使用的基本数学知识.秘钥的计算过程以及加密和解密的过程. 安全体系(零)—— 加解密算法.消息摘要.消息认证技术.数字签名与公钥证书 安全体系(一)—— DES算法详解 1.概述 RSA公钥加密算法是1977年由罗纳德·李维斯特(Ron Rivest).阿迪·萨莫尔(Adi Shamir)和伦纳德·阿德曼(Leonard Adleman)一起提出的.1987年首次公布,当时他们三人都在麻省理工学院工作.RSA算法以他们三人姓氏开头字母命名. RSA是目前最有影响力的公钥加密