Доступ до останніх x символів рядка в Bash


124

Я з’ясував, що за допомогою ${string:0:3}одного можна отримати доступ до перших 3 символів рядка. Чи існує настільки ж простий метод доступу до останніх трьох символів?

Відповіді:


235

Останні три символи string:

${string: -3}

або

${string:(-3)}

(майте на увазі простір між першою формою :та -3в)

Будь ласка, зверніться до розширення параметра оболонки в посібнику :

${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.

Оскільки ця відповідь має декілька регулярних поглядів, дозвольте мені додати можливість звернутися до коментаря Джона Рікса ; як він зазначає, якщо ваша струна має довжину менше 3, вона ${string: -3}розширюється до порожньої. Якщо в цьому випадку ви хочете розширення string, ви можете використовувати:

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

Для цього використовується ?:потрійний оператор if, який може бути використаний в арифметиці Shell ; оскільки, як задокументовано, зміщення є арифметичним виразом, це справедливо.


5
Зауважте, що це не працює, якщо ваш рядок коротший від негативного зміщення, яке ви надаєте. Ви просто отримуєте порожній рядок у таких випадках.
Джон Рікс

2
@gniourf_gniourf Як це можна використовувати разом із заміною команд? Я намагаюся витягнути останні три символи мого імені хоста deppfx@localhost:/tmp$ echo ${$(hostname): -3} -bash: ${$(hostname): -3}: bad substitution
deppfx

3
@deppfx: ви не можете в Bash. Використання тимчасової змінної: temp=$(hostname); echo "${temp: -3}". Bash також має HOSTNAMEзмінну (яка може відрізнятись від результату виходу hostname). Якщо ви хочете використовувати його, просто зробіть echo "${HOSTNAME: -3}".
gniourf_gniourf

Як ви можете використовувати це на виході з труби? Так , наприклад, some func | echo ${string: -3}- як я можу призначити вихід some funcна string?
Майкл Хейс

1
@MichaelHays: Просто призначте результат своєї функції: string=$(some func)а потім echo "${string: -3}".
gniourf_gniourf

48

Ви можете використовувати tail:

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

Дещо крутим способом отримати останні три символи було б сказати:

echo $foo | rev | cut -c1-3 | rev

2
"Хвіст" також корисний у тире, коли удари недоступні. Наприклад, розділ сценарію на початку.
vskubriev

13

Іншим способом вирішення є використання grep -oза допомогою невеликої магії регексу, щоб отримати три символи з наступним кінцем рядка:

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

Щоб додатково отримати від 1 до 3 останніх знаків, у разі рядків із меншими за 3 символи, ви можете використовувати egrepцей регекс:

$ 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

Ви також можете використовувати різні діапазони, наприклад 5,10отримати останні п'ять-десять символів.


7

Щоб узагальнити запитання та відповідь gniourf_gniourf (так як це я шукав), якщо ви хочете вирізати діапазон символів, скажімо, від 7-го від кінця до третього з кінця, ви можете використовувати цей синтаксис:

${string: -7:4}

Де 4 - довжина курсу (7-3).

Окрім того, хоча рішення gniourf_gniourf, очевидно, найкраще і найменше, я просто хотів додати альтернативне рішення, використовуючи розріз :

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

Це читабельніше, якщо ви робите це у два рядки, визначаючи довжину $ {# string} як окрему змінну.


Варіант, який не згадується в іншій відповіді, полягає у поєднанні цього cutпідходу обчислення спершу запуску / зупинки, а потім просто використання цих змінних у розширенні параметрів (також варто згадати, що cutі компенсації bash починаються відповідно з 1 та нуля, тому це знадобиться щоб розібратися в розрахунках, які я тут не роблю): start=$((${#string}-3)); stop=$((${#string}));а потім echo ${string: $start : $stop}протиecho $string | cut -c "$start"-"$stop"
Майкл

(О, неважливо - я бачу, що мій коментар не стосується проблеми занадто короткої струни, а потім взагалі не викликає виводу - як розрізання, так і розширення парам-рядків. Все-таки розбиваючи його на змінні ( без обов'язкового виклику додаткових команд) полегшує читання, і все-таки гарна ідея.)
michael
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.