Retaining value outside loop

Hi,

I m new to shell scripting. I did some research and understand that unix treats while and other loops as new shell and hence the variable loose its value outside of the loop.

I found solution for integer variable but in mycase this is a string variable.

here variable loc is a multiline string.

totalstr=""
echo "$loc" | while IFS= read -r line ; do
totalstr=$totalstr,$line;
done
echo "Final Output:$totalstr"

I wish the get the multiline string in variable totalstr this format: string1,string2,string3,string4

But currently it is blank as it looses scope. Can you let me know what should be done ?

Hi,
maybe so

totalstr=$(grep -o "\S\+" <<<"$loc" | paste -sd,)
echo "Final Output:$totalstr"

Tried this suggestion but i see nothing empty output:

more list.sh
#!/bin/bash

loc=$1

totalstr=""

totalstr=$(grep -o "\S\+" <<<"$loc" | paste -sd,)

echo "FINAL:$totalstr"

My system is Linux.

Please suggest.

Try double quoting $1 when assigning it to the loc variable, so <NL> won't be lost.

If your target is to replace <NL> chars in a variable by commas, try

totalstr=${loc//$'\n'/,}

or even

totalstr=${1//$'\n'/,}
1 Like

This is not completely wrong but also not completely right:

"UNIX" is completely indifferent about this topic. This is a property of the shell and hence the behavior you see depends on which shell you use (and how you configure it).

Most of todays shells have a common ancestor, the "Bourne Shell", which indeed shows the behavior you noticed. In the following sample:

process | while read line ; do
     var=<something>
done

the while-do..done-loop is the second part of a pipeline and therefore executed in a subshell. Therefore, whatever value you assign to "var", it will be lost outside the loop because after the loop the subshell is simply closed. Notice that defining it up front like this:

var=<value>
process | while read line ; do
     var=<something>
done

will not help: the variable "var" in the subshell may have the same name as the variable "var" in the script shell but they are not the same at all. The only way you can overcome this is by replacing the pipeline with a redirection:

while read line ; do
     var=<something>
done < $(process)

Today, the real Bourne Shell is rarely used at all (IIRC it is from the beginning of the eighties) but mostly one of its two main successors: Korn Shell (ksh) and the Bourne Again Shell (bash). Both are (almost) completely backwards compatible with the Bourne Shell which means every shell script written for Bourne Shell should run on Korn shell as well as bash.

In ksh this (rather unintuitive, IMHO) behavior of the Bourne Shell was abandoned and therefore the two examples above work absolutely identical.

i=1
while [ i -le 100 ] ; do
     (( i += 1 ))
done
print $i

This will produce "100" as output in ksh and not "1", like in the Bourne Shell.

In bash, however, the original behavior of the Bourne Shell was retained and hence would produce "1" too. But because that was unintuitive even for the bash programmers in bash 4 (? i am not sure about the exact version) the shell-option "lastpipe" was introduced. If you set that with

shopt -s lastpipe

bash will work exactly like ksh (in this regard). Notice, though, that you might still come across bash versions which will not understand this configuration. You will have to do it either this way (redirection):

totalstr=""
while IFS= read -r line ; do
     totalstr=$totalstr,$line;
done < $(echo "$loc")
echo "Final Output:$totalstr"

or you can do it this way (a so-called "here-document"):

totalstr=""
while IFS= read -r line ; do
     totalstr=$totalstr,$line;
done <<- EOF
     echo "$loc"
EOF
echo "Final Output:$totalstr"

I hope this helps.

bakunin

1 Like

Why use read for a variable? This is exactly what a for loop is for:

OLDIFS="$IFS"
IFS="
"
for line in $loc
do
        totalstr=$totalstr,$line;
done
IFS="$OLDIFS"

If your multiline variable is simple enough, i.e. no embedded whitespace besides newline, leading, and trailing -- you can omit the IFS stuff for the same result.

1 Like

A correction:

i=1
while [ i -le 100 ] ; do
  (( i += 1 ))
done
print $i

produces 101 in ksh and errors in Bourne shell.
While the following

i=1
while [ $i -le 100 ] ; do
  i=`expr $i + 1`
done
echo $i

runs on both,
but might produce 1 in an old Bourne shell because it always runs the while loop in a sub shell.
A later Bourne shell has got a fix, but seeing a redirection it still runs the sub shell.

i=1
while [ $i -le 100 ] ; do
  i=`expr $i + 1`
done </dev/null
echo $i

That means all Bourne shells do not work with the previously suggested <<EOF redirection.
The Posix shell has got it all fixed. And all Posix-compatible shells.
But a pipe really forces a sub shell. Because it is the last part of the pipe. Compare with a simple command

echo "a:b:c:" | IFS=":" read a b c

Only the ksh has got the (incompatible) modification to run the last part of the pipe in the current shell.

Back to the initial post - that works only in ksh.
The following is portable to all Posix-compatible shells:

totalstr=$(
# sub shell starts
  sep=""
  echo "$loc" |
  while IFS= read -r line ; do
    # the pipe might force a sub sub shell
    printf "%s" "$sep$line"
    sep=","
  done
# sub shell ends
)
echo "Final Output:$totalstr "

As you can see, a sub shell inherits everything from the current shell, but not vice versa.