Difference between revisions of "Guide:Core kinds"

From IokeWiki
Jump to: navigation, search
m (Reverted edits by Icotime (Talk) to last version by Olabini)
 
Line 1: Line 1:
=[http://imyqokyf.co.cc This Page Is Currently Under Construction And Will Be Available Shortly, Please Visit Reserve Copy Page]=
 
 
= Core kinds =
 
= Core kinds =
  
Line 8: Line 7:
 
One of the major parts of Ioke is the condition system. Unlike most other programming languages, Ioke doesn't have exceptions. Instead it has conditions, where errors are a specific kind of condition. The condition system comprises several different things. Specifically, the condition system uses the kinds Condition, Restart, Handler and Rescue. Restarts are mostly orthogonal to the rest of the system.
 
One of the major parts of Ioke is the condition system. Unlike most other programming languages, Ioke doesn't have exceptions. Instead it has conditions, where errors are a specific kind of condition. The condition system comprises several different things. Specifically, the condition system uses the kinds Condition, Restart, Handler and Rescue. Restarts are mostly orthogonal to the rest of the system.
  
The way the condition system works is this. When something happens, a program can elect to signal a condition. This can be done using "signal!", "warn!" or "error!". Both "warn!" and "error!" use "signal!" under the covers, but do some other things as well. A condition will always mimic Condition. Each of these three methods can be called in three different ways. First, you can call them with a text argument. In that case the signalled condition will be the default for that type. (The default for "signal!" is Condition Default. The default for "warn!" is Condition Warning Default and the default for "error!" is Condition Error Default.) A new mimic of the default condition will be created, and a cell called text will be set to the text argument. The second variation is to give an instance of an existing condition to one of the methods. In that case that condition will be signalled unmodified. Finally, the third version gives a condition mimic and one or more keyword arguments with data to set on that condition. In that case a mimic of the condition will be created, and then cells with data set based on the arguments.
+
The way the condition system works is this. When something happens, a program can elect to signal a condition. This can be done using "signal!", "warn!" or "error!". Both "warn!" and "error!" use "signal!" under the covers, but do some other things as well. A condition will always mimic Condition. Each of these three methods can be called in three different ways. First, you can call them with a text argument. In that case the signalled condition will be the default for that type. (The default for "signal!" is Condition Default. The default for "warn!" is Condition Warning Default and the default for "error!" is Condition Error Default.) A new mimic of the default condition will be created, and a cell called text will be set to the text argument. The second variation is to give an instance of an existing condition to one of the methods. In that case that condition will be signalled unmodified. Finally, the third version gives a condition mimic and one or more keyword arguments with data to set on that condition. In that case a mimic of the condition will be created, and then cells with data set based on the arguments.
  
 
If a signal is not handled, nothing happens.
 
If a signal is not handled, nothing happens.
Line 26: Line 25:
 
Both handlers, rescues and restarts are established inside a call to the bind macro. All arguments to this macro need to be either handlers, rescues or restarts, except for the last argument which should be the code to execute.
 
Both handlers, rescues and restarts are established inside a call to the bind macro. All arguments to this macro need to be either handlers, rescues or restarts, except for the last argument which should be the code to execute.
  
You create a new handler by calling the method "handle". You create a new rescue by calling the method "rescue". You create a new restart by calling the method "restart". These all take funky arguments, so refer to the reference to better understand how they work.
+
You create a new handler by calling the method "handle". You create a new rescue by calling the method "rescue". You create a new restart by calling the method "restart". These all take funky arguments, so refer to the reference to better understand how they work.
  
 
This small example doesn't necessarily show the power of conditions, but it can give an idea about how it works.
 
This small example doesn't necessarily show the power of conditions, but it can give an idea about how it works.
  
<source lang="ioke">;; to handle any problem
+
<source lang="ioke">;; to handle any problem
 
bind(
 
bind(
 
   rescue(fn(c, nil)), ;; do nothing in the rescue
 
   rescue(fn(c, nil)), ;; do nothing in the rescue
 
    
 
    
   error!(&quot;This is bad!!&quot;)
+
   error!("This is bad!!")
 
)
 
)
  
Line 41: Line 40:
 
   handle(fn(c, c println)),
 
   handle(fn(c, c println)),
 
    
 
    
   signal!(&quot;something&quot;)
+
   signal!("something")
   signal!(&quot;something more&quot;)
+
   signal!("something more")
  
   warn!(&quot;A warning!!&quot;)
+
   warn!("A warning!!")
 
)
 
)
  
