Odpowiedzi:
Możesz użyć formularza, w ${VAR/subs}
którym VAR
znajduje się większy ciąg i
subs
jest to podciąg, który próbujesz znaleźć:
my_string=abc
substring=ab
if [ "${my_string/$substring}" = "$my_string" ] ; then
echo "${substring} is not in ${my_string}"
else
echo "${substring} was found in ${my_string}"
fi
Działa ${VAR/subs}
to, ponieważ jest równe, $VAR
ale z pierwszym wystąpieniem ciągu subs
usuniętego, w szczególności jeśli $VAR
nie zawiera słowa subs
, nie zostanie zmodyfikowane.
ab was found in abc
, ale jeśli użyję substring=z
, dostajęz was found in abc
ab is not in abc
. Ale z was found in abc
… To zabawne: D
[[ "bcd" =~ "ab" ]]
[[ "abc" =~ "ab" ]]
nawiasy klamrowe są do testu, a ponieważ są to podwójne nawiasy klamrowe, może to oznaczać dodatkowe testy =~
.
Więc możesz użyć tego formularza
var1="ab"
var2="bcd"
if [[ "$var2" =~ "$var1" ]]; then
echo "pass"
else
echo "fail"
fi
Edycja: poprawione „= ~”, odwróciło się.
fail
z tymi parametrami:var2="abcd"
[[ $string =~ $substring ]]
. Zaktualizowałem odpowiedź.
Używanie wzorców nazw plików bash (zwanych również wzorcami „glob”)
substr=ab
[[ abc == *"$substr"* ]] && echo yes || echo no # yes
[[ bcd == *"$substr"* ]] && echo yes || echo no # no
Poniższe dwa podejścia będą działać w dowolnym środowisku zgodnym z POSIX, nie tylko w bash:
substr=ab
for s in abc bcd; do
if case ${s} in *"${substr}"*) true;; *) false;; esac; then
printf %s\\n "'${s}' contains '${substr}'"
else
printf %s\\n "'${s}' does not contain '${substr}'"
fi
done
substr=ab
for s in abc bcd; do
if printf %s\\n "${s}" | grep -qF "${substr}"; then
printf %s\\n "'${s}' contains '${substr}'"
else
printf %s\\n "'${s}' does not contain '${substr}'"
fi
done
Oba powyższe dane wyjściowe:
'abc' contains 'ab'
'bcd' does not contain 'ab'
Ta pierwsza ma tę zaletę, że nie spawnuje osobnego grep
procesu.
Zauważ, że używam printf %s\\n "${foo}"
zamiast, echo "${foo}"
ponieważ echo
może się mangować, ${foo}
jeśli zawiera odwrotne ukośniki.
xrandr
nazw monitorów przechowywanych w zmiennej. +1 i witamy w klubie 1K rep :)
To najbardziej przenośne rozwiązanie, działa nawet na starych pociskach Bourne'a i Korn
#!/bin/bash
case "abcd" in
*$1*) echo "It's a substring" ;;
*) echo "Not a substring" ;;
esac
Przykładowy przebieg:
$ ./case_substr.sh "ab"
It's a substring
$ ./case_substr.sh "whatever"
Not a substring
Pamiętaj, że nie musisz specjalnie używać echo
, możesz używać exit 1
i exit 0
oznaczać sukces lub porażkę.
Możemy również stworzyć funkcję (w razie potrzeby w dużych skryptach) z określonymi wartościami zwracanymi (0 w przypadku dopasowania, 1 w przypadku braku dopasowania):
$ ./substring_function.sh
ab is substring
$ cat substring_function.sh
#!/bin/sh
is_substring(){
case "$2" in
*$1*) return 0;;
*) return 1;;
esac
}
main(){
if is_substring "ab" "abcdefg"
then
echo "ab is substring"
fi
}
main $@
$ grep -q 'ab' <<< "abcd" && echo "it's a substring" || echo "not a substring"
it's a substring
To szczególne podejście jest przydatne w przypadku instrukcji if-else w bash
. Również w większości przenośne
$ awk '$0~/ab/{print "it is a substring"}' <<< "abcd"
it is a substring
$ python -c 'import sys;sys.stdout.write("it is a substring") if "ab" in sys.stdin.read() else exit(1)' <<< "abcd"
it is a substring
$ ruby -e ' puts "is substring" if ARGV[1].include? ARGV[0]' "ab" "abcdef"
is substring
Przeszkadza [[
i "
:
[[ $a == z* ]] # True if $a starts with an "z" (pattern matching).
[[ $a == "z*" ]] # True if $a is equal to z* (literal matching).
[ $a == z* ] # File globbing and word splitting take place.
[ "$a" == "z*" ] # True if $a is equal to z* (literal matching).
Tak jak powiedział @glenn_jackman, ale pamiętaj, że jeśli owiniesz cały drugi termin podwójnymi cudzysłowami, przełączy on test na dosłowne dopasowanie.
Podobne do odpowiedzi edwina, ale z ulepszoną przenośnością dla posix & ksh i odrobiną mniej hałasu niż Richard:
substring=ab
string=abc
if [ "$string" != "${string%$substring*}" ]; then
echo "$substring IS in $string"
else
echo "$substring is NOT in $string"
fi
string=bcd
if [ "$string" != "${string%$substring*}" ]; then
echo "$string contains $substring"
else
echo "$string does NOT contain $substring"
fi
Wynik:
abc contains ab
bcd does NOT contain ab
echo
instrukcji. Ponieważ dostajęab is not in abc