Combinatory logic: Difference between revisions

From HaskellWiki
m (1) typographic correction 2) swapping the order of mailing list references to make a subset-superset order)
m (simulating TeX's displaymath with MediaWiki's indented math (: <math>...</math>) instead of (unordered) lists)
Line 42: Line 42:
Let us begin with a notion of the ordered pair and denote it by <math>\Diamond_2</math>.
Let us begin with a notion of the ordered pair and denote it by <math>\Diamond_2</math>.
We know this construct well when defining operations for booleans
We know this construct well when defining operations for booleans
* <math>\mathbf{true} \equiv \mathbf K</math>
: <math>\mathbf{true} \equiv \mathbf K</math>
* <math>\mathbf{false} \equiv \mathbf{K_*}</math>
: <math>\mathbf{false} \equiv \mathbf{K_*}</math>
* <math>\mathbf{not} \equiv \mathbf{\Diamond_2}\;\mathbf{false}\;\mathbf{true}</math>
: <math>\mathbf{not} \equiv \mathbf{\Diamond_2}\;\mathbf{false}\;\mathbf{true}</math>
and Church numbers. I think, in generally, when defining datatypes in a continuation-passing way (e.g. Maybe or direct sum), then operations on so-defined datatypes often turn to be well-definable by some <math>\mathbf{\Diamond_n}</math>.
and Church numbers. I think, in generally, when defining datatypes in a continuation-passing way (e.g. Maybe or direct sum), then operations on so-defined datatypes often turn to be well-definable by some <math>\mathbf{\Diamond_n}</math>.


We define it with
We define it with
* <math>\mathbf{\Diamond_2} \equiv \lambda\;x\;y\;f\;.\;f\;x\;y</math>
: <math>\mathbf{\Diamond_2} \equiv \lambda\;x\;y\;f\;.\;f\;x\;y</math>
in <math>lambda</math>-calculus and
in <math>\lambda</math>-calculus and
* <math>\mathbf{\Diamond_2} \equiv\;\mathbf{C_{(1)}}\;\mathbf{C_*}</math>
: <math>\mathbf{\Diamond_2} \equiv\;\mathbf{C_{(1)}}\;\mathbf{C_*}</math>
in combinatory logic.
in combinatory logic.