Line 52: Line 51:
  
 
bind(
 
bind(
   rescue(C1, C2, fn(c, &quot;got an error: #{c}&quot; println)),
+
   rescue(C1, C2, fn(c, "got an error: #{c}" println)),
 
    
 
    
 
   error!(C1)
 
   error!(C1)
Line 70: Line 69:
 
    
 
    
 
   invokeRestart(:something, 1, 2, 3)
 
   invokeRestart(:something, 1, 2, 3)
)&lt;/source&gt;
+
)</source>
 
The code above shows several different things you can do with the condition system. It is a very powerful system, so I recommend trying to understand it. It lies at the core of many things in Ioke, and some parts will not make sense without a deep understanding of conditions. For more information on what such a system is capable of, look for documentation about the Common Lisp condition system, which has been a heavy influence on Ioke. Also, the debugger in IIk uses the condition system to implement its functionality.
 
The code above shows several different things you can do with the condition system. It is a very powerful system, so I recommend trying to understand it. It lies at the core of many things in Ioke, and some parts will not make sense without a deep understanding of conditions. For more information on what such a system is capable of, look for documentation about the Common Lisp condition system, which has been a heavy influence on Ioke. Also, the debugger in IIk uses the condition system to implement its functionality.
  
Line 77: Line 76:
 
Finally, one thing that you might miss if you're used to exceptions in other languages, is a construct that makes it possible to ensure that code gets executed, even if a non-local flow control happens. Don't despair, Ioke has one. It is called ensure, and works mostly like ensure in Ruby, and finally in Java. It takes one main code argument, followed by zero or more arguments that contain the code to always make sure executes. It looks like this:
 
Finally, one thing that you might miss if you're used to exceptions in other languages, is a construct that makes it possible to ensure that code gets executed, even if a non-local flow control happens. Don't despair, Ioke has one. It is called ensure, and works mostly like ensure in Ruby, and finally in Java. It takes one main code argument, followed by zero or more arguments that contain the code to always make sure executes. It looks like this:
  
&lt;source lang=&quot;ioke&quot;&gt;ensure(
+
<source lang="ioke">ensure(
 
   conn = Database open
 
   conn = Database open
 
   conn SELECT * FROM a_table,
 
   conn SELECT * FROM a_table,
 
   conn close!,
 
   conn close!,
 
   conn reallyClose!,
 
   conn reallyClose!,
   conn reallyReallyReallyClose!)&lt;/source&gt;
+
   conn reallyReallyReallyClose!)</source>
 
This code uses a hypothetical database library, opens up a connection, does something, and then in three different ensure blocks tries to ensure that it really is closed afterwards. The return value of the ensure block will still be the return value of the last expression in the main code. Non local flow control can happen inside of the ensure block, but exactly what will happen is undefined -- so avoid it, please.
 
This code uses a hypothetical database library, opens up a connection, does something, and then in three different ensure blocks tries to ensure that it really is closed afterwards. The return value of the ensure block will still be the return value of the last expression in the main code. Non local flow control can happen inside of the ensure block, but exactly what will happen is undefined -- so avoid it, please.
  
Line 93: Line 92:
 
You can do several things with Ioke text. These examples show some of the methods:
 
You can do several things with Ioke text. These examples show some of the methods:
  
&lt;source lang=&quot;ioke&quot;&gt;;; repeat a text several times
+
<source lang="ioke">;; repeat a text several times
&quot;foo&quot; * 3  ;; =&gt; &quot;foofoofoo&quot;
+
"foo" * 3  ;; => "foofoofoo"
  
 
;; concatenate two texts
 
;; concatenate two texts
&quot;foo&quot; + &quot;bar&quot; ;; =&gt; &quot;foobar&quot;
+
"foo" + "bar" ;; => "foobar"
  
 
;; get the character at a specific index
 
;; get the character at a specific index
&quot;foo&quot;[1]  ;; =&gt; 111
+
"foo"[1]  ;; => 111
  
 
;; get a subset of text
 
;; get a subset of text
&quot;foo&quot;[1..1]  ;; =&gt; &quot;o&quot;
+
"foo"[1..1]  ;; => "o"
&quot;foo&quot;[0..1]  ;; =&gt; &quot;fo&quot;
+
"foo"[0..1]  ;; => "fo"
&quot;foo&quot;[0...1]  ;; =&gt; &quot;f&quot;
+
"foo"[0...1]  ;; => "f"
  
&quot;foxtrot&quot;[1..-1] ;; =&gt; &quot;oxtrot&quot;
+
"foxtrot"[1..-1] ;; => "oxtrot"
&quot;foxtrot&quot;[1...-1] ;; =&gt; &quot;oxtro&quot;
+
"foxtrot"[1...-1] ;; => "oxtro"
  
 
;; is a text empty?
 
;; is a text empty?
&quot;foo&quot; empty?  ;; =&gt; false
+
"foo" empty?  ;; => false
&quot;&quot; empty?  ;; =&gt; true
+
"" empty?  ;; => true
  
 
;; the length of the text
 
;; the length of the text
&quot;foo&quot; length  ;; =&gt; 3
+
"foo" length  ;; => 3
&quot;&quot; length  ;; =&gt; 0
+
"" length  ;; => 0
  
 
;; replace the first occurrence of something with something else
 
;; replace the first occurrence of something with something else
&quot;hello fox fob folk&quot; replace(&quot;fo&quot;, &quot;ba&quot;)
+
"hello fox fob folk" replace("fo", "ba")
   ;; =&gt; &quot;hello bax fob folk&quot;
+
   ;; => "hello bax fob folk"
  
 
;; replace all occurrences of something with something else
 
;; replace all occurrences of something with something else
&quot;hello fox fob folk&quot; replaceAll(&quot;fo&quot;, &quot;ba&quot;)
+
"hello fox fob folk" replaceAll("fo", "ba")
   ;; =&gt; &quot;hello bax bab balk&quot;
+
   ;; => "hello bax bab balk"
  
 
;; split around a text
 
;; split around a text
&quot;foo bar bax&quot; split(&quot; &quot;)
+
"foo bar bax" split(" ")
   ;; =&gt; [&quot;foo&quot;, &quot;bar&quot;, &quot;bax&quot;]&lt;/source&gt;
+
   ;; => ["foo", "bar", "bax"]</source>
 
The Text kind contains lots of useful functionality like this. The purpose is to make it really easy to massage text of any kind.
 
The Text kind contains lots of useful functionality like this. The purpose is to make it really easy to massage text of any kind.
  
One important tool for doing that is the &amp;quot;format&amp;quot; method. This is a mix between C printf and Common Lisp format. At the moment, it only contains a small amount of functionality, but it can still be very convenient. Specifically you can print each element in a list directly by using format, instead of concatenating text yourself.
+
One important tool for doing that is the &quot;format&quot; method. This is a mix between C printf and Common Lisp format. At the moment, it only contains a small amount of functionality, but it can still be very convenient. Specifically you can print each element in a list directly by using format, instead of concatenating text yourself.
  
The &amp;quot;format&amp;quot; method takes format specifiers that begin with %, and then inserts one of its arguments in different ways depending on what kind of format specifier is used.
+
The &quot;format&quot; method takes format specifiers that begin with %, and then inserts one of its arguments in different ways depending on what kind of format specifier is used.
  
 
Some examples of format follow:
 
Some examples of format follow:
  
&lt;source lang=&quot;ioke&quot;&gt;;; insert simple value as text
+
<source lang="ioke">;; insert simple value as text
&quot;%s&quot; format(123) ;; =&gt; &quot;123&quot;
+
"%s" format(123) ;; => "123"
  
 
;; insert value right justified by 6
 
;; insert value right justified by 6
&quot;%6s&quot; format(123) ;; =&gt; &quot;   123&quot;
+
"%6s" format(123) ;; => "   123"
  
 
;; insert value left justified by 6
 
;; insert value left justified by 6
&quot;%-6s&quot; format(123) ;; =&gt; &quot;123  &quot;
+
"%-6s" format(123) ;; => "123  "
  
 
;; insert two values
 
;; insert two values
&quot;%s: %s&quot; format(123, 321) ;; =&gt; &quot;123: 321&quot;
+
"%s: %s" format(123, 321) ;; => "123: 321"
  
 
;; insert a list of values formatted the same
 
;; insert a list of values formatted the same
&quot;%[%s, %]\n&quot; format([1,2,3])
+
"%[%s, %]\n" format([1,2,3])
   ;; =&gt; &quot;1, 2, 3, \n&quot;
+
   ;; => "1, 2, 3, \n"
  
 
;; insert a dict of values formatted the same
 
;; insert a dict of values formatted the same
&quot;%:[%s =&gt; %s, %]\n&quot; format({sam: 3, toddy: 10})
+
"%:[%s => %s, %]\n" format({sam: 3, toddy: 10})
   ;; =&gt; &quot;sam =&gt; 3, toddy =&gt; 10, \n&quot;
+
   ;; => "sam => 3, toddy => 10, \n"
  
 
;; insert splatted values from a list
 
;; insert splatted values from a list
&quot;wow: %*[%s: %s %]&quot; format([[1,2],[2,3],[3,4]])
+
"wow: %*[%s: %s %]" format([[1,2],[2,3],[3,4]])
   ;; =&gt; &quot;wow: 1: 2 2: 3 3: 4 &quot;&lt;/source&gt;
+
   ;; => "wow: 1: 2 2: 3 3: 4 "</source>
  
 
== Numbers ==
 
== Numbers ==
Line 165: Line 164:
 
As mentioned in the section on syntax, Ioke supports decimal numbers, integers and ratios. A Ratio will be created when two integers can't be divided evenly. A Ratio will always use the GCD. In most cases Ratios, Decimals and Integers can interact with each other as would be expected. The one thing that might surprise people is that Ioke doesn't have any inexact floating point data type. Instead, decimals are exact and can have any size. This means they are well suited to represent such things as money, since operations will always have well defined results.
 
As mentioned in the section on syntax, Ioke supports decimal numbers, integers and ratios. A Ratio will be created when two integers can't be divided evenly. A Ratio will always use the GCD. In most cases Ratios, Decimals and Integers can interact with each other as would be expected. The one thing that might surprise people is that Ioke doesn't have any inexact floating point data type. Instead, decimals are exact and can have any size. This means they are well suited to represent such things as money, since operations will always have well defined results.
  
All expected math works fine on Ioke numbers. The reference for numbers more closely specify what is possible. One thing to notice is that the % operator implements modulus, not remainder. This might be unintuitive for some developers. What that means is that it is not an error to ask for the modulus of 0: &amp;quot;13 % 0&amp;quot;, since there is no division necessary in this operation.
+
All expected math works fine on Ioke numbers. The reference for numbers more closely specify what is possible. One thing to notice is that the % operator implements modulus, not remainder. This might be unintuitive for some developers. What that means is that it is not an error to ask for the modulus of 0: &quot;13 % 0&quot;, since there is no division necessary in this operation.
  
In addition to the above, Integers have the &amp;quot;times&amp;quot; method described earlier. It can also return the successor and predecessor of itself with the &amp;quot;pred&amp;quot; and &amp;quot;succ&amp;quot; methods.
+
In addition to the above, Integers have the &quot;times&quot; method described earlier. It can also return the successor and predecessor of itself with the &quot;pred&quot; and &quot;succ&quot; methods.
  
 
== Lists ==
 
== Lists ==
Line 173: Line 172:
 
Ioke has lists that expand to the size needed. These lists can be created using a simple literal syntax. They can contain any kind of element, including itself (although don't try to print such a list). Ioke List mixes in the Enumerable mixin, which gives it quite powerful capabilities.
 
Ioke has lists that expand to the size needed. These lists can be created using a simple literal syntax. They can contain any kind of element, including itself (although don't try to print such a list). Ioke List mixes in the Enumerable mixin, which gives it quite powerful capabilities.
  
A list can be created using the &amp;quot;list&amp;quot; or &amp;quot;[]&amp;quot; methods:
+
A list can be created using the &quot;list&quot; or &quot;[]&quot; methods:
  
&lt;source lang=&quot;ioke&quot;&gt;;; an empty list
+
<source lang="ioke">;; an empty list
 
[]
 
[]
  
Line 183: Line 182:
  
 
;; a list with different elements
 
;; a list with different elements
[1, &quot;one&quot;, :one]&lt;/source&gt;
+
[1, "one", :one]</source>
 
Except for the Enumerable methods, List also defines many other methods that can be highly useful. Some examples are shown below:
 
Except for the Enumerable methods, List also defines many other methods that can be highly useful. Some examples are shown below:
  
&lt;source lang=&quot;ioke&quot;&gt;l = [1, 2, 3]
+
<source lang="ioke">l = [1, 2, 3]
  
 
;; add two lists together
 
;; add two lists together
l + l ;; =&gt; [1, 2, 3, 1, 2, 3]
+
l + l ;; => [1, 2, 3, 1, 2, 3]
  
 
;; return the difference of two lists
 
;; return the difference of two lists
l - [2] ;; =&gt; [1, 3]
+
l - [2] ;; => [1, 3]
  
 
;; add a new value to a list
 
;; add a new value to a list
l &lt;&lt; 42
+
l << 42
l == [1, 2, 3, 42] ;; =&gt; true
+
l == [1, 2, 3, 42] ;; => true
  
 
;; get a specific element from the list
 
;; get a specific element from the list
l[0]  ;; =&gt; 1
+
l[0]  ;; => 1
  
 
;; -1 returns the last, -2 the next to last
 
;; -1 returns the last, -2 the next to last
l[-1] ;; =&gt; 42
+
l[-1] ;; => 42
  
 
;; an index outside the boundaries return nil
 
;; an index outside the boundaries return nil
l[10] ;; =&gt; nil
+
l[10] ;; => nil
  
 
;; assign a new value
 
;; assign a new value
 
l[3] = 40
 
l[3] = 40
l == [1,2,3,40] ;; =&gt; true
+
l == [1,2,3,40] ;; => true
  
 
l[-1] = 39
 
l[-1] = 39
l == [1,2,3,39] ;; =&gt; true
+
l == [1,2,3,39] ;; => true
  
 
;; assign an out of bounds value
 
;; assign an out of bounds value
 
l[10] = 13
 
l[10] = 13
 
l == [1,2,3,39,nil,nil,nil,nil,nil,nil,13]
 
l == [1,2,3,39,nil,nil,nil,nil,nil,nil,13]
   ;; =&gt; true
+
   ;; => true
  
 
;; at and at= is the same as [] and []=
 
;; at and at= is the same as [] and []=
l at(0) ;; =&gt; 1
+
l at(0) ;; => 1
  
 
;; empty the list
 
;; empty the list
 
l clear!
 
l clear!
l == [] ;; =&gt; true
+
l == [] ;; => true
  
 
;; follows the each protocol
 
;; follows the each protocol
Line 230: Line 229:
  
 
;; is empty?
 
;; is empty?
l empty? ;; =&gt; true
+
l empty? ;; => true
  
 
;; does it include an element?
 
;; does it include an element?
l include?(:foo) ;; =&gt; false
+
l include?(:foo) ;; => false
  
 
;; the last element
 
;; the last element
l last ;; =&gt; nil
+
l last ;; => nil
[1, 2] last ;; =&gt; 2
+
[1, 2] last ;; => 2
  
 
;; the length
 
;; the length
[1, 2] length ;; =&gt; 2
+
[1, 2] length ;; => 2
  
 
;; first value
 
;; first value
[1, 2] first ;; =&gt; 1
+
[1, 2] first ;; => 1
  
 
;; rest except for first
 
;; rest except for first
[1, 2, 3] rest ;; =&gt; [2, 3]
+
[1, 2, 3] rest ;; => [2, 3]
  
 
;; returns a new sorted list
 
;; returns a new sorted list
[3, 2, 1] sort ;; =&gt; [1, 2, 3]
+
[3, 2, 1] sort ;; => [1, 2, 3]
  
 
;; sorts in place
 
;; sorts in place
 
l = [3, 2, 1]
 
l = [3, 2, 1]
 
l sort!
 
l sort!
l == [1, 2, 3] ;; =&gt; true&lt;/source&gt;
+
l == [1, 2, 3] ;; => true</source>
  
 
== Dicts ==
 
== Dicts ==
Line 260: Line 259:
 
A Dict is a dictionary of key-value mappings. The mappings are unordered, and there can only ever be one key with the same value. Any kind of Ioke object can be used as a key. There is no problem with having the same value for different keys. The default implementation of Dict uses a hash-based implementation. That's not necessarily always true for all dicts. The iteration order is not necessarily stable either, so don't write code that depends on it.
 
A Dict is a dictionary of key-value mappings. The mappings are unordered, and there can only ever be one key with the same value. Any kind of Ioke object can be used as a key. There is no problem with having the same value for different keys. The default implementation of Dict uses a hash-based implementation. That's not necessarily always true for all dicts. The iteration order is not necessarily stable either, so don't write code that depends on it.
  
Creating a dict is done using either the &amp;quot;dict&amp;quot; or the &amp;quot;{}&amp;quot; methods. Both of these expect either keyword arguments or mimics of Pair. If keyword arguments, these keywords will be used as symbol keys. That's the most common thing, so it makes sense to have that happen automatically. Dicts also try to print themselves that way.
+
Creating a dict is done using either the &quot;dict&quot; or the &quot;{}&quot; methods. Both of these expect either keyword arguments or mimics of Pair. If keyword arguments, these keywords will be used as symbol keys. That's the most common thing, so it makes sense to have that happen automatically. Dicts also try to print themselves that way.
  
&lt;source lang=&quot;ioke&quot;&gt;dict(1 =&gt; 2, 3 =&gt; 4)
+
<source lang="ioke">dict(1 => 2, 3 => 4)
  
 
;; these two are the same
 
;; these two are the same
dict(foo: &quot;bar&quot;, baaz: &quot;quux&quot;)
+
dict(foo: "bar", baaz: "quux")
dict(:foo =&gt; &quot;bar&quot;, :baaz =&gt; &quot;quux&quot;)
+
dict(:foo => "bar", :baaz => "quux")
  
{1 =&gt; 2, 3 =&gt; 4}
+
{1 => 2, 3 => 4}
  
 
;; these two are the same
 
;; these two are the same
{foo: &quot;bar&quot;, baaz: &quot;quux&quot;}
+
{foo: "bar", baaz: "quux"}
{:foo =&gt; &quot;bar&quot;, :baaz =&gt; &quot;quux&quot;}
+
{:foo => "bar", :baaz => "quux"}
  
 
;; the formats can be combined:
 
;; the formats can be combined:
{1 =&gt; 2, foo: 42, &quot;bar&quot; =&gt; &quot;qux&quot;}&lt;/source&gt;
+
{1 => 2, foo: 42, "bar" => "qux"}</source>
The literal Pair syntax (using =&amp;gt;) will not necessarily instantiate real pairs for this.
+
The literal Pair syntax (using =&gt;) will not necessarily instantiate real pairs for this.
  
 
Dicts mix in Enumerable. When using each, what will be yielded are mimics of Pair, where the first value will be the key and the second will be value. Just like Lists, Dicts have several useful methods in themselves:
 
Dicts mix in Enumerable. When using each, what will be yielded are mimics of Pair, where the first value will be the key and the second will be value. Just like Lists, Dicts have several useful methods in themselves:
  
&lt;source lang=&quot;ioke&quot;&gt;d = {one: &quot;two&quot;, 3 =&gt; 4}
+
<source lang="ioke">d = {one: "two", 3 => 4}
  
;; lookup with [], &quot;at&quot; works the same
+
;; lookup with [], "at" works the same
d[:one] ;; =&gt; &quot;two&quot;
+
d[:one] ;; => "two"
d[:two] ;; =&gt; nil
+
d[:two] ;; => nil
d[3]    ;; =&gt; 4
+
d[3]    ;; => 4
d[4]    ;; =&gt; nil
+
d[4]    ;; => nil
  
 
;; assign values with []=
 
;; assign values with []=
d[:one] = &quot;three&quot;
+
d[:one] = "three"
d[:new] = &quot;wow!&quot;
+
d[:new] = "wow!"
  
d == {one: &quot;three&quot;, 3 =&gt; 4, new: &quot;wow!&quot;}
+
d == {one: "three", 3 => 4, new: "wow!"}
  
 
;; iterate over it
 
;; iterate over it
Line 298: Line 297:
  
 
;; get all keys
 
;; get all keys
d keys == set(:one, :new, 3)&lt;/source&gt;
+
d keys == set(:one, :new, 3)</source>
  
 
== Sets ==
 
== Sets ==
  
If you want an object that work like a mathematical set, Ioke provides such a kind for you. There is also literal syntax for sets from [[Ioke E]]. This syntax looks like this: &lt;code&gt;#{42, 55, 111}&lt;/code&gt; A set can be iterated over and it is Enumerable. You can add and remove elements, and check for membership.
+
If you want an object that work like a mathematical set, Ioke provides such a kind for you. There is also literal syntax for sets from [[Ioke E]]. This syntax looks like this: <code>#{42, 55, 111}</code> A set can be iterated over and it is Enumerable. You can add and remove elements, and check for membership.
  
&lt;source lang=&quot;ioke&quot;&gt;x = set(1,2,3,3,2,1)
+
<source lang="ioke">x = set(1,2,3,3,2,1)
  
x map(*2) sort ; =&gt; [2, 4, 6]
+
x map(*2) sort ; => [2, 4, 6]
  
x === 1 ; =&gt; true
+
x === 1 ; => true
x === 0 ; =&gt; false
+
x === 0 ; => false
  
 
x remove!(2)
 
x remove!(2)
x === 2 ; =&gt; false
+
x === 2 ; => false
  
x &lt;&lt; 4
+
x << 4
x === 4 ; =&gt; true&lt;/source&gt;
+
x === 4 ; => true</source>
  
 
Sets also have methods to do union and intersection. These mimic the mathematical operators for these operations:
 
Sets also have methods to do union and intersection. These mimic the mathematical operators for these operations:
  
&lt;source lang=&quot;ioke&quot;&gt;; intersection
+
<source lang="ioke">; intersection
 
(#{1,2,3,4} ∩ #{1,2,3,4,5}) should == #{1,2,3,4}
 
(#{1,2,3,4} ∩ #{1,2,3,4,5}) should == #{1,2,3,4}
  
 
;union
 
;union
 
(#{1,2} ∪ #{1,3}) should == #{1,2,3}
 
(#{1,2} ∪ #{1,3}) should == #{1,2,3}
&lt;/source&gt;
+
</source>
  
 
You can also test membership, nonmembership, subset and superset relations using the mathematical operators:
 
You can also test membership, nonmembership, subset and superset relations using the mathematical operators:
&lt;source lang=&quot;ioke&quot;&gt;; membership
+
<source lang="ioke">; membership
 
42 ∈ #{1,2,42,3}
 
42 ∈ #{1,2,42,3}
  
Line 337: Line 336:
  
 
; proper subset
 
; proper subset
#{&quot;bar&quot;, &quot;foo&quot;} ⊂ #{&quot;foo&quot;, &quot;bar&quot;, &quot;quux&quot;}
+
#{"bar", "foo"} ⊂ #{"foo", "bar", "quux"}
  
 
; superset
 
; superset
Line 343: Line 342:
  
 
; proper superset
 
; proper superset
#{&quot;bar&quot;, &quot;foo&quot;, &quot;quux&quot;} ⊃ #{&quot;foo&quot;, &quot;bar&quot;}
+
#{"bar", "foo", "quux"} ⊃ #{"foo", "bar"}
&lt;/source&gt;
+
</source>
  
 
== Ranges and Pairs ==
 
== Ranges and Pairs ==
Line 350: Line 349:
 
Both ranges and pairs tie two values together. They also have literal syntax to create them, since they are very useful in many circumstances.
 
Both ranges and pairs tie two values together. They also have literal syntax to create them, since they are very useful in many circumstances.
  
A Range defines two endpoints. A Range is Enumerable and you can also check for membership. It's also convenient to send Ranges to the &amp;quot;List []&amp;quot; method. A Range can be exclusive or inclusive. If it's inclusive it includes the end value, and if it is exclusive it doesn't.
+
A Range defines two endpoints. A Range is Enumerable and you can also check for membership. It's also convenient to send Ranges to the &quot;List []&quot; method. A Range can be exclusive or inclusive. If it's inclusive it includes the end value, and if it is exclusive it doesn't.
  
An addition to Ioke S is the possibility of inverted ranges. If the first value is larger than the second value, then the range is inverted. This puts slightly different demands on the objects inside of it. Specifically, if you want to iterate over the elements, the kind you're using need to have a method called 'pred' for predecessor, instead of 'succ' for successor. Membership can still be tested, as long as &amp;lt;=&amp;gt; is defined. So you can do something like this: (&amp;quot;foo&amp;quot;..&amp;quot;aoo&amp;quot;) === &amp;quot;boo&amp;quot;. It's mostly useful for iterating in the opposite direction, like with 10..1, for example.
+
An addition to Ioke S is the possibility of inverted ranges. If the first value is larger than the second value, then the range is inverted. This puts slightly different demands on the objects inside of it. Specifically, if you want to iterate over the elements, the kind you're using need to have a method called 'pred' for predecessor, instead of 'succ' for successor. Membership can still be tested, as long as &lt;=&gt; is defined. So you can do something like this: (&quot;foo&quot;..&quot;aoo&quot;) === &quot;boo&quot;. It's mostly useful for iterating in the opposite direction, like with 10..1, for example.
  
&lt;source lang=&quot;ioke&quot;&gt;;; literal syntax for inclusive range
+
<source lang="ioke">;; literal syntax for inclusive range
 
1..10
 
1..10
  
Line 361: Line 360:
  
 
;; check for membership
 
;; check for membership
(1..10) === 5 ;; =&gt; true
+
(1..10) === 5 ;; => true
(1..10) === 10 ;; =&gt; true
+
(1..10) === 10 ;; => true
(1..10) === 11 ;; =&gt; false
+
(1..10) === 11 ;; => false
  
(1...10) === 5 ;; =&gt; true
+
(1...10) === 5 ;; => true
(1...10) === 10 ;; =&gt; false
+
(1...10) === 10 ;; => false
(1...10) === 11 ;; =&gt; false
+
(1...10) === 11 ;; => false
  
 
;; get the from value
 
;; get the from value
(1..10) from == 1 ;; =&gt; true
+
(1..10) from == 1 ;; => true
(1...10) from == 1 ;; =&gt; true
+
(1...10) from == 1 ;; => true
  
 
;; get the to value
 
;; get the to value
(1..10) to == 10 ;; =&gt; true
+
(1..10) to == 10 ;; => true
(1...10) to == 10 ;; =&gt; true
+
(1...10) to == 10 ;; => true
  
 
;; is this range exclusive?
 
;; is this range exclusive?
(1..10) exclusive? ;; =&gt; false
+
(1..10) exclusive? ;; => false
(1...10) exclusive? ;; =&gt; true
+
(1...10) exclusive? ;; => true
  
 
;; is this range inclusive?
 
;; is this range inclusive?
(1..10) inclusive? ;; =&gt; true
+
(1..10) inclusive? ;; => true
(1...10) inclusive? ;; =&gt; false&lt;/source&gt;
+
(1...10) inclusive? ;; => false</source>
A Pair represents a combination of two values. They don't have to be of the same kind. They can have any kind of relationship. Since Pairs are often used to represent the elements of Dicts, it is very useful to refer to the first value as the &amp;quot;key&amp;quot;, and the second value as the &amp;quot;value&amp;quot;.
+
A Pair represents a combination of two values. They don't have to be of the same kind. They can have any kind of relationship. Since Pairs are often used to represent the elements of Dicts, it is very useful to refer to the first value as the &quot;key&quot;, and the second value as the &quot;value&quot;.
  
&lt;source lang=&quot;ioke&quot;&gt;;; literal syntax for a pair
+
<source lang="ioke">;; literal syntax for a pair
&quot;foo&quot; =&gt; &quot;bar&quot;
+
"foo" => "bar"
  
 
;; getting the first value
 
;; getting the first value
(&quot;foo&quot; =&gt; &quot;bar&quot;) first ;; =&gt; &quot;foo&quot;
+
("foo" => "bar") first ;; => "foo"
(&quot;foo&quot; =&gt; &quot;bar&quot;) key ;; =&gt; &quot;foo&quot;
+
("foo" => "bar") key ;; => "foo"
  
 
;; getting the second value
 
;; getting the second value
(&quot;foo&quot; =&gt; &quot;bar&quot;) second ;; =&gt; &quot;bar&quot;
+
("foo" => "bar") second ;; => "bar"
(&quot;foo&quot; =&gt; &quot;bar&quot;) value ;; =&gt; &quot;bar&quot;&lt;/source&gt;
+
("foo" => "bar") value ;; => "bar"</source>
  
 
== Tuples ==
 
== Tuples ==
  
A Pair is a special case of wanting to tie two values together. The general case of bundling several values together can be achieved by using Tuples. If you give anything else than one argument to the empty method, you will get back a Tuple. You can also create a new tuple explicitly using the &lt;code&gt;tuple&lt;/code&gt; method.
+
A Pair is a special case of wanting to tie two values together. The general case of bundling several values together can be achieved by using Tuples. If you give anything else than one argument to the empty method, you will get back a Tuple. You can also create a new tuple explicitly using the <code>tuple</code> method.
  
 
To extract elements out of the tuple the main way is using destructuring assignment. It is also possible to use a named extractor to get a given index.
 
To extract elements out of the tuple the main way is using destructuring assignment. It is also possible to use a named extractor to get a given index.
Line 405: Line 404:
 
Tuples are by design not Enumerable. This is a feature, not a misfeature. If you use tuples as something that should be enumerated over, you're using the wrong data structure. Also remember that Enumerable defines asTuple - so anything enumerable can be turned into a tuple if needed.
 
Tuples are by design not Enumerable. This is a feature, not a misfeature. If you use tuples as something that should be enumerated over, you're using the wrong data structure. Also remember that Enumerable defines asTuple - so anything enumerable can be turned into a tuple if needed.
  
&lt;source lang=&quot;ioke&quot;&gt;t = tuple(42, 55, 18)
+
<source lang="ioke">t = tuple(42, 55, 18)
 
(x, y, z) = t
 
(x, y, z) = t
 
t third should == 18
 
t third should == 18
t _3 should == 18&lt;/source&gt;
+
t _3 should == 18</source>
 
As you can see english-style names are defined for accessing elements. These will only be available up to nine elements. The underscore syntax for indexing will always be possible, though.
 
As you can see english-style names are defined for accessing elements. These will only be available up to nine elements. The underscore syntax for indexing will always be possible, though.
  
Line 416: Line 415:
  
 
You create a new one by calling the Struct method:
 
You create a new one by calling the Struct method:
&lt;source lang=&quot;ioke&quot;&gt;Person = Struct(:first_name, :sur_name, :age)&lt;/source&gt;
+
<source lang="ioke">Person = Struct(:first_name, :sur_name, :age)</source>
  
 
You can provide default values using keyword arguments instead:
 
You can provide default values using keyword arguments instead:
&lt;source lang=&quot;ioke&quot;&gt;Person = Struct(:first_name, :sur_name, age: -1)&lt;/source&gt;
+
<source lang="ioke">Person = Struct(:first_name, :sur_name, age: -1)</source>
  
 
Creating a new version of a Struct can be done using the object as a function:
 
Creating a new version of a Struct can be done using the object as a function:
&lt;source lang=&quot;ioke&quot;&gt;o = Person(&quot;Ola&quot;, &quot;Bini&quot;)&lt;/source&gt;
+
<source lang="ioke">o = Person("Ola", "Bini")</source>
  
 
It is also possible to provide these values using keywords - or even a mix:
 
It is also possible to provide these values using keywords - or even a mix:
&lt;source lang=&quot;ioke&quot;&gt;s = Person(first_name: &quot;Sam&quot;, sur_name: &quot;Aaron&quot;)
+
<source lang="ioke">s = Person(first_name: "Sam", sur_name: "Aaron")
b = Person(sur_name: &quot;Guthrie&quot;, &quot;Brian&quot;)&lt;/source&gt;
+
b = Person(sur_name: "Guthrie", "Brian")</source>
  
All of these objects will implement &lt;nowiki&gt;==&lt;/nowiki&gt;, hash, asText, notice and inspect. Accessors and mutators also work.
+
All of these objects will implement <nowiki>==</nowiki>, hash, asText, notice and inspect. Accessors and mutators also work.
  
You can get access to all the Struct attributes by calling &lt;code&gt;attributes&lt;/code&gt;. If you just want the names you can use &lt;code&gt;attributeNames&lt;/code&gt;. A Struct instance is also Sequenced over a combination of its keys and values.
+
You can get access to all the Struct attributes by calling <code>attributes</code>. If you just want the names you can use <code>attributeNames</code>. A Struct instance is also Sequenced over a combination of its keys and values.
  
 
== Enumerable ==
 
== Enumerable ==
Line 440: Line 439:
 
Mapping a collection into a another collection can be done using map or mapFn. These are aliased as collect and collectFn too.
 
Mapping a collection into a another collection can be done using map or mapFn. These are aliased as collect and collectFn too.
  
&lt;source lang=&quot;ioke&quot;&gt;l = [10, 20, 30, 40]
+
<source lang="ioke">l = [10, 20, 30, 40]
  
 
;; mapping into text
 
;; mapping into text
l map(asText) ;; =&gt; [&quot;10&quot;, &quot;20&quot;, &quot;30&quot;, &quot;40&quot;]
+
l map(asText) ;; => ["10", "20", "30", "40"]
l map(n, n asText) ;; =&gt; [&quot;10&quot;, &quot;20&quot;, &quot;30&quot;, &quot;40&quot;]
+
l map(n, n asText) ;; => ["10", "20", "30", "40"]
  
 
;; exponentiation
 
;; exponentiation
l map(**2) ;; =&gt; [100, 400, 900, 1600]
+
l map(**2) ;; => [100, 400, 900, 1600]
l map(n, n*n) ;; =&gt; [100, 400, 900, 1600]&lt;/source&gt;
+
l map(n, n*n) ;; => [100, 400, 900, 1600]</source>
 
Filtering the contents of a collection can be done using select, which is aliased as filter and findAll.
 
Filtering the contents of a collection can be done using select, which is aliased as filter and findAll.
  
&lt;source lang=&quot;ioke&quot;&gt;;; with no arguments, return all true things
+
<source lang="ioke">;; with no arguments, return all true things
[nil, false, 13, 42] select ;; =&gt; [13, 42]
+
[nil, false, 13, 42] select ;; => [13, 42]
  
 
l = [1, 2, 3, 4, 5, 6]
 
l = [1, 2, 3, 4, 5, 6]
  
 
;; all elements over 3
 
;; all elements over 3
l select(&gt;3) ;; =&gt; [4, 5, 6]
+
l select(>3) ;; => [4, 5, 6]
l select(n, n&gt;3) ;; =&gt; [4, 5, 6]&lt;/source&gt;
+
l select(n, n>3) ;; => [4, 5, 6]</source>
 
A very common operation is to create one object based on the contents of a collection. This operation has different names in different languages. In Ioke it is called inject, but it is aliased as reduce and fold.
 
A very common operation is to create one object based on the contents of a collection. This operation has different names in different languages. In Ioke it is called inject, but it is aliased as reduce and fold.
  
&lt;source lang=&quot;ioke&quot;&gt;l = [1, 2, 3, 4, 5]
+
<source lang="ioke">l = [1, 2, 3, 4, 5]
  
 
;; inject around a message chain
 
;; inject around a message chain
l inject(+) ;; =&gt; 15
+
l inject(+) ;; => 15
l inject(*) ;; =&gt; 120
+
l inject(*) ;; => 120
  
 
;; with one arg
 
;; with one arg
l inject(n, *n) ;; =&gt; 120
+
l inject(n, *n) ;; => 120
l inject(n, +n*2) ;; =&gt; 29
+
l inject(n, +n*2) ;; => 29
  
 
;; with two args
 
;; with two args
l inject(sum, n, sum*n) ;; =&gt; 120
+
l inject(sum, n, sum*n) ;; => 120
l inject(sum, n, sum*2 + n*3) ;; =&gt; 139
+
l inject(sum, n, sum*2 + n*3) ;; => 139
  
 
;; with three args
 
;; with three args
l inject(1, sum, n, sum*n) ;; =&gt; 120
+
l inject(1, sum, n, sum*n) ;; => 120
l inject(10, sum, n, sum*n) ;; =&gt; 1200&lt;/source&gt;
+
l inject(10, sum, n, sum*n) ;; => 1200</source>
  
Most of the Enumerable methods that returns lists also have equivalent methods that return dictionaries or sets. Say for example that you want to filter out some elements from a set and return a new set. The best way to achive this in Ioke is to use the &lt;code&gt;filter:set&lt;/code&gt; method. If you want the output to be a dictionary you use &lt;code&gt;filter:dict&lt;/code&gt; instead. All of these methods follow the same convention. The dictionary versions will work differently depending on if the values are Pairs or not - if they are, the resulting dictionary will use these pairs directly. Otherwise it will use the element as keys and set the values to nil:
+
Most of the Enumerable methods that returns lists also have equivalent methods that return dictionaries or sets. Say for example that you want to filter out some elements from a set and return a new set. The best way to achive this in Ioke is to use the <code>filter:set</code> method. If you want the output to be a dictionary you use <code>filter:dict</code> instead. All of these methods follow the same convention. The dictionary versions will work differently depending on if the values are Pairs or not - if they are, the resulting dictionary will use these pairs directly. Otherwise it will use the element as keys and set the values to nil:
  
&lt;source lang=&quot;ioke&quot;&gt;(1..15) filter:dict(&lt;5) should == {1 =&gt; nil, 2 =&gt; nil, 3 =&gt; nil, 4 =&gt; nil}
+
<source lang="ioke">(1..15) filter:dict(<5) should == {1 => nil, 2 => nil, 3 => nil, 4 => nil}
{1 =&gt; &quot;foo 1&quot;, 2 =&gt; &quot;foo 2&quot;, 3 =&gt; &quot;foo 3&quot;, 4 =&gt; &quot;foo 4&quot;} filter:dict(key&lt;3) should == {1 =&gt; &quot;foo 1&quot;, 2 =&gt; &quot;foo 2&quot;}&lt;/source&gt;
+
{1 => "foo 1", 2 => "foo 2", 3 => "foo 3", 4 => "foo 4"} filter:dict(key<3) should == {1 => "foo 1", 2 => "foo 2"}</source>
  
 
== Sequences ==
 
== Sequences ==
  
In addition to the internal iterator protocol of Enumerable, Ioke also supports external iterators in the form of sequences. These all rely on the object to iterate over having a &lt;code&gt;seq&lt;/code&gt; method that returns a Sequence object. If you have &lt;code&gt;seq&lt;/code&gt; you can mixin Mixins Sequenced - which makes you Enumerable without any effort. As such, the sequence protocol is much easier to implement than the Enumerable protocol, and gives more functionality.
+
In addition to the internal iterator protocol of Enumerable, Ioke also supports external iterators in the form of sequences. These all rely on the object to iterate over having a <code>seq</code> method that returns a Sequence object. If you have <code>seq</code> you can mixin Mixins Sequenced - which makes you Enumerable without any effort. As such, the sequence protocol is much easier to implement than the Enumerable protocol, and gives more functionality.
  
 
Ioke sequences are lazy, which means that they can represent several things that are not possible using internal iterators.
 
Ioke sequences are lazy, which means that they can represent several things that are not possible using internal iterators.
  
When mixing in &lt;code&gt;Mixins Sequenced&lt;/code&gt; you get these methods:
+
When mixing in <code>Mixins Sequenced</code> you get these methods:
  
 
* collected
 
* collected
Line 505: Line 504:
 
* zipped
 
* zipped
  
All of these methods will just call &lt;code&gt;seq&lt;/code&gt; and resend the current message to the result of that call, so it's simply a shorter way of calling the same methods on the sequence object. You also get an implementation of &lt;code&gt;each&lt;/code&gt; implemented in terms of &lt;code&gt;seq&lt;/code&gt;.
+
All of these methods will just call <code>seq</code> and resend the current message to the result of that call, so it's simply a shorter way of calling the same methods on the sequence object. You also get an implementation of <code>each</code> implemented in terms of <code>seq</code>.
  
A Sequence is expected to implement &lt;code&gt;next?&lt;/code&gt; and &lt;code&gt;next&lt;/code&gt;. Every time you call &lt;code&gt;next&lt;/code&gt; you will get the next object in the sequence. &lt;code&gt;next?&lt;/code&gt; will tell you whether there is anything more in the sequence to fetch.
+
A Sequence is expected to implement <code>next?</code> and <code>next</code>. Every time you call <code>next</code> you will get the next object in the sequence. <code>next?</code> will tell you whether there is anything more in the sequence to fetch.
  
 
Sequence is Enumerable - once you call an Enumerable method on a Sequence it will be fully realized until no more is needed.
 
Sequence is Enumerable - once you call an Enumerable method on a Sequence it will be fully realized until no more is needed.
Line 513: Line 512:
 
Sequence defines these methods and cells:
 
Sequence defines these methods and cells:
  
; &lt;nowiki&gt;%&lt;/nowiki&gt;
+
; <nowiki>%</nowiki>
: See &lt;code&gt;interpose&lt;/code&gt;
+
: See <code>interpose</code>
; &amp;
+
; &
: See &lt;code&gt;interleave&lt;/code&gt;
+
: See <code>interleave</code>
 
; +
 
; +
 
: Combines two sequences into one. It will first execute all elements of the first sequence, then all the elements of the second:
 
: Combines two sequences into one. It will first execute all elements of the first sequence, then all the elements of the second:
&lt;source lang=&quot;ioke&quot;&gt;((1..10) seq + Sequence infinity) mapped(*2) take(12) should == [2,4,6,8,10,12,14,16,18,20,0,2]&lt;/source&gt;
+
<source lang="ioke">((1..10) seq + Sequence infinity) mapped(*2) take(12) should == [2,4,6,8,10,12,14,16,18,20,0,2]</source>
 
; collected
 
; collected
: An alias of &lt;code&gt;mapped&lt;/code&gt;
+
: An alias of <code>mapped</code>
 
; dropped
 
; dropped
: The sequence version of &lt;code&gt;drop&lt;/code&gt;. Creates a new sequence that will drop the requested amount of elements and then continue returning the elements from the sequence:
+
: The sequence version of <code>drop</code>. Creates a new sequence that will drop the requested amount of elements and then continue returning the elements from the sequence:
&lt;source lang=&quot;ioke&quot;&gt;Sequence infinity dropped(5) take(5) should == [6,7,8,9,10]&lt;/source&gt;
+
<source lang="ioke">Sequence infinity dropped(5) take(5) should == [6,7,8,9,10]</source>
 
; droppedWhile
 
; droppedWhile
: The sequence version of &lt;code&gt;dropWhile&lt;/code&gt;. Creates a new sequence that will drop while the given argument code returns true:
+
: The sequence version of <code>dropWhile</code>. Creates a new sequence that will drop while the given argument code returns true:
&lt;source lang=&quot;ioke&quot;&gt;Sequence infinity droppedWhile(&lt;10) take(5) should == [10,11,12,13,14]&lt;/source&gt;
+
<source lang="ioke">Sequence infinity droppedWhile(<10) take(5) should == [10,11,12,13,14]</source>
 
; filtered
 
; filtered
: The sequence version of &lt;code&gt;filter&lt;/code&gt;. Will return a new sequence that filters all elements according to the argument code.
+
: The sequence version of <code>filter</code>. Will return a new sequence that filters all elements according to the argument code.
 
; grepped
 
; grepped
: The sequence version of &lt;code&gt;grep&lt;/code&gt;. Will return a new sequence that only yield the elements that match the argument.
+
: The sequence version of <code>grep</code>. Will return a new sequence that only yield the elements that match the argument.
 
; indexed
 
; indexed
 
: Takes two optional keyword arguments, from: and step:. These default to 0 and 1 respectively. Will yield a list for each element where the first is the index and the second is the element.  
 
: Takes two optional keyword arguments, from: and step:. These default to 0 and 1 respectively. Will yield a list for each element where the first is the index and the second is the element.  
Line 541: Line 540:
 
: Will return a new sequence that will interpose the given argument between each of the elements in the sequence.
 
: Will return a new sequence that will interpose the given argument between each of the elements in the sequence.
 
; mapped
 
; mapped
: The sequence version of &lt;code&gt;map&lt;/code&gt;. Will return a new sequence where each element is transformed by the argument code.
+
: The sequence version of <code>map</code>. Will return a new sequence where each element is transformed by the argument code.
 
; rejected
 
; rejected
: The sequence version of &lt;code&gt;reject&lt;/code&gt;. Will return a new sequence where only the element that doesn't match will be yielded.
+
: The sequence version of <code>reject</code>. Will return a new sequence where only the element that doesn't match will be yielded.
 
; selected
 
; selected
: An alias of &lt;code&gt;filtered&lt;/code&gt;.
+
: An alias of <code>filtered</code>.
 
; zipped
 
; zipped
: The sequence version of &lt;code&gt;zip&lt;/code&gt;. Will return a new sequence where each element will be a list composed of one element from the receiver and one element each from all the argument sequences or enumerables.
+
: The sequence version of <code>zip</code>. Will return a new sequence where each element will be a list composed of one element from the receiver and one element each from all the argument sequences or enumerables.
 
; ω
 
; ω
 
: Returns a new infinite sequence that begins at 0 with step 1.
 
: Returns a new infinite sequence that begins at 0 with step 1.
Line 555: Line 554:
 
A pretty common pattern when creating infinite sequences is that you want to create the new value from the last value. A sequence like that can be created by calling iterate with the starting values on a lexical block. As an example, you can create a generator of the fibonacci sequence like this:
 
A pretty common pattern when creating infinite sequences is that you want to create the new value from the last value. A sequence like that can be created by calling iterate with the starting values on a lexical block. As an example, you can create a generator of the fibonacci sequence like this:
  
&lt;source lang=&quot;ioke&quot;&gt;fibSeq = fn(a, b, [b, a + b]) iterate(0, 1)
+
<source lang="ioke">fibSeq = fn(a, b, [b, a + b]) iterate(0, 1)
 
fibSeq take(5) should == [[0, 1], [1, 1], [1, 2], [2, 3], [3, 5]]
 
fibSeq take(5) should == [[0, 1], [1, 1], [1, 2], [2, 3], [3, 5]]
fibSeq mapped(first) take(5) should == [8, 13, 21, 34, 55]&lt;/source&gt;
+
fibSeq mapped(first) take(5) should == [8, 13, 21, 34, 55]</source>
  
 
== Regexps ==
 
== Regexps ==
Line 571: Line 570:
 
You can also investigate a Regexp, by asking for its pattern (the pattern method), its flags (the flag method), and the named groups it defines. The names of the named groups can be inspected with the names method.
 
You can also investigate a Regexp, by asking for its pattern (the pattern method), its flags (the flag method), and the named groups it defines. The names of the named groups can be inspected with the names method.
  
A Regexp Match is specific to one specific match. You will always get a new one every time you try to match against something. A match has a target, which is the original text the regular expression was matched against. So if I do #/ .. / =~ &amp;quot;abc fo bar&amp;quot;, then the whole &amp;quot;abc fo bar&amp;quot; is the target. You get the target from a match by using the target method. A match can also be asked for the named groups the regular expression it was matched against support. This is also done with the names method. Match also has two methods beforeMatch and afterMatch, that returns the text before and after the match. The match method returns the text comprising the actual match. The captures method will return a list of all the captured groups in this match. The asList method returns the same things as captures, except it also includes the full match text.
+
A Regexp Match is specific to one specific match. You will always get a new one every time you try to match against something. A match has a target, which is the original text the regular expression was matched against. So if I do #/ .. / =~ &quot;abc fo bar&quot;, then the whole &quot;abc fo bar&quot; is the target. You get the target from a match by using the target method. A match can also be asked for the named groups the regular expression it was matched against support. This is also done with the names method. Match also has two methods beforeMatch and afterMatch, that returns the text before and after the match. The match method returns the text comprising the actual match. The captures method will return a list of all the captured groups in this match. The asList method returns the same things as captures, except it also includes the full match text.
  
 
To figure out the indices where groups start or end, you can use the start or end methods. These take an optional index that defaults to zero, where group zero is the full match. They can also take a text or symbol that should be the name of a named group. The offset method returns a pair of the beginning and end offset of the group asked for. It works the same as start and end, with regards to what argument it takes.
 
To figure out the indices where groups start or end, you can use the start or end methods. These take an optional index that defaults to zero, where group zero is the full match. They can also take a text or symbol that should be the name of a named group. The offset method returns a pair of the beginning and end offset of the group asked for. It works the same as start and end, with regards to what argument it takes.
Line 579: Line 578:
 
Finally, a Regexp Match implements pass. It does this in such a way that if you use named groups, you can extract the value for that group by calling a method named the same as that group name. An example of this:
 
Finally, a Regexp Match implements pass. It does this in such a way that if you use named groups, you can extract the value for that group by calling a method named the same as that group name. An example of this:
  
&lt;source lang=&quot;ioke&quot;&gt;number = &quot;555-12345&quot;
+
<source lang="ioke">number = "555-12345"
 
m = #/({areaCode}\d{3})-({localNumber}\d{5})/ =~ number
 
m = #/({areaCode}\d{3})-({localNumber}\d{5})/ =~ number
 
m areaCode println
 
m areaCode println
m localNumber println&lt;/source&gt;
+
m localNumber println</source>
  
 
== FileSystem ==
 
== FileSystem ==
Line 592: Line 591:
 
Ioke supports transforming an object into another object through the use of the method become!. After an object becomes another object, those two objects are indistinguishable from Ioke. This makes it really easy to do transparent proxies, futures and other things like that. To use it, do something like this:
 
Ioke supports transforming an object into another object through the use of the method become!. After an object becomes another object, those two objects are indistinguishable from Ioke. This makes it really easy to do transparent proxies, futures and other things like that. To use it, do something like this:
  
&lt;source lang=&quot;ioke&quot;&gt;x = &quot;foo&quot;
+
<source lang="ioke">x = "foo"
y = &quot;bar&quot;
+
y = "bar"
  
 
x become!(y)
 
x become!(y)
x same?(y) ; =&gt; true&lt;/source&gt;
+
x same?(y) ; => true</source>
 
In Ioke, objects can also be frozen. When an object is frozen it cannot be modified in any way. Any try at modifying a frozen object will result in a condition. This can be really helpful to track down problems in code, and make assertions about what should be possible. You freeze an Ioke object by calling freeze! on it. You can unfreeze it by calling thaw! on it. You can also check if something is frozen by calling frozen? on it.
 
In Ioke, objects can also be frozen. When an object is frozen it cannot be modified in any way. Any try at modifying a frozen object will result in a condition. This can be really helpful to track down problems in code, and make assertions about what should be possible. You freeze an Ioke object by calling freeze! on it. You can unfreeze it by calling thaw! on it. You can also check if something is frozen by calling frozen? on it.

Latest revision as of 15:26, 24 November 2010

Core kinds

Ioke obviously contains lots of different data types, but there are some that are much more important than others. In this chapter I'll take a look at these and talk about how to work with them correctly.

Conditions

One of the major parts of Ioke is the condition system. Unlike most other programming languages, Ioke doesn't have exceptions. Instead it has conditions, where errors are a specific kind of condition. The condition system comprises several different things. Specifically, the condition system uses the kinds Condition, Restart, Handler and Rescue. Restarts are mostly orthogonal to the rest of the system.

The way the condition system works is this. When something happens, a program can elect to signal a condition. This can be done using "signal!", "warn!" or "error!". Both "warn!" and "error!" use "signal!" under the covers, but do some other things as well. A condition will always mimic Condition. Each of these three methods can be called in three different ways. First, you can call them with a text argument. In that case the signalled condition will be the default for that type. (The default for "signal!" is Condition Default. The default for "warn!" is Condition Warning Default and the default for "error!" is Condition Error Default.) A new mimic of the default condition will be created, and a cell called text will be set to the text argument. The second variation is to give an instance of an existing condition to one of the methods. In that case that condition will be signalled unmodified. Finally, the third version gives a condition mimic and one or more keyword arguments with data to set on that condition. In that case a mimic of the condition will be created, and then cells with data set based on the arguments.

If a signal is not handled, nothing happens.

If a warning is not handled, a message will be printed with the text of that warning.

If an error is not handled, the debugger will be invoked - if a debugger is available. Otherwise the program will be terminated.

A Rescue allows conditions to unwind the stack to the place where the rescue is established. Combining rescues and conditions looks a lot like regular exception handling in other programming languages.

A Handler on the other hand will run code in the dynamic context of the place where the condition was signalled. A handler can invoke restarts to handle an error state at the place it happened, and as such doesn't have to actually unwind the stack anywhere. Any number of handlers can run - the last handler to run will be either the last handler, the handler that activates a restart, or the last handler before a valid rescue for that condition.

A restart is a way to allow ways of getting back to a valid state. Take the example of referring to a cell that doesn't exist. Before signalling a condition, Ioke will set up restarts so you can provide a value to use in the case a cell doesn't exist. This restart will use that new value and continue execution at the point where it would otherwise have failed.

A restart can have a name. If it doesn't have a name it can only be used interactively. You can use findRestart to get hold of the closest restart with a given name. You can invoke a given restart with invokeRestart, which takes either the name of a restart or a restart mimic. Finally, you can get all available restarts using availableRestarts.

Both handlers, rescues and restarts are established inside a call to the bind macro. All arguments to this macro need to be either handlers, rescues or restarts, except for the last argument which should be the code to execute.

You create a new handler by calling the method "handle". You create a new rescue by calling the method "rescue". You create a new restart by calling the method "restart". These all take funky arguments, so refer to the reference to better understand how they work.

This small example doesn't necessarily show the power of conditions, but it can give an idea about how it works.

;; to handle any problem
bind(
  rescue(fn(c, nil)), ;; do nothing in the rescue
  
  error!("This is bad!!")
)

;; to print all conditions happening, but not do anything
bind(
  handle(fn(c, c println)),
  
  signal!("something")
  signal!("something more")

  warn!("A warning!!")
)

;; rescue either of two conditions
C1 = Condition Error mimic
C2 = Condition Error mimic

bind(
  rescue(C1, C2, fn(c, "got an error: #{c}" println)),
  
  error!(C1)
)


;; invoke a restart when no such cell is signaled
bind(
  handle(Condition Error NoSuchCell, fn(c, invokeRestart(:useValue, 42))),

  blarg println) ;; will print 42


;; establish a restart
bind(
  restart(something, fn(+args, args println)),
  
  invokeRestart(:something, 1, 2, 3)
)

The code above shows several different things you can do with the condition system. It is a very powerful system, so I recommend trying to understand it. It lies at the core of many things in Ioke, and some parts will not make sense without a deep understanding of conditions. For more information on what such a system is capable of, look for documentation about the Common Lisp condition system, which has been a heavy influence on Ioke. Also, the debugger in IIk uses the condition system to implement its functionality.

There are many conditions defined in the core of Ioke, and they are used by the implementation to signal error conditions of different kinds. Refer to the reference to see which conditions are available.

Finally, one thing that you might miss if you're used to exceptions in other languages, is a construct that makes it possible to ensure that code gets executed, even if a non-local flow control happens. Don't despair, Ioke has one. It is called ensure, and works mostly like ensure in Ruby, and finally in Java. It takes one main code argument, followed by zero or more arguments that contain the code to always make sure executes. It looks like this:

ensure(
  conn = Database open
  conn SELECT * FROM a_table,
  conn close!,
  conn reallyClose!,
  conn reallyReallyReallyClose!)

This code uses a hypothetical database library, opens up a connection, does something, and then in three different ensure blocks tries to ensure that it really is closed afterwards. The return value of the ensure block will still be the return value of the last expression in the main code. Non local flow control can happen inside of the ensure block, but exactly what will happen is undefined -- so avoid it, please.

Text

In Ioke, the equivalent of Strings in other languages are called Text. This better describes the purpose of the type. Ioke Text is immutable. All operations that would change the text returns a new object instead. If you are used to Java strings or Ruby strings, then most operations available on Ioke Texts will not come as a surprise.

To create a new Text, you use the literal syntax as described in the syntax chapter. You can use interpolation to include dynamic data.

You can do several things with Ioke text. These examples show some of the methods:

;; repeat a text several times
"foo" * 3  ;; => "foofoofoo"

;; concatenate two texts
"foo" + "bar"  ;; => "foobar"

;; get the character at a specific index
"foo"[1]  ;; => 111

;; get a subset of text
"foo"[1..1]  ;; => "o"
"foo"[0..1]  ;; => "fo"
"foo"[0...1]  ;; => "f"

"foxtrot"[1..-1] ;; => "oxtrot"
"foxtrot"[1...-1] ;; => "oxtro"

;; is a text empty?
"foo" empty?  ;; => false
"" empty?  ;; => true

;; the length of the text
"foo" length  ;; => 3
"" length  ;; => 0

;; replace the first occurrence of something with something else
"hello fox fob folk" replace("fo", "ba")
  ;; => "hello bax fob folk"

;; replace all occurrences of something with something else
"hello fox fob folk" replaceAll("fo", "ba")
  ;; => "hello bax bab balk"

;; split around a text
"foo bar bax" split(" ")
  ;; => ["foo", "bar", "bax"]

The Text kind contains lots of useful functionality like this. The purpose is to make it really easy to massage text of any kind.

One important tool for doing that is the "format" method. This is a mix between C printf and Common Lisp format. At the moment, it only contains a small amount of functionality, but it can still be very convenient. Specifically you can print each element in a list directly by using format, instead of concatenating text yourself.

The "format" method takes format specifiers that begin with %, and then inserts one of its arguments in different ways depending on what kind of format specifier is used.

Some examples of format follow:

;; insert simple value as text
"%s" format(123) ;; => "123"

;; insert value right justified by 6
"%6s" format(123) ;; => "   123"

;; insert value left justified by 6
"%-6s" format(123) ;; => "123   "

;; insert two values
"%s: %s" format(123, 321) ;; => "123: 321"

;; insert a list of values formatted the same
"%[%s, %]\n" format([1,2,3])
  ;; => "1, 2, 3, \n"

;; insert a dict of values formatted the same
"%:[%s => %s, %]\n" format({sam: 3, toddy: 10})
  ;; => "sam => 3, toddy => 10, \n"

;; insert splatted values from a list
"wow: %*[%s: %s %]" format([[1,2],[2,3],[3,4]])
  ;; => "wow: 1: 2 2: 3 3: 4 "

Numbers

As mentioned in the section on syntax, Ioke supports decimal numbers, integers and ratios. A Ratio will be created when two integers can't be divided evenly. A Ratio will always use the GCD. In most cases Ratios, Decimals and Integers can interact with each other as would be expected. The one thing that might surprise people is that Ioke doesn't have any inexact floating point data type. Instead, decimals are exact and can have any size. This means they are well suited to represent such things as money, since operations will always have well defined results.

All expected math works fine on Ioke numbers. The reference for numbers more closely specify what is possible. One thing to notice is that the % operator implements modulus, not remainder. This might be unintuitive for some developers. What that means is that it is not an error to ask for the modulus of 0: "13 % 0", since there is no division necessary in this operation.

In addition to the above, Integers have the "times" method described earlier. It can also return the successor and predecessor of itself with the "pred" and "succ" methods.

Lists

Ioke has lists that expand to the size needed. These lists can be created using a simple literal syntax. They can contain any kind of element, including itself (although don't try to print such a list). Ioke List mixes in the Enumerable mixin, which gives it quite powerful capabilities.

A list can be created using the "list" or "[]" methods:

;; an empty list
[]

;; the same list created in two different ways
list(1, 2, 3)
[1, 2, 3]

;; a list with different elements
[1, "one", :one]

Except for the Enumerable methods, List also defines many other methods that can be highly useful. Some examples are shown below:

l = [1, 2, 3]

;; add two lists together
l + l ;; => [1, 2, 3, 1, 2, 3]

;; return the difference of two lists
l - [2] ;; => [1, 3]

;; add a new value to a list
l << 42
l == [1, 2, 3, 42] ;; => true

;; get a specific element from the list
l[0]  ;; => 1

;; -1 returns the last, -2 the next to last
l[-1] ;; => 42

;; an index outside the boundaries return nil
l[10] ;; => nil

;; assign a new value
l[3] = 40
l == [1,2,3,40] ;; => true

l[-1] = 39
l == [1,2,3,39] ;; => true

;; assign an out of bounds value
l[10] = 13
l == [1,2,3,39,nil,nil,nil,nil,nil,nil,13]
  ;; => true

;; at and at= is the same as [] and []=
l at(0) ;; => 1

;; empty the list
l clear!
l == [] ;; => true

;; follows the each protocol
l each(println)

;; is empty?
l empty? ;; => true

;; does it include an element?
l include?(:foo) ;; => false

;; the last element
l last ;; => nil
[1, 2] last ;; => 2

;; the length
[1, 2] length ;; => 2

;; first value
[1, 2] first ;; => 1

;; rest except for first
[1, 2, 3] rest ;; => [2, 3]

;; returns a new sorted list
[3, 2, 1] sort ;; => [1, 2, 3]

;; sorts in place
l = [3, 2, 1]
l sort!
l == [1, 2, 3] ;; => true

Dicts

A Dict is a dictionary of key-value mappings. The mappings are unordered, and there can only ever be one key with the same value. Any kind of Ioke object can be used as a key. There is no problem with having the same value for different keys. The default implementation of Dict uses a hash-based implementation. That's not necessarily always true for all dicts. The iteration order is not necessarily stable either, so don't write code that depends on it.

Creating a dict is done using either the "dict" or the "{}" methods. Both of these expect either keyword arguments or mimics of Pair. If keyword arguments, these keywords will be used as symbol keys. That's the most common thing, so it makes sense to have that happen automatically. Dicts also try to print themselves that way.

dict(1 => 2, 3 => 4)

;; these two are the same
dict(foo: "bar", baaz: "quux")
dict(:foo => "bar", :baaz => "quux")

{1 => 2, 3 => 4}

;; these two are the same
{foo: "bar", baaz: "quux"}
{:foo => "bar", :baaz => "quux"}

;; the formats can be combined:
{1 => 2, foo: 42, "bar" => "qux"}

The literal Pair syntax (using =>) will not necessarily instantiate real pairs for this.

Dicts mix in Enumerable. When using each, what will be yielded are mimics of Pair, where the first value will be the key and the second will be value. Just like Lists, Dicts have several useful methods in themselves:

d = {one: "two", 3 => 4}

;; lookup with [], "at" works the same
d[:one] ;; => "two"
d[:two] ;; => nil
d[3]    ;; => 4
d[4]    ;; => nil

;; assign values with []=
d[:one] = "three"
d[:new] = "wow!"

d == {one: "three", 3 => 4, new: "wow!"}

;; iterate over it
d each(value println)

;; get all keys
d keys == set(:one, :new, 3)

Sets

If you want an object that work like a mathematical set, Ioke provides such a kind for you. There is also literal syntax for sets from Ioke E. This syntax looks like this: #{42, 55, 111} A set can be iterated over and it is Enumerable. You can add and remove elements, and check for membership.

x = set(1,2,3,3,2,1)

x map(*2) sort ; => [2, 4, 6]

x === 1 ; => true
x === 0 ; => false

x remove!(2)
x === 2 ; => false

x << 4
x === 4 ; => true

Sets also have methods to do union and intersection. These mimic the mathematical operators for these operations:

; intersection
(#{1,2,3,4}  #{1,2,3,4,5}) should == #{1,2,3,4}

;union
(#{1,2}  #{1,3}) should == #{1,2,3}

You can also test membership, nonmembership, subset and superset relations using the mathematical operators:

; membership
42  #{1,2,42,3}

; nonmembership
42  #{1,2,3}

; subset
#{2}  #{1,2,3}

; proper subset
#{"bar", "foo"}  #{"foo", "bar", "quux"}

; superset
#{1,2,3}  #{2}

; proper superset
#{"bar", "foo", "quux"}  #{"foo", "bar"}

Ranges and Pairs

Both ranges and pairs tie two values together. They also have literal syntax to create them, since they are very useful in many circumstances.

A Range defines two endpoints. A Range is Enumerable and you can also check for membership. It's also convenient to send Ranges to the "List []" method. A Range can be exclusive or inclusive. If it's inclusive it includes the end value, and if it is exclusive it doesn't.

An addition to Ioke S is the possibility of inverted ranges. If the first value is larger than the second value, then the range is inverted. This puts slightly different demands on the objects inside of it. Specifically, if you want to iterate over the elements, the kind you're using need to have a method called 'pred' for predecessor, instead of 'succ' for successor. Membership can still be tested, as long as <=> is defined. So you can do something like this: ("foo".."aoo") === "boo". It's mostly useful for iterating in the opposite direction, like with 10..1, for example.

;; literal syntax for inclusive range
1..10

;; literal syntax for exclusive range
1...10

;; check for membership
(1..10) === 5 ;; => true
(1..10) === 10 ;; => true
(1..10) === 11 ;; => false

(1...10) === 5 ;; => true
(1...10) === 10 ;; => false
(1...10) === 11 ;; => false

;; get the from value
(1..10) from == 1 ;; => true
(1...10) from == 1 ;; => true

;; get the to value
(1..10) to == 10 ;; => true
(1...10) to == 10 ;; => true

;; is this range exclusive?
(1..10) exclusive? ;; => false
(1...10) exclusive? ;; => true

;; is this range inclusive?
(1..10) inclusive? ;; => true
(1...10) inclusive? ;; => false

A Pair represents a combination of two values. They don't have to be of the same kind. They can have any kind of relationship. Since Pairs are often used to represent the elements of Dicts, it is very useful to refer to the first value as the "key", and the second value as the "value".

;; literal syntax for a pair
"foo" => "bar"

;; getting the first value
("foo" => "bar") first ;; => "foo"
("foo" => "bar") key ;; => "foo"

;; getting the second value
("foo" => "bar") second ;; => "bar"
("foo" => "bar") value ;; => "bar"

Tuples

A Pair is a special case of wanting to tie two values together. The general case of bundling several values together can be achieved by using Tuples. If you give anything else than one argument to the empty method, you will get back a Tuple. You can also create a new tuple explicitly using the tuple method.

To extract elements out of the tuple the main way is using destructuring assignment. It is also possible to use a named extractor to get a given index.

Tuples are by design not Enumerable. This is a feature, not a misfeature. If you use tuples as something that should be enumerated over, you're using the wrong data structure. Also remember that Enumerable defines asTuple - so anything enumerable can be turned into a tuple if needed.

t = tuple(42, 55, 18)
(x, y, z) = t
t third should == 18
t _3 should == 18

As you can see english-style names are defined for accessing elements. These will only be available up to nine elements. The underscore syntax for indexing will always be possible, though.

Structs

It is pretty common to want to describe a type of object with set elements that already provide support for structural equality and hash encoding. Ioke structs provide this behavior.

You create a new one by calling the Struct method:

Person = Struct(:first_name, :sur_name, :age)

You can provide default values using keyword arguments instead:

Person = Struct(:first_name, :sur_name, age: -1)

Creating a new version of a Struct can be done using the object as a function:

o = Person("Ola", "Bini")

It is also possible to provide these values using keywords - or even a mix:

s = Person(first_name: "Sam", sur_name: "Aaron")
b = Person(sur_name: "Guthrie", "Brian")

All of these objects will implement ==, hash, asText, notice and inspect. Accessors and mutators also work.

You can get access to all the Struct attributes by calling attributes. If you just want the names you can use attributeNames. A Struct instance is also Sequenced over a combination of its keys and values.

Enumerable

One of the most important mixins in Ioke is Mixins Enumerable - the place where most of the collection functionality is available. The contract for any kind that wants to be Enumerable is that it should implement each in the manner described earlier. If it does that it can mixin Enumerable and get access to all the methods defined in it. I'm not going to show all the available methods, but just a few useful examples here. Note that any method name that ends with Fn takes a block instead of a raw message chain.

Almost all methods in Enumerable take variable amounts of arguments and do different things depending on how many arguments are provided. The general rule is that if there is only one argument, it should be a message chain, and if there are two or more arguments the last one should be code, and the rest should be names of arguments to use in that code.

Mapping a collection into a another collection can be done using map or mapFn. These are aliased as collect and collectFn too.

l = [10, 20, 30, 40]

;; mapping into text
l map(asText) ;; => ["10", "20", "30", "40"]
l map(n, n asText) ;; => ["10", "20", "30", "40"]

;; exponentiation
l map(**2) ;; => [100, 400, 900, 1600]
l map(n, n*n) ;; => [100, 400, 900, 1600]

Filtering the contents of a collection can be done using select, which is aliased as filter and findAll.

;; with no arguments, return all true things
[nil, false, 13, 42] select ;; => [13, 42]

l = [1, 2, 3, 4, 5, 6]

;; all elements over 3
l select(>3) ;; => [4, 5, 6]
l select(n, n>3) ;; => [4, 5, 6]

A very common operation is to create one object based on the contents of a collection. This operation has different names in different languages. In Ioke it is called inject, but it is aliased as reduce and fold.

l = [1, 2, 3, 4, 5]

;; inject around a message chain
l inject(+) ;; => 15
l inject(*) ;; => 120

;; with one arg
l inject(n, *n) ;; => 120
l inject(n, +n*2) ;; => 29

;; with two args
l inject(sum, n, sum*n) ;; => 120
l inject(sum, n, sum*2 + n*3) ;; => 139

;; with three args
l inject(1, sum, n, sum*n) ;; => 120
l inject(10, sum, n, sum*n) ;; => 1200

Most of the Enumerable methods that returns lists also have equivalent methods that return dictionaries or sets. Say for example that you want to filter out some elements from a set and return a new set. The best way to achive this in Ioke is to use the filter:set method. If you want the output to be a dictionary you use filter:dict instead. All of these methods follow the same convention. The dictionary versions will work differently depending on if the values are Pairs or not - if they are, the resulting dictionary will use these pairs directly. Otherwise it will use the element as keys and set the values to nil:

(1..15) filter:dict(<5) should == {1 => nil, 2 => nil, 3 => nil, 4 => nil}
{1 => "foo 1", 2 => "foo 2", 3 => "foo 3", 4 => "foo 4"} filter:dict(key<3) should == {1 => "foo 1", 2 => "foo 2"}

Sequences

In addition to the internal iterator protocol of Enumerable, Ioke also supports external iterators in the form of sequences. These all rely on the object to iterate over having a seq method that returns a Sequence object. If you have seq you can mixin Mixins Sequenced - which makes you Enumerable without any effort. As such, the sequence protocol is much easier to implement than the Enumerable protocol, and gives more functionality.

Ioke sequences are lazy, which means that they can represent several things that are not possible using internal iterators.

When mixing in Mixins Sequenced you get these methods:

  • collected
  • dropped
  • droppedWhile
  • filtered
  • grepped
  • indexed
  • interleave
  • interpose
  • mapped
  • rejected
  • selected
  • zipped

All of these methods will just call seq and resend the current message to the result of that call, so it's simply a shorter way of calling the same methods on the sequence object. You also get an implementation of each implemented in terms of seq.

A Sequence is expected to implement next? and next. Every time you call next you will get the next object in the sequence. next? will tell you whether there is anything more in the sequence to fetch.

Sequence is Enumerable - once you call an Enumerable method on a Sequence it will be fully realized until no more is needed.

Sequence defines these methods and cells:

%
See interpose
&
See interleave
+
Combines two sequences into one. It will first execute all elements of the first sequence, then all the elements of the second:
((1..10) seq + Sequence infinity) mapped(*2) take(12) should == [2,4,6,8,10,12,14,16,18,20,0,2]
collected
An alias of mapped
dropped
The sequence version of drop. Creates a new sequence that will drop the requested amount of elements and then continue returning the elements from the sequence:
Sequence infinity dropped(5) take(5) should == [6,7,8,9,10]
droppedWhile
The sequence version of dropWhile. Creates a new sequence that will drop while the given argument code returns true:
Sequence infinity droppedWhile(<10) take(5) should == [10,11,12,13,14]
filtered
The sequence version of filter. Will return a new sequence that filters all elements according to the argument code.
grepped
The sequence version of grep. Will return a new sequence that only yield the elements that match the argument.
indexed
Takes two optional keyword arguments, from: and step:. These default to 0 and 1 respectively. Will yield a list for each element where the first is the index and the second is the element.
infinity
Returns a new sequence that goes from zero and continues forever. It also takes from: and step: arguments that default to 0 and 1.
interleave
Returns a new sequence where each element of the receiver will be followed by one element from the argument enumerable or sequence. It will only continue as long as the first sequence lasts.
interpose
Will return a new sequence that will interpose the given argument between each of the elements in the sequence.
mapped
The sequence version of map. Will return a new sequence where each element is transformed by the argument code.
rejected
The sequence version of reject. Will return a new sequence where only the element that doesn't match will be yielded.
selected
An alias of filtered.
zipped
The sequence version of zip. Will return a new sequence where each element will be a list composed of one element from the receiver and one element each from all the argument sequences or enumerables.
ω
Returns a new infinite sequence that begins at 0 with step 1.
Returns a new infinite sequence of all the natural numbers.

A pretty common pattern when creating infinite sequences is that you want to create the new value from the last value. A sequence like that can be created by calling iterate with the starting values on a lexical block. As an example, you can create a generator of the fibonacci sequence like this:

fibSeq = fn(a, b, [b, a + b]) iterate(0, 1)
fibSeq take(5) should == [[0, 1], [1, 1], [1, 2], [2, 3], [3, 5]]
fibSeq mapped(first) take(5) should == [8, 13, 21, 34, 55]

Regexps

Regular expressions allow the matching of text against an abstract pattern. Ioke uses the JRegex engine to implement regular expressions. This means Ioke supports quite advanced expressions. Exactly what kind of regular expression syntax is supported can be found at http://jregex.sf.net. This section will describe how you interact with regular expressions from Ioke.

There are two kinds that are used when working with regular expression. First, Regexp, which represents an actual pattern. The second is Regexp Match, which contains information about a regular expression match. It is this match that can be used to extract most information about where and how a complicated expression matched.

The standard way of matching something is with the match method. This method is aliased as =~, which is the idiomatic usage. It takes anything that can be converted to a text and returns nil if it fails to match anything, or a Regexp Match if it matches.

You can create a new Regexp from a text by using the Regexp from method. You can quote all meta characters in a text by using the Regexp quote method. Finally, if you just want to get all the text pieces that match for a regular expression, you can use Regexp allMatches. It takes a text and returns a list of all the text pieces where the Regexp matched.

You can also investigate a Regexp, by asking for its pattern (the pattern method), its flags (the flag method), and the named groups it defines. The names of the named groups can be inspected with the names method.

A Regexp Match is specific to one specific match. You will always get a new one every time you try to match against something. A match has a target, which is the original text the regular expression was matched against. So if I do #/ .. / =~ "abc fo bar", then the whole "abc fo bar" is the target. You get the target from a match by using the target method. A match can also be asked for the named groups the regular expression it was matched against support. This is also done with the names method. Match also has two methods beforeMatch and afterMatch, that returns the text before and after the match. The match method returns the text comprising the actual match. The captures method will return a list of all the captured groups in this match. The asList method returns the same things as captures, except it also includes the full match text.

To figure out the indices where groups start or end, you can use the start or end methods. These take an optional index that defaults to zero, where group zero is the full match. They can also take a text or symbol that should be the name of a named group. The offset method returns a pair of the beginning and end offset of the group asked for. It works the same as start and end, with regards to what argument it takes.

You can use the [] method on Match to extract the one or several pieces of matches. If m is a match, then all of these are valid expressions: m[0]. m[1..3]. m[:foo]. m[-2], where the ranges return several groups, and the negative index returns indexed from the end of the list of captures.

Finally, a Regexp Match implements pass. It does this in such a way that if you use named groups, you can extract the value for that group by calling a method named the same as that group name. An example of this:

number = "555-12345"
m = #/({areaCode}\d{3})-({localNumber}\d{5})/ =~ number
m areaCode println
m localNumber println

FileSystem

The FileSystem kind allows access to functionality in the file system. It is the entry point to any manipulation of files and directories, and it can also be used to get listings of existing files. The reference includes good information about the existing methods in the FileSystem kind.

Other things

Ioke supports transforming an object into another object through the use of the method become!. After an object becomes another object, those two objects are indistinguishable from Ioke. This makes it really easy to do transparent proxies, futures and other things like that. To use it, do something like this:

x = "foo"
y = "bar"

x become!(y)
x same?(y) ; => true

In Ioke, objects can also be frozen. When an object is frozen it cannot be modified in any way. Any try at modifying a frozen object will result in a condition. This can be really helpful to track down problems in code, and make assertions about what should be possible. You freeze an Ioke object by calling freeze! on it. You can unfreeze it by calling thaw! on it. You can also check if something is frozen by calling frozen? on it.