用while read读取指定目录下含空格的文件名,却不能用数组接收?

sh/bash/dash/ksh/zsh等Shell脚本
回复
头像
phi
帖子: 15
注册时间: 2014-01-29 23:06
系统: Ub
送出感谢: 2 次
接收感谢: 0

用while read读取指定目录下含空格的文件名,却不能用数组接收?

#1

帖子 phi » 2018-03-03 7:08

ls --color=never ~/ |
while read filename
do
echo "$filename"
filenames+=("$filename")
done
echo "${#filenames[@]}"
echo "${filenames[@]}"

我打算用while read批量读取一个指定目录下含有空格的文件名,并用一个数组批量接收。
很奇怪!read读取是正常的,echo回显正常,但是数组却不能正常被赋值???

我试过同样的方法批量读取一个列表文件中包含空格的文件名却是正常的,直接读目录的文件列表却不行!
while read filename
do
echo "$filename"
filenames+=("$filename")
done < list
echo "${#filenames[@]}"
echo "${filenames[@]}"
头像
careone
帖子: 804
注册时间: 2007-12-17 21:41
送出感谢: 71 次
接收感谢: 24 次

Re: 用while read读取指定目录下含空格的文件名,却不能用数组接收?

#2

帖子 careone » 2018-03-05 22:12

数组中,带有空格的单个数组值,当时存入时,可能是能正常识别为一个值的,
但是在后面多次读取数组值时,会“健忘”,直接按空格来区分数组值。

通常保险一点的作法,是先把列表保存到文件,再读取文件的每一行,
当成一个单独的数值(不管这一行包含多少个空格)。

可以考虑强行指定 IFS 变量的值为换行符(\n),也许可以解决这个问题。
注意:$ 要写在单引号的前面,另外还要是单引号,不能是双引号

代码: 全选

IFS=$'\n'
---------
或者直接这样写成两行(注意两个单引号中间不要有空格或者制表符TAB):

代码: 全选

IFS='
'
魁拔不死,天下不宁。
魁拔既死,天下不平。
--------------
Careone <emacslocale^126.com>
https://sourceforge.net/projects/emacslocale/files/
头像
phi
帖子: 15
注册时间: 2014-01-29 23:06
系统: Ub
送出感谢: 2 次
接收感谢: 0

Re: 用while read读取指定目录下含空格的文件名,却不能用数组接收?

#3

帖子 phi » 2018-03-06 21:03

careone 写了:数组中,带有空格的单个数组值,当时存入时,可能是能正常识别为一个值的,
但是在后面多次读取数组值时,会“健忘”,直接按空格来区分数组值。

通常保险一点的作法,是先把列表保存到文件,再读取文件的每一行,
当成一个单独的数值(不管这一行包含多少个空格)。

可以考虑强行指定 IFS 变量的值为换行符(\n),也许可以解决这个问题。
注意:$ 要写在单引号的前面,另外还要是单引号,不能是双引号

代码: 全选

IFS=$'\n'
---------
或者直接这样写成两行(注意两个单引号中间不要有空格或者制表符TAB):

代码: 全选

IFS='
'

谢谢! :em11
头像
astolia
论坛版主
帖子: 3147
注册时间: 2008-09-18 13:11
送出感谢: 1 次
接收感谢: 527 次

Re: 用while read读取指定目录下含空格的文件名,却不能用数组接收?

#4

帖子 astolia » 2018-04-22 11:38

@phi ,虽然 @careone 的方案是可以用,但他把问题原因完全解释错了,所以解决方案也不是最佳的

代码: 全选

ls --color=never ~/ |
while read filename
do
echo "$filename"
filenames+=("$filename")
echo "${filenames[@]}"
done
echo "${#filenames[@]}"
echo "${filenames[@]}"
这样在循环里加一句输出,实际问题就清楚了。filenames在循环内还是有值的,到了循环外就为空了。
这是因为对于管道,会创建子shell执行,所以while中的一切都是发生在子shell里,里面对filenames的修改不会影响到当前shell中的filenames变量
对于这种需求,bash有个有专门的语法,详见man bash里的Process Substitution一节

代码: 全选

while read filename
do
echo "$filename"
filenames+=("$filename")
done < <(ls --color=never ~/)
echo "${#filenames[@]}"
echo "${filenames[@]}"
这些用户感谢了作者 astolia 于这个帖子 (主题 2):
careone (2018-04-22 21:13) • phi (2018-05-03 8:01)
评价: 7.41%
kidkey
帖子: 28
注册时间: 2017-09-21 14:46
系统: ubuntu16.04
送出感谢: 5 次
接收感谢: 0

Re: 用while read读取指定目录下含空格的文件名,却不能用数组接收?

#5

帖子 kidkey » 2018-05-02 11:33

我搞不清楚 为什么同样LZ的代码在centos6.9下可以正常实验,为什么ubuntu18.04就不行呢?
是shell不一样还是怎么?
Ubuntu下 根本就不让过

代码: 全选

filenames+=("$filename")
根本无法 sh -x来看.shellcheck 没解决我的疑问,但是好像验证了 astolia的解释

代码: 全选

kids@kids-virtual-machine:~/桌面/nice$ shellcheck test.sh 

In test.sh line 1:
filenames=()
^-- SC2148: Tips depend on target shell and yours is unknown. Add a shebang.


In test.sh line 2:
ls --color=never ~/ |
^-- SC2012: Use find instead of ls to better handle non-alphanumeric filenames.


In test.sh line 3:
while read filename
      ^-- SC2162: read without -r will mangle backslashes.


In test.sh line 6:
		filenames+=("$filename")
                ^-- SC2030: Modification of filenames is local (to subshell caused by pipeline).


In test.sh line 7:
		echo ${filenames[*]}
                     ^-- SC2086: Double quote to prevent globbing and word splitting.


In test.sh line 9:
echo "${#filenames[@]}"
      ^-- SC2031: filenames was modified in a subshell. That change might be lost.


In test.sh line 11:
echo "${filenames[@]}"
      ^-- SC2031: filenames was modified in a subshell. That change might be lost.

kids@kids-virtual-machine:~/桌面/nice$ 
头像
astolia
论坛版主
帖子: 3147
注册时间: 2008-09-18 13:11
送出感谢: 1 次
接收感谢: 527 次

Re: 用while read读取指定目录下含空格的文件名,却不能用数组接收?

#6

帖子 astolia » 2018-05-02 11:38

ubuntu的sh默认是dash喔,不是bash
这些用户感谢了作者 astolia 于这个帖子:
kidkey (2018-05-02 11:47)
评价: 3.7%
头像
lilydjwg
论坛版主
帖子: 4163
注册时间: 2009-04-11 23:46
系统: Arch Linux
送出感谢: 11 次
接收感谢: 127 次
联系:

Re: 用while read读取指定目录下含空格的文件名,却不能用数组接收?

#7

帖子 lilydjwg » 2018-05-02 12:08

注意不要混淆 bash 和 sh(POSIX 兼容 shell)。bash 的特性会多一些。shell 脚本的第一行应当按照其实际使用的语法来标注是 /bin/sh 还是 /bin/bash。
回复

回到 “Shell脚本”