Line 56: Line 56:
* as the <math>\langle\dots\rangle</math> construct can be generalized to any natural number <math>n</math> (the concept of <math>n</math>-tuple, see Barendregt's <math>\lambda</math> Calculus)
* as the <math>\langle\dots\rangle</math> construct can be generalized to any natural number <math>n</math> (the concept of <math>n</math>-tuple, see Barendregt's <math>\lambda</math> Calculus)
* and in this generalized scheme <math>\mathbf I</math> corresponds to the 0 case, <math>\mathbf{C_*}</math> to the 1 case, and the ordered pair construct <math>\Diamond_2</math> to the 2 case, as though defining
* and in this generalized scheme <math>\mathbf I</math> corresponds to the 0 case, <math>\mathbf{C_*}</math> to the 1 case, and the ordered pair construct <math>\Diamond_2</math> to the 2 case, as though defining
** <math>\mathbf{\Diamond_0} \equiv \mathbf I</math>
: <math>\mathbf{\Diamond_0} \equiv \mathbf I</math>
** <math>\mathbf{\Diamond_1} \equiv \mathbf{C_*}</math>
: <math>\mathbf{\Diamond_1} \equiv \mathbf{C_*}</math>
so we can write definition
so we can write definition
* <math>\mathbf{\Diamond_2} \equiv \mathbf{C_{(1)}}\;\mathbf{C_*}</math>
: <math>\mathbf{\Diamond_2} \equiv \mathbf{C_{(1)}}\;\mathbf{C_*}</math>
or the same
or the same
* <math>\mathbf{\Diamond_2} \equiv \mathbf C \cdot \mathbf{C_*}</math>
: <math>\mathbf{\Diamond_2} \equiv \mathbf C \cdot \mathbf{C_*}</math>
in a more interesting way:
in a more interesting way:
* <math>\mathbf{\Diamond_2} \equiv \mathbf C\cdot\mathbf{\Diamond_1}</math>
: <math>\mathbf{\Diamond_2} \equiv \mathbf C\cdot\mathbf{\Diamond_1}</math>
Is this generalizable? I do not know.
Is this generalizable? I do not know.
I know an analogy in the case of <math>\mathbf{left}</math>, <math>\mathbf{right}</math>, <math>\mathbf{just}</math>, <math>\mathbf{nothing}</math>.
I know an analogy in the case of <math>\mathbf{left}</math>, <math>\mathbf{right}</math>, <math>\mathbf{just}</math>, <math>\mathbf{nothing}</math>.
Line 73: Line 73:
A nice argument described in David Madore's [http://www.madore.org/~david/programs/unlambda/ Unlambda] page gives us a continuation-passig style like solution.
A nice argument described in David Madore's [http://www.madore.org/~david/programs/unlambda/ Unlambda] page gives us a continuation-passig style like solution.
We expect reductions like
We expect reductions like
* <math>\mathbf{left}\;x \to \lambda\;f\;g\;.\;f\;x</math>
: <math>\mathbf{left}\;x \to \lambda\;f\;g\;.\;f\;x</math>
* <math>\mathbf{right}\;x \to \lambda\;f\;g\;.\;g\;x</math>
: <math>\mathbf{right}\;x \to \lambda\;f\;g\;.\;g\;x</math>


so we define
so we define
* <math>\mathbf{left} \equiv \lambda\;x\;f\;g\;.\;f\;x</math>
: <math>\mathbf{left} \equiv \lambda\;x\;f\;g\;.\;f\;x</math>
* <math>\mathbf{right} \equiv \lambda\;x\;f\;g\;.\;g\;x</math>
: <math>\mathbf{right} \equiv \lambda\;x\;f\;g\;.\;g\;x</math>


now we translate it from <math>\lambda</math>-calculus into combinatory logic:
now we translate it from <math>\lambda</math>-calculus into combinatory logic:
* <math>\mathbf{left} \equiv \mathbf{K_{(2)}}\;\mathbf{C_*}</math>
: <math>\mathbf{left} \equiv \mathbf{K_{(2)}}\;\mathbf{C_*}</math>
* <math>\mathbf{right} \equiv \mathbf{K_{(1)}}\;\mathbf{C_*}</math>
: <math>\mathbf{right} \equiv \mathbf{K_{(1)}}\;\mathbf{C_*}</math>


Of course, we can recognize Haskell's direct sum construct
Of course, we can recognize Haskell's direct sum construct
Line 102: Line 102:
* ''j'' as just-continuation
* ''j'' as just-continuation
In a continuation passing style approach, if we want to implement something like the Maybe constuct in <math>\lambda</math>-calculus, then we may expect the following reductions:
In a continuation passing style approach, if we want to implement something like the Maybe constuct in <math>\lambda</math>-calculus, then we may expect the following reductions:
* <math>\mathbf{nothing} \equiv \lambda\;n\;j\;.\;n</math>
: <math>\mathbf{nothing} \equiv \lambda\;n\;j\;.\;n</math>
* <math>\mathbf{just}\;x\;\to\;\lambda\;n\;j\;.\;j\;x</math>
: <math>\mathbf{just}\;x\;\to\;\lambda\;n\;j\;.\;j\;x</math>
we know both of them well, one is just <math>\mathbf K</math>, and we remember the other too from the direct sum:
we know both of them well, one is just <math>\mathbf K</math>, and we remember the other too from the direct sum:
* <math>\mathbf{nothing} \equiv \mathbf K</math>
: <math>\mathbf{nothing} \equiv \mathbf K</math>
* <math>\mathbf{just} \equiv \mathbf{right}</math>
: <math>\mathbf{just} \equiv \mathbf{right}</math>
thus their definition is
thus their definition is
* <math>\mathbf{nothing} \equiv \mathbf K</math>
: <math>\mathbf{nothing} \equiv \mathbf K</math>
* <math>\mathbf{just} \equiv \mathbf{K_{(1)}}\;\mathbf{C_*}</math>
: <math>\mathbf{just} \equiv \mathbf{K_{(1)}}\;\mathbf{C_*}</math>
where both <math>\mathbf{just}</math> and <math>\mathbf{right}</math> have a common definition.
where both <math>\mathbf{just}</math> and <math>\mathbf{right}</math> have a common definition.
=== Catamorphisms for recursive datatypes ===
=== Catamorphisms for recursive datatypes ===
Line 121: Line 121:
and returns a value coming from a term consisting of applying cons-continuations and nil-continuations in the same shape as the correspondig list.
and returns a value coming from a term consisting of applying cons-continuations and nil-continuations in the same shape as the correspondig list.
E. g. in case of having defined
E. g. in case of having defined
* <math>\mathbf{oneTwoThree} \equiv \mathbf{cons}\;\mathbf1\;\left( \mathbf{cons}\;\mathbf2\;\left(\mathbf{cons}\;\mathbf3\;\mathbf{nil}\right) \right)</math>
: <math>\mathbf{oneTwoThree} \equiv \mathbf{cons}\;\mathbf1\;\left( \mathbf{cons}\;\mathbf2\;\left(\mathbf{cons}\;\mathbf3\;\mathbf{nil}\right) \right)</math>
the expression
the expression
* <math>\mathbf{oneTwoThree}\;\mathbf+\;\mathbf0</math>
: <math>\mathbf{oneTwoThree}\;\mathbf+\;\mathbf0</math>
reduces to
reduces to
* <math>\mathbf+\;\mathbf1\;\left(\mathbf+\;\mathbf2\; \left(\mathbf+\;\mathbf3\;\mathbf0\right)\right)</math>
: <math>\mathbf+\;\mathbf1\;\left(\mathbf+\;\mathbf2\; \left(\mathbf+\;\mathbf3\;\mathbf0\right)\right)</math>
But how to define <math>\mathbf{cons}</math> and <math>\mathbf{nil}</math>?
But how to define <math>\mathbf{cons}</math> and <math>\mathbf{nil}</math>?
In <math>\lambda</math>-calculus, we should like to see the following reductions:
In <math>\lambda</math>-calculus, we should like to see the following reductions:
* <math>\mathbf{nil}\;c\;n\;\to\;\;n</math>
: <math>\mathbf{nil}\;c\;n\;\to\;\;n</math>
* <math>\mathbf{cons}\;h\;t\; \to \;\lambda\;c\;n\;.\;c\;h\;\left(t\;c\;n\right)</math>
: <math>\mathbf{cons}\;h\;t\; \to \;\lambda\;c\;n\;.\;c\;h\;\left(t\;c\;n\right)</math>
Let us think of the variables as <math>h</math> denoting head, <math>t</math> denoting tail, <math>c</math> denoting cons-continuation, and <math>n</math> denoting nil-continuation.
Let us think of the variables as <math>h</math> denoting head, <math>t</math> denoting tail, <math>c</math> denoting cons-continuation, and <math>n</math> denoting nil-continuation.


Thus, we could achieve this goal with the following definitions:
Thus, we could achieve this goal with the following definitions:
* <math>\mathbf{nil} \equiv \lambda\;c\;n\;.\;n</math>
: <math>\mathbf{nil} \equiv \lambda\;c\;n\;.\;n</math>
* <math>\mathbf{cons} \equiv \lambda\;h\;t\;c\;n\;.\;c\;h\;\left(t\;c\;n\right)</math>
: <math>\mathbf{cons} \equiv \lambda\;h\;t\;c\;n\;.\;c\;h\;\left(t\;c\;n\right)</math>


Using the formulating combinators described in Haskell B. Curry's Combinatory Logic I, we can translate these definitions into combinatory logic without any  pain:
Using the formulating combinators described in Haskell B. Curry's Combinatory Logic I, we can translate these definitions into combinatory logic without any  pain:
* <math>\mathbf{nil} \equiv \mathbf K_*</math>
: <math>\mathbf{nil} \equiv \mathbf K_*</math>
* <math>\mathbf{cons} \equiv \mathbf B \left( \mathbf{\Phi}\;\mathbf B \right) \mathbf{C_*}</math>
: <math>\mathbf{cons} \equiv \mathbf B \left( \mathbf{\Phi}\;\mathbf B \right) \mathbf{C_*}</math>


Of course we could use the two parameters in the opposite order, but I am not sure  yet that it would provide a more easy way.
Of course we could use the two parameters in the opposite order, but I am not sure  yet that it would provide a more easy way.
Line 148: Line 148:
</haskell>
</haskell>
which corresponds in cominatory logic to reducing
which corresponds in cominatory logic to reducing
* <math>\mathbf{concat}\;l\equiv l\;\mathbf{append}\;\mathbf{nil}</math>
: <math>\mathbf{concat}\;l\equiv l\;\mathbf{append}\;\mathbf{nil}</math>
Let us use the ordered pair (direct product) construct:
Let us use the ordered pair (direct product) construct:
* <math>\mathbf{concat}\equiv \mathbf{\Diamond_2}\;\mathbf{append}\;\mathbf{nil}</math>
: <math>\mathbf{concat}\equiv \mathbf{\Diamond_2}\;\mathbf{append}\;\mathbf{nil}</math>
and if I use that nasty <math>\mathbf{centred}</math> (see later)
and if I use that nasty <math>\mathbf{centred}</math> (see later)
* <math>\mathbf{concat} \equiv \mathbf{centred}\;\mathbf{append}</math>
: <math>\mathbf{concat} \equiv \mathbf{centred}\;\mathbf{append}</math>


== Monads in Combinatory Logic? ==
== Monads in Combinatory Logic? ==
Line 167: Line 167:
</haskell>
</haskell>
in Haskell and
in Haskell and
* <math>\mathbf{maybe\!\!-\!\!return} \equiv \mathbf{just}</math>
: <math>\mathbf{maybe\!\!-\!\!return} \equiv \mathbf{just}</math>
in combinatory logic.
in combinatory logic.
===== map =====
===== map =====
Line 178: Line 178:
<math>\lambda</math>-calculus:
<math>\lambda</math>-calculus:
Expected reductions:
Expected reductions:
* <math>\mathbf{maybe\!\!-\!\!map}\;f\;p\;\to\;p\;\mathbf{nothing}\;\left(\mathbf{just_{(1)}}\;f\right)</math>
: <math>\mathbf{maybe\!\!-\!\!map}\;f\;p\;\to\;p\;\mathbf{nothing}\;\left(\mathbf{just_{(1)}}\;f\right)</math>


Definition:
Definition:
* <math>\mathbf{maybe\!\!-\!\!map}\equiv \lambda\;f\;p\;.\;p\;\mathbf{nothing}\;\left(\mathbf{just_{(1)}}\;f\right)</math>
: <math>\mathbf{maybe\!\!-\!\!map}\equiv \lambda\;f\;p\;.\;p\;\mathbf{nothing}\;\left(\mathbf{just_{(1)}}\;f\right)</math>


Combinatory logic: we expect the same reduction here too
Combinatory logic: we expect the same reduction here too
* <math>\mathbf{maybe\!\!-\!\!map}\;f\;p\;\to\;p\;\mathbf{nothing}\;\left(\mathbf{just_{(1)}}\;f\right)</math>
: <math>\mathbf{maybe\!\!-\!\!map}\;f\;p\;\to\;p\;\mathbf{nothing}\;\left(\mathbf{just_{(1)}}\;f\right)</math>
let us get rid of one parameter:
let us get rid of one parameter:
* <math>\mathbf{maybe\!\!-\!\!map}\;f\;\to\;\mathbf{\Diamond_2}\;\mathbf{nothing}\;\left(\mathbf{just_{(1)}}\;f\right)</math>
: <math>\mathbf{maybe\!\!-\!\!map}\;f\;\to\;\mathbf{\Diamond_2}\;\mathbf{nothing}\;\left(\mathbf{just_{(1)}}\;f\right)</math>
now we have the definition:
now we have the definition:
* <math>\mathbf{maybe\!\!-\!\!map}\equiv \mathbf{\Diamond_2}\;\mathbf{nothing}\;\cdot\;\mathbf{just_{(1)}}</math>
: <math>\mathbf{maybe\!\!-\!\!map}\equiv \mathbf{\Diamond_2}\;\mathbf{nothing}\;\cdot\;\mathbf{just_{(1)}}</math>


===== bind =====
===== bind =====
Line 197: Line 197:
</haskell>
</haskell>
<math>\lambda</math>-calculus: we expect
<math>\lambda</math>-calculus: we expect
* <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<}\;f\;p\;\to\;p\;\mathbf{nothing}\;f</math>
: <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<}\;f\;p\;\to\;p\;\mathbf{nothing}\;f</math>
achieved by defintion
achieved by defintion
* <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<} \equiv \lambda\;f\;p\;.\;p\;\mathbf{nothing}\;f</math>
: <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<} \equiv \lambda\;f\;p\;.\;p\;\mathbf{nothing}\;f</math>


