题目1543:无限完全二叉树的层次遍历

题目1543:无限完全二叉树的层次遍历

时间限制:1 秒

内存限制:128 兆

特殊判题:

题目描述:

有一棵无限完全二叉树,他的根节点是1/1,且任意一个节点p/q的左儿子节点和右儿子节点分别是,p/(p+q)和(p+q)/q。如下图:

它的层次遍历结果如下:
1/1, 1/2, 2/1, 1/3, 3/2, 2/3, 3/1,...
有如下两类问题:
1.找到层次遍历的第n个数字。如,n为2时,该数字为1/2;
2.给定一个数字p/q,输出它在层次遍历中的顺序,如p/q为1/2时,其顺序为2;

输入:

输入包含多组测试用例,输入的第一行为一个整数T,代表共有的测试用例数。
接下去T行每行代表一个测试用例,每个测试用例有如下两种类型
1.1 n。输出层次遍历中,第n个数字。
2.2 p q。输出p/q在层次遍历中的顺序。
1 ≤ n, p, q ≤ 2^64-1

输出:

对于每个测试用例,若其类型为1,输出两个整数p q,代表层次遍历中第n个数字为p/q。
若其类型为2,输出一个整数n,代表整数p/q在层次遍历的中的顺序n。
数据保证输出在[1,2^64-1]范围内。

样例输入:
4
1 2
2 1 2
1 5
2 3 2
样例输出:
1 2
2
3 2
5

递归的版本比较好懂一些。一定要注意细节,比如int随便替换LL就会RE。思想也就是从上到下,或者从下到上。
#include <iostream>
#include <stdio.h>
#include <queue>
#include <stdio.h>
#include <string.h>
#include <vector>
#include <queue>
#include <set>
#include <algorithm>
#include <map>
#include <stack>
#include <math.h>
#define Max(a,b) ((a)>(b)?(a):(b))
#define Min(a,b) ((a)<(b)?(a):(b))
using namespace std ;
typedef unsigned long long LL ;
struct Node{
   LL P ;
   LL Q ;
   Node(){} ;
   Node(LL p ,LL q):P(p),Q(q){} ;
};
stack<string>way;
void dfs(LL id){
      if(id<=1)
         return  ;
      if(id&1)
         way.push("R") ;
      else
         way.push("L") ;
      dfs(id>>1) ;
}
void gao_case1(LL N){
    while(!way.empty())
         way.pop() ;
    dfs(N)   ;
    Node now=Node((LL)1,(LL)1) ;
    LL p ,q ;
    while(!way.empty()){
        if(way.top()=="L"){
            p=now.P ;
            q=now.P+now.Q ;
        }
        else{
            p=now.P+now.Q ;
            q=now.Q ;
        }
        now=Node(p,q) ;
        way.pop() ;
    }
    cout<<now.P<<" "<<now.Q<<endl ;
}
void dfs2(Node now){
      if(now.P==1&&now.Q==1)
         return  ;
      LL p ,q  ;
      if(now.P<now.Q){
         p=now.P ;
         q=now.Q-now.P ;
         way.push("L") ;
      }
      else{
         p=now.P-now.Q ;
         q=now.Q ;
         way.push("R") ;
      }
      dfs2(Node(p,q)) ;
}
void gao_case2(LL p ,LL q){
    while(!way.empty())
         way.pop() ;
    dfs2(Node(p,q)) ;
    LL ans=1  ;
    while(!way.empty()){
         if(way.top()=="L")
            ans*=2 ;
         else
            ans=ans*2+1 ;
         way.pop() ;
    }
    cout<<ans<<endl ;
}
int main(){
    LL type  , N ,P ,Q ,T;
    cin>>T  ;
    while(T--){
        cin>>type ;
        if(type==1){
              cin>>N ;
              gao_case1(N)  ;
        }
        else{
            cin>>P>>Q  ;
            gao_case2(P,Q) ;
        }
    }
    return 0 ;
}

  

Long Long、__int64使用总结


  在32位环境下,int占32位,unsigned int占16位,long/unsigned long占32位
何时需要使用:
  long 和 int 范围是[-2^31,2^31),即-2147483648~2147483647,

      而unsigned范围是[0,2^32),即0~4294967295,所以常规的32位整数只能够处理40亿左右,当遇到比40亿大的多的数就要用到64位。
64位使用范围:
  不同的编译器对64位整数的扩展有所不同,VC使用__int64/unsigned __int64,范围是[-2^63, 2^63)和[0,2^64),即-9223372036854775808~9223372036854775807与 0~18446744073709551615(约1800亿亿)。
注意点:
1、编译器不同导致使用64位的申明方式不同;
2、long long / unsigned long long 一般是Linux下申明方式、如:G++
3、__int64 /unsigned __int64一般是Windows下使用64位的申明方式,如:VS
4、在赋值时需要注意加上ll进行显式赋值;
5、当进行64位与 32位的混合运算时,32位整数会被隐式转换成64位整数。
6、输出printf("");,long long使用%lld输出,__int64使用%I64d,无符号使用u替代d即可。
7、测试下来编译器一般都支持2种操作,不必太过纠结,怎么使用看个人喜欢。

原文地址:https://www.cnblogs.com/liyangtianmen/p/3354762.html