搜索
bottom↓
回复: 15

以下情况查找数据用什么数据结构比较高效呢?

[复制链接]

出5入4汤圆

发表于 2018-8-13 23:29:52 | 显示全部楼层 |阅读模式
1、先说一个简单的情况:
假设一群学生的学号范围是1-3000,想要记录或者查询每个人的成绩的话,主需要建立一个uint16  score[3001]的数组就行
2、问题来了:
如果学号范围是1-65535,并且一个班一共3000人,每个学生的学号都是1-65535中随机且唯一的数值。这样要随时记录并且查找的话,用uint16  score[65535]显然不合适,首先占用太大的ram,其次很多数组元素根本用不到。
请问有什么合适的数据结构能做到随时读写对应学号学生的成绩?
欢迎大家一起讨论

阿莫论坛20周年了!感谢大家的支持与爱护!!

月入3000的是反美的。收入3万是亲美的。收入30万是移民美国的。收入300万是取得绿卡后回国,教唆那些3000来反美的!

出0入442汤圆

发表于 2018-8-14 00:01:55 | 显示全部楼层
你内存足够大就不用管了。多占点内存换更少的编码时间和运行时间,何乐而不为呢?

出5入4汤圆

 楼主| 发表于 2018-8-14 00:53:49 来自手机 | 显示全部楼层
wye11083 发表于 2018-8-14 00:01
你内存足够大就不用管了。多占点内存换更少的编码时间和运行时间,何乐而不为呢? ...

mcu一共才多少ram哦,又不是pc有好几个g的内存

出0入0汤圆

发表于 2018-8-14 01:36:38 | 显示全部楼层
这个应该没办法吧?用结构至少要有学号跟成绩,合起来就快比直接存用的内存多了

出0入0汤圆

发表于 2018-8-14 03:32:22 来自手机 | 显示全部楼层
使用链表,用到一个就malloc一个,并添加到链表中

出0入93汤圆

发表于 2018-8-14 06:18:23 | 显示全部楼层
支持C++不?支持的话,最简单就是std::map
不支持的话,自己造一个map,或者两个数组:一个常量数组存放学号(不占用RAM)另一个记录成绩。
使用map速度快,学号排完序二分法查找学号最多只需要查找13次,但是占用RAM多出1倍;使用两个数组查找学号最多需要3000次,占用CPU。

出5入4汤圆

 楼主| 发表于 2018-8-14 06:54:28 来自手机 | 显示全部楼层
wistarky 发表于 2018-8-14 01:36
这个应该没办法吧?用结构至少要有学号跟成绩,合起来就快比直接存用的内存多了 ...

办法肯定有的,只是看时间和空间怎么妥协了

出5入4汤圆

 楼主| 发表于 2018-8-14 06:56:32 来自手机 | 显示全部楼层
undead 发表于 2018-8-14 03:32
使用链表,用到一个就malloc一个,并添加到链表中

查询某个学号对应的成绩会比较慢哎,比如存了n个学生,正好被查询学生的学号是链表尾巴,需要查询n次

出5入4汤圆

 楼主| 发表于 2018-8-14 06:57:06 来自手机 | 显示全部楼层
takashiki 发表于 2018-8-14 06:18
支持C++不?支持的话,最简单就是std::map
不支持的话,自己造一个map,或者两个数组:一个常量数组存放学 ...

您说的这个map是哈希表吗?

出0入93汤圆

发表于 2018-8-14 07:03:32 | 显示全部楼层
按6楼的补充如下,结合上面两种方案:
做成两个表,一个放学号的常量数组,预先排序好。另一个放成绩。使用时首先查找学号所在的位置,二分法最多查找13次,然后修改成绩表。
当然最简单方便的还是std::map,本来就是他该干的事情。std::map<uint16_t, uint16_t> Students; 读写30342号学生的成绩就是这样:Students[30342] = 成绩; 成绩 = Students[30342]; 没有用到的学号不占空间的

出0入93汤圆

发表于 2018-8-14 07:16:10 | 显示全部楼层
tim4146 发表于 2018-8-14 06:57
您说的这个map是哈希表吗?

大概不是吧……你就认为是吧,反正这些数据结构都差不多,我的原则能用好用就行了,懒得去纠结叫啥名称。C++的动态数组叫vector,array是静态的,找谁说理去。

出300入477汤圆

发表于 2018-8-14 08:14:18 来自手机 | 显示全部楼层
takashiki 发表于 2018-8-14 07:16
大概不是吧……你就认为是吧,反正这些数据结构都差不多,我的原则能用好用就行了,懒得去纠结叫啥名称。 ...

stl map不是哈希表,而是红黑树。完全两码事。当然都可以做为查找用的数据结构。
如果不想用c++只用c,请找别人写好的红黑树的库,别自己写,很少有人能写对的。

出0入0汤圆

发表于 2018-8-14 08:45:24 | 显示全部楼层
按学号从小到大存放,然后用二分法查找

出0入17汤圆

发表于 2021-2-25 16:11:18 | 显示全部楼层
本帖最后由 think_a_second 于 2021-2-25 16:31 编辑

纯C的情况下,小内存而追求效率时,同时使用链表和红黑树
1. 使用链表,方便增加、删除
2. 红黑树,用于查找,算法复杂度就跟排序后的二分法一样

出0入17汤圆

发表于 2021-2-25 22:00:36 | 显示全部楼层
think_a_second 发表于 2021-2-25 16:11
纯C的情况下,小内存而追求效率时,同时使用链表和红黑树
1. 使用链表,方便增加、删除
2. 红黑树,用于查 ...

https://www.amobbs.com/thread-5716767-1-1.html
看了这个帖子,只用红黑树就可以了

出0入31汤圆

发表于 2021-2-26 16:31:29 来自手机 | 显示全部楼层
二维数组,一纬学号,一纬成绩
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-4-27 06:36

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表