In combinatory logic the above expected reduction
In combinatory logic the above expected reduction
* <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<}\;f\;p\;\to\;p\;\mathbf{nothing}\;f</math>
: <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<}\;f\;p\;\to\;p\;\mathbf{nothing}\;f</math>
getting rid of the outest parameter
getting rid of the outest parameter
* <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<}\;f\;\to\;\mathbf{\Diamond_2}\;\mathbf{nothing}\;f</math>
: <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<}\;f\;\to\;\mathbf{\Diamond_2}\;\mathbf{nothing}\;f</math>
yielding definition
yielding definition
* <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<} \equiv \mathbf{\Diamond_2}\;\mathbf{nothing}</math>
: <math>\mathbf{maybe\!\!-\!\!=\!\!<\!\!<} \equiv \mathbf{\Diamond_2}\;\mathbf{nothing}</math>
and of course
and of course
* <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=} \equiv \mathbf C\;\mathbf{maybe\!\!-\!\!=\!\!<\!\!<}</math>
: <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=} \equiv \mathbf C\;\mathbf{maybe\!\!-\!\!=\!\!<\!\!<}</math>


But the other way (starting with a better chosen parameter order) is much better:
But the other way (starting with a better chosen parameter order) is much better:
* <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=}\;p\;f\;\to\;p\;\mathbf{nothing}\;f</math>
: <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=}\;p\;f\;\to\;p\;\mathbf{nothing}\;f</math>
* <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=}\;p\;\to\;p\;\mathbf{nothing}</math>
: <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=}\;p\;\to\;p\;\mathbf{nothing}</math>
yielding the much simplier and more efficient definition:
yielding the much simplier and more efficient definition:
* <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=} \equiv \mathbf{C_*}\;\mathbf{nothing}</math>
: <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=} \equiv \mathbf{C_*}\;\mathbf{nothing}</math>


