免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
楼主: VIP_fuck
打印 上一主题 下一主题

[算法] 请教一个算法问题 [复制链接]

论坛徽章:
36
子鼠
日期:2013-08-28 22:23:29黄金圣斗士
日期:2015-12-01 11:37:51程序设计版块每日发帖之星
日期:2015-12-14 06:20:00CU十四周年纪念徽章
日期:2015-12-22 16:50:40IT运维版块每日发帖之星
日期:2016-01-25 06:20:0015-16赛季CBA联赛之深圳
日期:2016-01-27 10:31:172016猴年福章徽章
日期:2016-02-18 15:30:3415-16赛季CBA联赛之福建
日期:2016-04-07 11:25:2215-16赛季CBA联赛之青岛
日期:2016-04-29 18:02:5915-16赛季CBA联赛之北控
日期:2016-06-20 17:38:50技术图书徽章
日期:2016-07-19 13:54:03程序设计版块每日发帖之星
日期:2016-08-21 06:20:00
11 [报告]
发表于 2015-06-07 21:24 |只看该作者
回复 8# windoze


    我说的root是第0层,3^0是1不。。

论坛徽章:
36
子鼠
日期:2013-08-28 22:23:29黄金圣斗士
日期:2015-12-01 11:37:51程序设计版块每日发帖之星
日期:2015-12-14 06:20:00CU十四周年纪念徽章
日期:2015-12-22 16:50:40IT运维版块每日发帖之星
日期:2016-01-25 06:20:0015-16赛季CBA联赛之深圳
日期:2016-01-27 10:31:172016猴年福章徽章
日期:2016-02-18 15:30:3415-16赛季CBA联赛之福建
日期:2016-04-07 11:25:2215-16赛季CBA联赛之青岛
日期:2016-04-29 18:02:5915-16赛季CBA联赛之北控
日期:2016-06-20 17:38:50技术图书徽章
日期:2016-07-19 13:54:03程序设计版块每日发帖之星
日期:2016-08-21 06:20:00
12 [报告]
发表于 2015-06-07 22:10 |只看该作者
  1. int getParentIdx(int idx)
  2. {
  3.         if (idx == 0)
  4.         {
  5.                 return 0;
  6.         }
  7.         int floor = 0;
  8.         int total = 0;
  9.         int start = 0;
  10.         int end = 0;
  11.         int parentStart = 0;
  12.         int parentEnd = 0;
  13.         int sub = 0;
  14.         int parent = 0;

  15.         while (true)
  16.         {
  17.                 total = pow(3.0, floor);
  18.                 if (floor == 0)
  19.                 {
  20.                         parentStart = parentEnd = 0;
  21.                 }
  22.                 else
  23.                 {
  24.                         start = end + 1;
  25.                         end = end + total;
  26.                         if (idx >= start && idx <= end)
  27.                         {
  28.                                 sub = idx - start;
  29.                                 parent = parentEnd - sub / 3;
  30.                                 return parent;
  31.                         }
  32.                 }
  33.                 parentStart = start;
  34.                 parentEnd = end;

  35.                 floor++;
  36.         }
  37. }

  38. void testCTTree()
  39. {
  40.         printf("2  -> parent: %d\n", getParentIdx(2));
  41.         printf("3  -> parent: %d\n", getParentIdx(3));
  42.         printf("5  -> parent: %d\n", getParentIdx(5));
  43.         printf("7  -> parent: %d\n", getParentIdx(7));
  44.         printf("11  -> parent: %d\n", getParentIdx(11));
  45.         printf("15  -> parent: %d\n", getParentIdx(15));
  46.         printf("17 -> parent: %d\n", getParentIdx(17));
  47. }
复制代码
输出:
  1. 2  -> parent: 0
  2. 3  -> parent: 0
  3. 5  -> parent: 3
  4. 7  -> parent: 2
  5. 11  -> parent: 1
  6. 15  -> parent: 12
  7. 17 -> parent: 11
复制代码
然后套到我之前给的那个循环里就ok了

