This is an alternate compiler built on top of Pharo's Opal compiler. It can be the compiler of choice for any class. All of the syntax proposed here is otherwise syntax errors that can be recognized by this compiler and turned into legal Pharo Smalltalk. However, the source would still contain these syntax elements... only the internal AST is affected. Only if you lost your source code and recovered it from reverse-compilation you would lose the syntactic forms.
This adds a compose or pipe syntax to Pharo (we call it the parrot operator :>
). It can significantly reduce the need for parentheses and make functional code more readable. It works with PharoJS and can be very convenient when using some Javascript
libraries such as D3.
For example:
foo
" self new foo >>> 42 "
^ 17 negated
:> min: -53
:> abs
:> < 100
:> and: [ 4 > 2 ]
:> and: [ 5 < 10 ]
:> ifTrue: [ 42 ] ifFalse: [ 99 ]
The precedence is the same as cascade, so you can intermix them and could say something like:
x := OrderedCollection new
add: 42;
add: 17;
yourself
:> collect: #negated
:> add: 35;
add: 99;
yourself
:> with: #(1 2 3 4) collect: [:l :r| l+r ]
:> max
It's somewhat common in the functional and array-processing world. Several of the extensions here support it. Here is a link to an old article about it: http://smalltalkthoughts.blogspot.com/2011/08/point-free-programming-in-smalltalk.html
A proposed extension adds a syntax to support composing blocks or symbols with combinators. This can be very convenient when used with the parrot operator, as well as curry:
and some other messages from the Pharo-Functional part of this repository. This reduces the amount of visual noise from value:
and value:value:
messages. Anywhere a unary messsage can go, you can instead put an expression in parentheses or put a block. Anywhere a binary message can go, you can do the same, but follow the close-parenthesis with a colon (:
). This is extremely useful with the combinators described later.
For example:
x (...)
([...] value: x)
x (...) + y
([...] value: x) + y
x (...): y
([...] value: x value: y)
x (#sort <|> #=)
You can do the same with unary or binary blocks. Because we know the arity of blocks the trailing :
isn't used for block operators
x [:w|...]
([:w|...] value: x)
x [:w:z|...] y
([:w:z|...] value: x value: y)
If you had a unary or binary block in a variable, you could use it with the parenthesis syntax:
a: aBlock
^ 42 (aBlock): 17
Initializing variables at the point of declaration is best-practise in most of the programming world. Note that if a :=
or =
appears in a declaration, the expression must be terminated with a .
and it can't be omitted on the last initialization:
| x := 42. y = x+5. z|
is legal, but
| x := 42. y = x+5. z = 17 |
is illegal, because the trailing |
would be be ambigous.
Note that :=
declares and initializes a mutable variable and =
declares and initializes an immutable variable.
Arrays have a literal syntax {1 . 2 . 3}
, but other collections don't. This extension would recognize :className
immediately after the {
and if the className were a collection it would translate, e.g.
{:Set 3 . 4 . 5 . 3}
{:Dictionary #a->1 . #b->2}
to
Set with: 3 with: 4 with: 5 with: 3
Dictionary with: #a->1 with: #b->2
If there are more than 6 values, it will use withAll:
and an array.
There isn't a convenient way to return multiple values from a method, or even to extract multiple values from a collection. We propose:
:| a b c | := some-collection
which would destructure the 3 elements of a SequenceableCollection or would extract the value of keys #a
#b
etc. if it was a Dictionary, with anything else being a runtime error. This is conveniently done by converting that to:
([:temp|
a := temp firstNamed: #a.
b := temp secondNamed: #b.
c := temp thirdNames: #c] value: some-collection)
These firstNamed:
, etc. methods would be trivial to implement in those collection classes and also other classes could handle them if appropriate. A nice estension might be for this syntax to also define those local variables in the nearest enclosing scope if they aren't already accessible.
While these might be convenient, they really would need a more generalized syntax, and even this simple version is hard to compile.
curry:
is very convenient to partially apply parameters to a binary operator. And of course the binary operator could be any of the above operators:
x := (3+).
y := (*4).
would become:
x := [:temp| 3+temp].
y := [:temp| temp*4].
These are very convenient to use with the parrot operator:
x :> + 3 :> (...): 7 :> [...] y :> (4-) :> - 4
You can load into a Pharo image Playground with:
Metacello new
baseline: 'PharoFunctional';
repository: 'github://dvmason/Pharo-Functional:master';
load: #compiler
Then for any class heirarchy where you want to use the extended syntax, add a trait to the class (the uses: line), like:
RBScannerTest subclass: #ComposeExampleTest
uses: ComposeSyntax
instanceVariableNames: ''
classVariableNames: ''
package: 'CompileWithCompose-Tests'
Or, on the class-side define the following method:
compilerClass
" Answer a compiler class appropriate for source methods of this class and subclasses. "
^ ComposeCompiler
You can use this second approach if you want to add it to the entire image (including in playgrounds), by defining this in Object class.
Functional support for Pharo
This project is to add to Smalltalk additional data structures and methods that are familiar to functional programmers
It includes Tuple class, which is a read-only Array.
It makes Symbols work as unary, binary, ternary, or quatrenary operators
It adds a unary and binary map:
and map:map:
to symbols, so:
#negated map: #(1 2 3)
equals ``#(-1 -2 -3)`#+ map: #(1 2 3) map: #(-1 0 1)
equals ``#(0 2 4)`
It also includes ZippedCollection which is a collection of Pairs.
collect:
and do:
on ZippedCollection will call the block/symbol with either a Pair (if it's 1-argument) or the pair of elements from the two collections (if it's 2-argument)
SequenceableCollections can be zipped together with either zip:
or `>==<
It also includes a curry:
method that allows composing a value and a binary operator or block, so:
(5 curry: [:l :r | l-r]) value: 4
equals1
([:l :r | l-r] curry: 4) value: 9
equals5
(5 curry: #-) value: 4
equals1
(#- curry: 4) value: 9
equals5
(where the latter 2 depend on Symbols working as binary operators)
If programming in a class with CompileWithCompose
enabled, you can use this as, for example:
4 (5 curry: #-)
equals1
x := #- curry: 4. 9 (x)
equals5
addArrays := #+ curry: #map:map:. #(1 2 3) (addArrays): #(-1 0 1)
equals#(0 2 4)
- note the(...):
because the curried function requires 2 values - i.e. it's a binary message.
Slices allow a view on a data structure that can be iterated over. This is an extension of slices as seen in Rust and Zig. This works well for iterators while minimizing the copying of data.
#(1 2 3 4 5 6 7 8) sliceFrom: 3 to: 7
:> reduceWith: #(true false true true false)
:> collect: #negated
would give #(-3 -5 -6)
without creating any garbage (i.e. the collect is the first place that allocates some memory.
It also includes a chain method that allows similar code to CompileWithCompose
but with no additional syntax (unfortunately quite a bit slower because it requires a DNU and perform
for each chained message:
foo
" self new foo >>> 42 "
^ 17 chain
negated
; min: -53
; abs
; < 100
; and: [ 4 > 2 ]
; and: [ 5 < 10 ]
; ifTrue: [ 42 ] ifFalse: [ 99 ]
Combinators allow combining blocks and symbols to produce new operators. This is more useful when combined with using expressions are operators (see above)
This is the identity combinator (I). It simply returns the passed value. This is useful in combination with other combinators.
This is the B combinator. It combines two blocks/symbols, applying the second after the first, so a |> b
is equivalent to [:x| b value: (a value: x)]
. It is an operator equivalent to the :>
parrot syntax.
This is the Q combinator known as the Queer Bird. a <| b
is equivalent to [:x | a value: (b value: x) ]
If the left argument is binary, this is the Psi or 'over' operator and a <|> b
is equivalent to [:x :y | a value: (b value: x) value: (b value: y) ]
.
Otherwise this is the B1 or 'atop' operator and a <|> b
is equivalent to [:x :y | a value: (b value: x value: y) ]
This is the backHook or the S combinator, depending on the arity of the right argument. a <*> b
is equivalent to either [:x| b value: (a value: x) value: x]
or [:x| a value: x value: (b value: x)]
This is the D combinator also known as the Dove bird. Depending on the arity of the left argument, a <-> b
is equivalent to either [:x :y| a value: x value: (b value: y) ]
or [:x :y| b value: (a value: x) value: y ]
This is the fold operator. It returns a block that acts like putting the operator between elements of the argument collection on the right.
This is the scan operator. It returns a block that acts like putting the operator between elements of the argument collection on the right like fold
but returns a collection.
#sorted <|> #=
tests if a parameter is already sorted
To load, in a Pharo9 image Playground do:
Metacello new
repository: 'github://dvmason/Pharo-Functional:master';
baseline: 'PharoFunctional';
load