We know already that <math>\mathbf{C_*}</math> can be seen as as a member of the scheme of tuples: <math>\mathbf{\Diamond_n}</math> for <math>n=1</math> case.
We know already that <math>\mathbf{C_*}</math> can be seen as as a member of the scheme of tuples: <math>\mathbf{\Diamond_n}</math> for <math>n=1</math> case.
As the tupe construction is a usual guest at things like this (we shall meet it at list and other maybe-operations like <math>\mathbf{maybe\!\!-\!\!join}</math>),
As the tupe construction is a usual guest at things like this (we shall meet it at list and other maybe-operations like <math>\mathbf{maybe\!\!-\!\!join}</math>),
so us express the above definition with <math>\mathbf{C_*}</math> denoted as <math>\Diamond_1</math>:
so us express the above definition with <math>\mathbf{C_*}</math> denoted as <math>\Diamond_1</math>:
* <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=} \equiv \mathbf{\Diamond_1}\;\mathbf{nothing}</math>
: <math>\mathbf{maybe\!\!-\!\!\!>\!\!>\!\!=} \equiv \mathbf{\Diamond_1}\;\mathbf{nothing}</math>
hoping that this will enable us some interesting generalization in the future.
hoping that this will enable us some interesting generalization in the future.


Line 225: Line 225:


===== join =====
===== join =====
* <math>\mathbf{maybe\!\!-\!\!join} \equiv \mathbf{\Diamond_2}\;\mathbf{nothing}\;\mathbf I</math>
: <math>\mathbf{maybe\!\!-\!\!join} \equiv \mathbf{\Diamond_2}\;\mathbf{nothing}\;\mathbf I</math>
We should think of changing the architecture if we suspect that we could avoid <math>\mathbf I</math> and solve the problem with a more simple construct.
We should think of changing the architecture if we suspect that we could avoid <math>\mathbf I</math> and solve the problem with a more simple construct.


Line 234: Line 234:


we could name
we could name
* <math>\mathbf{list\!\!-\!\!join} \equiv \mathbf{concat}
: <math>\mathbf{list\!\!-\!\!join} \equiv \mathbf{concat}
</math>
</math>
Now let us see mapping a list, concatenating a list, binding a list.
Now let us see mapping a list, concatenating a list, binding a list.
Mapping and binding have a common property: yielding nil for nil.
Mapping and binding have a common property: yielding nil for nil.
I shall say these operations are ''centred'': their definition would contain a <math>\mathbf C\;\mathbf{\Diamond_2}\;\mathbf{nil}</math> subexpression. Thus I shall give a name to this subexpression:
I shall say these operations are ''centred'': their definition would contain a <math>\mathbf C\;\mathbf{\Diamond_2}\;\mathbf{nil}</math> subexpression. Thus I shall give a name to this subexpression:
* <math>\mathbf{centred} \equiv \mathbf C\;\mathbf{\Diamond_2}\;\mathbf{nil}</math>
: <math>\mathbf{centred} \equiv \mathbf C\;\mathbf{\Diamond_2}\;\mathbf{nil}</math>


Now let us define map and bind for lists:
Now let us define map and bind for lists:
* <math>\mathbf{list\!\!-\!\!map} \equiv \mathbf{centred}_{(1)}\;\mathbf{cons}_{(1)}</math>
: <math>\mathbf{list\!\!-\!\!map} \equiv \mathbf{centred}_{(1)}\;\mathbf{cons}_{(1)}</math>
* <math>\mathbf{list\!\!-\!\!=\!\!<\!\!<} \equiv \mathbf{centred}_{(1)}\;\mathbf{append}_{(1)}</math>
: <math>\mathbf{list\!\!-\!\!=\!\!<\!\!<} \equiv \mathbf{centred}_{(1)}\;\mathbf{append}_{(1)}</math>
now we see it was worth of defining a common <math>\mathbf{centred}</math>.
now we see it was worth of defining a common <math>\mathbf{centred}</math>.
But to tell the truth, it may be a trap. <math>\mathbf{centred}</math> breaks a symmetry: we should always define the cons and nil part of the foldr construct on the same level, always together. Modularization should be pointed towards this direction, and not to run forward into the T-street of <math>\mathbf{centred}</math>.
But to tell the truth, it may be a trap. <math>\mathbf{centred}</math> breaks a symmetry: we should always define the cons and nil part of the foldr construct on the same level, always together. Modularization should be pointed towards this direction, and not to run forward into the T-street of <math>\mathbf{centred}</math>.


Another remark: of course we can get the monadic bind for lists
Another remark: of course we can get the monadic bind for lists
* <math>\mathbf{list\!\!-\!\!\!>\!\!>\!\!=} \equiv \mathbf{C}\;\mathbf{list\!\!-\!\!=\!\!<\!\!<}</math>
: <math>\mathbf{list\!\!-\!\!\!>\!\!>\!\!=} \equiv \mathbf{C}\;\mathbf{list\!\!-\!\!=\!\!<\!\!<}</math>
But we used <math>\mathbf{append}</math> here. How do we define it? It is surprisingly simple. Let us think how we would define it in Haskell by <hask>foldr</hask> function, if it was not defined already as <hask>++</hask> defined in Prelude:
But we used <math>\mathbf{append}</math> here. How do we define it? It is surprisingly simple. Let us think how we would define it in Haskell by <hask>foldr</hask> function, if it was not defined already as <hask>++</hask> defined in Prelude:
In defining
In defining
Line 264: Line 264:
</haskell>
</haskell>
let us se how we should reduce its corresponding expression in Combinatory Logic:
let us se how we should reduce its corresponding expression in Combinatory Logic:
* <math>\mathbf{append}\;l\;m \to l\;\mathbf{cons}\;m</math>
: <math>\mathbf{append}\;l\;m \to l\;\mathbf{cons}\;m</math>
thus
thus
* <math>\mathbf{append}\,l\,m = l\,\mathbf{cons}\,m</math>
: <math>\mathbf{append}\,l\,m = l\,\mathbf{cons}\,m</math>
* <math>\mathbf{append}\;l =\!\!_1\;l\;\mathbf{cons}</math>
: <math>\mathbf{append}\;l =\!\!_1\;l\;\mathbf{cons}</math>
* <math>\mathbf{append} \equiv \mathbf{C_*}\;\mathbf{cons}
: <math>\mathbf{append} \equiv \mathbf{C_*}\;\mathbf{cons}
</math>
</math>


