Delphi 高效处理大数据量的字典数据的查询问题

一、需求

系统中存在基本字典数据,且数据量大,需要根据各种条件查询某个字典数据,该如何高效实现?

例如:系统中的菜品字典数据,需要根据菜品ID、菜品编码、菜品名+菜品规格等条件查找菜品

二、思路

1、使用索引的方法

将每个查询条件都建立一个TStringList,并添加条件值,并确保其与菜品对象 TMenuItem 在TList中的位置一样。 查询时,根据条件值找到TStringList中的位置 index,然后直接通过TList[index] 获取菜品对象。

该方法查询时不用遍历整个 TList ,从而提高查询效率。

2、使用哈希方法

将每个查询条件添加到 TDictionary 字典 TFMenuItemHash(TDictionary<string, TMenuItem>),Key 值为 【条件前缀+菜品对应ID等值】,Value 值为 TMenu 对象地址。查询时,通过FMenuItemHash.TryGetValue 获取。

注意:程序结束时,需要遍历 TFMenuItemHash,释放 TFMenuItemHash 对应的 TMenu 对象

三、代码实现

这里只展示思路2对应的代码实现。

unit uMenuItem;interfaceusesWinapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, System.Generics.Collections;//定义查询条件类型  按菜品ID  菜品Key  菜品编码  菜品名称  菜品名称+规格  菜品ID+规格
typeTMenuItemListType = (mltID, mltKey, mltCode, mltName, mltNameUnit, mltIDUnit, mltKeyUnit);
constMenuItemListTypeArry: array [TMenuItemListType] of string = ('mltID', 'mltKey', 'mltCode', 'mltName', 'mltNameUnit', 'mltIDUnit', 'mltKeyUnit');type//菜品类TMenuItem = class//...菜品ID、菜品名等属性end;//菜品字典列表类TMenuItemLst = classprivateFMenuItemHash: TDictionary<string, TMenuItem>;//添加Key值到 FMenuItemHash 中procedure AddMenuItem(MenuItemType: TMenuItemListType; AKey: string; AMenuItem: TMenuItem);//从 FMenuItemHash 中获取 key 值对应的 TMenuItem 对象function GetMenuItem(MenuItemType: TMenuItemListType; AKey: string): TMenuItem;function InitData(): boolean;function GetCount: integer;function GetItems(Index: integer): TMenuItem;protectedFList: TList<TMenuItem>;publicconstructor Create(); overload;destructor Destroy; override;property Count: integer read GetCount;property Items[Index: integer]: TMenuItem read GetItems; default;procedure Clear;procedure ClearFoodCannotSaleList;procedure Delete(iIndex: integer);function IndexOf(AMenuItem: TMenuItem): integer;function Refresh(): Boolean;//按菜品ID查询function GetMenuItemByFoodID(AFoodID: int64): TMenuItem;//按菜品规格查询function GetMenuItemByUnitKey(AUnitKey: string): TMenuItem;//按菜品编码查询function GetMenuItemByFoodCode(AFoodCode: string): TMenuItem;//按菜品名称查询function GetMenuItemByFoodName(AFoodName: string): TMenuItem;//按菜品名称+规格查询function GetMenuItemByFoodNameUnit(AFoodName, AFoodUnit: string): TMenuItem;//按菜品ID+规格查询function GetMenuItemByFoodIDUnit(AFoodID: int64; AFoodUnit: string): TMenuItem;//按菜品Key+规格查询function GetMenuItemByFoodKeyUnit(AFoodKey, AFoodUnit: string): TMenuItem;
end;implementation{ TMenuItemLst }procedure TMenuItemLst.Clear;
vari: integer;
beginfor i := 0 to FList.Count - 1 dobeginFList[i].Free;end;FList.Clear;FMenuItemHash.Clear;
end;constructor TMenuItemLst.Create;
beginFList := TList<TMenuItem>.Create;FMenuItemHash := TDictionary<string, TMenuItem>.Create;
end;procedure TMenuItemLst.Delete(iIndex: integer);
beginTMenuItem(FList[iIndex]).Free;FList.Delete(iIndex);
end;destructor TMenuItemLst.Destroy;
beginClear;FList.Free;FMenuItemHash.Free;inherited;
end;function TMenuItemLst.GetCount: integer;
beginResult := FList.Count;
end;procedure TMenuItemLst.AddMenuItem(MenuItemType: TMenuItemListType;AKey: string; AMenuItem: TMenuItem);
beginFMenuItemHash.AddOrSetValue(MenuItemListTypeArry[MenuItemType] + AKey, AMenuItem);
end;function TMenuItemLst.GetMenuItem(MenuItemType: TMenuItemListType;AKey: string): TMenuItem;
beginFMenuItemHash.TryGetValue(MenuItemListTypeArry[MenuItemType] + AKey, Result);
end;function TMenuItemLst.GetItems(Index: integer): TMenuItem;
beginResult := nil;if (Index >= 0) and (Index < FList.Count) thenResult := TMenuItem(FList[Index]);
end;function TMenuItemLst.GetMenuItemByFoodCode(AFoodCode: string): TMenuItem;
beginResult := GetMenuItem(mltCode, AFoodCode);
end;function TMenuItemLst.GetMenuItemByFoodID(AFoodID: int64): TMenuItem;
beginResult := GetMenuItem(mltID, AFoodID.ToString);
end;function TMenuItemLst.GetMenuItemByFoodIDUnit(AFoodID: int64;AFoodUnit: string): TMenuItem;
beginResult := GetMenuItem(mltIDUnit, AFoodID.ToString + AFoodUnit);
end;function TMenuItemLst.GetMenuItemByFoodKeyUnit(AFoodKey,AFoodUnit: string): TMenuItem;
beginResult := GetMenuItem(mltKeyUnit, AFoodKey + AFoodUnit);
end;function TMenuItemLst.GetMenuItemByFoodName(AFoodName: string): TMenuItem;
beginResult := GetMenuItem(mltName, AFoodName);
end;function TMenuItemLst.GetMenuItemByFoodNameUnit(AFoodName,AFoodUnit: string): TMenuItem;
beginResult := GetMenuItem(mltNameUnit, AFoodName + AFoodUnit);
end;function TMenuItemLst.GetMenuItemByUnitKey(AUnitKey: string): TMenuItem;
beginResult := GetMenuItem(mltKey, AUnitKey);
end;function TMenuItemLst.IndexOf(AMenuItem: TMenuItem): integer;
beginResult := FList.IndexOf(AmenuItem);
end;function TMenuItemLst.InitData(): boolean;
vari: integer;menuItem: TMenuItem;foodID, foodUnitKey, foodCode, foodName, foodUnit: string;
beginresult := false;trytry//循环数据集, 假如已从数据库中查询了所有菜品字典数据for i := 0 to 10000 dobegin//初始化菜品对象,在 TMenuItemLst 类释放时,需要循环释放 TMenuItemmenuItem := TMenuItem.Create();foodID := menuItem.FoodID;foodUnitKey := menuItem.FoodUnitKey;foodCode := menuItem.FoodCode;foodName := menuItem.FoodName;foodUnit := menuItem.FoodUnit;if foodID <> '' then           //添加菜品ID的Key值AddMenuItem(mltID, foodID, menuItem);if foodUnitKey <> '' then      //添加菜品Key的Key值AddMenuItem(mltKey, foodUnitKey, menuItem);if FoodCode <> '' then         //添加菜品编码的Key值AddMenuItem(mltCode, foodCode, menuItem);if foodName <> '' thenAddMenuItem(mltName, foodName, menuItem);if (foodName <> '') or (foodUnit <> '') thenAddMenuItem(mltNameUnit, foodName + foodUnit, menuItem);if (foodID <> '') or (foodUnit <> '') thenAddMenuItem(mltIDUnit, foodID + foodUnit, menuItem);if (foodUnitKey <> '') or (foodUnit <> '') thenAddMenuItem(mltKeyUnit, foodUnitKey + foodUnit, menuItem);FList.Add(menuItem);end;result := true;excepton E: Exception dobeginraise Exception.Create(E.Message);end;end;finallyend;
end;function TMenuItemLst.Refresh(): Boolean;
beginend;end.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/12590.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

