- •Table of Contents
- •Chapter 1. Why Shell Programming?
- •2.1. Invoking the script
- •2.2. Preliminary Exercises
- •Part 2. Basics
- •Chapter 3. Exit and Exit Status
- •Chapter 4. Special Characters
- •Chapter 5. Introduction to Variables and Parameters
- •5.1. Variable Substitution
- •5.2. Variable Assignment
- •5.3. Bash Variables Are Untyped
- •5.4. Special Variable Types
- •Chapter 6. Quoting
- •Chapter 7. Tests
- •7.1. Test Constructs
- •7.2. File test operators
- •7.3. Comparison operators (binary)
- •7.4. Nested if/then Condition Tests
- •7.5. Testing Your Knowledge of Tests
- •Chapter 8. Operations and Related Topics
- •8.1. Operators
- •8.2. Numerical Constants
- •Part 3. Beyond the Basics
- •Chapter 9. Variables Revisited
- •9.1. Internal Variables
- •9.2. Manipulating Strings
- •9.2.1. Manipulating strings using awk
- •9.2.2. Further Discussion
- •9.3. Parameter Substitution
- •9.4. Typing variables: declare or typeset
- •9.5. Indirect References to Variables
- •9.6. $RANDOM: generate random integer
- •9.7. The Double Parentheses Construct
- •Chapter 10. Loops and Branches
- •10.1. Loops
- •10.2. Nested Loops
- •10.3. Loop Control
- •10.4. Testing and Branching
- •Chapter 11. Internal Commands and Builtins
- •11.1. Job Control Commands
- •Chapter 12. External Filters, Programs and Commands
- •12.1. Basic Commands
- •12.2. Complex Commands
- •12.3. Time / Date Commands
- •12.4. Text Processing Commands
- •12.5. File and Archiving Commands
- •12.6. Communications Commands
- •12.7. Terminal Control Commands
- •12.8. Math Commands
- •12.9. Miscellaneous Commands
- •Chapter 13. System and Administrative Commands
- •Chapter 14. Command Substitution
- •Chapter 15. Arithmetic Expansion
- •Chapter 16. I/O Redirection
- •16.1. Using exec
- •16.2. Redirecting Code Blocks
- •16.3. Applications
- •Chapter 17. Here Documents
- •Chapter 18. Recess Time
- •Part 4. Advanced Topics
- •Chapter 19. Regular Expressions
- •19.1. A Brief Introduction to Regular Expressions
- •19.2. Globbing
- •Chapter 20. Subshells
- •Chapter 21. Restricted Shells
- •Chapter 22. Process Substitution
- •Chapter 23. Functions
- •23.1. Complex Functions and Function Complexities
- •23.2. Local Variables
- •23.2.1. Local variables make recursion possible.
- •Chapter 24. Aliases
- •Chapter 25. List Constructs
- •Chapter 26. Arrays
- •Chapter 27. Files
- •Chapter 28. /dev and /proc
- •28.2. /proc
- •Chapter 29. Of Zeros and Nulls
- •Chapter 30. Debugging
- •Chapter 31. Options
- •Chapter 32. Gotchas
- •Chapter 33. Scripting With Style
- •33.1. Unofficial Shell Scripting Stylesheet
- •Chapter 34. Miscellany
- •34.2. Shell Wrappers
- •34.3. Tests and Comparisons: Alternatives
- •34.4. Optimizations
- •34.5. Assorted Tips
- •34.6. Oddities
- •34.7. Portability Issues
- •34.8. Shell Scripting Under Windows
- •Chapter 35. Bash, version 2
- •Chapter 36. Endnotes
- •36.1. Author's Note
- •36.2. About the Author
- •36.3. Tools Used to Produce This Book
- •36.3.1. Hardware
- •36.3.2. Software and Printware
- •36.4. Credits
- •Bibliography
- •Appendix A. Contributed Scripts
- •Appendix C. Exit Codes With Special Meanings
- •Appendix D. A Detailed Introduction to I/O and I/O Redirection
- •Appendix E. Localization
- •Appendix F. History Commands
- •Appendix G. A Sample .bashrc File
- •Appendix H. Converting DOS Batch Files to Shell Scripts
- •Appendix I. Exercises
- •Appendix J. Copyright
Advanced Bash−Scripting Guide
exit 0
There is an alternative, and perhaps less confusing method of redirecting a function's stdin. This involves redirecting the stdin to an embedded bracketed code block within the function.
#Instead of: Function ()
{
...
}< file
#Try this: Function ()
{
{
...
}< file
}
#Similarly,
Function |
() |
# This works. |
{ |
|
|
{ |
|
|
echo $* |
|
|
} | tr |
a b |
|
} |
|
|
Function |
() |
# This doesn't work. |
{ |
|
|
echo $* |
|
|
} | tr a |
b |
# A nested code block is mandatory here. |
#Thanks, S.C.
23.2.Local Variables
What makes a variable "local"?
local variables
A variable declared as local is one that is visible only within the block of code in which it appears. It has local "scope". In a function, a local variable has meaning only within that function block.
Example 23−8. Local variable visibility
#!/bin/bash |
|
func () |
|
{ |
|
local loc_var=23 |
# Declared local. |
echo |
|
echo "\"loc_var\" in function = $loc_var" |
|
global_var=999 |
# Not declared local. |
|
|
23.2. Local Variables |
254 |
Advanced Bash−Scripting Guide
echo "\"global_var\" in function = $global_var"
}
func
# Now, see if local 'a' exists outside function.
echo
echo "\"loc_var\" outside function = $loc_var"
# "loc_var" outside function =
# Nope, $loc_var not visible globally. echo "\"global_var\" outside function = $global_var"
# "global_var" outside function = 999
# $global_var is visible globally.
echo
exit 0
Before a function is called, all variables declared within the function are invisible outside the body of the function, not just those explicitly declared as local.
#!/bin/bash |
|
func () |
|
{ |
|
global_var=37 |
# Visible only within the function block |
|
#+ before the function has been called. |
} |
# END OF FUNCTION |
echo "global_var = $global_var" # global_var =
# Function "func" has not yet been called, #+ so $global_var is not visible here.
func
echo "global_var = $global_var" # global_var = 37
#Has been set by function call.
23.2.1.Local variables make recursion possible.
Local variables permit recursion, [52] but this practice generally involves much computational overhead and
is definitely not recommended in a shell script. [53]
Example 23−9. Recursion, using a local variable
#!/bin/bash |
|
# |
factorial |
# |
−−−−−−−−− |
#Does bash permit recursion?
#Well, yes, but...
#You gotta have rocks in your head to try it.
23.2.1. Local variables make recursion possible. |
255 |
Advanced Bash−Scripting Guide
MAX_ARG=5
E_WRONG_ARGS=65
E_RANGE_ERR=66
if [ −z "$1" ] then
echo "Usage: `basename $0` number" exit $E_WRONG_ARGS
fi
if [ "$1" −gt $MAX_ARG ] then
echo "Out of range (5 is maximum)."
#Let's get real now.
#If you want greater range than this,
#rewrite it in a real programming language. exit $E_RANGE_ERR
fi
fact ()
{
local number=$1
#Variable "number" must be declared as local,
#otherwise this doesn't work.
if [ "$number" −eq 0 ] then
|
factorial=1 |
# Factorial of 0 = 1. |
else |
|
|
|
let "decrnum = |
number − 1" |
|
fact $decrnum |
# Recursive function call. |
|
let "factorial |
= $number * $?" |
fi |
|
|
return $factorial |
|
|
} |
|
|
fact |
$1 |
|
echo |
"Factorial of |
$1 is $?." |
exit |
0 |
|
See also Example A−11 for an example of recursion in a script. Be aware that recursion is resource−intensive and executes slowly, and is therefore generally not appropriate to use in a script.
23.2.1. Local variables make recursion possible. |
256 |