Line 288: Line 288:
</haskell>
</haskell>
so we can see how to do it in combinatory logic:
so we can see how to do it in combinatory logic:
* <math>\mathbf{list\!\!-\!\!return} \equiv \mathbf C\;\mathbf{cons}\;\mathbf{nil}</math>
: <math>\mathbf{list\!\!-\!\!return} \equiv \mathbf C\;\mathbf{cons}\;\mathbf{nil}</math>


=== How to AOP with monads in Combinatory Logic? ===
=== How to AOP with monads in Combinatory Logic? ===
Line 298: Line 298:
Of course my simple CL interpreter does not know anything on type classes, overloading. But there is a rather restricted andstatic possibility provided by the concept of ''definition'' itself:
Of course my simple CL interpreter does not know anything on type classes, overloading. But there is a rather restricted andstatic possibility provided by the concept of ''definition'' itself:


* <math>\mathbf{work} \equiv \mathbf{A\!\!-\!\!\!>\!\!>\!\!>\!\!=}\;\mathbf{subwork\!\!-\!\!1}\;\mathbf{parametrized\!\!-\!\!subwork\!\!-\!\!2}</math>
: <math>\mathbf{work} \equiv \mathbf{A\!\!-\!\!\!>\!\!>\!\!>\!\!=}\;\mathbf{subwork\!\!-\!\!1}\;\mathbf{parametrized\!\!-\!\!subwork\!\!-\!\!2}</math>


and later we can change the binding mode named A e.g. from a failure-handling Maybe-like one to a more general indeterminism-handling list-like one, then we can do that simply by replacing definition
and later we can change the binding mode named A e.g. from a failure-handling Maybe-like one to a more general indeterminism-handling list-like one, then we can do that simply by replacing definition
* <math>\mathbf{A\!\!\!-\!\!>\!\!>\!\!>\!\!=} \equiv \mathbf{maybe\!\!\!-\!\!>\!\!>\!\!>\!\!=}</math>
: <math>\mathbf{A\!\!\!-\!\!>\!\!>\!\!>\!\!=} \equiv \mathbf{maybe\!\!\!-\!\!>\!\!>\!\!>\!\!=}</math>
with definition
with definition
* <math>\mathbf{A\!\!\!-\!\!>\!\!>\!\!>\!\!=} \equiv \mathbf{list\!\!\!-\!\!>\!\!>\!\!>\!\!=}
: <math>\mathbf{A\!\!\!-\!\!>\!\!>\!\!>\!\!=} \equiv \mathbf{list\!\!\!-\!\!>\!\!>\!\!>\!\!=}
</math>
</math>


Line 330: Line 330:


But sometimes, representing CL terms by themselves is not enough. Let us imagine a tutoring program! Let the topic be combinatory logic, the language of implementation -- combinatory logic, too. How should the tutoring program ask the pupil questions like:
But sometimes, representing CL terms by themselves is not enough. Let us imagine a tutoring program! Let the topic be combinatory logic, the language of implementation -- combinatory logic, too. How should the tutoring program ask the pupil questions like:
* Tell me if the following two expresions have the same normal form:
: Tell me if the following two expresions have the same normal form:
** <math>\mathbf K\;\mathbf{24}\;\mathbf{48}</math>
: <math>\mathbf K\;\mathbf{24}\;\mathbf{48}</math>
** <math>\mathbf{24}</math>
: <math>\mathbf{24}</math>


The problem is that our program is simply ''unable'' to distinguish between CL terms which have the same normal form (in fact, equivalence cannot be defined generally either). If we represent CL terms by themselves, we simply ''loose'' a lot of information, including loosing any possibility to make distinctions between equivalent terms.  
The problem is that our program is simply ''unable'' to distinguish between CL terms which have the same normal form (in fact, equivalence cannot be defined generally either). If we represent CL terms by themselves, we simply ''loose'' a lot of information, including loosing any possibility to make distinctions between equivalent terms.  
Line 378: Line 378:


After having solved the representation (quoting) problem, we can do many things. We can define meta-concepts, e.g.
After having solved the representation (quoting) problem, we can do many things. We can define meta-concepts, e.g.
;<math>\equiv</math> (the notion of ''same terms'')
; <math>\equiv</math> (the notion of ''same terms'')
: by bool tree equality
: by bool tree equality
; <math>=</math> (equivalence made by ''reduction'')
; <math>=</math> (equivalence made by ''reduction'')
Line 490: Line 490:
<math>\mathbf G\;S\;(\lambda x . T)</math>
<math>\mathbf G\;S\;(\lambda x . T)</math>
in Illative Combinatory Logic. I think e.g. the followings should correspond to each other:
in Illative Combinatory Logic. I think e.g. the followings should correspond to each other:
* <math>\mathbf{realNullvector} :\;\;\;\forall n: \mathbf{Nat} \Rightarrow \mathbf{RealVector}\;n</math>
: <math>\mathbf{realNullvector} :\;\;\;\forall n: \mathbf{Nat} \Rightarrow \mathbf{RealVector}\;n</math>
* <math>\mathbf G\;\,\mathbf{Nat}\;\,\mathbf{RealVector}\;\,\mathbf{realNullvector}</math>
: <math>\mathbf G\;\,\mathbf{Nat}\;\,\mathbf{RealVector}\;\,\mathbf{realNullvector}</math>