uview-plus在uniapp项目中单选和复选框不显示问题

在我的uniapp小程序项目中&#xff0c;我使用了vue3ts的组合&#xff0c;ui组件库使用了uview-plus这个组件库&#xff0c;但是在使用个别组件的时候&#xff0c;没有显示出效果&#xff0c;就像单选或者复选框&#xff0c;官方效果&#xff1a; 但是当我用到自己项目中的时候&…

[晕事]今天做了件晕事33 c++ mangle 函数名

作为一名c的程序员&#xff0c;开始使用c的时候&#xff0c;如果工程里同时有c和c&#xff0c;而且函数相互调用。 这个时候&#xff0c;需要注意&#xff0c; https://mzhan017.blog.csdn.net/article/details/121950739。 这个最好是在提示的时候&#xff0c;直接按照mangle…

【全开源】Java同城信息付费系统家政服务房屋租赁房屋买卖房屋装修信息发布平台小程序APP公众号源码

同城信息付费系统&#xff1a;家政服务的新篇章 在快节奏的现代生活中&#xff0c;家政服务已成为许多家庭不可或缺的一部分。然而&#xff0c;如何快速、准确地找到合适、可靠的家政服务人员&#xff0c;一直是困扰着许多家庭的问题。为了解决这一难题&#xff0c;我们推出了…