论坛徽章:
36
子鼠
日期:2013-08-28 22:23:29黄金圣斗士
日期:2015-12-01 11:37:51程序设计版块每日发帖之星
日期:2015-12-14 06:20:00CU十四周年纪念徽章
日期:2015-12-22 16:50:40IT运维版块每日发帖之星
日期:2016-01-25 06:20:0015-16赛季CBA联赛之深圳
日期:2016-01-27 10:31:172016猴年福章徽章
日期:2016-02-18 15:30:3415-16赛季CBA联赛之福建
日期:2016-04-07 11:25:2215-16赛季CBA联赛之青岛
日期:2016-04-29 18:02:5915-16赛季CBA联赛之北控
日期:2016-06-20 17:38:50技术图书徽章
日期:2016-07-19 13:54:03程序设计版块每日发帖之星
日期:2016-08-21 06:20:00
13 [报告]
发表于 2015-06-07 22:10 |只看该作者
  1. int getParentIdx(int idx)
  2. {
  3.         if (idx == 0)
  4.         {
  5.                 return 0;
  6.         }
  7.         int floor = 0;
  8.         int total = 0;
  9.         int start = 0;
  10.         int end = 0;
  11.         int parentStart = 0;
  12.         int parentEnd = 0;
  13.         int sub = 0;
  14.         int parent = 0;

  15.         while (true)
  16.         {
  17.                 total = pow(3.0, floor);
  18.                 if (floor == 0)
  19.                 {
  20.                         parentStart = parentEnd = 0;
  21.                 }
  22.                 else
  23.                 {
  24.                         start = end + 1;
  25.                         end = end + total;
  26.                         if (idx >= start && idx <= end)
  27.                         {
  28.                                 sub = idx - start;
  29.                                 parent = parentEnd - sub / 3;
  30.                                 return parent;
  31.                         }
  32.                 }
  33.                 parentStart = start;
  34.                 parentEnd = end;

  35.                 floor++;
  36.         }
  37. }

  38. void testCTTree()
  39. {
  40.         printf("2  -> parent: %d\n", getParentIdx(2));
  41.         printf("3  -> parent: %d\n", getParentIdx(3));
  42.         printf("5  -> parent: %d\n", getParentIdx(5));
  43.         printf("7  -> parent: %d\n", getParentIdx(7));
  44.         printf("11  -> parent: %d\n", getParentIdx(11));
  45.         printf("15  -> parent: %d\n", getParentIdx(15));
  46.         printf("17 -> parent: %d\n", getParentIdx(17));
  47. }
复制代码
输出:
  1. 2  -> parent: 0
  2. 3  -> parent: 0
  3. 5  -> parent: 3
  4. 7  -> parent: 2
  5. 11  -> parent: 1
  6. 15  -> parent: 12
  7. 17 -> parent: 11
复制代码
然后套到我之前给的那个循环里就ok了

论坛徽章:
36
子鼠
日期:2013-08-28 22:23:29黄金圣斗士
日期:2015-12-01 11:37:51程序设计版块每日发帖之星
日期:2015-12-14 06:20:00CU十四周年纪念徽章
日期:2015-12-22 16:50:40IT运维版块每日发帖之星
日期:2016-01-25 06:20:0015-16赛季CBA联赛之深圳
日期:2016-01-27 10:31:172016猴年福章徽章
日期:2016-02-18 15:30:3415-16赛季CBA联赛之福建
日期:2016-04-07 11:25:2215-16赛季CBA联赛之青岛
日期:2016-04-29 18:02:5915-16赛季CBA联赛之北控
日期:2016-06-20 17:38:50技术图书徽章
日期:2016-07-19 13:54:03程序设计版块每日发帖之星
日期:2016-08-21 06:20:00
14 [报告]
发表于 2015-06-07 22:35 |只看该作者
考虑运行效率,初始化的时候生成个数组吧,数组元素保存每层的起点idx和终点idx,然后getParentIdx的时候查表再加个二分,要省些

论坛徽章:
12
2015年辞旧岁徽章
日期:2015-03-03 16:54:1515-16赛季CBA联赛之同曦
日期:2017-03-17 19:13:162016科比退役纪念章
日期:2016-11-07 08:28:12luobin
日期:2016-06-17 17:46:36wusuopu
日期:2016-06-17 17:43:4515-16赛季CBA联赛之福建
日期:2016-01-14 12:49:22程序设计版块每日发帖之星
日期:2015-12-13 06:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:002015年亚洲杯之科威特
日期:2015-03-24 14:21:272015年迎新春徽章
日期:2015-03-04 09:57:092016科比退役纪念章
日期:2018-04-10 16:20:18
15 [报告]
发表于 2015-06-07 22:52 |只看该作者
回复 13# cokeboL


    能把你的公式写一下吗,判断某一个节点的子节点是哪个,


   正在看代码,,,

论坛徽章:
12
2015年辞旧岁徽章
日期:2015-03-03 16:54:1515-16赛季CBA联赛之同曦
日期:2017-03-17 19:13:162016科比退役纪念章
日期:2016-11-07 08:28:12luobin
日期:2016-06-17 17:46:36wusuopu
日期:2016-06-17 17:43:4515-16赛季CBA联赛之福建
日期:2016-01-14 12:49:22程序设计版块每日发帖之星
日期:2015-12-13 06:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:002015年亚洲杯之科威特
日期:2015-03-24 14:21:272015年迎新春徽章
日期:2015-03-04 09:57:092016科比退役纪念章
日期:2018-04-10 16:20:18
16 [报告]
发表于 2015-06-07 23:33 |只看该作者
回复 14# cokeboL


    服。

    用数据测试,正确。

    我的思路比较死,是考虑每个节点的子节点怎么计算,而且,只考虑加,乘,除,唯独没有考虑减法。

论坛徽章:
1
亥猪
日期:2014-09-10 11:43:17
17 [报告]
发表于 2015-06-08 06:52 |只看该作者
很规律的排列,献丑了。
  1. #include <stdio.h>

  2. void getAncestors(int * ancestor, int layer)
  3. {
  4.         static int from, to;
  5.        
  6.         if(*ancestor == 0){ from = to = 0; return;}
  7.         ancestor[1] = (ancestor[0] - 1) / 3;
  8.         getAncestors(ancestor + 1, !layer);
  9.         from = from * 3 + 1;
  10.         to = (to + 1) * 3;
  11.         if(layer) *ancestor = from + to - *ancestor;
  12. }

  13. int commonAncestor(int a, int b)
  14. {
  15.         int La[24] = {a}, Lb[24] = {b}, *pa, *pb;
  16.        
  17.         getAncestors(La, 0);
  18.         getAncestors(Lb, 0);
  19.         for(pa = La, pb = Lb; *pa != *pb; *pa > *pb ? pa++ : pb++);
  20.         return *pa;
  21. }

  22. int main()
  23. {
  24.         int a, b;
  25.        
  26.         scanf("%d %d", &a, &b);
  27.         printf("%d\n", commonAncestor(a, b));
  28.         return 0;
  29. }
复制代码

论坛徽章:
12
2015年辞旧岁徽章
日期:2015-03-03 16:54:1515-16赛季CBA联赛之同曦
日期:2017-03-17 19:13:162016科比退役纪念章
日期:2016-11-07 08:28:12luobin
日期:2016-06-17 17:46:36wusuopu
日期:2016-06-17 17:43:4515-16赛季CBA联赛之福建
日期:2016-01-14 12:49:22程序设计版块每日发帖之星
日期:2015-12-13 06:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:002015年亚洲杯之科威特
日期:2015-03-24 14:21:272015年迎新春徽章
日期:2015-03-04 09:57:092016科比退役纪念章
日期:2018-04-10 16:20:18
18 [报告]
发表于 2015-06-08 08:14 |只看该作者
回复 17# Kurosaki_Ichigo


   

论坛牛人够多

我自己只是无限接近这个结果,但是始终都差那么一点,惭愧。

论坛徽章:
36
子鼠
日期:2013-08-28 22:23:29黄金圣斗士
日期:2015-12-01 11:37:51程序设计版块每日发帖之星
日期:2015-12-14 06:20:00CU十四周年纪念徽章
日期:2015-12-22 16:50:40IT运维版块每日发帖之星
日期:2016-01-25 06:20:0015-16赛季CBA联赛之深圳
日期:2016-01-27 10:31:172016猴年福章徽章
日期:2016-02-18 15:30:3415-16赛季CBA联赛之福建
日期:2016-04-07 11:25:2215-16赛季CBA联赛之青岛
日期:2016-04-29 18:02:5915-16赛季CBA联赛之北控
日期:2016-06-20 17:38:50技术图书徽章
日期:2016-07-19 13:54:03程序设计版块每日发帖之星
日期:2016-08-21 06:20:00
19 [报告]
发表于 2015-06-08 09:00 |只看该作者
真要实现的话,这种树用数组存储比较好,idx就是child下标,生成的时候父子关系就确定

论坛徽章:
59
2015年亚洲杯之约旦
日期:2015-01-27 21:27:392015年亚洲杯之日本
日期:2015-02-06 22:09:41拜羊年徽章
日期:2015-03-03 16:15:432015年辞旧岁徽章
日期:2015-03-03 16:54:152015年迎新春徽章
日期:2015-03-04 09:50:282015元宵节徽章
日期:2015-03-06 15:50:392015年亚洲杯之阿联酋
日期:2015-03-19 17:39:302015年亚洲杯之中国
日期:2015-03-23 18:52:23巳蛇
日期:2014-12-14 22:44:03双子座
日期:2014-12-10 21:39:16处女座
日期:2014-12-02 08:03:17天蝎座
日期:2014-07-21 19:08:47
20 [报告]
发表于 2015-06-08 10:23 |只看该作者
这个不是满三叉树么, 用数学公式能算出来啊。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP