我有一个由项[I]定义的数据结构--包含-->子项[k]。我想写一个小的bash脚本,解析子项,项在命中特定子项时执行操作。
我的sampletext.properties文件如下:
Item1.subitem1=tom
Item1.subitem2=bob
Item2.subitem1=alice
Item2.subitem2=cindy
为了克服Bash中缺少二维数组的问题,我编写了两个函数,一个用于获取键的值,另一个用于计算字符串的出现次数。
然后,我构建了一个用于解析项的for循环,并尝试包含一个IF子句来命中";bob";然后说点什么。不幸的是,脚本没有看到bob,IF子句似乎不起作用:
#!/bin/bash
# The input text file
PROPERTY_FILE=sampletext.properties
#Defining a function to get the value of a key
function getProperty {
PROP_KEY=$1
PROP_VALUE=`cat $PROPERTY_FILE | grep "$PROP_KEY" | cut -d'=' -f2 | tr -d 'n'`
echo $PROP_VALUE
}
# Counting the number of subitem1 entries
function countsubitem1 {
toolsnumber=`cat $PROPERTY_FILE | grep "Item[0-9].subitem1" | wc -l`
echo $toolsnumber
}
countsubitem1
for (( i=1; i<=$(countsubitem1); i++))
do
subitem1=`getProperty "Item$i.subitem1"`
subitem2=`getProperty "Item$i.subitem2"`
echo item$i:$(getProperty "Item$i.subitem1") # Testing the getProperty function again
echo subitem2:$subitem2 # Testing the for loop and the getProperty function
if [[ "$subitem2" == "bob" ]]; then
echo "I see bob!"
else
echo "No bob here!"
fi
done
启动脚本时得到的内容:
2
item1:tom
subitem2:bob
No bob here!
item2:alice
subitem2:cindy
No bob here!
我的期望:
./samplebash.sh
2
item1:tom
subitem2:bob
I see bob!
item2:alice
subitem2:cindy
No bob here!
我试着对IF条款进行了几次修改:
if [ "$subitem2" == "bob" ]; then
if [ '$subitem2' == 'bob' ]; then
但我总是听到"这里没有波波头!"消息
请考虑以下代码:
readarray -t arr < "$PROPERTY_FILE"
getitem () {
for i in "${arr[@]}"; {
read item subitem value <<< ${i//[.=]/ }
case ${1:-1}${2:-2} in $item$subitem|${item}2|1$subitem)
echo "item=$item subitem=$subitem value=$value";;
esac
}
}
使用
$ getitem Item1
item=Item1 subitem=subitem1 value=tom
item=Item1 subitem=subitem2 value=bob
$ getitem Item1 subitem2
item=Item1 subitem=subitem2 value=bob
$ getitem '' subitem1
item=Item1 subitem=subitem1 value=tom
item=Item2 subitem=subitem1 value=alice
$ getitem Item2
item=Item2 subitem=subitem1 value=alice
使用关联数组
#!/bin/bash
# The input text file
PROPERTY_FILE=sampletext.properties
# Load the properties file
#
declare -A properties
loadProperties() {
local key value
if [[ "${#properties[@]}" -eq 0 ]]
then
# Lazy load the properties file
# Strip Windows-style CR as we go
#
while IFS='=' read -r key value
do
properties[$key]="$value"
done < <(tr -d 'r' <"$PROPERTY_FILE")
fi
}
# Get the value of a key
#
getProperty() {
local key="$1"
loadProperties
echo "${properties[$key]}"
}
# Count the number of "subitem1" items
#
countsubitem1() {
loadProperties
printf "%sn" "${!properties[@]}" | awk -F. '$2 == "subitem1"' | wc -l
}
现在您可以使用函数并获得预期值
countsubitem1
2
getProperty Item1.subitem2
bob
i=1
subitem1="$(getProperty "Item$i.subitem1")"
printf "%sn" "$subitem1"
tom
不要使用不推荐使用的function
关键字来声明函数。目标是使用小写的变量名,这样您的变量名就不会与$PATH
等标准变量发生冲突。使用$( ... )
而不是反勾号来括起需要执行的值。
感谢@William Pursell的提示!
添加一个| tr -d 'r'
清除了字符串,并且它被正确解析。
我现在获得:
item1subitem1:tom
subitem2:bob
I see bob!
item2subitem2:alice
subitem2:cindy
No bob here!
代码:
#!/bin/bash
# The input text file
PROPERTY_FILE=sampletext.properties
#Defining a function to get the value of a key
function getProperty {
PROP_KEY=$1
PROP_VALUE=`cat $PROPERTY_FILE | grep "$PROP_KEY" | cut -d'=' -f2 | tr -d 'n' | tr -d 'r'`
echo $PROP_VALUE
}
# Counting the number of subitem1 entries
function countsubitem1 {
toolsnumber=`cat $PROPERTY_FILE | grep "Item[0-9].subitem1" | wc -l`
echo $toolsnumber
}
for (( i=1; i<=$(countsubitem1); i++))
do
subitem1=`getProperty "Item$i.subitem1"`
subitem2=`getProperty "Item$i.subitem2"`
echo item"$i"subitem$i:$(getProperty "Item$i.subitem1") # Testing the getProperty function again
echo subitem2:$subitem2 # Testing the for loop and the getProperty function
if [[ "$subitem2" == "bob" ]]; then
echo "I see bob!"
else
echo "No bob here!"
fi
done