Revision as of 09:47, 23 March 2006

General

Albeit having precursors, it was Moses Schönfinkel who explored first combinatory logic. Later it has been continued by Haskell B. Curry. It was developed to be a theory for the foundation of mathematics [Bun:NatICL], and it has also relevance in Linguistics too.

General materials:

Portals and other large-scale resources

Implementing CL

Base

Some thoughts on base combinators and on the relatedness of their rules to other topics

  • Conal Elliott's reply to thread zips and maps
  • Intuitionisitc fragment of propositional logic
  • Records in function: in set theory and database theory, we regard functions as consisting of more elementary parts, records: a function f can be regarded as the set of all its records. A record is a pair of a key and its value, and for funtions we expect unicity (and sometimes stress this requirement by writing xx instead of x,y).Sometimes I think of S as having a taste of record selection: Scfx selects a record determinated by key x in function f (as in a database), and returns the found record (i.e. corresponding key and value) contained in the c container (continuation). Is this thought just a toy or can it be brought further? Does it explain why S and K can constitute a base?
  • Also bracket abstraction gives us a natural way to understand the seemingly rather unintuitive and artificial S combinator better

Programming in CL

I think many thoughts from John Hughes' Why Functional Programming Matters can be applied to programming in Combinatory Logic. And almost all concepts used in the Haskell world (catamorphisms etc.) helps us a lot here too. Combinatory logic is a powerful and concise programming language. I wonder how functional logic programming could be done by using the concepts of Illative combinatory logic, too.

Datatypes

Continuation passing for polynomial datatypes

Direct product

Let us begin with a notion of the ordered pair and denote it by 2. We know this construct well when defining operations for booleans

trueK
falseK*
not2falsetrue

and Church numbers. I think, in generally, when defining datatypes in a continuation-passing way (e.g. Maybe or direct sum), then operations on so-defined datatypes often turn to be well-definable by some n.

We define it with

2λxyf.fxy

in λ-calculus and

2C(1)C*

in combinatory logic.

A nice generalization scheme:

  • as the construct can be generalized to any natural number n (the concept of n-tuple, see Barendregt's λ Calculus)
  • and in this generalized scheme I corresponds to the 0 case, C* to the 1 case, and the ordered pair construct 2 to the 2 case, as though defining
0I
1C*

so we can write definition

2C(1)C*

or the same

2CC*

in a more interesting way:

2C1

Is this generalizable? I do not know. I know an analogy in the case of left, right, just, nothing.

Direct sum

The notion of ordered pair mentioned above really enables us to deal with direct products. What about it dual concept? How to make direct sums in Combinatory Logic? And after we have implemented it, how can we see that it is really a dual concept of direct product?

A nice argument described in David Madore's Unlambda page gives us a continuation-passig style like solution. We expect reductions like

leftxλfg.fx
rightxλfg.gx

so we define

leftλxfg.fx
rightλxfg.gx

now we translate it from λ-calculus into combinatory logic:

leftK(2)C*
rightK(1)C*

Of course, we can recognize Haskell's direct sum construct

 Either (Left, Right)

implemented in an analogous way.

Maybe

Let us remember Haskell's maybe:

 maybe :: a' -> (a -> a') -> Maybe a -> a'
 maybe n j Nothing = n
 maybe n j (Just x) = j x

thinking of

  • n as nothing-continuation
  • j as just-continuation

In a continuation passing style approach, if we want to implement something like the Maybe constuct in λ-calculus, then we may expect the following reductions:

nothingλnj.n
justxλnj.jx

we know both of them well, one is just K, and we remember the other too from the direct sum:

nothingK
justright

thus their definition is

nothingK
justK(1)C*

where both just and right have a common definition.

Catamorphisms for recursive datatypes

List