5.14号模拟前端面试10问

1. setState是同步还是异步 setState是React中用于更新组件状态的方法。在React中&#xff0c;setState可以是同步的&#xff0c;也可以是异步的&#xff0c;具体取决于调用场景和React的内部实现。 在大部分情况下&#xff0c;setState是异步的。当setState被调用时&#xf…

一图流解释Java中线程状态的转换

目录 一.Java中的几大线程状态 二.线程之间的相互转换 ▐ NEW --> RUNNABLE ▐ RUNNABLE <--> WAITING ▐ RUNNABLE <--> Timed Waiting ▐ RUNNABLE<--> BLOCKED ▐ RUNNABLE<-->TERMINATED 一.Java中的几大线程状态 简单来说线程可以处于…

PCIE协议-2-事务层规范-Message Request Rules-Vendor_Defined Messages

2.2.8.6 厂商定义消息 厂商定义消息允许扩展PCI Express消息功能&#xff0c;可以作为PCI Express规范的一般扩展&#xff0c;也可以是厂商特定的扩展。本节通用地定义了与这些消息相关的规则。 厂商定义消息&#xff08;见表2-25&#xff09;使用图2-28中显示的头标格式。re…

银行核心背后的落地工程体系丨混沌测试的场景设计与实战演练

本文作者&#xff1a; 张显华、窦智浩、卢进文 与集中式架构相比&#xff0c;分布式架构的系统复杂性呈指数级增长&#xff0c;混沌工程在信创转型、分布式架构转型、小机下移等过程中有效保障了生产的稳定性。本文分享了 TiDB 分布式数据库在银行核心业务系统落地中进行混沌测…

【Axure视频教程】中继器表格查询数据并标记变色

今天教大家在Axure制作中继器表格数据査询并标记变色的原型模板&#xff0c;我们在输入框里输入对应的区间&#xff0c;然后点击标注按钮&#xff0c;可以快速找到中继器里符合条件的值&#xff0c;并将他高亮显示&#xff0c;该板的表格是用中继器制作的&#xff0c;所以使用也…

问题—前端调用接口url多加一个/,本地可以调通,测试环境报错302,分开调两个接口

问题背景 接口url前面多加一个/ &#xff0c;npm run serve 起项目&#xff0c;本地调用正常 npm run build 打包到测试环境&#xff0c;接口出现问题&#xff0c;分开调用接口&#xff0c;且报302错误 问题原因&#xff1a; 本地开发环境和测试环境的URL处理方式不同 本地使…

动态规划-两个数组的dp问题3

文章目录 1. 两个字符串的最小ASCII删除和&#xff08;712&#xff09;2. 最长重复子数组&#xff08;718&#xff09; 1. 两个字符串的最小ASCII删除和&#xff08;712&#xff09; 题目描述&#xff1a; 状态表示&#xff1a; 根据经验以及题目要求&#xff0c;建立二维数…

