- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我在 answer to a previous question 中认为可以在 Haskell 中表示 primitive recursive functions (PRF) 和 ⊥ 或 undefined
的单个额外值的联合。这个论点是基于对原始递归函数的公理结构的直接翻译;它需要许多语言扩展和关于函数数量的类型级推理。是否可以用更惯用的 Haskell 来表示一组等效的原始递归函数?
理想情况下,PRF 的惯用表示应该能够满足以下所有条件:
Category
实例 undefined
函数都是所有输入的 undefined
。这将 PRF 集限制为单个不可避免的额外值 ⊥ 而不是包含多个部分递归函数。这意味着 while
循环或类似部分递归函数的任何定义都应该是 undefined
。 Category
定律,
Arrow
没有
arr
(实际上它与
arr
相反),以及仅适用于自然数的有限循环形式。
最佳答案
Haskell 中有一个非常简单的原始递归函数表示。这是一个函数的 newtype
,我们将断言它是一个正确的构造原始递归函数。我们不导出构造函数以防止构造可能是部分递归的任意函数。这种技术称为 smart constructor 。
module Data.PRF (
-- We don't export the PRF constructor
PRF (runPRF),
) where
newtype PRF b c = PRF {runPRF :: b -> c}
PRF
s。
Category
实例将提供 PRF 所需的扩展组合的组合部分。
import Prelude hiding (id, (.), fst, snd, succ)
import Control.Category
instance Category PRF where
id = PRF id
PRF f . PRF g = PRF $ f `seq` g `seq` (f . g)
seq
s 要求
f
和
g
是弱头范式,然后才能从中计算出任何结果;如果任一函数是
undefined
,那么组合也将是
undefined
。
fst
和
snd
。结合类似
Arrow
的
(&&&)
来构建元组,我们可以满足扩展投影的所有要求。 PRF 就像“没有
arr
的箭头”;
arr
将允许将任意部分递归函数制成
PRFs
。我们将定义
ArrowLike
类别的类。
class Category a => ArrowLike a where
fst :: a (b, d) b
snd :: a (d, b) b
(&&&) :: a b c -> a b c' -> a b (c,c')
first :: a b c -> a (b, d) (c, d)
first = (*** id)
second :: a b c -> a (d,b) (d,c)
second = (id ***)
(***) :: a b c -> a b' c' -> a (b,b') (c,c')
f *** g = (f . fst) &&& (g . snd)
fst
和
snd
代替了
arr
。当与扇出
ArrowLike
结合使用时,它们是描述
(&&&)
行为所需的唯一函数。
ArrowLike
提供
PRF
实例之前,我们会说普通函数
(->)
是如何
ArrowLike
import qualified Prelude (fst, snd)
instance ArrowLike (->) where
fst = Prelude.fst
snd = Prelude.snd
f &&& g = \b -> (f b, g b)
PRF
s,我们将使用我们在
(.)
定义中为
Category
实例使用的相同归纳步骤,并要求两个函数都处于弱头范式。
instance ArrowLike PRF where
fst = PRF fst
snd = PRF snd
PRF f &&& PRF g = PRF $ f `seq` g `seq` (f &&& g)
class ArrowLike a => PrimRec a where
zero :: a b Nat
succ :: a Nat Nat
prec :: a e c -> a (c, (Nat,e)) c -> a (Nat, e) c
Nat
是
data Nat = Z | S Nat
给出的自然数。我选择将常量函数
zero
和后继函数视为原始递归的一部分,可以解构或检查它们构造的
Nat
值的唯一方法是使用
prec
。用
zero
替换
const :: c -> a b c
很诱人;这将是一个致命的缺陷,因为有人可以通过
infinity = S infinity
引入
const
来将
prec
变成无限循环。
(->)
支持原始递归。
instance PrimRec (->) where
zero = const Z
succ = S
prec f g = go
where
go (Z, d) = f d
go (S n, d) = g (go (n, d), (n, d))
PRF
和
(.)
相同的归纳技巧为
(&&&)
定义原始递归。
instance PrimRec PRF where
zero = PRF zero
succ = PRF succ
prec (PRF f) (PRF g) = PRF $ f `seq` g `seq` prec f g
Category
,具有构造和解构元组和自然数的能力。
add
。
import Prelude hiding (id, (.), fst, snd, succ)
import Control.Category
import Data.PRF
add :: PrimRec a => a (Nat, Nat) Nat
add = prec id (succ . fst)
match
,它有助于构建原始递归函数,这些函数根据自然数是否为零进行分支。
match :: PrimRec a => a b c -> a (Nat, b) c -> a (Nat, b) c
match fz fs = prec fz (fs . snd)
match
我们可以轻松快速地测试一个值是否为
Z
以及最终是否为奇数
one :: PrimRec a => a b Nat
one = succ . zero
nonZero :: PrimRec a => a Nat Nat
nonZero = match zero one . (id &&& id)
isZero :: PrimRec a => a Nat Nat
isZero = match one zero . (id &&& id)
isOdd :: PrimRec a => a Nat Nat
isOdd = prec zero (isZero . fst) . (id &&& id)
PRF
将是
undefined
。
while :: PrimRec a => a s Nat -> a s s -> a s s
while test step = goTest
where
goTest = goMatch . (test &&& id)
goMatch = match id (goStep . snd)
goStep = goTest . step
infiniteLoop
仅在奇数输入时无法终止。
infiniteLoop :: PrimRec a => a Nat Nat
infiniteLoop = while isOdd (succ . succ)
import System.IO
mseq :: Monad m => a -> m a
mseq a = a `seq` return a
run :: Show b => PRF a b -> a -> IO ()
run f i =
do
putStrLn "Compiling function"
hFlush stdout
f' <- mseq $ runPRF f
putStrLn "Running function"
hFlush stdout
n <- mseq $ f' i
print n
PRF
方便地定义的
match
。
run isOdd (S $ S $ S Z)
Compiling function
Running function
S Z
infiniteLoop
定义的函数一般都是
undefined
,而不仅仅是奇数。
run infiniteLoop (Z)
Compiling function
关于haskell - Haskell 中原始递归函数的良好表示,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27387930/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!