Вырезание всех символов после последнего /

Как я сокращаю все символы после последнего '/'?

Этот текст

xxxx/x/xx/xx/xxxx/x/yyyyy
xxx/xxxx/xxxxx/yyy

должен возвратиться

xxxx/x/xx/xx/xxxx/x
xxx/xxxx/xxxxx
10
задан 27 February 2018 в 20:57

7 ответов

Если Вы хотите добраться "cutted часть"

yyy
yyyyy

Можно использовать

sed 's|.*/||' 

например.

echo "xxx/xxxx/xxxxx/yyy" | sed 's|.*/||'
echo "xxxx/x/xx/xx/xxxx/x/yyyyy" | sed 's|.*\/||'

вывод

yyy
yyyyy

(Примечание: Это использует способность sed использовать разделитель кроме/, в этом случае |, в команде s),


Если Вы хотите получить начало строки:

xxx/xxxx/xxxxx
xxxx/x/xx/xx/xxxx/x

Можно использовать

sed 's|\(.*\)/.*|\1|'

например.

echo "xxx/xxxx/xxxxx/yyy" | sed 's|\(.*\)/.*|\1|'
echo "xxxx/x/xx/xx/xxxx/x/yyyyy" | sed 's|\(.*\)/.*|\1|'

вывод

xxx/xxxx/xxxxx
xxxx/x/xx/xx/xxxx/x
12
ответ дан 23 November 2019 в 04:15

Если Вы отключаете концы строк,dirname мог бы отвечать всем требованиям:

$ dirname xxxx/x/xx/xx/xxxx/x/yyyyy
xxxx/x/xx/xx/xxxx/x
$ _

При попытке изолировать последнюю часть строки, использовать echo /$(basename "$str").

$ str=xxxx/x/xx/xx/xxxx/x/yyyyy
$ echo /$(basename "$str")
/yyyyy
$ _
18
ответ дан 23 November 2019 в 04:15

Расширение параметра в bash

Можно использовать расширение параметра в bash, в этом случае

  • ${parameter%word} где word /*
  • ${parameter##word} где word */

Примеры:

Удалите последнюю часть

$ asdf="xxx/xxxx/xxxxx/yyy"
$ echo ${asdf%/*}
xxx/xxxx/xxxxx

Это описано в man bash:

${parameter%word}
${parameter%%word}
      Remove matching suffix pattern.  The word is expanded to produce
      a pattern just as in pathname expansion.  If the pattern matches
      a  trailing portion of the expanded value of parameter, then the
      result of the expansion is the expanded value of parameter  with
      the  shortest  matching  pattern (the ``%'' case) or the longest
      matching pattern (the ``%%'' case) deleted.  If parameter  is  @
      or  *,  the  pattern  removal operation is applied to each posi‐
      tional parameter in turn, and the  expansion  is  the  resultant
      list.   If  parameter is an array variable subscripted with @ or
      *, the pattern removal operation is applied to  each  member  of
      the array in turn, and the expansion is the resultant list.

Удалите все кроме последней части

$ asdf="xxx/xxxx/xxxxx/yyy"
$ echo ${asdf##*/}
yyy

Можно добавить наклонную черту как так

$ echo /${asdf##*/}
/yyy

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

Это описано в man bash:

${parameter#word}
${parameter##word}
      Remove matching prefix pattern.  The word is expanded to produce
      a pattern just as in pathname expansion.  If the pattern matches
      the  beginning of the value of parameter, then the result of the
      expansion is the expanded value of parameter with  the  shortest
      matching  pattern  (the ``#'' case) or the longest matching pat‐
      tern (the ``##'' case) deleted.  If parameter is  @  or  *,  the
      pattern  removal operation is applied to each positional parame‐
      ter in turn, and the expansion is the resultant list.  If param‐
      eter  is  an array variable subscripted with @ or *, the pattern
      removal operation is applied to each  member  of  the  array  in
      turn, and the expansion is the resultant list.
15
ответ дан 23 November 2019 в 04:15

Иначе должен использовать grep чтобы только отобразить последнюю наклонную черту на строку и независимо от того, что следует за ним:

$ grep -o '/[^/]*$' example.txt
/yyy
/yyyyy

Объяснение:

-o говорит grep только показать часть соответствия строки вместо целой строки.

Шаблон /[^/]*$ соответствует литеральной наклонной черте / сопровождаемый любым символом за исключением наклонной черты [^/] любое количество раз * до конца строки $.

9
ответ дан 23 November 2019 в 04:15

Просто, потому что другие отправили более “нормальные” ответы, вот несколько глупый:

rev | cut -d/ -f1 | rev

rev инвертирует символы в каждой строке, например. abcd/ef/g становится g/fe/dcba. Затем cut отключает первый сегмент. Наконец это инвертируется снова.

7
ответ дан 23 November 2019 в 04:15

Классическое решение с awk, это рассматривает / как разделитель полей и для ввода и вывода и для наборов последнее поле к пустой строке (который действительно "разыменовывает" количества полей NF переменная):

$ echo "xxx/xxxx/xxxxx/yyy"|awk  'BEGIN{OFS=FS="/"};{$NF="";print $0}'
xxx/xxxx/xxxxx/

Короче, как fedorqui указанный в комментариях:

$ echo "xxx/xxxx/xxxxx/yyy"|awk  'BEGIN{OFS=FS="/"};NF--'
xxx/xxxx/xxxxx/

Вариация на это должна была бы поместить путь к awkсреда выполнения, которая накопит при инфраструктуре, но делает awk более подробная часть:

mypath="xxx/xxxx/xxxxx/yyy" awk  'BEGIN{OFS=FS="/"; sub(/\/([^\/]*)$/,"",ENVIRON["mypath"]);print(ENVIRON["mypath"]) };'
3
ответ дан 23 November 2019 в 04:15

Добавление к ответу egmont, потому что вопрос был отредактирован...

Используйте - дополнение, если Вы хотите удалить первое поле с-f1.

echo "xxxx/x/xx/xx/xxxx/x/yyyyy"|rev|cut -d/ -f1 --complement|rev
xxxx/x/xx/xx/xxxx/x

echo "xxxx/x/xx/xx/xxxx/x/yyyyy"|rev|cut -d/ -f1|rev
yyyyy

Кроме того, вопрос не вполне соглашается с тем, что должно произойти с исходными данными, не содержащими любые наклонные черты. xxxx => xxxx или xxxx => ничто

2
ответ дан 23 November 2019 в 04:15

Другие вопросы по тегам:

Похожие вопросы: