免费注册 查看新帖 |

Chinaunix

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

[算法] 大链表查找,求解决方法 [复制链接]

论坛徽章:
6
酉鸡
日期:2013-11-04 15:30:02巳蛇
日期:2014-01-23 10:36:23双鱼座
日期:2014-01-23 13:08:332015亚冠之鹿岛鹿角
日期:2015-09-03 14:36:002015亚冠之武里南联
日期:2015-09-18 10:48:1315-16赛季CBA联赛之山西
日期:2016-05-05 00:05:33
11 [报告]
发表于 2014-08-08 23:41 |只看该作者
本帖最后由 Dannysd 于 2014-08-08 23:47 编辑

回复 10# starwing83



   当然,如果是用 struct test *next这样的链表应该是不行的,用二级指针不是可以当作二维数组那样访问嘛~  

论坛徽章:
5
狮子座
日期:2013-08-20 10:12:24午马
日期:2013-11-23 18:04:102015年辞旧岁徽章
日期:2015-03-03 16:54:152015亚冠之德黑兰石油
日期:2015-06-29 18:11:1115-16赛季CBA联赛之新疆
日期:2024-02-21 10:00:53
12 [报告]
发表于 2014-08-09 00:26 |只看该作者
回复 11# Dannysd


    亲,二级指针也得链表node本身在内存里是连续的,即使链表本身的存储是连续的,在排序之后连续的元素也不一定是排序的连续元素了(因为只更改了next指针的值,并没有拷贝移动元素本身——这也是链表排序的优势),因此这时候你用二级指针直接就是作死。

一般的链表,通过malloc/free分配,你用二级指针操作除非是想crash想着急了

论坛徽章:
89
水瓶座
日期:2014-04-01 08:53:31天蝎座
日期:2014-04-01 08:53:53天秤座
日期:2014-04-01 08:54:02射手座
日期:2014-04-01 08:54:15子鼠
日期:2014-04-01 08:55:35辰龙
日期:2014-04-01 08:56:36未羊
日期:2014-04-01 08:56:27戌狗
日期:2014-04-01 08:56:13亥猪
日期:2014-04-01 08:56:02亥猪
日期:2014-04-08 08:38:58程序设计版块每日发帖之星
日期:2016-01-05 06:20:00程序设计版块每日发帖之星
日期:2016-01-07 06:20:00
13 [报告]
发表于 2014-08-09 08:22 |只看该作者
如果只有链表,那就没有好讲的了。

如果想提高查询性能,那就一定要有一个辅助的数据结构,或者干脆转化为其他的存储方式,比如hash table。