旧手机-基于Termux配置服务器(不用拿去换钢盆了)

Hi&#xff0c;大家好&#xff0c;我是抢老婆酸奶的小肥仔。 大家用手机这么多年了&#xff0c;手上或多或少都有一两个被替换下来的旧手机&#xff0c;也不用拿去换啥钢盆了&#xff0c;使用Termux可以将旧手机改造成一个服务器。 不多说&#xff0c;直接开干。 1、安装app…

Swagger --学习笔记

什么是Swagger&#xff1f; Swagger是API设计工具集&#xff0c;用于帮助开发者设计、构建和文档化RESTful Web服务。它通过标准化的格式来描述API接口&#xff0c;使得创建、维护和使用API变得更加清晰和简便&#xff0c;同时提供工具支持API的交互式文档、编辑器以及代码的自…

深入理解JVM:介绍JVM的工作原理,包括类加载机制,内存模型,垃圾回收机制等

类加载机制&#xff1a; JVM的类加载机制主要包括加载、连接&#xff08;验证、准备和解析&#xff09;、初始化、使用和卸载五个阶段。第一个阶段是加载需求的.class文件到内存中。第二个阶段是完成对字节码的验证&#xff0c;为类变量分配内存并初始化为对应类型默认值。第三…

别再找了!吐血整理ChatGPT 3.5/4.0新手使用手册

引领科技潮流的ChatGPT早已名声在外&#xff0c;如今获取ChatGPT已变得触手可及&#xff0c;但很多人还多次提问如何使用chatgpt&#xff0c;为了避免陷入误区&#xff0c;本文旨在为广大ChatGPT爱好者提供一份实用的指南。 因此&#xff0c;帮助大家更好地掌握其使用技巧&…

python的字典学习笔记

来自WeTab AI PRO 1.两个元素数量相同的元组&#xff0c;将其中一个元组的每个元素作为键&#xff0c;将另一个元组的每个元素作为值&#xff0c;生成字典 要根据两个元素数量相同的元组生成字典&#xff0c;其中一个元组的每个元素作为键&#xff0c;另一个元组的每个元素作…

leetcode.环形链表问题

目录 题目1 示例 解题思路 代码实现 补充 题目2 示例 解题思路 代码实现 题目1 该题链接&#xff1a;https://leetcode.cn/problems/linked-list-cycle/description/ 示例 解题思路 要创建两个指针一个是快指针(fast)&#xff0c;另一个慢指针(slow)。快指针走两步慢指…

Linux修改终端命令颜色

1.在家目录中修改.bashrc文件 cd ~ vim .bashrc2.找到PS1相关段落&#xff0c;把其他的注释掉&#xff0c;填上该行代码&#xff0c;修改为自己设置的颜色 (具体颜色查看参考文章) 提供两种颜色&#xff0c;其他的自学调色盘吧(下文有)~ (祝你愉快) ①浅蓝色 深蓝 PS1\[\03…

华医网在线教育系统,经过疫情,知识付费行业的机遇!

因为知识付费的行业入门门槛很低&#xff0c;所以产品的质量也参差不齐&#xff0c;因为疫情的影响&#xff0c;知识付费行业迎来了一波机遇。 因为知识付费门槛低&#xff0c;所以持续创造优质内容才是王道&#xff1b;现在成熟的知识付费平台和产品受到了大部分用户的喜欢&am…

WebLogic Serer的服务策略

服务策略 Oracle提供终身服务策略。以下按照服务级别,分别说明了各级别的服务期间以及服务内容。 Premier Support 服务期间 制品出荷~开始后的第5年 服务内容 提供程序的更新版提供新做成的补丁提供安全情报提供Critical Patch Update税务法律合规提供升级脚本大多数新产…

CSRF 攻击实验:Token 不存在绕过验证

前言 CSRF&#xff08;Cross-Site Request Forgery&#xff09;&#xff0c;也称为XSRF&#xff0c;是一种安全漏洞&#xff0c;攻击者通过欺骗用户在受信任网站上执行非自愿的操作&#xff0c;以实现未经授权的请求。 CSRF攻击利用了网站对用户提交的请求缺乏充分验证和防范…