Удаление определенных значений из массива в ksh - PullRequest
6 голосов
/ 21 января 2010

У меня есть настроенный .profile, который я использую в ksh и ниже, это функция, которую я создал, чтобы пропускать туда и обратно каталоги с чрезмерно сложными или длинными именами.

Как видите, пути хранятся в массиве (BOOKMARKS[]), чтобы отслеживать их и ссылаться на них позже. Я хочу иметь возможность удалять определенные значения из массива, используя оператор case (или OPTARG, если необходимо), чтобы я мог просто набрать bmk -d #, чтобы удалить путь по связанному индексу.

Я возился с array +A and -A, но он просто завел мой массив (то, что осталось в закомментированном коде, может не быть симпатичным ... Я не корректировал его).

Какие-либо предложения / советы о том, как создать эту функциональность? Спасибо!

# To bookmark the current directory you are in for easy navigation back and forth from multiple non-aliased directories
# Use like 'bmk' (sets the current directory to a bookmark number) to go back to this directory, i.e. type 'bmk 3' (for the 3rd)
# To find out what directories are linked to which numbers, type 'bmk -l' (lowercase L)
# For every new directory bookmarked, the number will increase so the first time you run 'bmk' it will be 1 then 2,3,4...etc. for every consecutive run therea
fter
# TODO: finish -d (delete bookmark entry) function
make_bookmark()
{
        if [[ $# -eq 0 ]]; then
                BOOKMARKS[${COUNTER}]=${PWD}
                (( COUNTER=COUNTER+1 ))
        else
                case $1 in
                        -l)     NUM_OF_ELEMENTS=${#BOOKMARKS[*]}

                                while [[ ${COUNTER} -lt ${NUM_OF_ELEMENTS} ]]
                                do
                                        (( ACTUAL_NUM=i+1 ))
                                        echo ${ACTUAL_NUM}":"${BOOKMARKS[${i}]}
                                        (( COUNTER=COUNTER+1 ))
                                done
                                break ;;


                       #-d)    ACTUAL_NUM=$2
                                #(( REMOVE=${ACTUAL_NUM}-1 ))
                                #echo "Removing path ${BOOKMARKS[${REMOVE}]} from 'bmk'..."
                                #NUM_OF_ELEMENTS=${#BOOKMARKS[*]}

                                #while [[ ${NUM_OF_ELEMENTS} -gt 0 ]]
                                #do
                                        #if [[ ${NUM_OF_ELEMENTS} -ne ${ACTUAL_NUM} ]]; then
                                        #       TEMP_ARR=$(echo "${BOOKMARKS[*]}")
                                        #       (( NUM_OF_ELEMENTS=${NUM_OF_ELEMENTS}-1 ))
                                        #fi
                                        #echo $TEMP_ARR
                                #done
                                #break
                                #for VALUE in ${TEMP_ARR}
                                #do
                                #       set +A BOOKMARK ${TEMP_ARR}
                                #done
                                #echo ${BOOKMARK[*]}

                                #break ;;

                        *)      (( INDEX=$1-1 ))
                                cd ${BOOKMARKS[${INDEX}]}
                                break ;;
                esac
        fi
}

Ответы [ 3 ]

5 голосов
/ 21 января 2010

Массивы в оболочке Korn (и Bash, и другие) редки, поэтому, если вы используете unset для удаления членов массива, вы не сможете использовать размер массива в качестве индекса до последнего член и другие ограничения.

Вот несколько полезных фрагментов (второй цикл for - это то, что вы можете использовать сразу):

array=(1 2 3)
unset array[2]
echo ${array[2]}          # null
indices=(${!array[@]})    # create an array of the indices of "array"
size=${#indices[@]}       # the size of "array" is the number of indices into it
size=${#array[@]}         # same
echo ${array[@]: -1}      # you can use slices to get array elements, -1 is the last one, etc.
for element in ${array[@]}; do    # iterate over the array without an index

for index in ${indices[@]}        # iterate over the array WITH an index
do
    echo "Index: ${index}, Element: ${array[index]}"
done

for index in ${!array[@]}         # iterate over the array WITH an index, directly

Последний может устранить необходимость в счетчике.

Вот еще несколько полезных техник:

array+=("new element")    # append a new element without referring to an index
((counter++))             # shorter than ((counter=counter+1)) or ((counter+=1))
if [[ $var == 3 ]]        # you can use the more "natural" comparison operators inside double square brackets
while [[ $var < 11 ]]     # another example
echo ${array[${index}-1]  # math inside an array subscript

Все это предполагает, что ksh93, некоторые вещи могут не работать в более ранних версиях.

2 голосов
/ 21 января 2010

вы можете использовать unset. например, чтобы удалить элемент массива 1

unset array[0]

чтобы удалить весь массив

unset array
1 голос
/ 29 августа 2016

Несколько предостережений относительно предыдущего ответа:

Первое: я вижу эту ошибку все время. Когда вы предоставляете элемент массива для «unset», вы должны заключить его в кавычки. Обратите внимание:

$ echo foo > ./a2
$ ls a[2]
a2
$ a2="Do not delete this"
$ a=(this is not an array)
$ unset -v a[2]
$ echo "a2=${a2-UNSET}, a[]=${a[@]}"
a2=UNSET a[]=this is not an array

Что случилось? Подстановка. Вы, очевидно, хотели удалить элемент 2 из [], но синтаксис оболочки был таким, какой он есть, оболочка сначала проверила текущий каталог на наличие файла, который соответствует шаблону glob"a [2]". Если он находит совпадение, он заменяет шаблон глобуса на это имя файла, и вы решаете, какую переменную удалить, основываясь на том, какие файлы существуют в вашем текущем каталоге.

Это глубоко глупо. Но это не то, что кто-то удосужился исправить, по-видимому, и ошибка появляется во всех видах документации и примеров кода за последние 3 десятилетия.

Следующая проблема связана с тем, что элементы вашего ассоциативного массива легко вставлять с любым ключом, который вам нравится. Но сложнее удалить этих элементов:

typeset -A assoc
key="foo] bar"
assoc[$key]=3    #No problem!
unset -v "assoc[$key]"    #Problem!

В bash вы можете сделать это:

unset -v "assoc[\$key]"

В Korn Shell вы должны сделать это:

unset -v "assoc[foo\]\ bar]"

Так что все становится немного сложнее в случае, когда ваши ключи содержат синтаксические символы.

...