论坛徽章:
6
酉鸡
日期:2013-11-04 15:30:02巳蛇
日期:2014-01-23 10:36:23双鱼座
日期:2014-01-23 13:08:332015亚冠之鹿岛鹿角
日期:2015-09-03 14:36:002015亚冠之武里南联
日期:2015-09-18 10:48:1315-16赛季CBA联赛之山西
日期:2016-05-05 00:05:33
14 [报告]
发表于 2014-08-09 23:06 |只看该作者
回复 12# starwing83


    我表达不太清楚,请大神参考一下scandir的实现(其中一个版本),如下
  1. /*
  2. * scandir, alphasort - scan a directory
  3. *
  4. * implementation for systems that do not have it in libc
  5. */

  6. #include "../config.h"

  7. #ifndef        HAVE_SCANDIR

  8. #include <sys/types.h>
  9. #include <dirent.h>
  10. #include <stdlib.h>
  11. #include <stddef.h>
  12. #include <string.h>

  13. /*
  14. * convenience helper function for scandir's |compar()| function:
  15. * sort directory entries using strcoll(3)
  16. */
  17. int
  18. alphasort(const void *_a, const void *_b)
  19. {
  20.     struct dirent **a = (struct dirent **)_a;
  21.     struct dirent **b = (struct dirent **)_b;
  22.     return strcoll((*a)->d_name, (*b)->d_name);
  23. }


  24. #define strverscmp(a,b) strcoll(a,b) /* for now */

  25. /*
  26. * convenience helper function for scandir's |compar()| function:
  27. * sort directory entries using GNU |strverscmp()|
  28. */
  29. int
  30. versionsort(const void *_a, const void *_b)
  31. {
  32.     struct dirent **a = (struct dirent **)_a;
  33.     struct dirent **b = (struct dirent **)_b;
  34.     return strverscmp((*a)->d_name, (*b)->d_name);
  35. }

  36. /*
  37. * The scandir() function reads the directory dirname and builds an
  38. * array of pointers to directory entries using malloc(3).  It returns
  39. * the number of entries in the array.  A pointer to the array of
  40. * directory entries is stored in the location referenced by namelist.
  41. *
  42. * The select parameter is a pointer to a user supplied subroutine
  43. * which is called by scandir() to select which entries are to be
  44. * included in the array.  The select routine is passed a pointer to
  45. * a directory entry and should return a non-zero value if the
  46. * directory entry is to be included in the array.  If select is null,
  47. * then all the directory entries will be included.
  48. *
  49. * The compar parameter is a pointer to a user supplied subroutine
  50. * which is passed to qsort(3) to sort the completed array.  If this
  51. * pointer is null, the array is not sorted.
  52. */
  53. int
  54. scandir(const char *dirname,
  55.         struct dirent ***ret_namelist,
  56.         int (*select)(const struct dirent *),
  57.         int (*compar)(const struct dirent **, const struct dirent **))
  58. {
  59.     int i, len;
  60.     int used, allocated;
  61.     DIR *dir;
  62.     struct dirent *ent, *ent2;
  63.     struct dirent **namelist = NULL;

  64.     if ((dir = opendir(dirname)) == NULL)
  65.         return -1;

  66.     used = 0;
  67.     allocated = 2;
  68.     namelist = malloc(allocated * sizeof(struct dirent *));
  69.     if (!namelist)
  70.         goto error;

  71.     while ((ent = readdir(dir)) != NULL) {

  72.         if (select != NULL && !select(ent))
  73.             continue;

  74.         /* duplicate struct direct for this entry */
  75.         len = offsetof(struct dirent, d_name) + strlen(ent->d_name) + 1;
  76.         if ((ent2 = malloc(len)) == NULL)
  77.             return -1;
  78.        
  79.         if (used >= allocated) {
  80.             allocated *= 2;
  81.             namelist = realloc(namelist, allocated * sizeof(struct dirent *));
  82.             if (!namelist)
  83.                 goto error;
  84.         }
  85.         memcpy(ent2, ent, len);
  86.         namelist[used++] = ent2;
  87.     }
  88.     closedir(dir);

  89.     if (compar)
  90.         qsort(namelist, used, sizeof(struct dirent *),
  91.               (int (*)(const void *, const void *)) compar);

  92.     *ret_namelist = namelist;
  93.     return used;


  94. error:
  95.     if (namelist) {
  96.         for (i = 0; i < used; i++)
  97.             free(namelist[i]);
  98.         free(namelist);
  99.     }
  100.     return -1;
  101. }
  102. #endif


  103. #if        STANDALONE_MAIN
  104. int
  105. main(int argc, char **argv)
  106. {
  107.         struct dirent **namelist;
  108.         int i, n;

  109.         n = scandir("/etc", &namelist, NULL, alphasort);

  110.         for (i = 0; i < n; i++)
  111.                 printf("%s\n", namelist[i]->d_name);
  112. }
  113. #endif
复制代码

论坛徽章:
224
2022北京冬奥会纪念版徽章
日期:2015-08-10 16:30:32操作系统版块每日发帖之星
日期:2016-02-18 06:20:00操作系统版块每日发帖之星
日期:2016-03-01 06:20:00操作系统版块每日发帖之星
日期:2016-03-02 06:20:0015-16赛季CBA联赛之上海
日期:2019-09-20 12:29:3219周年集字徽章-周
日期:2019-10-01 20:47:4815-16赛季CBA联赛之八一
日期:2020-10-23 18:30:5320周年集字徽章-20	
日期:2020-10-28 14:14:2615-16赛季CBA联赛之广夏
日期:2023-02-25 16:26:26CU十四周年纪念徽章
日期:2023-04-13 12:23:1015-16赛季CBA联赛之四川
日期:2023-07-25 16:53:45操作系统版块每日发帖之星
日期:2016-05-10 19:22:58
15 [报告]
发表于 2014-08-10 12:33 |只看该作者
回复 3# folklore


    map是红黑树,比left+right二叉树性能要稳定多了

论坛徽章:
0
16 [报告]
发表于 2014-08-13 11:27 |只看该作者
linux下经常用的list_entry是最快的。
根据地址直接查到,不管你是千万还是亿万。
libuv里有简单的例子:

#define container_of(ptr, type, member) \
  ((type *) ((char *) (ptr) - offsetof(type, member)))

struct timer_ctx* ctx = container_of(handle, struct timer_ctx, handle);

原理就是记住某个结构体里的某个变量的内存地址。自行搜索一下,http://blog.csdn.net/jiatingqiang/article/details/6437496 这里也有。
这也是为什么指针对c/c++如此重要的原因。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP