From c8298b0bfc70f95f940a6e100b855f09cfd16e5f Mon Sep 17 00:00:00 2001 From: Glenn Jackman Date: Fri, 10 Jan 2025 08:10:27 -0500 Subject: [PATCH] Concept: functions (#715) * Concept: functions * reword sentence about local variable "overwriting" global variable * Use wording from the manual * add FUNCNAME and FUNCNEST notes --- concepts/README.md | 12 +- concepts/functions/.meta/config.json | 10 ++ concepts/functions/about.md | 245 +++++++++++++++++++++++++++ concepts/functions/introduction.md | 245 +++++++++++++++++++++++++++ concepts/functions/links.json | 10 ++ config.json | 5 + 6 files changed, 520 insertions(+), 7 deletions(-) create mode 100644 concepts/functions/.meta/config.json create mode 100644 concepts/functions/about.md create mode 100644 concepts/functions/introduction.md create mode 100644 concepts/functions/links.json diff --git a/concepts/README.md b/concepts/README.md index a4416ec0..2444df01 100644 --- a/concepts/README.md +++ b/concepts/README.md @@ -45,19 +45,17 @@ The [plan](http://forum.exercism.org/t/bash-syllabus-planning/11952) - for elem in elements ... - arithmetic for -6. conditionals 2 +6. pipelines and command lists - boolean operators `&&` `||` - how `A && B || C` != `if A; then B; else C; fi` - -7. arrays + +7. functions + +8. arrays - numeric and associative - iteration - namerefs -8. functions - -9. pipelines and subshells - ... - brace expansions and how it's different from patterns `/path/to/{foo,bar,baz}.txt` diff --git a/concepts/functions/.meta/config.json b/concepts/functions/.meta/config.json new file mode 100644 index 00000000..b871b9f7 --- /dev/null +++ b/concepts/functions/.meta/config.json @@ -0,0 +1,10 @@ +{ + "authors": [ + "glennj" + ], + "contributors": [ + "kotp", + "IsaacG" + ], + "blurb": "Functions in Bash programs." +} diff --git a/concepts/functions/about.md b/concepts/functions/about.md new file mode 100644 index 00000000..92d6a3d7 --- /dev/null +++ b/concepts/functions/about.md @@ -0,0 +1,245 @@ +# Functions + +Many Bash scripts are written in a strictly imperative style: execute one command, then execute another command, and so on. +Sometimes you need to group together a sequence of commands that conceptually perform a single purpose. +This is where _functions_ come in. + +## Defining a Function + +You define a function like this: + +```bash +my_function () { + COMMANDS +} +``` + +The empty set of parentheses simply denotes that you are defining a function. +Nothing goes inside them. + +## Function Parameters + +Functions, once defined, act like any other command (builtin or not). +Like any command, you can provide _arguments_ to your functions. +Inside the functions, you access the arguments using the _positional parameters_, `$1`, `$2`, etc. +(Recall, we learned about positional parameters in the [Variables][variables] concept.) + +~~~~exercism/advanced +The special parameter `$0` is not changed inside a function; it is still the name of the executing script. +The currently executing function can access its name with the `$FUNCNAME` variable. + +See [3.4.2 Special Parameters][special] in the manual. + +[special]: https://www.gnu.org/software/bash/manual/bash.html#Special-Parameters +~~~~ + +## Variables + +You can define variables inside a function. +If you declare the variables with the `local` command, the _scope_ of the variable is limited to the current function (and to any functions called by it). +Otherwise, the variable is placed in the _global scope_. + +Local variables can have the same name as a global variable. +In that case, the local variable "shadows" the global variable. +For instance, a local variable declared in a function hides a global variable of the same name: references and assignments refer to the local variable, leaving the global variable unmodified. +When the function returns, the global variable is once again visible. + +```bash +x=5 + +myfunc () { + local x=100 + echo "in my function, $x == 100" +} + +echo "in the global scope, $x == 5" + +myfunc + +echo "back in the global scope, $x == 5" +``` + +This outputs + +```none +in the global scope, 5 == 5 +in my function, 100 == 100 +back in the global scope, 5 == 5 +``` + +Inside a function, you can access variables from the _caller_'s scope. +That means you can use global variables, as well as local variables that were declared in the caller (or in some function that calls the caller). + +~~~~exercism/advanced +Technically, "global" is not the right word to use. +To expand a variable in a function, Bash will traverse up the call stack, as far as the global scope, to find a function where that variable name has been declared. + +This example is adapted from the [Shell Functions][man-funcs] section of the manual: + +```bash +func1() { + local var='func1 local' + func2 +} + +func2() { + echo "In func2, var = $var" +} + +var=global +func1 +func2 +``` + +The output is: + +```none +In func2, var = func1 local +In func2, var = global +``` + +Similarly, _assigning_ a value to a variable will assign it _in the scope where it was declared_. +This "action at a distance" can create hard-to-follow code, as it is not always obvious where a variable was assigned a value. +~~~~ + +~~~~exercism/advanced +The call stack can be examined using [the `FUNCNAME` array variable][funcname]. + +[funcname]: https://www.gnu.org/software/bash/manual/bash.html#index-FUNCNAME +~~~~ + +## Return Values + +A function, like any command, has an _exit status_. +By default, the status of a function is the exit status of the _last command executed_. + +You can use the `return` command to return from a function with a specific exit status. + +```bash +check_password () { + if [[ $1 == "secret" ]]; then + return 0 + else + return 1 + fi +} + +read -sp "Enter your password: " pass + +if check_password "$pass"; then + echo "Correct!" +else + echo "Wrong password." +fi +``` + +Using `return` with no arguments returns the status of the last command executed. + +~~~~exercism/note +Note that the `check_password` function can be simplified to: + +```bash +check_password () { [[ $1 == "secret" ]]; } +``` + +1. The `[[...]]` conditional construct has an exit status: `0` for "true", `1` for "false. +2. The `{...}` grouping construct must have either a newline or a semicolon before the ending brace. +~~~~ + +## Function Output + +The return status of a function is just a number. +How can a function produce output? + +Your function can print to standard output. +Use the familiar _command substitution_ to capture it: + +```bash +d6 () { echo "$(( 1 + RANDOM % 6 ))"; } + +die=$( d6 ) +echo "You rolled a $die." +``` + +### Using Both the Output and the Status + +The exit status of a function is available to use even when you are capturing the output. + +```bash +roll () { + local n=$1 + if (( 4 <= n && n <= 20 )); then + echo "$(( 1 + RANDOM % n ))" # exit status is 0 + else + return 1 + fi +} + +read -p "How many faces does your die have? " faces +if die=$( roll "$faces" ); then + echo "You rolled a $die." +else + echo "I can't roll a die with $faces faces." +fi +``` + +## Recursion + +Functions can call themselves recursively. +By default, there is no limit to the depth of recursion. + +An example: + +```bash +fibonacci() { + local n=$1 + if (( n <= 1 )); then + echo "1" + else + local a=$(fibonacci "$(( n - 1 ))") + local b=$(fibonacci "$(( n - 2 ))") + echo "$(( a + b ))" + fi +} + +for i in {1..10}; do fibonacci "$i"; done +# => 1 +# => 2 +# => 3 +# => 5 +# => 8 +# => 13 +# => 21 +# => 34 +# => 55 +# => 89 +``` + +~~~~exercism/advanced +The recursion depth can be controlled with [the `FUNCNEST` variable][funcnest]. + +```bash +bash -c ' + recur() { + echo $1 + recur $(($1 + 1)) + } + FUNCNEST=5 + recur 1 +' +``` + +```none +1 +2 +3 +4 +5 +environment: line 1: recur: maximum function nesting level exceeded (5) +``` + +[funcnest]: https://www.gnu.org/software/bash/manual/bash.html#index-FUNCNEST +~~~~ + +[variables]: https://exercism.org/tracks/bash/concepts/variables +[man-funcs]: https://www.gnu.org/software/bash/manual/bash.html#Shell-Functions diff --git a/concepts/functions/introduction.md b/concepts/functions/introduction.md new file mode 100644 index 00000000..92d6a3d7 --- /dev/null +++ b/concepts/functions/introduction.md @@ -0,0 +1,245 @@ +# Functions + +Many Bash scripts are written in a strictly imperative style: execute one command, then execute another command, and so on. +Sometimes you need to group together a sequence of commands that conceptually perform a single purpose. +This is where _functions_ come in. + +## Defining a Function + +You define a function like this: + +```bash +my_function () { + COMMANDS +} +``` + +The empty set of parentheses simply denotes that you are defining a function. +Nothing goes inside them. + +## Function Parameters + +Functions, once defined, act like any other command (builtin or not). +Like any command, you can provide _arguments_ to your functions. +Inside the functions, you access the arguments using the _positional parameters_, `$1`, `$2`, etc. +(Recall, we learned about positional parameters in the [Variables][variables] concept.) + +~~~~exercism/advanced +The special parameter `$0` is not changed inside a function; it is still the name of the executing script. +The currently executing function can access its name with the `$FUNCNAME` variable. + +See [3.4.2 Special Parameters][special] in the manual. + +[special]: https://www.gnu.org/software/bash/manual/bash.html#Special-Parameters +~~~~ + +## Variables + +You can define variables inside a function. +If you declare the variables with the `local` command, the _scope_ of the variable is limited to the current function (and to any functions called by it). +Otherwise, the variable is placed in the _global scope_. + +Local variables can have the same name as a global variable. +In that case, the local variable "shadows" the global variable. +For instance, a local variable declared in a function hides a global variable of the same name: references and assignments refer to the local variable, leaving the global variable unmodified. +When the function returns, the global variable is once again visible. + +```bash +x=5 + +myfunc () { + local x=100 + echo "in my function, $x == 100" +} + +echo "in the global scope, $x == 5" + +myfunc + +echo "back in the global scope, $x == 5" +``` + +This outputs + +```none +in the global scope, 5 == 5 +in my function, 100 == 100 +back in the global scope, 5 == 5 +``` + +Inside a function, you can access variables from the _caller_'s scope. +That means you can use global variables, as well as local variables that were declared in the caller (or in some function that calls the caller). + +~~~~exercism/advanced +Technically, "global" is not the right word to use. +To expand a variable in a function, Bash will traverse up the call stack, as far as the global scope, to find a function where that variable name has been declared. + +This example is adapted from the [Shell Functions][man-funcs] section of the manual: + +```bash +func1() { + local var='func1 local' + func2 +} + +func2() { + echo "In func2, var = $var" +} + +var=global +func1 +func2 +``` + +The output is: + +```none +In func2, var = func1 local +In func2, var = global +``` + +Similarly, _assigning_ a value to a variable will assign it _in the scope where it was declared_. +This "action at a distance" can create hard-to-follow code, as it is not always obvious where a variable was assigned a value. +~~~~ + +~~~~exercism/advanced +The call stack can be examined using [the `FUNCNAME` array variable][funcname]. + +[funcname]: https://www.gnu.org/software/bash/manual/bash.html#index-FUNCNAME +~~~~ + +## Return Values + +A function, like any command, has an _exit status_. +By default, the status of a function is the exit status of the _last command executed_. + +You can use the `return` command to return from a function with a specific exit status. + +```bash +check_password () { + if [[ $1 == "secret" ]]; then + return 0 + else + return 1 + fi +} + +read -sp "Enter your password: " pass + +if check_password "$pass"; then + echo "Correct!" +else + echo "Wrong password." +fi +``` + +Using `return` with no arguments returns the status of the last command executed. + +~~~~exercism/note +Note that the `check_password` function can be simplified to: + +```bash +check_password () { [[ $1 == "secret" ]]; } +``` + +1. The `[[...]]` conditional construct has an exit status: `0` for "true", `1` for "false. +2. The `{...}` grouping construct must have either a newline or a semicolon before the ending brace. +~~~~ + +## Function Output + +The return status of a function is just a number. +How can a function produce output? + +Your function can print to standard output. +Use the familiar _command substitution_ to capture it: + +```bash +d6 () { echo "$(( 1 + RANDOM % 6 ))"; } + +die=$( d6 ) +echo "You rolled a $die." +``` + +### Using Both the Output and the Status + +The exit status of a function is available to use even when you are capturing the output. + +```bash +roll () { + local n=$1 + if (( 4 <= n && n <= 20 )); then + echo "$(( 1 + RANDOM % n ))" # exit status is 0 + else + return 1 + fi +} + +read -p "How many faces does your die have? " faces +if die=$( roll "$faces" ); then + echo "You rolled a $die." +else + echo "I can't roll a die with $faces faces." +fi +``` + +## Recursion + +Functions can call themselves recursively. +By default, there is no limit to the depth of recursion. + +An example: + +```bash +fibonacci() { + local n=$1 + if (( n <= 1 )); then + echo "1" + else + local a=$(fibonacci "$(( n - 1 ))") + local b=$(fibonacci "$(( n - 2 ))") + echo "$(( a + b ))" + fi +} + +for i in {1..10}; do fibonacci "$i"; done +# => 1 +# => 2 +# => 3 +# => 5 +# => 8 +# => 13 +# => 21 +# => 34 +# => 55 +# => 89 +``` + +~~~~exercism/advanced +The recursion depth can be controlled with [the `FUNCNEST` variable][funcnest]. + +```bash +bash -c ' + recur() { + echo $1 + recur $(($1 + 1)) + } + FUNCNEST=5 + recur 1 +' +``` + +```none +1 +2 +3 +4 +5 +environment: line 1: recur: maximum function nesting level exceeded (5) +``` + +[funcnest]: https://www.gnu.org/software/bash/manual/bash.html#index-FUNCNEST +~~~~ + +[variables]: https://exercism.org/tracks/bash/concepts/variables +[man-funcs]: https://www.gnu.org/software/bash/manual/bash.html#Shell-Functions diff --git a/concepts/functions/links.json b/concepts/functions/links.json new file mode 100644 index 00000000..929529d1 --- /dev/null +++ b/concepts/functions/links.json @@ -0,0 +1,10 @@ +[ + { + "url": "https://www.gnu.org/software/bash/manual/bash.html#Shell-Functions", + "description": "\"Functions\" in the bash manual" + }, + { + "url": "https://mywiki.wooledge.org/BashGuide/CompoundCommands#Functions", + "description": "\"Functions\" in the Bash Guide" + } +] diff --git a/config.json b/config.json index 00451a2c..1032f526 100644 --- a/config.json +++ b/config.json @@ -1248,6 +1248,11 @@ "uuid": "f64a17aa-cbdb-49de-b50c-f3bf14a4e03d", "slug": "pipelines", "name": "Pipelines and Command Lists" + }, + { + "uuid": "44349c3b-8b64-456a-afa9-aa7fc0e7d618", + "slug": "functions", + "name": "Functions" } ], "key_features": [