←
examples/advanced/05_nd_recursion.cat
#!/usr/bin/env catnip
# RUN: catnip docs/examples/advanced/05_nd_recursion.cat
# ND-Recursion : syntaxe des opérateurs
#
# Ces exemples illustrent la forme implémentée.
b = import('builtins')
is_list = (x) => { b.isinstance(x, b.list) }
# ~~ : ND-recursion
# Forme combinateur : seed initial + lambda avec recur
~~(0, (v, recur) => {
if v < 10 { recur(v + 1) }
else { v }
})
# Forme déclaration : fonction ND-recursive
countdown = ~~(n, recur) => {
if n > 0 { recur(n - 1) }
else { "done" }
}
# Forme broadcast simple (equivalent a map)
data = list(1, 2, 3)
data.[~~(v, _) => { v * 2 }]
# Forme broadcast récursive (traverse les structures)
nested = list(1, list(2, 3), 4)
nested.[~~(v, recur) => {
if is_list(v) { v.[~>(x) => { recur(x) }] }
else { v * 2 }
}]
# ~> : ND-map (lift)
# Lift une fonction dans le contexte ND
lifted_abs = ~> abs
# Forme broadcast : map en contexte ND
data.[~> abs]
# ~[] : topos vide
# Element neutre des operations ND
empty = ~[]
# Modes d'exécution
# Mode sequential (défaut) : debug, petits calculs
pragma('nd_mode', ND.sequential)
~~(10, (n, r) => { if n <= 1 { 1 } else { n * r(n - 1) } })
# Mode threads : memoization partagee, I/O bound
pragma('nd_mode', ND.thread)
pragma('nd_memoize', True)
~~(20, (n, r) => { if n <= 1 { n } else { r(n - 1) + r(n - 2) } })
# Mode processes : vrai parallelisme, CPU bound
pragma('nd_mode', ND.process)
pragma('nd_workers', 4)
list(5, 6, 7, 8).[~~(n, r) => {
if n <= 1 { 1 }
else { n * r(n - 1) }
}]