Jak sprawdzić, czy ciąg zawiera podciąg w Bash


2564

Mam ciąg w Bash:

string="My string"

Jak mogę sprawdzić, czy zawiera inny ciąg?

if [ $string ?? 'foo' ]; then
  echo "It's there!"
fi

Gdzie ??jest mój nieznany operator. Czy używam echa i grep?

if echo "$string" | grep 'foo'; then
  echo "It's there!"
fi

To wygląda trochę niezdarnie.


4
Cześć, jeśli puste ciągi są fałszywe, dlaczego uważasz to za niezdarne? To był jedyny sposób, który działał dla mnie, pomimo zaproponowanych rozwiązań.
ericson.cepeda

1
Możesz użyć exprpolecenia tutaj
cli__

4
Oto jeden z pocisków posix: stackoverflow.com/questions/2829613/...
patrz

Odpowiedzi:


3647

Możesz użyć odpowiedzi Marcusa (* symboli wieloznacznych) poza instrukcją case, jeśli używasz podwójnych nawiasów:

string='My long string'
if [[ $string == *"My long"* ]]; then
  echo "It's there!"
fi

Zauważ, że spacje w ciągu igły muszą być umieszczone między podwójnymi cudzysłowami, a *symbole wieloznaczne powinny znajdować się na zewnątrz. Zauważ też, że używany jest prosty operator porównania (tj. ==), A nie operator wyrażenia regularnego =~.


127
Pamiętaj też, że możesz odwrócić porównanie, po prostu przełączając na! = W teście. Dziękuję za odpowiedź!
Quinn Taylor

63
@Jonik: Być może brakuje ci shebang lub masz go jako #!/bin/sh. Spróbuj #!/bin/bashzamiast tego.
Wstrzymano do odwołania.

10
Pozostaw odstęp między nawiasami a zawartością.
Paul Price

17
Nie musisz cytować zmiennych wewnątrz [[]].
Działa

12
@Orwellophile Ostrożnie! Możesz pominąć podwójne cudzysłowy tylko w pierwszym argumencie do [[. Zobacz ideone.com/ZSy1gZ
nyuszika7h

611

Jeśli wolisz podejście wyrażenia regularnego:

string='My string';

if [[ $string =~ "My" ]]
then
   echo "It's there!"
fi

2
Musiałem zastąpić wyrażenie regularne egrep w skrypcie bash, działało to doskonale!
blast_hardcheese

114
=~Operator już przeszukuje cały ciąg na mecz; te .*„s są tu obcy. Również cytaty są na ogół lepsze niż ukośniki odwrotne:[[ $string =~ "My s" ]]
bukzor

16
@bukzor Cytaty przestały działać tutaj od wersji Bash 3.2+: tiswww.case.edu/php/chet/bash/FAQ E14) . Prawdopodobnie najlepiej przypisać do zmiennej (używając cudzysłowów), a następnie porównać. W ten sposób:re="My s"; if [[ $string =~ $re ]]
seanf

36
Sprawdź, czy NIE zawiera ciągu: if [[ ! "abc" =~ "d" ]]prawda.
KrisWebDev

1
Pamiętaj, że kolejność wykonywania testu ma znaczenie. Wynik następnego wiersza kodu zostanie przesłany do przodu 2.number=2; if [[ "1, 2, 4, 5" = *${number}* ]]; then echo forward $number; fi; if [[ *${number}* = "1, 2, 4, 5" ]]; then echo backwards $number; fi;
Douwe van der Leest

356

Nie jestem pewien co do użycia instrukcji if, ale możesz uzyskać podobny efekt dzięki instrukcji case:

case "$string" in 
  *foo*)
    # Do stuff
    ;;
esac

76
Jest to prawdopodobnie najlepsze rozwiązanie, ponieważ można je przenosić na powłoki POSIX. (aka no bashisms)
technosaurus

25
@ Technosaurus Uważam, że dość dziwne jest krytykowanie „bashism” w pytaniu, które ma tylko tag bash :)
PP

39
@PP To nie tyle krytyka, co preferencja bardziej uniwersalnego rozwiązania niż bardziej ograniczonego. Proszę wziąć pod uwagę, że lata później ludzie (tacy jak ja) wpadną do nas, aby poszukać odpowiedzi i być może z przyjemnością znajdziemy taką, która będzie przydatna w szerszym zakresie niż pierwotne pytanie. Jak mówią w świecie Open Source: „wybór jest dobry!”
Carl Smotricz

2
@technosaurus, FWIW [[ $string == *foo* ]]działa również w niektórych wersjach sh zgodnych z POSIX (np. /usr/xpg4/bin/shw Solarisie 10) i ksh (> = 88)
maxschlepzig

3
Busybox ash nie obsługuje gwiazdek w [[... przełącznik skrzynek działał!
Ray Foss,

232

stringContain warianty (kompatybilne lub niezależne od wielkości liter)

Ponieważ te odpowiedzi dotyczące przepełnienia stosu mówią głównie o Bash , na samym dole tego postu opublikowałem niezależną od wielkości funkcję Bash ...

W każdym razie jest mój

Kompatybilna odpowiedź

Ponieważ jest już wiele odpowiedzi za pomocą funkcji specyficznych dla Bash, istnieje sposób działania w słabszych powłokach, takich jak BusyBox :

[ -z "${string##*$reqsubstr*}" ]

W praktyce może to dać:

string='echo "My string"'
for reqsubstr in 'o "M' 'alt' 'str';do
  if [ -z "${string##*$reqsubstr*}" ] ;then
      echo "String '$string' contain substring: '$reqsubstr'."
    else
      echo "String '$string' don't contain substring: '$reqsubstr'."
    fi
  done

Zostało to przetestowane pod Bash, Dash , KornShell ( ksh) i ash (BusyBox), a wynik jest zawsze:

String 'echo "My string"' contain substring: 'o "M'.
String 'echo "My string"' don't contain substring: 'alt'.
String 'echo "My string"' contain substring: 'str'.

W jedną funkcję

Jak pyta @EeroAaltonen tutaj jest wersja tego samego demo, przetestowana pod tymi samymi powłokami:

myfunc() {
    reqsubstr="$1"
    shift
    string="$@"
    if [ -z "${string##*$reqsubstr*}" ] ;then
        echo "String '$string' contain substring: '$reqsubstr'.";
      else
        echo "String '$string' don't contain substring: '$reqsubstr'."
    fi
}

Następnie:

$ myfunc 'o "M' 'echo "My String"'
String 'echo "My String"' contain substring 'o "M'.

$ myfunc 'alt' 'echo "My String"'
String 'echo "My String"' don't contain substring 'alt'.

Uwaga: musisz uciec lub podwójnie dołączyć cudzysłowy i / lub podwójne cudzysłowy:

$ myfunc 'o "M' echo "My String"
String 'echo My String' don't contain substring: 'o "M'.

$ myfunc 'o "M' echo \"My String\"
String 'echo "My String"' contain substring: 'o "M'.

Prosta funkcja

Zostało to przetestowane pod BusyBox, Dash i oczywiście Bash:

stringContain() { [ -z "${2##*$1*}" ]; }

Więc teraz:

$ if stringContain 'o "M3' 'echo "My String"';then echo yes;else echo no;fi
no
$ if stringContain 'o "M' 'echo "My String"';then echo yes;else echo no;fi
yes

... Lub jeśli przesłany ciąg może być pusty, jak wskazał @Sjlver, funkcja wyglądałaby następująco:

stringContain() { [ -z "${2##*$1*}" ] && [ -z "$1" -o -n "$2" ]; }

lub jak sugeruje komentarz Adriana Güntera , unikając -oprzełączników:

stringContain() { [ -z "${2##*$1*}" ] && { [ -z "$1" ] || [ -n "$2" ];};}

Ostateczna (prosta) funkcja:

I odwracanie testów, aby były potencjalnie szybsze:

stringContain() { [ -z "$1" ] || { [ -z "${2##*$1*}" ] && [ -n "$2" ];};}

Z pustymi łańcuchami:

$ if stringContain '' ''; then echo yes; else echo no; fi
yes
$ if stringContain 'o "M' ''; then echo yes; else echo no; fi
no

Niezależny od wielkości liter (tylko Bash!)

Aby przetestować ciągi bez dbania o wielkość liter, po prostu przekonwertuj każdy ciąg na małe litery:

stringContain() {
    local _lc=${2,,}
    [ -z "$1" ] || { [ -z "${_lc##*${1,,}*}" ] && [ -n "$2" ] ;} ;}

Czek:

stringContain 'o "M3' 'echo "my string"' && echo yes || echo no
no
stringContain 'o "My' 'echo "my string"' && echo yes || echo no
yes
if stringContain '' ''; then echo yes; else echo no; fi
yes
if stringContain 'o "M' ''; then echo yes; else echo no; fi
no

1
Byłoby jeszcze lepiej, gdybyś mógł wymyślić jakiś sposób na przypisanie tej funkcji.
Eero Aaltonen,

2
@EeroAaltonen Jak znaleźć moją (dodaną) funkcję?
F. Hauri

2
Wiem! odnaleźć . -nazwa „*” | xargs grep "myfunc" 2> / dev / null
eggmatters

6
To wspaniałe, ponieważ jest tak kompatybilne. Jeden błąd: Nie działa, jeśli łańcuch stogu siana jest pusty. Prawidłowa wersja to string_contains() { [ -z "${2##*$1*}" ] && [ -n "$2" -o -z "$1" ]; } Ostatnia myśl: czy pusty ciąg zawiera pusty ciąg? Wersja powyżej rzeczy tak (z powodu tej -o -z "$1"części).
Sjlver

3
+1. Bardzo dobre! Dla mnie zmieniłem zamówienie stringContain () {[-z "$ {1 ## * $ 2 *}"] && [-z "2 $" -o -n "$ 1"]; }; „Wyszukaj gdzie” „Znajdź co”. Pracuj w busyboksie. Zaakceptowana odpowiedź powyżej nie działa w busyboksie.
user3439968

148

Powinieneś pamiętać, że skrypty powłoki to mniej język, a raczej zbiór poleceń. Instynktownie myślisz, że ten „język” wymaga, abyś podążał za ifznakiem „a” [lub „a” [[. Oba są tylko poleceniami, które zwracają status wyjścia wskazujący powodzenie lub niepowodzenie (tak jak każde inne polecenie). Z tego powodu użyję grep, a nie [polecenia.

Po prostu zrób:

if grep -q foo <<<"$string"; then
    echo "It's there"
fi

Teraz, gdy myślisz o iftestowaniu statusu wyjścia polecenia, które następuje po nim (wraz z dwukropkiem), dlaczego nie ponownie rozważyć źródła testowanego ciągu?

## Instead of this
filetype="$(file -b "$1")"
if grep -q "tar archive" <<<"$filetype"; then
#...

## Simply do this
if file -b "$1" | grep -q "tar archive"; then
#...

Ta -qopcja powoduje, że grep nic nie wypisuje, ponieważ chcemy tylko kodu powrotu. <<<sprawia, że ​​powłoka rozszerza następne słowo i używa go jako danych wejściowych do polecenia, wersji jednego wiersza <<dokumentu tutaj (nie jestem pewien, czy jest to standard czy baszizm).


7
nazywa się je tutaj strunami (3.6.7) Wierzę, że to bashizm
alex.pilon

10
można również zastosować proces podstawiania if grep -q foo <(echo somefoothing); then
larsr,

Pamiętaj, że nie echomożna tego przenieść, jeśli przekazujesz zmienną, użyj printf '%s' "$stringzamiast tego.
nyuszika7h

5
Koszt tego jest bardzo drogi: zrobienie grep -q foo <<<"$mystring"widelca implikuje 1 i jest bashizmem i echo $mystring | grep -q fooimplikuje 2 widelce (jeden do fajki, a drugi do biegania /path/to/grep)
F. Hauri

1
@BrunoBronosky echobez flag może nadal mieć nieoczekiwane problemy z przenośnością, jeśli ciąg argumentu zawiera sekwencje odwrotnego ukośnika. echo "nope\c"oczekuje się, że na niektórych platformach będzie działać tak, jak echo -e "nope"na innych. printf '%s' "nope"vsprintf '%s\n' 'nope\c'
tripleee

92

Najlepsza jest zaakceptowana odpowiedź, ale ponieważ jest na to więcej niż jeden sposób, oto inne rozwiązanie:

if [ "$string" != "${string/foo/}" ]; then
    echo "It's there!"
fi

${var/search/replace}jest $varz pierwszą instancją searchzastąpioną przez replace, jeśli zostanie znaleziona (nie zmienia się $var). Jeśli spróbujesz zastąpić foonic, a łańcuch się zmienił, to oczywiście fooznaleziono.


5
powyższe rozwiązanie ephemient:> `if [" $ string "! =" $ {string / foo /} "]; potem echo „Jest tam!” fi` jest użyteczny, gdy używasz popiołu z BusyBox . Akceptowane rozwiązanie nie działa z BusyBox, ponieważ niektóre wyrażenia regularne bash nie są zaimplementowane.
TPoschel,

1
nierówność różnicy. Dość dziwna myśl! Uwielbiam to
nitinr708

1
chyba że twój ciąg jest „foo”
venimus

Sam napisałem to samo rozwiązanie (ponieważ mój tłumacz nie przyjąłby najlepszych odpowiedzi), a potem zacząłem szukać lepszego, ale znalazłem to!
BuvinJ

56

Istnieje więc wiele przydatnych rozwiązań tego pytania - ale które jest najszybsze / zużywa najmniej zasobów?

Powtórzone testy przy użyciu tej ramki:

/usr/bin/time bash -c 'a=two;b=onetwothree; x=100000; while [ $x -gt 0 ]; do TEST ; x=$(($x-1)); done'

Wymiana TESTU za każdym razem:

[[ $b =~ $a ]]           2.92 user 0.06 system 0:02.99 elapsed 99% CPU

[ "${b/$a//}" = "$b" ]   3.16 user 0.07 system 0:03.25 elapsed 99% CPU

[[ $b == *$a* ]]         1.85 user 0.04 system 0:01.90 elapsed 99% CPU

case $b in *$a):;;esac   1.80 user 0.02 system 0:01.83 elapsed 99% CPU

doContain $a $b          4.27 user 0.11 system 0:04.41 elapsed 99%CPU

(doContain był w odpowiedzi F. Houri)

A dla chichotów:

echo $b|grep -q $a       12.68 user 30.86 system 3:42.40 elapsed 19% CPU !ouch!

Zatem prosta opcja podstawienia jest przewidywalnie wygrana, czy to w rozszerzonym teście, czy w przypadku. Obudowa jest przenośna.

Wypompowanie do 100000 greps jest przewidywalnie bolesne! Stara zasada korzystania z zewnętrznych narzędzi bez potrzeby ma zastosowanie.


4
Neat Benchmark. Przekonał mnie do użycia [[ $b == *$a* ]].
Szalony fizyk

2
Jeśli czytam to poprawnie, casewygrywa przy najmniejszym całkowitym zużyciu czasu. Po tym jednak brakuje Ci gwiazdki $b in *$a. Otrzymuję nieco szybsze wyniki [[ $b == *$a* ]]niż casez poprawionym błędem, ale oczywiście może to zależeć również od innych czynników.
tripleee

1
ideone.com/5roEVt naprawił mój eksperyment z kilkoma dodatkowymi błędami i testami dla innego scenariusza (gdzie łańcuch nie jest faktycznie obecny w dłuższym łańcuchu). Wyniki są w dużej mierze podobne; [[ $b == *$a* ]]jest szybki i caseprawie tak szybki (i przyjemnie zgodny z POSIX).
tripleee

28

Działa to również:

if printf -- '%s' "$haystack" | egrep -q -- "$needle"
then
  printf "Found needle in haystack"
fi

A negatywny test to:

if ! printf -- '%s' "$haystack" | egrep -q -- "$needle"
then
  echo "Did not find needle in haystack"
fi

Podejrzewam, że ten styl jest nieco bardziej klasyczny - mniej zależny od cech powłoki Bash.

--Argument jest czysta paranoja POSIX, używane do zabezpieczone przed ciągach podobnych do opcji, takich jak --abcczy -a.

Uwaga: W ciasnej pętli kod ten będzie znacznie wolniejszy niż przy użyciu wewnętrznych funkcji powłoki Bash, ponieważ jeden (lub dwa) osobne procesy zostaną utworzone i połączone za pomocą potoków.


5
... ale OP nie mówi, która wersja bash; np. starsze bash (takie jak solaris często) mogą nie zawierać tych nowszych funkcji bash. (Natrafiłem na ten konkretny problem (nie zaimplementowano dopasowania wzorca bashu) na solaris w / bash 2.0)
Michael

2
echojest nieprzenośny, powinieneś użyć printf '%s' "$haystackzamiast tego.
nyuszika7h

2
Nie, po prostu unikaj echocałkowicie niczego poza dosłownym tekstem bez znaków ucieczki, które nie zaczynają się od -. Może to działać dla Ciebie, ale nie jest przenośne. Nawet bash echozachowa się inaczej w zależności od tego, czy xpg_echoopcja jest ustawiona. PS: Zapomniałem zamknąć podwójny cytat w poprzednim komentarzu.
nyuszika7h

1
@kevinarpe Nie jestem pewien, --nie jest wymieniony w specyfikacji POSIX dlaprintf , ale i tak powinieneś użyć printf '%s' "$anything", aby uniknąć problemów, jeśli $anythingzawiera %znak.
nyuszika7h

1
@kevinarpe Na tej podstawie prawdopodobnie tak jest.
nyuszika7h

21

Przykłady Bash 4+. Uwaga: niestosowanie cudzysłowów spowoduje problemy, gdy słowa zawierają spacje itp. Zawsze cytuj w Bash, IMO.

Oto kilka przykładów Bash 4+:

Przykład 1, sprawdź ciąg „tak” (bez rozróżniania wielkości liter):

    if [[ "${str,,}" == *"yes"* ]] ;then

Przykład 2, sprawdź ciąg „tak” (bez rozróżniania wielkości liter):

    if [[ "$(echo "$str" | tr '[:upper:]' '[:lower:]')" == *"yes"* ]] ;then

Przykład 3, sprawdź ciąg „tak” (rozróżniana jest wielkość liter):

     if [[ "${str}" == *"yes"* ]] ;then

Przykład 4, sprawdź ciąg „tak” (rozróżniana jest wielkość liter):

     if [[ "${str}" =~ "yes" ]] ;then

Przykład 5, dopasowanie ścisłe (wielkość liter ma znaczenie):

     if [[ "${str}" == "yes" ]] ;then

Przykład 6, dopasowanie ścisłe (bez rozróżniania wielkości liter):

     if [[ "${str,,}" == "yes" ]] ;then

Przykład 7, dopasowanie ścisłe:

     if [ "$a" = "$b" ] ;then

Przykład 8, dopasowanie do znaku wieloznacznego .ext (bez rozróżniania wielkości liter):

     if echo "$a" | egrep -iq "\.(mp[3-4]|txt|css|jpg|png)" ; then

Cieszyć się.


17

Co powiesz na to:

text="   <tag>bmnmn</tag>  "
if [[ "$text" =~ "<tag>" ]]; then
   echo "matched"
else
   echo "not matched"
fi

2
= ~ służy do dopasowywania wyrażeń regularnych, a zatem jest zbyt potężne, aby mogło służyć OP.

16

Jak wspomniał Paweł w swoim porównaniu wyników:

if echo "abcdefg" | grep -q "bcdef"; then
    echo "String contains is true."
else
    echo "String contains is not true."
fi

Jest to zgodne z POSIX, takie jak „case” $ string ”w” odpowiedzi dostarczonej przez Marcusa , ale jest nieco łatwiejsze do odczytania niż odpowiedź instrukcji case. Zauważ też, że będzie to znacznie wolniejsze niż użycie instrukcji case. Jak zauważył Paul, nie używaj go w pętli.



9
[[ $string == *foo* ]] && echo "It's there" || echo "Couldn't find"

Dodam, że echo "Couldn't findinstrukcja na końcu jest fajną sztuczką, aby zwrócić 0 statusów wyjścia dla tych pasujących poleceń.
nicodjimenez

@nicodjimenez nie można już kierować statusu wyjścia za pomocą tego rozwiązania. Status wyjścia jest pochłaniany przez komunikaty o stanie ...
Jahid

1
Dokładnie o to mi chodziło ... Jeśli go nie masz || echo "Couldn't find", zwrócisz status wyjścia z błędu, jeśli nie ma zgodności, czego możesz nie chcieć, jeśli korzystasz z potoku CI, na przykład, aby wszystkie komendy zwróciły bez błędów statusy wyjścia
nicodjimenez 17.10.17

9

Jeden jest:

[ $(expr $mystring : ".*${search}.*") -ne 0 ] && echo 'yes' ||  echo 'no'

2
exprjest jednym z tych noży szwajcarskich, które zwykle potrafią zrobić wszystko, co trzeba, kiedy już wymyślisz, jak to zrobić, ale po wdrożeniu nigdy nie będziesz pamiętać, dlaczego i jak to robi, więc nigdy więcej go nie dotykaj i miej nadzieję, że nigdy nie przestanie robić tego, co robi.
Michael

@AloisMahdal Nigdy nie głosowałem w dół, po prostu postuluję, dlaczego otrzymano głosy negatywne. Uwaga ostrzegawcza. Używam expr, w rzadkich przypadkach, gdy przenośność uniemożliwia użycie bash (np. Niespójne zachowanie w starszych wersjach), tr (niespójne wszędzie) lub sed (czasami zbyt wolno). Ale z własnego doświadczenia, za każdym razem, gdy czytam ponownie te expr-ismy, muszę wrócić do strony podręcznika. Chciałbym więc skomentować, że każde użycie exprkomentowane ...
Michael

1
Był czas, kiedy jedyne, co miałeś, to oryginalna skorupa Bourne'a. Brakowało niektórych często wymaganych funkcji, więc narzędzia takie jak expri testzostały zaimplementowane do ich wykonania. W dzisiejszych czasach są zwykle lepsze narzędzia, wiele z nich wbudowanych w każdą nowoczesną powłokę. Chyba testwciąż tam jest, ale chyba nikogo nie brakuje expr.
tripleee

6

Lubię sed.

substr="foo"
nonsub="$(echo "$string" | sed "s/$substr//")"
hassub=0 ; [ "$string" != "$nonsub" ] && hassub=1

Edycja, logika:

  • Użyj sed, aby usunąć wystąpienie podłańcucha z łańcucha

  • Jeśli nowy ciąg różni się od starego, istnieje podciąg


2
Dodaj wyjaśnienie. Nadanie podstawowej logice jest ważniejsze niż podanie kodu, ponieważ pomaga OP i innym czytelnikom samodzielnie rozwiązać ten i podobne problemy
Zulan

5

grep -q jest przydatny do tego celu.

To samo przy użyciu awk:

string="unix-bash 2389"
character="@"
printf '%s' "$string" | awk -vc="$character" '{ if (gsub(c, "")) { print "Found" } else { print "Not Found" } }'

Wynik:

Nie znaleziono

string="unix-bash 2389"
character="-"
printf '%s' "$string" | awk -vc="$character" '{ if (gsub(c, "")) { print "Found" } else { print "Not Found" } }'

Wynik:

Znaleziony

Oryginalne źródło: http://unstableme.blogspot.com/2008/06/bash-search-letter-in-string-awk.html


3
echojest nieprzenośny, powinieneś użyć printf '%s' "$string"zamiast tego. Edytuję odpowiedź, ponieważ wydaje się, że użytkownik już nie istnieje.
nyuszika7h

W jaki sposób echonie jest przenośny, @ nyuszika7h?
starbeamrainbowlabs

5

Okazało się, że potrzebuję tej funkcji dość często, więc używam domowej powłoki w moim .bashrcpodobnym, co pozwala mi używać go tak często, jak potrzebuję, z łatwą do zapamiętania nazwą:

function stringinstring()
{
    case "$2" in
       *"$1"*)
          return 0
       ;;
    esac
    return 1
}

Aby sprawdzić, czy $string1(powiedzmy, abc ) jest zawarty w $string2(powiedzmy, 123abcABC ), muszę po prostu uruchomić stringinstring "$string1" "$string2"i sprawdzić wartość zwracaną, na przykład

stringinstring "$str1" "$str2"  &&  echo YES  ||  echo NO

[["$ str" == $ substr ]] && echo TAK || echo NIE
elyase,

Jestem pewien, że xhack jest wymagany tylko w przypadku bardzo starych pocisków.
nyuszika7h

5

Mój plik .bash_profile i sposób użycia grep:

Jeśli zmienna środowiskowa PATH zawiera moje dwa binkatalogi, nie dołączaj ich,

# .bash_profile
# Get the aliases and functions
if [ -f ~/.bashrc ]; then
    . ~/.bashrc
fi

U=~/.local.bin:~/bin

if ! echo "$PATH" | grep -q "home"; then
    export PATH=$PATH:${U}
fi

1
Czy to odpowiedź?
codeforester

głosować. dlaczego warto zadać sobie trud, aby dowiedzieć się, jak działa Bash, gdy grep, o wiele potężniejszy, będzie więcej niż prawdopodobne. również przedłużyć go nieco dalej, dopasowując z listą wzorców: grep -q -E 'pattern1|...|patternN'.
Mohamed Bana,

4

Rozszerzenie odpowiedzi na pytanie tutaj Jak sprawdzić, czy łańcuch zawiera inny ciąg w POSIX sh? :

To rozwiązanie działa ze znakami specjalnymi:

# contains(string, substring)
#
# Returns 0 if the specified string contains the specified substring,
# otherwise returns 1.
contains() {
    string="$1"
    substring="$2"

    if echo "$string" | $(type -p ggrep grep | head -1) -F -- "$substring" >/dev/null; then
        return 0    # $substring is in $string
    else
        return 1    # $substring is not in $string
    fi
}

contains "abcd" "e" || echo "abcd does not contain e"
contains "abcd" "ab" && echo "abcd contains ab"
contains "abcd" "bc" && echo "abcd contains bc"
contains "abcd" "cd" && echo "abcd contains cd"
contains "abcd" "abcd" && echo "abcd contains abcd"
contains "" "" && echo "empty string contains empty string"
contains "a" "" && echo "a contains empty string"
contains "" "a" || echo "empty string does not contain a"
contains "abcd efgh" "cd ef" && echo "abcd efgh contains cd ef"
contains "abcd efgh" " " && echo "abcd efgh contains a space"

contains "abcd [efg] hij" "[efg]" && echo "abcd [efg] hij contains [efg]"
contains "abcd [efg] hij" "[effg]" || echo "abcd [efg] hij does not contain [effg]"

contains "abcd *efg* hij" "*efg*" && echo "abcd *efg* hij contains *efg*"
contains "abcd *efg* hij" "d *efg* h" && echo "abcd *efg* hij contains d *efg* h"
contains "abcd *efg* hij" "*effg*" || echo "abcd *efg* hij does not contain *effg*"

Test contains "-n" "n"nie działa tutaj, ponieważ echo -npołknie -nopcję jako opcję! Popularną poprawką jest użycie printf "%s\n" "$string"zamiast tego.
joeytwiddle

doskonale rozwiązany ciąg zawierał spacje
dengApro

4

Ponieważ pytanie POSIX / BusyBox jest zamknięte bez podania właściwej odpowiedzi (IMHO), opublikuję odpowiedź tutaj.

Najkrótsza możliwa odpowiedź to:

[ ${_string_##*$_substring_*} ] || echo Substring found!

lub

[ "${_string_##*$_substring_*}" ] || echo 'Substring found!'

Zauważ, że podwójny skrót jest obowiązkowy w przypadku niektórych powłok ( ash). Powyżej oceni, [ stringvalue ]kiedy nie zostanie znaleziony podciąg. Nie zwraca błędu. Po znalezieniu podłańcucha wynik jest pusty i ocenia [ ]. Spowoduje to wyrzucenie kodu błędu 1, ponieważ ciąg jest całkowicie podstawiony (z powodu* ).

Najkrótsza bardziej powszechna składnia:

[ -z "${_string_##*$_substring_*}" ] && echo 'Substring found!'

lub

[ -n "${_string_##*$_substring_*}" ] || echo 'Substring found!'

Inny:

[ "${_string_##$_substring_}" != "$_string_" ] && echo 'Substring found!'

lub

[ "${_string_##$_substring_}" = "$_string_" ] || echo 'Substring found!'

Zwróć uwagę na pojedynczy znak równości!


3

Dokładne dopasowanie słów:

string='My long string'
exactSearch='long'

if grep -E -q "\b${exactSearch}\b" <<<${string} >/dev/null 2>&1
  then
    echo "It's there"
  fi

3

Spróbuj oobash.

Jest to biblioteka ciągów w stylu OO dla Bash 4. Obsługuje niemieckie umlauty. Jest napisane w języku Bash.

Wiele funkcji dostępne są: -base64Decode, -base64Encode, -capitalize, -center, -charAt, -concat, -contains, -count, -endsWith, -equals, -equalsIgnoreCase, -reverse, -hashCode, -indexOf, -isAlnum, -isAlpha, -isAscii, -isDigit, -isEmpty, -isHexDigit, -isLowerCase, -isSpace, -isPrintable, -isUpperCase, -isVisible, -lastIndexOf, -length, -matches, -replaceAll, -replaceFirst, -startsWith, -substring, -swapCase, -toLowerCase, -toString, -toUpperCase, -trim, i -zfill.

Spójrz na zawiera przykład:

[Desktop]$ String a testXccc
[Desktop]$ a.contains tX
true
[Desktop]$ a.contains XtX
false

oobash jest dostępny na Sourceforge.net .


2

Używam tej funkcji (jedna zależność nie jest uwzględniona, ale oczywista). Przechodzi testy pokazane poniżej. Jeśli funkcja zwraca wartość> 0, łańcuch został znaleziony. Zamiast tego możesz równie łatwo zwrócić 1 lub 0.

function str_instr {
   # Return position of ```str``` within ```string```.
   # >>> str_instr "str" "string"
   # str: String to search for.
   # string: String to search.
   typeset str string x
   # Behavior here is not the same in bash vs ksh unless we escape special characters.
   str="$(str_escape_special_characters "${1}")"
   string="${2}"
   x="${string%%$str*}"
   if [[ "${x}" != "${string}" ]]; then
      echo "${#x} + 1" | bc -l
   else
      echo 0
   fi
}

function test_str_instr {
   str_instr "(" "'foo@host (dev,web)'" | assert_eq 11
   str_instr ")" "'foo@host (dev,web)'" | assert_eq 19
   str_instr "[" "'foo@host [dev,web]'" | assert_eq 11
   str_instr "]" "'foo@host [dev,web]'" | assert_eq 19
   str_instr "a" "abc" | assert_eq 1
   str_instr "z" "abc" | assert_eq 0
   str_instr "Eggs" "Green Eggs And Ham" | assert_eq 7
   str_instr "a" "" | assert_eq 0
   str_instr "" "" | assert_eq 0
   str_instr " " "Green Eggs" | assert_eq 6
   str_instr " " " Green "  | assert_eq 1
}

Napięcie! Jaka jest zależność?
Peter Mortensen

Funkcja „str_escape_special_characters”. Znajduje się w pliku GitHub arcshell_str.sh. arcshell.io cię tam zaprowadzi.
Ethan Post


0
msg="message"

function check {
    echo $msg | egrep [abc] 1> /dev/null

    if [ $? -ne 1 ];
    then 
        echo "found" 
    else 
        echo "not found" 
    fi
}

check

Znajduje to każde wystąpienie a, b lub c


0

Ogólny przykład stogu siana z igłą podąża za zmiennymi

#!/bin/bash

needle="a_needle"
haystack="a_needle another_needle a_third_needle"
if [[ $haystack == *"$needle"* ]]; then
    echo "needle found"
else
    echo "needle NOT found"
fi
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.