我将一个对象存储在一个由许多整数属性描述的数据库中。真实的对象有点复杂,但现在让我们假设我将汽车存储在我的数据库中。每辆汽车都有很多整数属性来描述汽车(即最大速度、轴距、最大功率等),用户可以搜索这些属性。用户为每个对象定义一个首选范围,并且由于有很多属性,所以很可能不会有任何汽车匹配所有属性范围。因此,查询必须返回按最佳匹配排序的汽车数量。
目前我使用以下查询在 MySQL 中实现了这一点:
SELECT *, SQRT( POW((a < min_a)*(min_a - a) + (a > max_a)*(a - max_a), 2) +
POW((b < min_b)*(min_b - b) + (b > max_b)*(b - max_b), 2) +
... ) AS match
WHERE a < (min_a - max_allowable_deviation) AND a > (max_a + max_allowable_deviation) AND ...
ORDER BY match ASC
其中 a 和 b 是对象的属性,min_a、max_a、min_b 和 max_b 是用户定义的值。基本上,匹配是所需范围与属性实际值之间的平方差之和的平方根。值为 0 表示完美匹配。
该表包含几百万条记录,引入 WHERE 子句只是为了限制执行计算的记录数。索引放在所有可查询的记录上,查询大约需要 500 毫秒。我想改进这个数字,我正在研究改进这个查询的方法。
此外,我想知道是否会有更适合执行此工作的不同数据库。此外,我非常想改用 NoSQL 数据库,因为它有更灵活的数据方案选项。我一直在研究 MongoDB,但找不到有效(快速)解决此问题的方法。
有没有比 MySQL 更适合这项工作的数据库?
最佳答案
看看R-trees . (有关特定变体的页面会详细介绍并提供伪代码)。这些数据结构允许您通过边界矩形进行查询,这就是您在每个属性上按范围搜索的问题所在。
将您的汽车视为 n 维空间中的点,其中 n 是描述您的汽车的属性数。然后给定 n 个范围,每个范围描述一个属性,问题是找到包含在该 n 维超矩形中的所有点。 R 树有效地支持此查询。 MySQL 为其空间数据类型实现了 R 树,但 MySQL 仅支持二维空间,这对您来说不够。我不知道有任何现成的支持 n 维 R 树的常见数据库,但是您可以使用一些对用户定义的树数据结构有良好支持的数据库并自己实现 R 树最重要的是。例如,您可以使用子指针为 MongoDB 中的 R 树节点定义一个结构。然后,您将在自己的代码中实现 R 树算法,同时让 MongoDB 负责存储数据。
此外,还有这个 C++ header file R树的实现,但目前它只是一个内存结构。尽管如果您的数据集只有几百万行,那么在启动时加载此内存结构并在添加新车时更新它似乎是可行的(我认为这种情况很少见)。
关于mysql - 选择哪个数据库来查找最佳匹配记录?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6804085/
很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
在我的应用程序中,我需要能够找到所有数字子字符串,然后扫描每个子字符串,找到第一个匹配范围(例如5到15之间)的子字符串,并将该实例替换为另一个字符串“X”。我的测试字符串s="1foo100bar10gee1"我的初始模式是1个或多个数字的任何字符串,例如,re=Regexp.new(/\d+/)matches=s.scan(re)给出["1","100","10","1"]如果我想用“X”替换第N个匹配项,并且只替换第N个匹配项,我该怎么做?例如,如果我想替换第三个匹配项“10”(匹配项[2]),我不能只说s[matches[2]]="X"因为它做了两次替换“1fooX0barXg
如何匹配未被反斜杠转义的平衡定界符对(其本身未被反斜杠转义)(无需考虑嵌套)?例如对于反引号,我试过了,但是转义的反引号没有像转义那样工作。regex=/(?!$1:"how\\"#expected"how\\`are"上面的正则表达式不考虑由反斜杠转义并位于反引号前面的反斜杠,但我愿意考虑。StackOverflow如何做到这一点?这样做的目的并不复杂。我有文档文本,其中包括内联代码的反引号,就像StackOverflow一样,我想在HTML文件中显示它,内联代码用一些spanMaterial装饰。不会有嵌套,但转义反引号或转义反斜杠可能出现在任何地方。
Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/
我刚刚被困在这个问题上一段时间了。以这个基地为例:moduleTopclassTestendmoduleFooendend稍后,我可以通过这样做在Foo中定义扩展Test的类:moduleTopmoduleFooclassSomeTest但是,如果我尝试通过使用::指定模块来最小化缩进:moduleTop::FooclassFailure这失败了:NameError:uninitializedconstantTop::Foo::Test这是一个错误,还是仅仅是Ruby解析变量名的方式的逻辑结果? 最佳答案 Isthisabug,or
我有一个驼峰式字符串,例如:JustAString。我想按照以下规则形成长度为4的字符串:抓取所有大写字母;如果超过4个大写字母,只保留前4个;如果少于4个大写字母,则将最后大写字母后的字母大写并添加字母,直到长度变为4。以下是可能发生的3种情况:ThisIsMyString将产生TIMS(大写字母);ThisIsOneVeryLongString将产生TIOV(前4个大写字母);MyString将生成MSTR(大写字母+tr大写)。我设法用这个片段解决了前两种情况:str.scan(/[A-Z]/).first(4).join但是,我不太确定如何最好地修改上面的代码片段以处理最后一种
有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳
我有两个Rails模型,即Invoice和Invoice_details。一个Invoice_details属于Invoice,一个Invoice有多个Invoice_details。我无法使用accepts_nested_attributes_forinInvoice通过Invoice模型保存Invoice_details。我收到以下错误:(0.2ms)BEGIN(0.2ms)ROLLBACKCompleted422UnprocessableEntityin25ms(ActiveRecord:4.0ms)ActiveRecord::RecordInvalid(Validationfa
我正在尝试解析一个CSV文件并使用SQL命令自动为其创建一个表。CSV中的第一行给出了列标题。但我需要推断每个列的类型。Ruby中是否有任何函数可以找到每个字段中内容的类型。例如,CSV行:"12012","Test","1233.22","12:21:22","10/10/2009"应该产生像这样的类型['integer','string','float','time','date']谢谢! 最佳答案 require'time'defto_something(str)if(num=Integer(str)rescueFloat(s