Let us define the concept of list by its catamorphism (see Haskell's foldrfunction): a list (each concrete list) is a function taking two arguments

  • a two-parameter function argument (cons-continuation)
  • a zero-parameter function argument (nil-continuation)

and returns a value coming from a term consisting of applying cons-continuations and nil-continuations in the same shape as the correspondig list. E. g. in case of having defined

oneTwoThreecons1(cons2(cons3nil))

the expression

oneTwoThree+0

reduces to

+1(+2(+30))

But how to define cons and nil? In λ-calculus, we should like to see the following reductions:

nilcnn
conshtλcn.ch(tcn)

Let us think of the variables as h denoting head, t denoting tail, c denoting cons-continuation, and n denoting nil-continuation.

Thus, we could achieve this goal with the following definitions:

nilλcn.n
consλhtcn.ch(tcn)

Using the formulating combinators described in Haskell B. Curry's Combinatory Logic I, we can translate these definitions into combinatory logic without any pain:

nilK*
consB(ΦB)C*

Of course we could use the two parameters in the opposite order, but I am not sure yet that it would provide a more easy way.

A little practice: let us define concat. In Haskell, we can do that by

 concat = foldr (++) []

which corresponds in cominatory logic to reducing

concatllappendnil

Let us use the ordered pair (direct product) construct:

concat2appendnil

and if I use that nasty centred (see later)

concatcentredappend

Monads in Combinatory Logic?

Concrete monads

Maybe as a monad

return

Implementing the return monadic method for the Maybe monad is rather straightforward, both in Haskell and CL:

 instance Monad Maybe
         return = Just
         ...

in Haskell and

maybereturnjust

in combinatory logic.

map

Haskell:

 instance Functor Maybe where
         map f = maybe Nothing (Just . f)

λ-calculus: Expected reductions:

maybemapfppnothing(just(1)f)

Definition:

maybemapλfp.pnothing(just(1)f)

Combinatory logic: we expect the same reduction here too

maybemapfppnothing(just(1)f)

let us get rid of one parameter:

maybemapf2nothing(just(1)f)

now we have the definition:

maybemap2nothingjust(1)
bind

Haskell:

 instance Monad Maybe (>>=) where
         (>>=) f p = maybe Nothing f

λ-calculus: we expect

maybe=<<fppnothingf

achieved by defintion

maybe=<<λfp.pnothingf

In combinatory logic the above expected reduction

maybe=<<fppnothingf

getting rid of the outest parameter

maybe=<<f2nothingf

yielding definition

maybe=<<2nothing

and of course

maybe>>=Cmaybe=<<

But the other way (starting with a better chosen parameter order) is much better:

maybe>>=pfpnothingf
maybe>>=ppnothing

yielding the much simplier and more efficient definition:

maybe>>=C*nothing

We know already that C* can be seen as as a member of the scheme of tuples: n for n=1 case. As the tupe construction is a usual guest at things like this (we shall meet it at list and other maybe-operations like maybejoin), so us express the above definition with C* denoted as 1:

maybe>>=1nothing

hoping that this will enable us some interesting generalization in the future.

But why we have not made a more brave genralization, and express monadic bind from monadic join and map? Later in the list monad, we shall see that it may be better to avoid this for sake of deforestation. Here a maybe similar problem will appear: the problem of superfluous I.

join
maybejoin2nothingI

We should think of changing the architecture if we suspect that we could avoid I and solve the problem with a more simple construct.


The list as a monad

Let us think of our list-operations as implementing monadic methods of the list monad. We can express this by definitions too, e.g.

we could name

listjoinconcat

Now let us see mapping a list, concatenating a list, binding a list. Mapping and binding have a common property: yielding nil for nil. I shall say these operations are centred: their definition would contain a C2nil subexpression. Thus I shall give a name to this subexpression:

centredC2nil

Now let us define map and bind for lists:

listmapcentred(1)cons(1)
list=<<centred(1)append(1)

now we see it was worth of defining a common centred. But to tell the truth, it may be a trap. centred breaks a symmetry: we should always define the cons and nil part of the foldr construct on the same level, always together. Modularization should be pointed towards this direction, and not to run forward into the T-street of centred.

Another remark: of course we can get the monadic bind for lists

list>>=Clist=<<

But we used append here. How do we define it? It is surprisingly simple. Let us think how we would define it in Haskell by foldr function, if it was not defined already as ++ defined in Prelude: In defining

 (++) list1 list2

we can do it by foldr:

 (++) [] list2 = list2
 (++) (a : as) list2 = a : (++) as list2

thus

 (++) list1 list2 = foldr (:) list2 list1

let us se how we should reduce its corresponding expression in Combinatory Logic:

appendlmlconsm

thus

appendlm=lconsm
appendl=1lcons
appendC*cons

Thus, we have defined monadic bind for lists. I shall call this the deforested bind for lists. Of course, we could define it another way too: by concat and map, which corresponds to defining monadic bind from monadic map and monadic join. But I think this way forces my CL-interpreter to manage temporary lists, so I gave rather the deforested definition.

Defining the other monadic operation: return for lists is easy:

 instance Monad [] where
         return = (: [])

in Haskell -- we know,

 (: [])

translates to

 return = flip (:) []

so we can see how to do it in combinatory logic:

listreturnCconsnil

How to AOP with monads in Combinatory Logic?

We have defined monadic list in CL. Of course we can make monadic Maybe, binary tree, Error monad with direct sum constructs...

But separation of concerns by monads is more than having a bunch of special monads. It requires other possibilities too: e.g. being able to use monads generally, which can become any concrete mondads.

Of course my simple CL interpreter does not know anything on type classes, overloading. But there is a rather restricted andstatic possibility provided by the concept of definition itself:

workA>>>=subwork1parametrizedsubwork2

and later we can change the binding mode named A e.g. from a failure-handling Maybe-like one to a more general indeterminism-handling list-like one, then we can do that simply by replacing definition

A>>>=maybe>>>=

with definition

A>>>=list>>>=

Self-replication, quines, reflective programming

Background

David Madore's Quines (self-replicating programs) and Shin-Cheng Mu's many writings, including a Haskell quine give us woderful insights on mathematical logic, programming, self-reference. Wikipedia's quine page and John Bethencourt's quine quine. See also the writings of Raymond Smullyan, Hofstadter, also his current research project on a self-watching cognitive architecture, Manfred Eigen and Ruthild Winkler: Laws of the Game / How the Principles of Nature Govern Chance, and Karl Sigmund's Games of Life, and Reflective programming (see Reflection '96 and P. Maes & D. Nardi: Meta-Level Architectures and Reflection). G.J. Chaitin especially his Understandable Papers on Incompleteness, especially The Unknowable (the book is available on this page, just roll the page bellow that big colored photos). The book begins with the limits of mathematics: Goldel's undecidable, Turing's uncompatiblity, Chaitin's randomness); but (or exactly that's why?) it ends with writing on the future and beuty of science.

I must read Autopoesis and The Tree of Knowledge carefully from Maturana and Varela to say if their topics are releted to here.

Self-replication

Quines: the idea of self-replication can be conveyed by the concept of a program, which is able to print its own list. But pure λ-calculus and combinatory logic does not know any notion of printing! We should like the capture the essence of self-replication, wethout resorting to the imperative world.

Representation, qoutation -- the DNS

Let us introduce the concept of representing combinatory logic terms. How could we do that? For example, by binary trees. The leaves should represent the base combinators, and the branches mean application.

And how to represent combintory logic terms -- in combinatory logic itself? The first thought could be, that it is not a problem. Each combinatory logic term could be represented by itself.

Sometimes this idea works. The huge power of higher order functions is exactly in being able to treat datas programs and vice versa. Sometimes we are enabled to do things, which could be done in other languages only by carefully designing a representation, a specific language.

But sometimes, representing CL terms by themselves is not enough. Let us imagine a tutoring program! Let the topic be combinatory logic, the language of implementation -- combinatory logic, too. How should the tutoring program ask the pupil questions like:

Tell me if the following two expresions have the same normal form:
K2448
24

The problem is that our program is simply unable to distinguish between CL terms which have the same normal form (in fact, equivalence cannot be defined generally either). If we represent CL terms by themselves, we simply loose a lot of information, including loosing any possibility to make distinctions between equivalent terms.

We see that there is something that relates to make a distinction between target language and metalanguage (See Imre Ruzsa, or Haskell B. Curry)

