How to Construct an Array of Arrays in Bash [duplicate] - arrays
I'm wondering how to declare a 2D array in bash and then initialize to 0.
In C it looks like this:
int a[4][5] = {0};
And how do I assign a value to an element? As in C:
a[2][3] = 3;
You can simulate them for example with hashes, but need care about the leading zeroes and many other things. The next demonstration works, but it is far from optimal solution.
#!/bin/bash
declare -A matrix
num_rows=4
num_columns=5
for ((i=1;i<=num_rows;i++)) do
for ((j=1;j<=num_columns;j++)) do
matrix[$i,$j]=$RANDOM
done
done
f1="%$((${#num_rows}+1))s"
f2=" %9s"
printf "$f1" ''
for ((i=1;i<=num_rows;i++)) do
printf "$f2" $i
done
echo
for ((j=1;j<=num_columns;j++)) do
printf "$f1" $j
for ((i=1;i<=num_rows;i++)) do
printf "$f2" ${matrix[$i,$j]}
done
echo
done
the above example creates a 4x5 matrix with random numbers and print it transposed, with the example result
1 2 3 4
1 18006 31193 16110 23297
2 26229 19869 1140 19837
3 8192 2181 25512 2318
4 3269 25516 18701 7977
5 31775 17358 4468 30345
The principle is: Creating one associative array where the index is an string like 3,4. The benefits:
it's possible to use for any-dimension arrays ;) like: 30,40,2 for 3 dimensional.
the syntax is close to "C" like arrays ${matrix[2,3]}
Bash doesn't have multi-dimensional array. But you can simulate a somewhat similar effect with associative arrays. The following is an example of associative array pretending to be used as multi-dimensional array:
declare -A arr
arr[0,0]=0
arr[0,1]=1
arr[1,0]=2
arr[1,1]=3
echo "${arr[0,0]} ${arr[0,1]}" # will print 0 1
If you don't declare the array as associative (with -A), the above won't work. For example, if you omit the declare -A arr line, the echo will print 2 3 instead of 0 1, because 0,0, 1,0 and such will be taken as arithmetic expression and evaluated to 0 (the value to the right of the comma operator).
Bash does not support multidimensional arrays.
You can simulate it though by using indirect expansion:
#!/bin/bash
declare -a a0=(1 2 3 4)
declare -a a1=(5 6 7 8)
var="a1[1]"
echo ${!var} # outputs 6
Assignments are also possible with this method:
let $var=55
echo ${a1[1]} # outputs 55
Edit 1: To read such an array from a file, with each row on a line, and values delimited by space, use this:
idx=0
while read -a a$idx; do
let idx++;
done </tmp/some_file
Edit 2: To declare and initialize a0..a3[0..4] to 0, you could run:
for i in {0..3}; do
eval "declare -a a$i=( $(for j in {0..4}; do echo 0; done) )"
done
Another approach is you can represent each row as a string, i.e. mapping the 2D array into an 1D array. Then, all you need to do is unpack and repack the row's string whenever you make an edit:
# Init a 4x5 matrix
a=("00 01 02 03 04" "10 11 12 13 14" "20 21 22 23 24" "30 31 32 33 34")
aset() {
row=$1
col=$2
value=$3
IFS=' ' read -r -a rowdata <<< "${a[$row]}"
rowdata[$col]=$value
a[$row]="${rowdata[#]}"
}
aget() {
row=$1
col=$2
IFS=' ' read -r -a rowdata <<< "${a[$row]}"
echo ${rowdata[$col]}
}
aprint() {
for rowdata in "${a[#]}"; do
echo $rowdata
done
}
echo "Matrix before change"
aprint
# Outputs: a[2][3] == 23
echo "a[2][3] == $( aget 2 3 )"
echo "a[2][3] = 9999"
aset 2 3 9999
# Show result
echo "Matrix after change"
aprint
Outputs:
Matrix before change
00 01 02 03 04
10 11 12 13 14
20 21 22 23 24
30 31 32 33 34
a[2][3] == 23
a[2][3] = 9999
Matrix after change
00 01 02 03 04
10 11 12 13 14
20 21 22 9999 24
30 31 32 33 34
You can also approach this in a much less smarter fashion
q=()
q+=( 1-2 )
q+=( a-b )
for set in ${q[#]};
do
echo ${set%%-*}
echo ${set##*-}
done
of course a 22 line solution or indirection is probably the better way to go and why not sprinkle eval every where to .
2D array can be achieved in bash by declaring 1D array and then elements can be accessed using (r * col_size) + c). Below logic delcares 1D array (str_2d_arr) and prints as 2D array.
col_size=3
str_2d_arr=()
str_2d_arr+=('abc' '200' 'xyz')
str_2d_arr+=('def' '300' 'ccc')
str_2d_arr+=('aaa' '400' 'ddd')
echo "Print 2D array"
col_count=0
for elem in ${str_2d_arr[#]}; do
if [ ${col_count} -eq ${col_size} ]; then
echo ""
col_count=0
fi
echo -e "$elem \c"
((col_count++))
done
echo ""
Output is
Print 2D array
abc 200 xyz
def 300 ccc
aaa 400 ddd
Below logic is very useful to get each row from the above declared 1D array str_2d_arr.
# Get nth row and update to 2nd arg
get_row_n()
{
row=$1
local -n a=$2
start_idx=$((row * col_size))
for ((i = 0; i < ${col_size}; i++)); do
idx=$((start_idx + i))
a+=(${str_2d_arr[${idx}]})
done
}
arr=()
get_row_n 0 arr
echo "Row 0"
for e in ${arr[#]}; do
echo -e "$e \c"
done
echo ""
Output is
Row 0
abc 200 xyz
A way to simulate arrays in bash (it can be adapted for any number of dimensions of an array):
#!/bin/bash
## The following functions implement vectors (arrays) operations in bash:
## Definition of a vector <v>:
## v_0 - variable that stores the number of elements of the vector
## v_1..v_n, where n=v_0 - variables that store the values of the vector elements
VectorAddElementNext () {
# Vector Add Element Next
# Adds the string contained in variable $2 in the next element position (vector length + 1) in vector $1
local elem_value
local vector_length
local elem_name
eval elem_value=\"\$$2\"
eval vector_length=\$$1\_0
if [ -z "$vector_length" ]; then
vector_length=$((0))
fi
vector_length=$(( vector_length + 1 ))
elem_name=$1_$vector_length
eval $elem_name=\"\$elem_value\"
eval $1_0=$vector_length
}
VectorAddElementDVNext () {
# Vector Add Element Direct Value Next
# Adds the string $2 in the next element position (vector length + 1) in vector $1
local elem_value
local vector_length
local elem_name
eval elem_value="$2"
eval vector_length=\$$1\_0
if [ -z "$vector_length" ]; then
vector_length=$((0))
fi
vector_length=$(( vector_length + 1 ))
elem_name=$1_$vector_length
eval $elem_name=\"\$elem_value\"
eval $1_0=$vector_length
}
VectorAddElement () {
# Vector Add Element
# Adds the string contained in the variable $3 in the position contained in $2 (variable or direct value) in the vector $1
local elem_value
local elem_position
local vector_length
local elem_name
eval elem_value=\"\$$3\"
elem_position=$(($2))
eval vector_length=\$$1\_0
if [ -z "$vector_length" ]; then
vector_length=$((0))
fi
if [ $elem_position -ge $vector_length ]; then
vector_length=$elem_position
fi
elem_name=$1_$elem_position
eval $elem_name=\"\$elem_value\"
if [ ! $elem_position -eq 0 ]; then
eval $1_0=$vector_length
fi
}
VectorAddElementDV () {
# Vector Add Element
# Adds the string $3 in the position $2 (variable or direct value) in the vector $1
local elem_value
local elem_position
local vector_length
local elem_name
eval elem_value="$3"
elem_position=$(($2))
eval vector_length=\$$1\_0
if [ -z "$vector_length" ]; then
vector_length=$((0))
fi
if [ $elem_position -ge $vector_length ]; then
vector_length=$elem_position
fi
elem_name=$1_$elem_position
eval $elem_name=\"\$elem_value\"
if [ ! $elem_position -eq 0 ]; then
eval $1_0=$vector_length
fi
}
VectorPrint () {
# Vector Print
# Prints all the elements names and values of the vector $1 on sepparate lines
local vector_length
vector_length=$(($1_0))
if [ "$vector_length" = "0" ]; then
echo "Vector \"$1\" is empty!"
else
echo "Vector \"$1\":"
for ((i=1; i<=$vector_length; i++)); do
eval echo \"[$i]: \\\"\$$1\_$i\\\"\"
###OR: eval printf \'\%s\\\n\' \"[\$i]: \\\"\$$1\_$i\\\"\"
done
fi
}
VectorDestroy () {
# Vector Destroy
# Empties all the elements values of the vector $1
local vector_length
vector_length=$(($1_0))
if [ ! "$vector_length" = "0" ]; then
for ((i=1; i<=$vector_length; i++)); do
unset $1_$i
done
unset $1_0
fi
}
##################
### MAIN START ###
##################
## Setting vector 'params' with all the parameters received by the script:
for ((i=1; i<=$#; i++)); do
eval param="\${$i}"
VectorAddElementNext params param
done
# Printing the vector 'params':
VectorPrint params
read temp
## Setting vector 'params2' with the elements of the vector 'params' in reversed order:
if [ -n "$params_0" ]; then
for ((i=1; i<=$params_0; i++)); do
count=$((params_0-i+1))
VectorAddElement params2 count params_$i
done
fi
# Printing the vector 'params2':
VectorPrint params2
read temp
## Getting the values of 'params2'`s elements and printing them:
if [ -n "$params2_0" ]; then
echo "Printing the elements of the vector 'params2':"
for ((i=1; i<=$params2_0; i++)); do
eval current_elem_value=\"\$params2\_$i\"
echo "params2_$i=\"$current_elem_value\""
done
else
echo "Vector 'params2' is empty!"
fi
read temp
## Creating a two dimensional array ('a'):
for ((i=1; i<=10; i++)); do
VectorAddElement a 0 i
for ((j=1; j<=8; j++)); do
value=$(( 8 * ( i - 1 ) + j ))
VectorAddElementDV a_$i $j $value
done
done
## Manually printing the two dimensional array ('a'):
echo "Printing the two-dimensional array 'a':"
if [ -n "$a_0" ]; then
for ((i=1; i<=$a_0; i++)); do
eval current_vector_lenght=\$a\_$i\_0
if [ -n "$current_vector_lenght" ]; then
for ((j=1; j<=$current_vector_lenght; j++)); do
eval value=\"\$a\_$i\_$j\"
printf "$value "
done
fi
printf "\n"
done
fi
################
### MAIN END ###
################
If each row of the matrix is the same size, then you can simply use a linear array and multiplication.
That is,
a=()
for (( i=0; i<4; ++i )); do
for (( j=0; j<5; ++j )); do
a[i*5+j]=0
done
done
Then your a[2][3] = 3 becomes
a[2*5+3] = 3
This approach might be worth turning into a set of functions, but since you can't pass arrays to or return arrays from functions, you would have to use pass-by-name and sometimes eval. So I tend to file multidimensional arrays under "things bash is simply Not Meant To Do".
One can simply define two functions to write ($4 is the assigned value) and read a matrix with arbitrary name ($1) and indexes ($2 and $3) exploiting eval and indirect referencing.
#!/bin/bash
matrix_write () {
eval $1"_"$2"_"$3=$4
# aux=$1"_"$2"_"$3 # Alternative way
# let $aux=$4 # ---
}
matrix_read () {
aux=$1"_"$2"_"$3
echo ${!aux}
}
for ((i=1;i<10;i=i+1)); do
for ((j=1;j<10;j=j+1)); do
matrix_write a $i $j $[$i*10+$j]
done
done
for ((i=1;i<10;i=i+1)); do
for ((j=1;j<10;j=j+1)); do
echo "a_"$i"_"$j"="$(matrix_read a $i $j)
done
done
Mark Reed suggested a very good solution for 2D arrays (matrix)! They always can be converted in a 1D array (vector). Although Bash doesn't have a native support for 2D arrays, it's not that hard to create a simple ADT around the mentioned principle.
Here is a barebone example with no argument checks, etc, just to keep the solution clear: the array's size is set as two first elements in the instance (documentation for the Bash module that implements a matrix ADT, https://github.com/vorakl/bash-libs/blob/master/src.docs/content/pages/matrix.rst )
#!/bin/bash
matrix_init() {
# matrix_init instance x y data ...
declare -n self=$1
declare -i width=$2 height=$3
shift 3;
self=(${width} ${height} "$#")
}
matrix_get() {
# matrix_get instance x y
declare -n self=$1
declare -i x=$2 y=$3
declare -i width=${self[0]} height=${self[1]}
echo "${self[2+y*width+x]}"
}
matrix_set() {
# matrix_set instance x y data
declare -n self=$1
declare -i x=$2 y=$3
declare data="$4"
declare -i width=${self[0]} height=${self[1]}
self[2+y*width+x]="${data}"
}
matrix_destroy() {
# matrix_destroy instance
declare -n self=$1
unset self
}
# my_matrix[3][2]=( (one, two, three), ("1 1" "2 2" "3 3") )
matrix_init my_matrix \
3 2 \
one two three \
"1 1" "2 2" "3 3"
# print my_matrix[2][0]
matrix_get my_matrix 2 0
# print my_matrix[1][1]
matrix_get my_matrix 1 1
# my_matrix[1][1]="4 4 4"
matrix_set my_matrix 1 1 "4 4 4"
# print my_matrix[1][1]
matrix_get my_matrix 1 1
# remove my_matrix
matrix_destroy my_matrix
For simulating a 2-dimensional array, I first load the first n-elements (the elements of the first column)
local pano_array=()
i=0
for line in $(grep "filename" "$file")
do
url=$(extract_url_from_xml $line)
pano_array[i]="$url"
i=$((i+1))
done
To add the second column, I define the size of the first column and calculate the values in an offset variable
array_len="${#pano_array[#]}"
i=0
while [[ $i -lt $array_len ]]
do
url="${pano_array[$i]}"
offset=$(($array_len+i))
found_file=$(get_file $url)
pano_array[$offset]=$found_file
i=$((i+1))
done
The below code will definitely work provided if you are working on a Mac you have bash version 4. Not only can you declare 0 but this is more of a universal approach to dynamically accepting values.
2D Array
declare -A arr
echo "Enter the row"
read r
echo "Enter the column"
read c
i=0
j=0
echo "Enter the elements"
while [ $i -lt $r ]
do
j=0
while [ $j -lt $c ]
do
echo $i $j
read m
arr[${i},${j}]=$m
j=`expr $j + 1`
done
i=`expr $i + 1`
done
i=0
j=0
while [ $i -lt $r ]
do
j=0
while [ $j -lt $c ]
do
echo -n ${arr[${i},${j}]} " "
j=`expr $j + 1`
done
echo ""
i=`expr $i + 1`
done
Related
how to using Dynamic Value in bash script [duplicate]
I'm wondering how to declare a 2D array in bash and then initialize to 0. In C it looks like this: int a[4][5] = {0}; And how do I assign a value to an element? As in C: a[2][3] = 3;
You can simulate them for example with hashes, but need care about the leading zeroes and many other things. The next demonstration works, but it is far from optimal solution. #!/bin/bash declare -A matrix num_rows=4 num_columns=5 for ((i=1;i<=num_rows;i++)) do for ((j=1;j<=num_columns;j++)) do matrix[$i,$j]=$RANDOM done done f1="%$((${#num_rows}+1))s" f2=" %9s" printf "$f1" '' for ((i=1;i<=num_rows;i++)) do printf "$f2" $i done echo for ((j=1;j<=num_columns;j++)) do printf "$f1" $j for ((i=1;i<=num_rows;i++)) do printf "$f2" ${matrix[$i,$j]} done echo done the above example creates a 4x5 matrix with random numbers and print it transposed, with the example result 1 2 3 4 1 18006 31193 16110 23297 2 26229 19869 1140 19837 3 8192 2181 25512 2318 4 3269 25516 18701 7977 5 31775 17358 4468 30345 The principle is: Creating one associative array where the index is an string like 3,4. The benefits: it's possible to use for any-dimension arrays ;) like: 30,40,2 for 3 dimensional. the syntax is close to "C" like arrays ${matrix[2,3]}
Bash doesn't have multi-dimensional array. But you can simulate a somewhat similar effect with associative arrays. The following is an example of associative array pretending to be used as multi-dimensional array: declare -A arr arr[0,0]=0 arr[0,1]=1 arr[1,0]=2 arr[1,1]=3 echo "${arr[0,0]} ${arr[0,1]}" # will print 0 1 If you don't declare the array as associative (with -A), the above won't work. For example, if you omit the declare -A arr line, the echo will print 2 3 instead of 0 1, because 0,0, 1,0 and such will be taken as arithmetic expression and evaluated to 0 (the value to the right of the comma operator).
Bash does not support multidimensional arrays. You can simulate it though by using indirect expansion: #!/bin/bash declare -a a0=(1 2 3 4) declare -a a1=(5 6 7 8) var="a1[1]" echo ${!var} # outputs 6 Assignments are also possible with this method: let $var=55 echo ${a1[1]} # outputs 55 Edit 1: To read such an array from a file, with each row on a line, and values delimited by space, use this: idx=0 while read -a a$idx; do let idx++; done </tmp/some_file Edit 2: To declare and initialize a0..a3[0..4] to 0, you could run: for i in {0..3}; do eval "declare -a a$i=( $(for j in {0..4}; do echo 0; done) )" done
Another approach is you can represent each row as a string, i.e. mapping the 2D array into an 1D array. Then, all you need to do is unpack and repack the row's string whenever you make an edit: # Init a 4x5 matrix a=("00 01 02 03 04" "10 11 12 13 14" "20 21 22 23 24" "30 31 32 33 34") aset() { row=$1 col=$2 value=$3 IFS=' ' read -r -a rowdata <<< "${a[$row]}" rowdata[$col]=$value a[$row]="${rowdata[#]}" } aget() { row=$1 col=$2 IFS=' ' read -r -a rowdata <<< "${a[$row]}" echo ${rowdata[$col]} } aprint() { for rowdata in "${a[#]}"; do echo $rowdata done } echo "Matrix before change" aprint # Outputs: a[2][3] == 23 echo "a[2][3] == $( aget 2 3 )" echo "a[2][3] = 9999" aset 2 3 9999 # Show result echo "Matrix after change" aprint Outputs: Matrix before change 00 01 02 03 04 10 11 12 13 14 20 21 22 23 24 30 31 32 33 34 a[2][3] == 23 a[2][3] = 9999 Matrix after change 00 01 02 03 04 10 11 12 13 14 20 21 22 9999 24 30 31 32 33 34
You can also approach this in a much less smarter fashion q=() q+=( 1-2 ) q+=( a-b ) for set in ${q[#]}; do echo ${set%%-*} echo ${set##*-} done of course a 22 line solution or indirection is probably the better way to go and why not sprinkle eval every where to .
2D array can be achieved in bash by declaring 1D array and then elements can be accessed using (r * col_size) + c). Below logic delcares 1D array (str_2d_arr) and prints as 2D array. col_size=3 str_2d_arr=() str_2d_arr+=('abc' '200' 'xyz') str_2d_arr+=('def' '300' 'ccc') str_2d_arr+=('aaa' '400' 'ddd') echo "Print 2D array" col_count=0 for elem in ${str_2d_arr[#]}; do if [ ${col_count} -eq ${col_size} ]; then echo "" col_count=0 fi echo -e "$elem \c" ((col_count++)) done echo "" Output is Print 2D array abc 200 xyz def 300 ccc aaa 400 ddd Below logic is very useful to get each row from the above declared 1D array str_2d_arr. # Get nth row and update to 2nd arg get_row_n() { row=$1 local -n a=$2 start_idx=$((row * col_size)) for ((i = 0; i < ${col_size}; i++)); do idx=$((start_idx + i)) a+=(${str_2d_arr[${idx}]}) done } arr=() get_row_n 0 arr echo "Row 0" for e in ${arr[#]}; do echo -e "$e \c" done echo "" Output is Row 0 abc 200 xyz
A way to simulate arrays in bash (it can be adapted for any number of dimensions of an array): #!/bin/bash ## The following functions implement vectors (arrays) operations in bash: ## Definition of a vector <v>: ## v_0 - variable that stores the number of elements of the vector ## v_1..v_n, where n=v_0 - variables that store the values of the vector elements VectorAddElementNext () { # Vector Add Element Next # Adds the string contained in variable $2 in the next element position (vector length + 1) in vector $1 local elem_value local vector_length local elem_name eval elem_value=\"\$$2\" eval vector_length=\$$1\_0 if [ -z "$vector_length" ]; then vector_length=$((0)) fi vector_length=$(( vector_length + 1 )) elem_name=$1_$vector_length eval $elem_name=\"\$elem_value\" eval $1_0=$vector_length } VectorAddElementDVNext () { # Vector Add Element Direct Value Next # Adds the string $2 in the next element position (vector length + 1) in vector $1 local elem_value local vector_length local elem_name eval elem_value="$2" eval vector_length=\$$1\_0 if [ -z "$vector_length" ]; then vector_length=$((0)) fi vector_length=$(( vector_length + 1 )) elem_name=$1_$vector_length eval $elem_name=\"\$elem_value\" eval $1_0=$vector_length } VectorAddElement () { # Vector Add Element # Adds the string contained in the variable $3 in the position contained in $2 (variable or direct value) in the vector $1 local elem_value local elem_position local vector_length local elem_name eval elem_value=\"\$$3\" elem_position=$(($2)) eval vector_length=\$$1\_0 if [ -z "$vector_length" ]; then vector_length=$((0)) fi if [ $elem_position -ge $vector_length ]; then vector_length=$elem_position fi elem_name=$1_$elem_position eval $elem_name=\"\$elem_value\" if [ ! $elem_position -eq 0 ]; then eval $1_0=$vector_length fi } VectorAddElementDV () { # Vector Add Element # Adds the string $3 in the position $2 (variable or direct value) in the vector $1 local elem_value local elem_position local vector_length local elem_name eval elem_value="$3" elem_position=$(($2)) eval vector_length=\$$1\_0 if [ -z "$vector_length" ]; then vector_length=$((0)) fi if [ $elem_position -ge $vector_length ]; then vector_length=$elem_position fi elem_name=$1_$elem_position eval $elem_name=\"\$elem_value\" if [ ! $elem_position -eq 0 ]; then eval $1_0=$vector_length fi } VectorPrint () { # Vector Print # Prints all the elements names and values of the vector $1 on sepparate lines local vector_length vector_length=$(($1_0)) if [ "$vector_length" = "0" ]; then echo "Vector \"$1\" is empty!" else echo "Vector \"$1\":" for ((i=1; i<=$vector_length; i++)); do eval echo \"[$i]: \\\"\$$1\_$i\\\"\" ###OR: eval printf \'\%s\\\n\' \"[\$i]: \\\"\$$1\_$i\\\"\" done fi } VectorDestroy () { # Vector Destroy # Empties all the elements values of the vector $1 local vector_length vector_length=$(($1_0)) if [ ! "$vector_length" = "0" ]; then for ((i=1; i<=$vector_length; i++)); do unset $1_$i done unset $1_0 fi } ################## ### MAIN START ### ################## ## Setting vector 'params' with all the parameters received by the script: for ((i=1; i<=$#; i++)); do eval param="\${$i}" VectorAddElementNext params param done # Printing the vector 'params': VectorPrint params read temp ## Setting vector 'params2' with the elements of the vector 'params' in reversed order: if [ -n "$params_0" ]; then for ((i=1; i<=$params_0; i++)); do count=$((params_0-i+1)) VectorAddElement params2 count params_$i done fi # Printing the vector 'params2': VectorPrint params2 read temp ## Getting the values of 'params2'`s elements and printing them: if [ -n "$params2_0" ]; then echo "Printing the elements of the vector 'params2':" for ((i=1; i<=$params2_0; i++)); do eval current_elem_value=\"\$params2\_$i\" echo "params2_$i=\"$current_elem_value\"" done else echo "Vector 'params2' is empty!" fi read temp ## Creating a two dimensional array ('a'): for ((i=1; i<=10; i++)); do VectorAddElement a 0 i for ((j=1; j<=8; j++)); do value=$(( 8 * ( i - 1 ) + j )) VectorAddElementDV a_$i $j $value done done ## Manually printing the two dimensional array ('a'): echo "Printing the two-dimensional array 'a':" if [ -n "$a_0" ]; then for ((i=1; i<=$a_0; i++)); do eval current_vector_lenght=\$a\_$i\_0 if [ -n "$current_vector_lenght" ]; then for ((j=1; j<=$current_vector_lenght; j++)); do eval value=\"\$a\_$i\_$j\" printf "$value " done fi printf "\n" done fi ################ ### MAIN END ### ################
If each row of the matrix is the same size, then you can simply use a linear array and multiplication. That is, a=() for (( i=0; i<4; ++i )); do for (( j=0; j<5; ++j )); do a[i*5+j]=0 done done Then your a[2][3] = 3 becomes a[2*5+3] = 3 This approach might be worth turning into a set of functions, but since you can't pass arrays to or return arrays from functions, you would have to use pass-by-name and sometimes eval. So I tend to file multidimensional arrays under "things bash is simply Not Meant To Do".
One can simply define two functions to write ($4 is the assigned value) and read a matrix with arbitrary name ($1) and indexes ($2 and $3) exploiting eval and indirect referencing. #!/bin/bash matrix_write () { eval $1"_"$2"_"$3=$4 # aux=$1"_"$2"_"$3 # Alternative way # let $aux=$4 # --- } matrix_read () { aux=$1"_"$2"_"$3 echo ${!aux} } for ((i=1;i<10;i=i+1)); do for ((j=1;j<10;j=j+1)); do matrix_write a $i $j $[$i*10+$j] done done for ((i=1;i<10;i=i+1)); do for ((j=1;j<10;j=j+1)); do echo "a_"$i"_"$j"="$(matrix_read a $i $j) done done
Mark Reed suggested a very good solution for 2D arrays (matrix)! They always can be converted in a 1D array (vector). Although Bash doesn't have a native support for 2D arrays, it's not that hard to create a simple ADT around the mentioned principle. Here is a barebone example with no argument checks, etc, just to keep the solution clear: the array's size is set as two first elements in the instance (documentation for the Bash module that implements a matrix ADT, https://github.com/vorakl/bash-libs/blob/master/src.docs/content/pages/matrix.rst ) #!/bin/bash matrix_init() { # matrix_init instance x y data ... declare -n self=$1 declare -i width=$2 height=$3 shift 3; self=(${width} ${height} "$#") } matrix_get() { # matrix_get instance x y declare -n self=$1 declare -i x=$2 y=$3 declare -i width=${self[0]} height=${self[1]} echo "${self[2+y*width+x]}" } matrix_set() { # matrix_set instance x y data declare -n self=$1 declare -i x=$2 y=$3 declare data="$4" declare -i width=${self[0]} height=${self[1]} self[2+y*width+x]="${data}" } matrix_destroy() { # matrix_destroy instance declare -n self=$1 unset self } # my_matrix[3][2]=( (one, two, three), ("1 1" "2 2" "3 3") ) matrix_init my_matrix \ 3 2 \ one two three \ "1 1" "2 2" "3 3" # print my_matrix[2][0] matrix_get my_matrix 2 0 # print my_matrix[1][1] matrix_get my_matrix 1 1 # my_matrix[1][1]="4 4 4" matrix_set my_matrix 1 1 "4 4 4" # print my_matrix[1][1] matrix_get my_matrix 1 1 # remove my_matrix matrix_destroy my_matrix
For simulating a 2-dimensional array, I first load the first n-elements (the elements of the first column) local pano_array=() i=0 for line in $(grep "filename" "$file") do url=$(extract_url_from_xml $line) pano_array[i]="$url" i=$((i+1)) done To add the second column, I define the size of the first column and calculate the values in an offset variable array_len="${#pano_array[#]}" i=0 while [[ $i -lt $array_len ]] do url="${pano_array[$i]}" offset=$(($array_len+i)) found_file=$(get_file $url) pano_array[$offset]=$found_file i=$((i+1)) done
The below code will definitely work provided if you are working on a Mac you have bash version 4. Not only can you declare 0 but this is more of a universal approach to dynamically accepting values. 2D Array declare -A arr echo "Enter the row" read r echo "Enter the column" read c i=0 j=0 echo "Enter the elements" while [ $i -lt $r ] do j=0 while [ $j -lt $c ] do echo $i $j read m arr[${i},${j}]=$m j=`expr $j + 1` done i=`expr $i + 1` done i=0 j=0 while [ $i -lt $r ] do j=0 while [ $j -lt $c ] do echo -n ${arr[${i},${j}]} " " j=`expr $j + 1` done echo "" i=`expr $i + 1` done
How read dynamic arrays in bash <4.3 version
I have a problem reading the contents of dynamic arrays. Here I create dynamic arrays: j=0 tab="project_" for i in "${unique[#]}" do echo -n $i eval ${tab}${i}=\(\) #eval $tab$i[0]="test" while read text do if [ "${i}" == "${text}" ]; then echo " - OK " eval ${tab}${i}=\(\) k=0 z=0 x=0 for k in "${array[#]}" do z=`expr ${z} + 1` if [[ "${i}" == "${k}" ]]; then eval $tab$i[x]+=${array[`expr $z + 1`]} x=`expr ${x} + 1` fi done fi done < "$allProjects" for name in "${tab}${i[#]}" do echo "$name" for n in "${name[#]}" do #eval echo "\${${tab}${i}[#]}" #this line will work, but I do not want to use eval echo ${n} # <-- Here I would like to read it with # the echo command, but it does not work # I get the name of a dynamic array, but # I want to get its contents. done done #eval echo \${#${tab}${i}[#]} #eval echo "\${${tab}${i}[#]}" done Maybe it's confusing because it's a snippet of code. The premise is to create dynaic arrays based on values from another array and add the word project_ to the beginning. Later on to these dynamically created arrays enter values from yet another array.
How to read first n elements from array
sum=0 read n read -a array for i in "${array[#]}" do sum=$(( $sum + $i )) done echo $sum I am new in scripting. Here I want to know what can I use to add condition so that reading array will be limited to n times. Ex: 3 1 2 3 4 here it should add only 1 2 3 and it should not take 4.
I prefer the bash C for loop, it doesn't require conditionals to escape the loop and looks neater. #! /bin/bash sum=0 read n read -a array for ((x=0; x<n; x++)) do sum=$(( sum + array[x] )) done echo "$sum"
You can avoid the loop altogether and use parameter substitution for slicing the array like this: $ cat script.bash sum=0 read -p "Enter n: " n read -p "Enter space separated array elements: " -a array echo "${array[#]:0:$n}" | sed 's/\s/+/g' | bc $ ./script.bash Enter n: 3 Enter space separated array elements: 1 2 3 4 6 $ echo "${array[#]:0:$n}" | sed 's/\s/+/g' | bc this statement dynamically slices the array and substitutes space \s with + to form the addition expression which is then piped to bc (calculator)
sum=0 i=0 read n read -a array while [[ $i -lt $n ]] do sum=$(( $sum + ${array[$i]})) let i++ done echo $sum I tried this and it worked. In btw thanks everyone
sum=0 read n j=0 read -a array for i in "${array[#]}" do if [[ $j -ge $n ]];then break else sum=$(( $sum + $i )) (( j = j + 1 )) fi done echo $sum
#!/bin/bash param=$1 sum=0 for i in {0..$param} do let " sum = $sum + ${array[$i]} " done echo "RESULT: $sum" and execute ./scipt.sh n
read N elements from an array and delete them
I am storing a file list in an array . What I'd like to do is loop through a process that will read N elements of an array and delete all the elements it just read. The Exception is the last iteration , when you come to the last iteration of the loop - whatever remains in the array - just split that out . th=$1 ar=('f1' 'f2' 'f3' 'f4' 'f5' 'f6') for ((i=1; i<=$th; i++)); do <stuff> if [ "$i" -eq "$th" ] then # if its the last iteration of the loop.Whatever remains in the array - spit that out echo " `echo ${ar[*]}`" >> somefile # it its anything short of the last iteration.Read N elements of the array at a time # and then delete them else echo " `echo ${ar[*]:0:$N}` " >> somefile for ((x=0; x<=$N; x++)) ; do unset ar[$x] done fi The results are very erratic. Even when I use this approach and test if separately for ((x=0; x<=$N; x++)) ; do unset ar[$x] done It will delete the WHOLE array EXCEPT the $Nth element I am new to arrays in shell. Any help is gladly appreciated
Try the following: #! /bin/bash th=3 N=2 ar=('f1 f' 'f2' 'f3' 'f4 x' 'f5' 'f6' 'f7') for ((i=0; i<$th; i++)); do if (( $i == $(($th - 1)) )) ; then echo "${ar[*]}" else echo "${ar[*]:0:$N}" ar=( "${ar[#]:$N}" ) fi done Output: f1 f f2 f3 f4 x f5 f6 f7 Note: Arrays in bash are zero based. Indices are not adjusted after unset ar[$x], therefore it would be easier to reconstruct the array as ar=( "${ar[#]:$N}" ) to force new indices to start at zero.. Update: Or you could avoid the reconstruction of the array using: #! /bin/bash th=3 N=2 ar=('f1 f' 'f2' 'f3' 'f4 x' 'f5' 'f6' 'f7') for ((i=0; i<$th; i++)); do if (( $i == $(($th - 1)) )) ; then echo "${ar[*]:$(($i * $N))}" else echo "${ar[*]:$(($i * $N)):$N}" fi done
How to pass array as an argument to a function in Bash
As we know, in bash programming the way to pass arguments is $1, ..., $N. However, I found it not easy to pass an array as an argument to a function which receives more than one argument. Here is one example: f(){ x=($1) y=$2 for i in "${x[#]}" do echo $i done .... } a=("jfaldsj jflajds" "LAST") b=NOEFLDJF f "${a[#]}" $b f "${a[*]}" $b As described, function freceives two arguments: the first is assigned to x which is an array, the second to y. f can be called in two ways. The first way use the "${a[#]}" as the first argument, and the result is: jfaldsj jflajds The second way use the "${a[*]}" as the first argument, and the result is: jfaldsj jflajds LAST Neither result is as I wished. So, is there anyone having any idea about how to pass array between functions correctly?
You cannot pass an array, you can only pass its elements (i.e. the expanded array). #!/bin/bash function f() { a=("$#") ((last_idx=${#a[#]} - 1)) b=${a[last_idx]} unset a[last_idx] for i in "${a[#]}" ; do echo "$i" done echo "b: $b" } x=("one two" "LAST") b='even more' f "${x[#]}" "$b" echo =============== f "${x[*]}" "$b" The other possibility would be to pass the array by name: #!/bin/bash function f() { name=$1[#] b=$2 a=("${!name}") for i in "${a[#]}" ; do echo "$i" done echo "b: $b" } x=("one two" "LAST") b='even more' f x "$b"
You can pass an array by name reference to a function in bash (since version 4.3+), by setting the -n attribute: show_value () # array index { local -n myarray=$1 local idx=$2 echo "${myarray[$idx]}" } This works for indexed arrays: $ shadock=(ga bu zo meu) $ show_value shadock 2 zo It also works for associative arrays: $ declare -A days=([monday]=eggs [tuesday]=bread [sunday]=jam) $ show_value days sunday jam See also nameref or declare -n in the man page.
You could pass the "scalar" value first. That would simplify things: f(){ b=$1 shift a=("$#") for i in "${a[#]}" do echo $i done .... } a=("jfaldsj jflajds" "LAST") b=NOEFLDJF f "$b" "${a[#]}" At this point, you might as well use the array-ish positional params directly f(){ b=$1 shift for i in "$#" # or simply "for i; do" do echo $i done .... } f "$b" "${a[#]}"
This will solve the issue of passing array to function: #!/bin/bash foo() { string=$1 array=($#) echo "array is ${array[#]}" echo "array is ${array[1]}" return } array=( one two three ) foo ${array[#]} colors=( red green blue ) foo ${colors[#]}
Try like this function parseArray { array=("$#") for data in "${array[#]}" do echo ${data} done } array=("value" "value1") parseArray "${array[#]}"
Pass the array as a function array() { echo "apple pear" } printArray() { local argArray="${1}" local array=($($argArray)) # where the magic happens. careful of the surrounding brackets. for arrElement in "${array[#]}"; do echo "${arrElement}" done } printArray array
Here is an example where I receive 2 bash arrays into a function, as well as additional arguments after them. This pattern can be continued indefinitely for any number of bash arrays and any number of additional arguments, accommodating any input argument order, so long as the length of each bash array comes just before the elements of that array. Function definition for print_two_arrays_plus_extra_args: # Print all elements of a bash array. # General form: # print_one_array array1 # Example usage: # print_one_array "${array1[#]}" print_one_array() { for element in "$#"; do printf " %s\n" "$element" done } # Print all elements of two bash arrays, plus two extra args at the end. # General form (notice length MUST come before the array in order # to be able to parse the args!): # print_two_arrays_plus_extra_args array1_len array1 array2_len array2 \ # extra_arg1 extra_arg2 # Example usage: # print_two_arrays_plus_extra_args "${#array1[#]}" "${array1[#]}" \ # "${#array2[#]}" "${array2[#]}" "hello" "world" print_two_arrays_plus_extra_args() { i=1 # Read array1_len into a variable array1_len="${#:$i:1}" ((i++)) # Read array1 into a new array array1=("${#:$i:$array1_len}") ((i += $array1_len)) # Read array2_len into a variable array2_len="${#:$i:1}" ((i++)) # Read array2 into a new array array2=("${#:$i:$array2_len}") ((i += $array2_len)) # You can now read the extra arguments all at once and gather them into a # new array like this: extra_args_array=("${#:$i}") # OR you can read the extra arguments individually into their own variables # one-by-one like this extra_arg1="${#:$i:1}" ((i++)) extra_arg2="${#:$i:1}" ((i++)) # Print the output echo "array1:" print_one_array "${array1[#]}" echo "array2:" print_one_array "${array2[#]}" echo "extra_arg1 = $extra_arg1" echo "extra_arg2 = $extra_arg2" echo "extra_args_array:" print_one_array "${extra_args_array[#]}" } Example usage: array1=() array1+=("one") array1+=("two") array1+=("three") array2=("four" "five" "six" "seven" "eight") echo "Printing array1 and array2 plus some extra args" # Note that `"${#array1[#]}"` is the array length (number of elements # in the array), and `"${array1[#]}"` is the array (all of the elements # in the array) print_two_arrays_plus_extra_args "${#array1[#]}" "${array1[#]}" \ "${#array2[#]}" "${array2[#]}" "hello" "world" Example Output: Printing array1 and array2 plus some extra args array1: one two three array2: four five six seven eight extra_arg1 = hello extra_arg2 = world extra_args_array: hello world For further examples and detailed explanations of how this works, see my longer answer on this topic here: Passing arrays as parameters in bash
You can also create a json file with an array, and then parse that json file with jq For example: my-array.json: { "array": ["item1","item2"] } script.sh: ARRAY=$(jq -r '."array"' $1 | tr -d '[],"') And then call the script like: script.sh ./path-to-json/my-array.json