2018-07-20 20:27:15 -07:00
|
|
|
#!/usr/bin/env nomsu -V2.5.4.3
|
2017-10-04 18:04:33 -07:00
|
|
|
# How do I...
|
|
|
|
# Write a comment? Put a # and go till the end of the line
|
2018-05-15 18:55:55 -07:00
|
|
|
# How do I write a multi-line comment?
|
|
|
|
After a comment line, any indented text
|
|
|
|
is considered part of the comment
|
|
|
|
(including any deeper-level indented text)
|
|
|
|
The comment ends when the indentation ends
|
2018-05-09 20:35:47 -07:00
|
|
|
# How do I import a file?
|
|
|
|
use "core/control_flow.nom"
|
|
|
|
|
|
|
|
# How do I import all the files in a directory?
|
|
|
|
use "core"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I print stuff?
|
|
|
|
say "Hello world!"
|
|
|
|
|
|
|
|
# How do I set a variable?
|
2018-07-18 01:27:56 -07:00
|
|
|
%foobar = 1
|
|
|
|
%text = "Hello world"
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
# Expressions that are more than just literal values require parentheses:
|
2018-07-18 01:27:56 -07:00
|
|
|
%foobar = (2 + 3)
|
|
|
|
%one_two = 12
|
|
|
|
say %one_two
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I modify a variable?
|
2018-07-18 01:27:56 -07:00
|
|
|
%foobar = (%foobar + 1)
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# Or, as a shorthand, you can do this to increment a variable:
|
2018-07-18 01:27:56 -07:00
|
|
|
%foobar += 1
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I define a mutli-line string?
|
2018-06-05 16:42:07 -07:00
|
|
|
# In Nomsu, "strings" are called "text", and multi-line text looks like:
|
2018-07-18 01:27:56 -07:00
|
|
|
%mutli_text = ".."
|
2018-06-05 16:42:07 -07:00
|
|
|
Start with "..", then put indented lines below it. The indented lines will not include
|
|
|
|
the indentation, except when the lines are indented more than 4 spaces relative
|
|
|
|
to the "..".
|
|
|
|
<- E.g. the 2 spaces here will be included as part of the text.
|
|
|
|
But this line will have no leading spaces.
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-06-05 16:42:07 -07:00
|
|
|
The text will continue until the indentation ends, skipping trailing newlines.
|
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-06-05 16:42:07 -07:00
|
|
|
# How do I put values inside text? (AKA string formatting, string interpolation)
|
|
|
|
say ".."
|
|
|
|
Text can contain a backslash followed by a variable, list, dict, or parenthesized
|
|
|
|
expression. This escaped value will be converted to readable text, like so:
|
|
|
|
The value of %foobar is \%foobar, isn't that nice?
|
2018-07-17 23:08:13 -07:00
|
|
|
These are some numbers: \[1 + 1, 2 + 1, 3 + 1]
|
2018-01-03 19:26:41 -08:00
|
|
|
The sum of 2 and 4 is \(2 + 4).
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-06-05 16:42:07 -07:00
|
|
|
A backslash not followed by any of these, and not at the end of a line
|
2018-07-17 23:08:13 -07:00
|
|
|
like this: \\ will just be treated as a backslash.
|
|
|
|
|
2018-06-05 16:42:07 -07:00
|
|
|
Or, two backlashes will be treated as a single backslash, no matter what follows,
|
|
|
|
like this: \\%foobar <- won't insert any values
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-07-20 20:27:15 -07:00
|
|
|
If you need to split a long line without inserting a newline, you can end a line with backslash and start the next line \
|
|
|
|
..with two periods, like that.
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-07-20 20:27:15 -07:00
|
|
|
Similarly, you can put a long interpolated indented value like: \(..)
|
|
|
|
1 + 2 + 3 + 4 + 5 + 6 + 7 + 8 + 9
|
|
|
|
.. between a backslash and two periods.
|
2018-06-05 16:42:07 -07:00
|
|
|
|
2018-07-20 20:13:20 -07:00
|
|
|
say "Single-line text can contain escape sequences like \", \\, \000, and \n"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I define a list?
|
2018-07-18 01:27:56 -07:00
|
|
|
%my_list = [1, 2, "hello"]
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-15 18:55:55 -07:00
|
|
|
# Really long lists can use [..] followed by a bunch of indented values delimited
|
2017-10-04 18:04:33 -07:00
|
|
|
by commas and/or newlines
|
2018-07-18 01:27:56 -07:00
|
|
|
%my_really_long_list = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I use a list?
|
2018-07-18 01:27:56 -07:00
|
|
|
%my_list = ["first item", "second item", "third item"]
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
# Lists are 1-indexed because they're implemented as Lua tables, so this prints "first item"
|
2018-07-18 01:27:56 -07:00
|
|
|
say %my_list.1
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
# List entries can be modified like this:
|
2018-07-18 01:27:56 -07:00
|
|
|
%my_list.1 = "ONE!!!"
|
|
|
|
say (length of %my_list)
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I define a dictionary/hash map?
|
2018-07-20 20:27:15 -07:00
|
|
|
%my_dict = {x:99, y:101}
|
|
|
|
%my_dict = {x:101, y:2, "99 bottles":99, 653:292}
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I use a dict?
|
2017-10-04 18:04:33 -07:00
|
|
|
# Dicts are also implemented as Lua tables, so they're accessed and modified the same way as lists
|
2018-07-18 01:27:56 -07:00
|
|
|
say %my_dict.x
|
|
|
|
%my_dict.x = 9999
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I do conditional branching?
|
2018-07-20 20:27:15 -07:00
|
|
|
if (1 < 10):
|
|
|
|
say "1 is indeed < 10"
|
|
|
|
|
|
|
|
if (1 > 10):
|
|
|
|
say "this won't print"
|
|
|
|
..else:
|
|
|
|
say "this will print"
|
2018-07-18 01:27:56 -07:00
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# There's no "elseif", so for longer conditionals, a "when" branch is the best option
|
2018-07-20 20:27:15 -07:00
|
|
|
if:
|
|
|
|
(3 > 6) (3 > 5) (3 > 4):
|
|
|
|
say "this won't print"
|
|
|
|
|
|
|
|
(3 > 3):
|
|
|
|
say "this won't print"
|
|
|
|
|
|
|
|
(3 > 2):
|
|
|
|
say "this will print"
|
|
|
|
|
|
|
|
else:
|
|
|
|
say "this is the default case"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I do a switch statement?
|
2018-07-20 20:27:15 -07:00
|
|
|
if 3 is:
|
|
|
|
0 1 2:
|
|
|
|
say "this won't print"
|
|
|
|
|
|
|
|
3:
|
|
|
|
say "this will print"
|
|
|
|
|
|
|
|
else:
|
|
|
|
say "this won't print"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I loop over a list (a foreach loop)?
|
2018-07-18 01:27:56 -07:00
|
|
|
%list = [1, 2, 3]
|
2018-07-20 20:27:15 -07:00
|
|
|
for %x in %list:
|
|
|
|
say "For %x loop #\%x"
|
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I loop over a number range?
|
2017-10-04 18:04:33 -07:00
|
|
|
# This is inclusive, so it will loop over 1,2, and 3
|
2018-07-20 20:27:15 -07:00
|
|
|
for %i in 1 to 3:
|
|
|
|
say "For %i in 1 to 3 loop #\%i"
|
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
# This will print 0,2, and 4
|
2018-07-18 01:27:56 -07:00
|
|
|
for %even in 0 to 5 by 2: say "Even #\%even"
|
2018-07-20 20:27:15 -07:00
|
|
|
for %backwards in 3 to 1 by -1:
|
|
|
|
say "Backwards #\%backwards"
|
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I do a 'while' loop?
|
2018-07-18 01:27:56 -07:00
|
|
|
%x = 1
|
2018-07-17 23:08:13 -07:00
|
|
|
repeat while (%x <= 3):
|
2018-01-03 19:26:41 -08:00
|
|
|
say "repeat while loop #\%x"
|
2018-07-18 01:27:56 -07:00
|
|
|
%x += 1
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-07-18 01:27:56 -07:00
|
|
|
%x = 1
|
2018-07-17 23:08:13 -07:00
|
|
|
repeat until (%x > 3):
|
2018-01-03 19:26:41 -08:00
|
|
|
say "repeat until loop #\%x"
|
2018-07-18 01:27:56 -07:00
|
|
|
%x += 1
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I do an infinite loop?
|
2018-07-18 01:27:56 -07:00
|
|
|
%x = 1
|
2018-07-17 23:08:13 -07:00
|
|
|
repeat:
|
2018-01-03 19:26:41 -08:00
|
|
|
say "repeat loop #\%x"
|
2018-07-18 01:27:56 -07:00
|
|
|
%x += 1
|
|
|
|
if (%x > 3): stop repeating
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I do a 'goto'?
|
2018-07-17 23:08:13 -07:00
|
|
|
do:
|
2018-07-18 01:27:56 -07:00
|
|
|
%x = 1
|
2018-01-25 17:43:57 -08:00
|
|
|
=== %again ===
|
2018-01-03 19:26:41 -08:00
|
|
|
say "GOTO loop #\%x"
|
2018-07-18 01:27:56 -07:00
|
|
|
%x += 1
|
|
|
|
if (%x <= 3): go to %again
|
2017-10-04 18:04:33 -07:00
|
|
|
say "finished going to"
|
|
|
|
|
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# How do I define a function/method/procedure?
|
|
|
|
# In nomsu, they're called "action"s, and they can be declared like this:
|
2018-07-17 23:08:13 -07:00
|
|
|
action [say both %first and also %second]:
|
2017-10-04 18:04:33 -07:00
|
|
|
say %first
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
# Function arguments are accessed just like variables
|
|
|
|
say %second
|
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# Actions can use "return" to return a value early
|
2018-07-17 23:08:13 -07:00
|
|
|
action [first fibonacci above %n]:
|
2018-07-18 01:27:56 -07:00
|
|
|
%f1 = 0
|
|
|
|
%f2 = 1
|
2018-07-17 23:08:13 -07:00
|
|
|
repeat:
|
2018-07-18 01:27:56 -07:00
|
|
|
%tmp = (%f1 + %f2)
|
|
|
|
%f1 = %f2
|
|
|
|
%f2 = %tmp
|
|
|
|
if (%f2 > %n): return %f2
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
say (first fibonacci above 10)
|
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# Actions can have aliases, which may or may not have the arguments in different order
|
2018-01-19 18:13:02 -08:00
|
|
|
action [..]
|
2018-07-18 01:27:56 -07:00
|
|
|
I hate %worse_things more than %better_things
|
|
|
|
I think %worse_things are worse than %better_things
|
|
|
|
I like %better_things more than %worse_things
|
2018-07-20 20:27:15 -07:00
|
|
|
..:
|
|
|
|
say "\(%better_things capitalized) rule and \%worse_things drool!"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
I like "dogs" more than "cats"
|
|
|
|
I think "chihuahuas" are worse than "corgis"
|
|
|
|
|
2018-05-15 18:55:55 -07:00
|
|
|
# Actions can have parts of the action's name spread throughout.
|
2018-05-10 22:47:03 -07:00
|
|
|
Everything that's not a literal value is treated as part of the action's name
|
2017-10-04 18:04:33 -07:00
|
|
|
say both "Hello" and also "again!"
|
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# Actions can even start with a parameter
|
2018-07-18 01:27:56 -07:00
|
|
|
action [%what_she_said is what she said]:
|
|
|
|
say %what_she_said
|
2017-10-04 18:04:33 -07:00
|
|
|
say "-- she said"
|
|
|
|
|
|
|
|
"Howdy pardner" is what she said
|
|
|
|
|
2018-05-15 18:55:55 -07:00
|
|
|
# The language only reserves []{}().,:;% as special characters, so actions
|
2017-10-04 18:04:33 -07:00
|
|
|
can have really funky names!
|
2018-07-18 01:27:56 -07:00
|
|
|
action [>> %foo_bar $$$^ --> % @&_~-^-~_~-^ %1 !]:
|
|
|
|
say %foo_bar
|
2017-10-30 14:08:23 -07:00
|
|
|
say %
|
2017-10-04 18:04:33 -07:00
|
|
|
say %1
|
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
>> "wow" $$$^ --> "so flexible!" @&_~-^-~_~-^ "even numbers can be variables!" !
|
2017-10-13 19:41:58 -07:00
|
|
|
|
2018-05-09 20:35:47 -07:00
|
|
|
# There's also full unicode support
|
2018-07-20 20:27:15 -07:00
|
|
|
%こんにちは = "こんにちは"
|
2018-07-17 23:08:13 -07:00
|
|
|
action [% と言う] "\%世界"
|
2018-05-09 20:35:47 -07:00
|
|
|
say (%こんにちは と言う)
|
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# Math and logic operations are just treated the same as actions in the syntax
|
2017-10-04 18:04:33 -07:00
|
|
|
say (2 + 3)
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-15 18:55:55 -07:00
|
|
|
# So you can define your own operators, although they will need to be parenthesized to
|
2018-05-09 20:35:47 -07:00
|
|
|
play nicely with other operators
|
2018-07-17 23:08:13 -07:00
|
|
|
action [%a ++ %b] (2 * (%a + %b))
|
2018-05-09 20:35:47 -07:00
|
|
|
say (1 ++ (2 * 3))
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-05-09 20:35:47 -07:00
|
|
|
# How do I do grouping?
|
2017-10-04 18:04:33 -07:00
|
|
|
# Expressions can be grouped by enclosing parentheses:
|
|
|
|
say (2 + 3)
|
2018-07-17 23:08:13 -07:00
|
|
|
|
|
|
|
# Or by (..) followed by an indented region
|
2018-07-18 01:27:56 -07:00
|
|
|
say (2 + 3)
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
# If you need to keep going after an indented region, you can start the next line with ".."
|
2018-07-20 20:27:15 -07:00
|
|
|
say both "Very long first argument that needs its own line" and also ".."
|
|
|
|
short second arg
|
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
action [my favorite number] (21 + 2)
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# This can be nested:
|
2018-07-18 01:27:56 -07:00
|
|
|
say both (my favorite number) and also "foo"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# Macros:
|
2017-10-19 17:00:10 -07:00
|
|
|
# The "lua> %" and "=lua %" macros can be used to write raw lua code:
|
2018-07-20 20:27:15 -07:00
|
|
|
action [say the time] (lua> "io.write(\"The OS time is: \", os.time(), \"\\n\");")
|
2017-10-04 18:04:33 -07:00
|
|
|
say the time
|
2017-10-19 17:00:10 -07:00
|
|
|
say "Math expression result is: \(=lua "(1 + 2*3 + 3*4)^2")"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-01-19 18:13:02 -08:00
|
|
|
# Variables can be accessed via \%varname
|
2018-07-17 23:08:13 -07:00
|
|
|
action [square root of %n] (=lua "math.sqrt(\%n)")
|
2017-10-04 18:04:33 -07:00
|
|
|
say "The square root of 2 is \(square root of 2)"
|
|
|
|
|
2018-06-20 15:22:03 -07:00
|
|
|
# Macros can be defined to transform one bit of nomsu code into another using "parse % as %":
|
2018-07-18 01:27:56 -07:00
|
|
|
parse [if %condition is untrue %body] as (if (not %condition) %body)
|
2018-06-20 15:22:03 -07:00
|
|
|
|
|
|
|
# Or to transform nomsu code into custom lua code using "compile % to %"
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [if %condition on opposite day %body] to (..)
|
2018-07-20 20:27:15 -07:00
|
|
|
Lua ".."
|
|
|
|
if not \(%condition as lua expr) then
|
|
|
|
\(%body as lua statements)
|
|
|
|
end
|
2018-06-20 15:22:03 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-06-20 15:22:03 -07:00
|
|
|
# Constants can be defined as macros
|
2018-07-17 23:08:13 -07:00
|
|
|
parse [TWENTY] as 20
|
|
|
|
|
2018-06-20 15:22:03 -07:00
|
|
|
# When they're invoked, they'll need parentheses just like a function call
|
2018-07-17 23:08:13 -07:00
|
|
|
parse [TWENTY ONE] as 21
|
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2018-06-20 15:22:03 -07:00
|
|
|
|
|
|
|
# If you need to use compile-time actions in the same file that they're defined in, you
|
|
|
|
can add a line of tildes (3 or more), and the file will be split into chunks, and each
|
|
|
|
chunk will run before the next one compiles. (note that each chunk has its own scope)
|
2018-07-17 23:08:13 -07:00
|
|
|
if (1 > (TWENTY)) is untrue:
|
2017-10-04 18:04:33 -07:00
|
|
|
say "Nomsu parsing macros work!"
|
|
|
|
say "It looks like a keyword, but there's no magic here!"
|
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
if (1 > (TWENTY)) on opposite day:
|
2017-10-04 18:04:33 -07:00
|
|
|
say "Lua compiling macros work!"
|
|
|
|
say "It looks like a keyword, but there's no magic here!"
|
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
|
|
|
|
# How do I use an action as a value?
|
2018-05-15 18:55:55 -07:00
|
|
|
# Well... it's always *possible* to fall back to Lua behavior for something like this:
|
2018-07-18 01:27:56 -07:00
|
|
|
action [best of %items according to %key_fn]:
|
2018-07-20 20:27:15 -07:00
|
|
|
set {%best:nil, %best_key:nil}
|
2018-07-17 23:08:13 -07:00
|
|
|
for %item in %items:
|
2018-07-18 01:27:56 -07:00
|
|
|
%key = (=lua "\%key_fn(\%item)")
|
|
|
|
if ((%best is (nil)) or (%key > %best_key)):
|
2018-07-20 20:27:15 -07:00
|
|
|
set {%best:%item, %best_key:%key}
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
return %best
|
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
say (best of [2, -3, 4, -8] according to ([%x] -> (%x * %x)))
|
2018-05-10 22:47:03 -07:00
|
|
|
|
2018-05-15 18:55:55 -07:00
|
|
|
# But nomsu was mostly designed so that you don't *need* to. Instead of creating a
|
2018-05-10 22:47:03 -07:00
|
|
|
one-off function to pass to another function and get called a bunch of times, you
|
|
|
|
could use a macro to generate a single block of code that inlines the expression you
|
|
|
|
want to use:
|
2018-07-18 01:27:56 -07:00
|
|
|
parse [best of %items where %item_var has score %key_expr] as (..)
|
2018-07-17 23:08:13 -07:00
|
|
|
result of:
|
2018-07-20 20:27:15 -07:00
|
|
|
set {%best:nil, %best_key:nil}
|
2018-07-18 01:27:56 -07:00
|
|
|
for %item_var in %items:
|
|
|
|
%key = %key_expr
|
|
|
|
if ((%best is (nil)) or (%key > %best_key)):
|
2018-07-20 20:27:15 -07:00
|
|
|
set {%best:%item_var, %best_key:%key}
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-06-20 15:22:03 -07:00
|
|
|
return %best
|
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2018-06-20 15:22:03 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
say (best of [2, -3, 4, -8] where %x has score (%x * %x))
|
|
|
|
# The first example generates code that looks like:
|
|
|
|
|
2018-06-20 15:22:03 -07:00
|
|
|
A_best_of_1_according_to_2 = function(items, key_fn)
|
|
|
|
local best, best_key = nil, nil
|
|
|
|
for _, item in ipairs(items) do
|
|
|
|
local key = key_fn(item)
|
|
|
|
if best == nil or key > best_key then
|
|
|
|
best, best_key = item, key
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return best
|
|
|
|
end
|
|
|
|
print(A_best_of_1_according_to_2({2,-3,4,-8}, function(x)
|
|
|
|
return x * x
|
|
|
|
end))
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-06-20 15:22:03 -07:00
|
|
|
But the second example produces something more like:
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-06-20 15:22:03 -07:00
|
|
|
print((function()
|
|
|
|
local best, best_key = nil, nil
|
|
|
|
for _, x in ipairs({1,-2,3,-4}) do
|
|
|
|
local key = x * x
|
|
|
|
if best == nil or key > best_key then
|
|
|
|
best, best_key = x, key
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return best
|
|
|
|
end)())
|
|
|
|
|
|
|
|
Notice that the second example has inlined the key function, so that in the inner loop,
|
|
|
|
the code only has to do a multiplication, instead of calling a function that does the
|
|
|
|
multiplication. In addition to being more efficient, it's also more powerful and
|
|
|
|
flexible for the language to allow you to define syntax that manipulates expressions.
|