วิธีการแบ่งอาร์เรย์ใน Bash


197

มองหาส่วน "Array" ในหน้า man (1) bash ฉันไม่พบวิธีที่จะแบ่งอาร์เรย์

ดังนั้นฉันจึงได้ฟังก์ชั่นที่ซับซ้อนมากเกินไปนี้:

#!/bin/bash

# @brief: slice a bash array
# @arg1:  output-name
# @arg2:  input-name
# @args:  seq args
# ----------------------------------------------
function slice() {
   local output=$1
   local input=$2
   shift 2
   local indexes=$(seq $*)

   local -i i
   local tmp=$(for i in $indexes 
                 do echo "$(eval echo \"\${$input[$i]}\")" 
               done)

   local IFS=$'\n'
   eval $output="( \$tmp )"
}

ใช้แบบนี้:

$ A=( foo bar "a  b c" 42 )
$ slice B A 1 2
$ echo "${B[0]}"  # bar
$ echo "${B[1]}"  # a  b c

มีวิธีที่ดีกว่าในการทำเช่นนี้?


ฉันกำลังค้นหาวิธีตัดส่วนท้ายของอาร์เรย์และกำกับที่นี่ คำตอบคือไม่ได้พบได้ที่นี่และมันจะเป็นที่ซ้ำกันจะทำเช่นนั้นเพราะผมพบคำตอบได้ที่นี่stackoverflow.com/questions/44939747/... แนวคิดพื้นฐานคือเราสามารถมีนิพจน์ทางคณิตศาสตร์เช่น $ {# array [@]} - (2 + 7) ซึ่งคาดว่าความยาวในการสร้าง $ {array: offset: length} ไม่มีคำตอบที่ให้ไว้ที่นี่แสดงให้เห็นว่า
Dominic108

คำตอบ:


313

ดูส่วนการขยายพารามิเตอร์ในmanหน้าBash A[@]ส่งคืนเนื้อหาของอาเรย์:1:2ใช้เวลาความยาว 2 เริ่มต้นที่ดัชนี 1

A=( foo bar "a  b c" 42 )
B=("${A[@]:1:2}")
C=("${A[@]:1}")       # slice to the end of the array
echo "${B[@]}"        # bar a  b c
echo "${B[1]}"        # a  b c
echo "${C[@]}"        # bar a  b c 42
echo "${C[@]: -2:2}"  # a  b c 42 # The space before the - is necesssary

โปรดทราบว่าข้อเท็จจริงที่ว่า "ab c" เป็นองค์ประกอบอาร์เรย์หนึ่ง (และมีพื้นที่พิเศษ) จะถูกเก็บรักษาไว้


2
เย็น. ฉันดูในส่วนอาเรย์และไม่เห็นมัน
เฉินประกาศ

36
เฉินโง่ทำไมมันจะอยู่ในส่วนของอาร์เรย์? * sarc
deltaray

1
@AquariusPower: idx=(${!A[@]}); echo ${idx[@]:1}สร้างอาร์เรย์ของดัชนีและเชือดมัน:
หยุดชั่วคราวจนกว่าจะมีการแจ้งให้ทราบต่อไป

7
@Feermurmel: เพียงแค่ทำโดยไม่ต้องวงเล็บเหลี่ยมดัชนี:${@:1:2}
หยุดชั่วคราวจนกว่าจะมีประกาศ

5
@DennisWilliamson ฉันพบว่าฉันต้องการแปลง$@เป็นอาร์เรย์ที่เหมาะสมก่อนที่จะทำสิ่งนี้หรือข้อโต้แย้งที่มีช่องว่างจะได้รับการแบ่ง:ARGS=( "$@" ); ARGS_AFTER_FIRST=( "${ARGS[@]:1}" )
Heath Borders

47

นอกจากนี้ยังมีทางลัดที่สะดวกในการรับองค์ประกอบทั้งหมดของอาร์เรย์โดยเริ่มจากดัชนีที่ระบุ ตัวอย่างเช่น "$ {A [@]: 1}" จะเป็น "tail" ของอาร์เรย์นั่นคืออาร์เรย์ที่ไม่มีองค์ประกอบแรก

version=4.7.1
A=( ${version//\./ } )
echo "${A[@]}"    # 4 7 1
B=( "${A[@]:1}" )
echo "${B[@]}"    # 7 1

8
และในขณะที่คุณอยู่ที่:echo "${A[@]::1}" # 4
เฉิน Levy

7
นี้เป็นที่ดี "${${@}[@]:1}"แต่มันควรจะตั้งข้อสังเกตว่าถ้าใช้ในการทำงานจะต้องมีการเปลี่ยนแปลงเล็กน้อยในการอ่าน
Alex Gray

@AlexGray: นั่นทำให้ฉัน "การทดแทนที่ไม่ดี" ที่นี่ แต่${@:2}ทำงานได้ดี
Nick Matteo

3

Array slicing เหมือนใน Python (จากไลบรารีrebash ):

array_slice() {
    local __doc__='
    Returns a slice of an array (similar to Python).

    From the Python documentation:
    One way to remember how slices work is to think of the indices as pointing
    between elements, with the left edge of the first character numbered 0.
    Then the right edge of the last element of an array of length n has
    index n, for example:
    ```
    +---+---+---+---+---+---+
    | 0 | 1 | 2 | 3 | 4 | 5 |
    +---+---+---+---+---+---+
    0   1   2   3   4   5   6
    -6  -5  -4  -3  -2  -1
    ```

    >>> local a=(0 1 2 3 4 5)
    >>> echo $(array.slice 1:-2 "${a[@]}")
    1 2 3
    >>> local a=(0 1 2 3 4 5)
    >>> echo $(array.slice 0:1 "${a[@]}")
    0
    >>> local a=(0 1 2 3 4 5)
    >>> [ -z "$(array.slice 1:1 "${a[@]}")" ] && echo empty
    empty
    >>> local a=(0 1 2 3 4 5)
    >>> [ -z "$(array.slice 2:1 "${a[@]}")" ] && echo empty
    empty
    >>> local a=(0 1 2 3 4 5)
    >>> [ -z "$(array.slice -2:-3 "${a[@]}")" ] && echo empty
    empty
    >>> [ -z "$(array.slice -2:-2 "${a[@]}")" ] && echo empty
    empty

    Slice indices have useful defaults; an omitted first index defaults to
    zero, an omitted second index defaults to the size of the string being
    sliced.
    >>> local a=(0 1 2 3 4 5)
    >>> # from the beginning to position 2 (excluded)
    >>> echo $(array.slice 0:2 "${a[@]}")
    >>> echo $(array.slice :2 "${a[@]}")
    0 1
    0 1

    >>> local a=(0 1 2 3 4 5)
    >>> # from position 3 (included) to the end
    >>> echo $(array.slice 3:"${#a[@]}" "${a[@]}")
    >>> echo $(array.slice 3: "${a[@]}")
    3 4 5
    3 4 5

    >>> local a=(0 1 2 3 4 5)
    >>> # from the second-last (included) to the end
    >>> echo $(array.slice -2:"${#a[@]}" "${a[@]}")
    >>> echo $(array.slice -2: "${a[@]}")
    4 5
    4 5

    >>> local a=(0 1 2 3 4 5)
    >>> echo $(array.slice -4:-2 "${a[@]}")
    2 3

    If no range is given, it works like normal array indices.
    >>> local a=(0 1 2 3 4 5)
    >>> echo $(array.slice -1 "${a[@]}")
    5
    >>> local a=(0 1 2 3 4 5)
    >>> echo $(array.slice -2 "${a[@]}")
    4
    >>> local a=(0 1 2 3 4 5)
    >>> echo $(array.slice 0 "${a[@]}")
    0
    >>> local a=(0 1 2 3 4 5)
    >>> echo $(array.slice 1 "${a[@]}")
    1
    >>> local a=(0 1 2 3 4 5)
    >>> array.slice 6 "${a[@]}"; echo $?
    1
    >>> local a=(0 1 2 3 4 5)
    >>> array.slice -7 "${a[@]}"; echo $?
    1
    '
    local start end array_length length
    if [[ $1 == *:* ]]; then
        IFS=":"; read -r start end <<<"$1"
        shift
        array_length="$#"
        # defaults
        [ -z "$end" ] && end=$array_length
        [ -z "$start" ] && start=0
        (( start < 0 )) && let "start=(( array_length + start ))"
        (( end < 0 )) && let "end=(( array_length + end ))"
    else
        start="$1"
        shift
        array_length="$#"
        (( start < 0 )) && let "start=(( array_length + start ))"
        let "end=(( start + 1 ))"
    fi
    let "length=(( end - start ))"
    (( start < 0 )) && return 1
    # check bounds
    (( length < 0 )) && return 1
    (( start < 0 )) && return 1
    (( start >= array_length )) && return 1
    # parameters start with $1, so add 1 to $start
    let "start=(( start + 1 ))"
    echo "${@: $start:$length}"
}
alias array.slice="array_slice"

1

ให้บอกว่าฉันกำลังอ่านอาเรย์จากผู้ใช้แล้วฉันต้องการที่จะเห็นองค์ประกอบ 3 ถึง 7 ทั้งรวม

cnt=0
while read var;
    do
    myarr[cnt]=$var
    cnt=$((cnt+1)) 
    done


echo ${myarr[@]:3:5}

4
ไวยากรณ์ส่วนแบ่งในรหัสของคุณเหมือนกับคำตอบที่ยอมรับได้ 8 ปี คำตอบของคุณไม่ได้เพิ่มอะไรใหม่
melpomene
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.