In this example, the distinction is:

  • We deal with combinatory logic expressions because our program has to teach them: it is related to it just like a vocabulary program is related to English.
  • But we deal with programming logic expressions because our program is implemented in them. Just like VIM is related to C++.

We said CL terms are eventually trees. Let us represent them with trees then -- now let us think of trees not as of term trees, but as datatypes which we must construct by hand, in a similar way as we defined Maybes, direct sums, direct products, lists.

K
leaftrue
S
leaffalse
(ab)
branchαβ

where let α denote the representation of a and β that of b

Let us make a distinction between term trees and datatype trees. A Haskell example:

  • many Haskell expressions can be regarded as term trees
  • but only special Haskell expressions can be seen as datatype trees: those who are constructed from Branch and Leaf in an appropriate way

Similarly,

  • all CL expressions can be regarded as term trees.
  • but CL expressions which can be revered as datatype trees must obey a huge amount of constraints: they may consist only of subexpressions leaf, branch, true, false subexpressions in an apprporiate way.

(In fact, all CL expressions can be regarded as datatype trees too: CL is a total thing, we can us each CL expression in a same way as a datatype tree: we can apply it leaf- and branch-continuation arguments. Something will always happen. At worst it will diverge -- but lazy trees can diverge too, amd they are inarguably datatype trees. But now let us ignore all these facts, and let us define the notion of quotations in the restictive way: let the definition require to be structured in a predefined way.)

We use datatype trees for representing other expressions. Let us call CL expressions which can represent (another CL expreesion) quotations. Quotations are exactly the datatype trees, but

  • the world quotation refers to their function,
  • the world datatype tree refers to their implemetation, structure

This means a datatype tree

  • is not only a tree regarded only as a term tree,
  • but on a higher level: itself a recursive datatype implemented in CL, it is appropiately consisting of leaf,branch and true, false subexpressions so that we can reason about it in CL itself

How do quotations relate to all CL expressions?

  • In one direction, informally, we could say, quotations make a very proper subset of all CL expressions (attention: cardinality is the same!). Not every CL expressions are datatype trees.
  • But the reverse is not true: all CL expressions can be quoted! Foreach CL expressionther is a (unique) CL expression who quotes it!

We can define a quote function on the set of all CL expressions. But of it is an conceptually outside function, not a CL combinator itself! (that is why I do not typest it boldface. Is it an example of what Curry called epitheory?).

After having solved the representation (quoting) problem, we can do many things. We can define meta-concepts, e.g.

(the notion of same terms)
by bool tree equality
= (equivalence made by reduction)
by building a metacircular interpreter

We can write our tutor program too. But let us discuss more clean and theoretical questions.

Concept of self-replication generalized -- pure functional quines

How can be the concept of quine transferred to combinatory logic? In the bellow definition, let us think of

  • A's as actions, programs
  • and Q's as quotations, representations
A quine is a CL term A this means quines are pure CL concepts, no imperative compromises
for whose normal form A0 this means quines are run
there exists an equivalent CL-term Q where datatypes in CL arealmost never defined in their normal form (not even ordered pairs are!). They save us from loosing information, but they almost never do that literary. I faced this as problems in nice rewritings when I wanted to implement CL with computer algebra services
Q is a quotation, which manifests itself in the fact that Q is a datatype tree (not only term tree) with boolean leafs,
Q quotes A and Q is exactly the representation of A

So a quine is a program which is run, then rewrited as a quotation and so we get the representation of the original program.

Of course the first three requirements can be contracted in two. Thus, a quine is a CL-term which is equivalent to its own representation (if we mean representation as treated here).

A metacircular interpeter

We have seen that we can represent CL expressions in CL itself, which enables us to do some meta things (see the into of this section, especially Reflective programming, e.g. Reflection '96). The first idea could be: to implement CL in itself!

Implementing lazy evaluation

The most important subtask to achieve this goal is to implement the algorithm of lazy evaluation. I confess I simply lack almost any knowledge on algorithms for implementing lazy evaluation. In my Haskell programs, when they must implement lazy evaluation, I use the following hand-made algorithm:


 module Reduce where
 
 import Term
 import Tree
 import Base
 
 eval :: Term -> Term
 eval (Branch function argument) = apply function argument
 eval atom = atom
 
 apply :: Term -> Term -> Term
 apply (Branch f a) b = curry' f a b
 apply atom argument = strictApply atom argument
 
 curry' :: Term -> Term -> Term -> Term
 curry' (Leaf K) f x = eval f
 curry' (Branch f a) b c = lazy f a b c
 curry' s a b = strictCurry s a b
 
 lazy :: Term -> Term -> Term -> Term -> Term
 lazy (Leaf S) c f x = curry' c x (Branch f x)
 lazy k_or_compound x y z = curry' k_or_compound x y `apply` z
 
 strictApply :: Term -> Term -> Term
 strictApply f a = f `Branch` eval a
 
 strictCurry :: Term -> Term -> Term -> Term
 strictCurry f a b = strictApply f a `strictApply` b

 module Term where
 
 import Base
 import Tree

 type Term = Tree Base

 module Tree where
 
 data Tree a = Leaf a | Branch (Tree a) (Tree a)

 module Base where
 
 data Base = K | S

and it seems hard to me hard to implement in CL. Almost all of these functions are mutual recursive definitions, and it looks hard for me to formulate the fixpont. Of coure I could find another algorithm. The main problem is that reducing CL trees is not so simple: the S rule requires lookahead in 2 levels. Maybe once I find another one with monads, arrows, or attribute grammars...

Illative Combinatory Logic

I think combinator G can be thought of as something analogous to Dependent types: it seems to me that the dependent type construct x:ST of Epigram corresponds to GS(λx.T) in Illative Combinatory Logic. I think e.g. the followings should correspond to each other:

realNullvector:n:NatRealVectorn
GNatRealVectorrealNullvector


My dream is making something in Illative Combinatory Logic. Maybe it could be theroretical base for a functional logic language?

References

  • [Bun:NatICL] Martin W. Bunder: The naturalness of Illative Combinatory Logic as a Basis for Mathematics, see in Dedicated to H.B.Curry on the occasion of his 80th birthday.