- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我知道在 Haskell 中构建 Map
的几种不同方法:
fromList
从列表构建它fromAscList
从排序列表构建它Map
是一个 Monoid
(或 Semigroup
)和 concat singletons
的事实。<据我了解,#1 的摊销复杂度为 O(n*log(n)),而 #2 的复杂度为 O(n)。我猜#3 应该大致相当于#1,并且可能会发生融合。
摊销很重要,因为 Haskell 默认情况下是懒惰的,即使从 Map 进行查找的时间复杂度为 O(log(n)),但实际上它可以与 Map 的构造交错它本身的时间复杂度为 O(n * log(n)),这在实践中可以使查找时间为 O(n * log(n)) (特别是如果您每次需要时都构建 map )。如果您使用硬编码 map ,这也可能会附加
例如,我是否认为 lookup 'b' (fromList [('a', 1), ('b', 2)])
实际上相当于 d在列表中查找而不使用中间映射?
那么 #1 和 #3 之间有区别吗?或者对列表进行排序和调用 fromList
之间有区别吗?
另外,如果我需要一个映射仅计算一次,我是否需要确保 GHC 不会内联它,以便它在函数之间共享?
我意识到这个问题可能有点模糊,实际上对应于我最近遇到的不同用例。
第一个对应于“静态连接”。我有一个管理项目的应用程序,每个项目代码都可以按样式和变体进行拆分(例如 'T-Shirt-Red' => ('T-Shirt', 'Red')。拆分基于规则(和regexp) 并且非常慢。为了不总是重新计算规则,只需执行一次并存储在数据库表中。我有一些纯函数需要能够拆分项目代码,所以我向它们传递了一个函数 Text -> (Text, Text)
,该函数实际上是部分应用于Map的查找,代码类似这样
getSplitter :: Handler (Text -> (Text, Text))
getSplitter = do
sku'style'vars <- runDB $ rawSQL "SELECT sku, style, var FROM style_cache " [] -- load the split table
let sku'map = fromList [ (sku, (style,var))
| (sku, style, var) <- sku'style'vars
]
return $ flip lookup sku'map
通过按 sku
对项目进行排序并使用 fromDistinctAscList
(实际上比 fromAscList
更快),可以轻松加快此速度。但是,我仍然有一些关于如何在不同请求之间缓存它的问题。
第二种情况是在两个表之间进行手动联接,我通常会同时执行一些操作
do
sku'infos <- selectList [] [] -- load item info
let skuInfo = fromList sku'infos
orderLines <- selectList [] [] -- load orders
-- find infos corresponding to order items
mapM (\o -> (o, lookup (orderSku o) skuInfo) orderLines
我可以再次在 SQL 中对 sku'infos 进行排序并使用 fromDistinctAscList。
第三种情况是获取与不同表中的类别项相关的杂项信息时。
例如,我希望能够按类别比较销售(销售表)和购买(购买表)。
在纯 SQL 中我会做一些事情
SELECT style, sum(sales.amount), sum(purchase.amount)
FROM style_cache
LEFT JOIN sales USING(sku)
LEFT JOIN purchases USING(sku)
GROUP by style
但是,这是一个简化的示例,实际上,聚合要复杂得多,并且必须在 Haskell 中完成以及连接。为此,我分别加载每个表(对 sql 中的内容进行分组)并返回Map Style SalesInfo
、Map Style PurchaseInfo
等...并将它们合并。该表相当大,我意识到我最终将所有内容加载到内存中,而我可能可以通过手动“压缩”内容来提高效率,但我不确定如何实现。
最佳答案
我不确定我是否理解这个问题背后的全部动机,但我会发表一些评论:
Map
是脊柱严格的——这意味着 Map
的树结构并且 key 本身在每个 Map
上都被强制(至少足够远以进行所有必要的比较)手术。所以我期望 Data.Map.lookup k (fromList xs)
进行 O(n*log(n)) 比较(n xs
的长度),而我期望 Prelude.lookup k xs
进行 O(n) 比较(实际上只是相等检查,但通常这与比较的复杂性几乎相同)。fromAscList . sort
确实比 fromList
更快,这是 Data.Map
中的性能错误并且库应该更改为定义 fromList = fromAscList . sort
。如果是这样的话,我会感到非常惊讶。人们花了相当多的时间来优化 containers
,所以我不希望看到任何水果挂得那么低。关于haskell - 在 Haskell 中构建(惰性)映射的有效方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52016512/
我想了解 Ruby 方法 methods() 是如何工作的。 我尝试使用“ruby 方法”在 Google 上搜索,但这不是我需要的。 我也看过 ruby-doc.org,但我没有找到这种方法。
Test 方法 对指定的字符串执行一个正则表达式搜索,并返回一个 Boolean 值指示是否找到匹配的模式。 object.Test(string) 参数 object 必选项。总是一个
Replace 方法 替换在正则表达式查找中找到的文本。 object.Replace(string1, string2) 参数 object 必选项。总是一个 RegExp 对象的名称。
Raise 方法 生成运行时错误 object.Raise(number, source, description, helpfile, helpcontext) 参数 object 应为
Execute 方法 对指定的字符串执行正则表达式搜索。 object.Execute(string) 参数 object 必选项。总是一个 RegExp 对象的名称。 string
Clear 方法 清除 Err 对象的所有属性设置。 object.Clear object 应为 Err 对象的名称。 说明 在错误处理后,使用 Clear 显式地清除 Err 对象。此
CopyFile 方法 将一个或多个文件从某位置复制到另一位置。 object.CopyFile source, destination[, overwrite] 参数 object 必选
Copy 方法 将指定的文件或文件夹从某位置复制到另一位置。 object.Copy destination[, overwrite] 参数 object 必选项。应为 File 或 F
Close 方法 关闭打开的 TextStream 文件。 object.Close object 应为 TextStream 对象的名称。 说明 下面例子举例说明如何使用 Close 方
BuildPath 方法 向现有路径后添加名称。 object.BuildPath(path, name) 参数 object 必选项。应为 FileSystemObject 对象的名称
GetFolder 方法 返回与指定的路径中某文件夹相应的 Folder 对象。 object.GetFolder(folderspec) 参数 object 必选项。应为 FileSy
GetFileName 方法 返回指定路径(不是指定驱动器路径部分)的最后一个文件或文件夹。 object.GetFileName(pathspec) 参数 object 必选项。应为
GetFile 方法 返回与指定路径中某文件相应的 File 对象。 object.GetFile(filespec) 参数 object 必选项。应为 FileSystemObject
GetExtensionName 方法 返回字符串,该字符串包含路径最后一个组成部分的扩展名。 object.GetExtensionName(path) 参数 object 必选项。应
GetDriveName 方法 返回包含指定路径中驱动器名的字符串。 object.GetDriveName(path) 参数 object 必选项。应为 FileSystemObjec
GetDrive 方法 返回与指定的路径中驱动器相对应的 Drive 对象。 object.GetDrive drivespec 参数 object 必选项。应为 FileSystemO
GetBaseName 方法 返回字符串,其中包含文件的基本名 (不带扩展名), 或者提供的路径说明中的文件夹。 object.GetBaseName(path) 参数 object 必
GetAbsolutePathName 方法 从提供的指定路径中返回完整且含义明确的路径。 object.GetAbsolutePathName(pathspec) 参数 object
FolderExists 方法 如果指定的文件夹存在,则返回 True;否则返回 False。 object.FolderExists(folderspec) 参数 object 必选项
FileExists 方法 如果指定的文件存在返回 True;否则返回 False。 object.FileExists(filespec) 参数 object 必选项。应为 FileS
我是一名优秀的程序员,十分优秀!