AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 77922852
Accepted
salferdez
salferdez
Asked: 2024-02-02 03:25:02 +0800 CST2024-02-02 03:25:02 +0800 CST 2024-02-02 03:25:02 +0800 CST

提高斯特林数计算效率

  • 772

我一直在尝试探索Haskell中递归的可能性,但在面对大递归时如何提高效率我不太了解。今天,我想做一个计算第二类斯特林数的函数:

factorial :: Integer -> Integer
factorial n = helper n 1
    where
        helper 1 a = a
        helper n a = helper (n-1) (a*n)


stirling :: Integer -> Integer -> Integer
stirling n 1 = 1
stirling n 2 = ((2 ^n)- 2) `div` 2
stirling n k
    | k > n      = 0
    | k == n     = 1
    | k == (n-1) = (factorial n) `div` (factorial 2 * (factorial(n-2)))
    | otherwise  = stirling (n-1) (k-1) + k *(stirling (n-1) k)

我一直在尝试创建一个辅助函数,就像阶乘定义中一样,可以避免 Haskell 的惰性求值占用过多内存资源带来的不便。我想做同样的事情,在每次调用中计算乘积

k *(stirling (n-1) k)

递归,但我没有掌握它的窍门。有任何想法吗?

PD:代码已经运行得相当不错,但我是 Haskell 的新手,我仍然不知道可能性是什么。欢迎任何改进(尽管我认为不使用任何库;我想在进入之前学习基础知识)。

performance
  • 2 2 个回答
  • 39 Views

2 个回答

  • Voted
  1. Noughtmare
    2024-02-02T04:31:11+08:002024-02-02T04:31:11+08:00

    我现在无法写出非常详细的解释,但您可以通过以下方式使其更快:

    • 使用数学简化一些公式
    • 使用移位代替除法
    • 用于Int输入而不是Integer

    然后你会得到这个:

    import Data.Bits
    
    stirling :: Int -> Int -> Integer
    stirling n 1 = 1
    stirling n 2 = (1 `unsafeShiftL` (n - 1)) - 1
    stirling n k
        | k > n        = 0
        | k == n       = 1
        | k == (n - 1) = fromIntegral ((n * (n - 1)) `unsafeShiftR` 1)
        | otherwise    = stirling (n - 1) (k - 1) + fromIntegral k * stirling (n - 1) k
    

    我接下来要尝试的是动态编程以避免冗余计算。

    并且您使用累加器样式使非尾递归调用之一成为尾递归,就像在此博客文章中所做的那样:https ://www.haskell.org/ghc/blog/20190728-free-variable-traversals.html

    • 0
  2. Best Answer
    Daniel Wagner
    2024-02-02T04:41:31+08:002024-02-02T04:41:31+08:00

    至少在我的测试中,您的实现似乎已经具有相当的内存效率。例如:

    % time ./test salferdez 32 16
    1194461517469807833782085
    11.50user 0.00system 0:11.48elapsed 100%CPU (0avgtext+0avgdata 8704maxresident)k
    0inputs+0outputs (0major+1267minor)pagefaults 0swaps
    

    更改参数(向上或向下)似乎不会对 8MB 驻留发生太大变化,因此我预计这只是 Haskell 程序的最低启动成本。另一方面,它似乎很慢。参数甚至比我在这里使用的参数稍大一些——比如,将它们都加倍——花费的时间比我愿意等待的时间还要长。直接直接实现维基百科中的公式似乎要快得多,同时保留良好的内存配置文件:

    % time ./test dmwit 32 16
    1194461517469807833782076
    0.00user 0.00system 0:00.01elapsed 16%CPU (0avgtext+0avgdata 4352maxresident)k
    0inputs+0outputs (0major+206minor)pagefaults 0swaps
    % time ./test dmwit 10000 5000
    <snip'd very very long output>
    10.62user 0.02system 0:10.62elapsed 100%CPU (0avgtext+0avgdata 16716maxresident)k
    0inputs+0outputs (1major+26643minor)pagefaults 0swaps
    

    (我没有在这些运行和之前的运行之间重新编译,因此您可以确定我们都获得了相同级别的优化。)

    另一方面,我得到的答案确实与你略有不同。也许您可以在我们的一个或另一个代码中发现错误。无论如何,这是来自维基百科*的公式,这是我对其最直接翻译的看法:

    stirling n k = sum [(-1)^(k-i)*i^n`div`(product [1..k-i]*product[1..i]) | i <- [0..k]]
    

    不要忘记使用 GHC 进行编译-O2,并让 GHC 完成所有思考如何使其快速且内存高效的肮脏工作。

    * StackOverflow 确实有一种直接包含图像而不是链接图像的机制,但它似乎对我不起作用。不知道为什么!

    • 0

相关问题

  • Haskell 速度问题,执行程序的两个部分比单独执行任一部分花费的时间要长得多

  • 为什么优化后的 g 脚本代码比“低效”的代码慢?

  • 如何让JMeter用户不等待响应

  • 如何解释两个处理器之间巨大的执行速度差异?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行?

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    何时应使用 std::inplace_vector 而不是 std::vector?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Marko Smith

    我正在尝试仅使用海龟随机和数学模块来制作吃豆人游戏

    • 1 个回答
  • Martin Hope
    Aleksandr Dubinsky 为什么 InetAddress 上的 switch 模式匹配会失败,并出现“未涵盖所有可能的输入值”? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge 为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini 具有指定基础类型但没有枚举器的“枚举类”的用途是什么? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer 何时应使用 std::inplace_vector 而不是 std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB 为什么 GCC 生成有条件执行 SIMD 实现的代码? 2024-02-17 06:17:14 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve