Mengakses karakter x terakhir dari sebuah string di Bash

125

Saya menemukan bahwa dengan ${string:0:3}seseorang dapat mengakses 3 karakter pertama dari sebuah string. Apakah ada metode yang sama mudahnya untuk mengakses tiga karakter terakhir?

aldorado
sumber

Jawaban:

236

Tiga karakter terakhir dari string:

${string: -3}

atau

${string:(-3)}

(perhatikan jarak antara :dan -3dalam bentuk pertama).

Silakan merujuk ke Ekspansi Parameter Shell di manual referensi :

${parameter:offset}
${parameter:offset:length}

Expands to up to length characters of parameter starting at the character
specified by offset. If length is omitted, expands to the substring of parameter
starting at the character specified by offset. length and offset are arithmetic
expressions (see Shell Arithmetic). This is referred to as Substring Expansion.

If offset evaluates to a number less than zero, the value is used as an offset
from the end of the value of parameter. If length evaluates to a number less than
zero, and parameter is not ‘@’ and not an indexed or associative array, it is
interpreted as an offset from the end of the value of parameter rather than a
number of characters, and the expansion is the characters between the two
offsets. If parameter is ‘@’, the result is length positional parameters
beginning at offset. If parameter is an indexed array name subscripted by ‘@’ or
‘*’, the result is the length members of the array beginning with
${parameter[offset]}. A negative offset is taken relative to one greater than the
maximum index of the specified array. Substring expansion applied to an
associative array produces undefined results.

Note that a negative offset must be separated from the colon by at least one
space to avoid being confused with the ‘:-’ expansion. Substring indexing is
zero-based unless the positional parameters are used, in which case the indexing
starts at 1 by default. If offset is 0, and the positional parameters are used,
$@ is prefixed to the list.

Karena jawaban ini sering dilihat, izinkan saya menambahkan kemungkinan untuk membahas komentar John Rix ; seperti yang dia sebutkan, jika string Anda memiliki panjang kurang dari 3, ${string: -3}memuai menjadi string kosong. Jika, dalam kasus ini, Anda menginginkan perluasan string, Anda dapat menggunakan:

${string:${#string}<3?0:-3}

Ini menggunakan ?:terner if operator, yang dapat digunakan dalam Aritmatika Shell ; karena seperti yang didokumentasikan, offset adalah ekspresi aritmatika, ini valid.

gniourf_gniourf
sumber
5
Perhatikan bahwa ini tidak berfungsi jika string Anda lebih pendek dari offset negatif yang Anda berikan. Anda hanya mendapatkan string kosong dalam contoh seperti itu.
John Rix
2
@gniourf_gniourf Bagaimana menggunakan ini bersama dengan substitusi perintah? Saya mencoba mengekstrak tiga karakter terakhir dari nama host saya deppfx@localhost:/tmp$ echo ${$(hostname): -3} -bash: ${$(hostname): -3}: bad substitution
deppfx
3
@deppfx: Anda tidak bisa di Bash. Menggunakan variabel sementara: temp=$(hostname); echo "${temp: -3}". Bash juga memiliki HOSTNAMEvariabel (yang mungkin atau mungkin tidak berbeda dari keluaran hostname). Jika Anda ingin menggunakannya, lakukan saja echo "${HOSTNAME: -3}".
gniourf_gniourf
Bagaimana Anda bisa menggunakan ini dari keluaran pipa? Misalnya, some func | echo ${string: -3}- bagaimana cara menetapkan keluaran some funcke string?
Michael Hays
1
@MichaelHays: Cukup tetapkan output dari fungsi Anda: string=$(some func)dan kemudian echo "${string: -3}".
gniourf_gniourf
48

Anda dapat menggunakan tail:

$ foo="1234567890"
$ echo -n $foo | tail -c 3
890

Cara yang agak tidak pasti untuk mendapatkan tiga karakter terakhir adalah dengan mengatakan:

echo $foo | rev | cut -c1-3 | rev
devnull
sumber
2
The "tail" juga berguna di dasbor, saat bash tidak tersedia. Misalnya bagian skrip pemula.
vskubriev
13

Solusi lain adalah menggunakan grep -osedikit sihir regex untuk mendapatkan tiga karakter yang diikuti oleh akhir baris:

$ foo=1234567890
$ echo $foo | grep -o ...$
890

Untuk membuatnya secara opsional mendapatkan 1 hingga 3 karakter terakhir, jika string dengan kurang dari 3 karakter, Anda dapat menggunakan egrepdengan regex ini:

$ echo a | egrep -o '.{1,3}$'
a
$ echo ab | egrep -o '.{1,3}$'
ab
$ echo abc | egrep -o '.{1,3}$'
abc
$ echo abcd | egrep -o '.{1,3}$'
bcd

Anda juga dapat menggunakan rentang yang berbeda, seperti 5,10untuk mendapatkan lima hingga sepuluh karakter terakhir.

Aurelio Jargas
sumber
7

Untuk menggeneralisasi pertanyaan dan jawaban dari gniourf_gniourf (karena inilah yang saya cari), jika Anda ingin memotong berbagai karakter dari, katakanlah, 7 dari akhir ke 3 dari akhir, Anda dapat menggunakan sintaks ini:

${string: -7:4}

Dimana 4 adalah panjang jalur (7-3).

Selain itu, meskipun solusi gniourf_gniourf jelas merupakan yang terbaik dan paling rapi, saya hanya ingin menambahkan solusi alternatif menggunakan cut :

echo $string | cut -c $((${#string}-2))-$((${#string}))

Ini lebih mudah dibaca jika Anda melakukannya dalam dua baris dengan mendefinisikan panjang $ {# string} sebagai variabel terpisah.

Adrian Tompkins
sumber
varian yang tidak disebutkan dalam jawaban lain adalah menggabungkan cutpendekatan ini untuk menghitung start / stop terlebih dahulu dan kemudian hanya menggunakan variabel-variabel ini dalam ekspansi parameter (juga perlu disebutkan bahwa cutdan offset bash masing-masing dimulai dari 1 dan nol, jadi ini perlu untuk dimasukkan ke dalam perhitungan, yang tidak saya lakukan di sini): start=$((${#string}-3)); stop=$((${#string}));dan kemudian echo ${string: $start : $stop}vsecho $string | cut -c "$start"-"$stop"
michael
(oh, sudahlah - saya melihat bahwa komentar saya tidak membahas masalah string yang terlalu pendek, dan kemudian tidak menyebabkan keluaran sama sekali - baik pemotongan maupun perluasan parameter string. Namun, memecahnya menjadi variabel ( tanpa harus memanggil perintah tambahan) membuatnya lebih mudah untuk dibaca & akan tetap menjadi ide yang bagus, saya pikir.)
michael