Bash에서 문자열에 하위 문자열이 포함되어 있는지 확인하는 방법
Bash에 문자열이 있습니다.
string="My string"
다른 문자열이 포함되어 있는지 테스트하려면 어떻게 해야 합니까?
if [ $string ?? 'foo' ]; then
echo "It's there!"
fi
에▁where디??
알 수 없는 제 교환원입니다.사용하시겠습니까?echo
그리고.grep
?
if echo "$string" | grep 'foo'; then
echo "It's there!"
fi
그것은 좀 어설프게 보입니다.
대괄호를 두 개 사용하는 경우 사례 문 밖에서도 Marcus의 답변(* 와일드카드)을 사용할 수 있습니다.
string='My long string'
if [[ $string == *"My long"* ]]; then
echo "It's there!"
fi
큰따옴표와 큰따옴표 .*
와일드카드는 외부에 있어야 합니다. 단순 또간한비즉연사가용다니됩자산교단한즉,==
연산자 regex가 )=~
.
정규식 접근 방식을 선호하는 경우:
string='My string';
if [[ $string =~ "My" ]]; then
echo "It's there!"
fi
if 문을 사용하는 것에 대해서는 잘 모르겠지만 사례 문을 사용하면 비슷한 효과를 얻을 수 있습니다.
case "$string" in
*foo*)
# Do stuff
;;
esac
stringContain
가능 구분 안 함)
스택 오버플로 답변에서 Bash에 대해 주로 알 수 있듯이, 이 게시물의 맨 아래에 사례 독립적인 Bash 기능을 게시했습니다.
어쨌든, 내 것은.
호환되는 답변
Bash 관련 기능을 사용하는 답변이 이미 많기 때문에 BusyBox와 같은 기능이 떨어지는 셸에서 작업하는 방법이 있습니다.
[ -z "${string##*$reqsubstr*}" ]
실제로 이는 다음과 같은 이점을 제공됩니다.
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
이 테스트는 Bash, Dash, KornShell에서 수행되었습니다(ksh
) 및 재(BusyBox), 결과는 항상 다음과 같습니다.
String 'echo "My string"' contain substring: 'o "M'.
String 'echo "My string"' don't contain substring: 'alt'.
String 'echo "My string"' contain substring: 'str'.
하나의 함수로
@EeroAaltonen이 질문한 것처럼, 여기 동일한 셸에서 테스트된 동일한 데모 버전이 있습니다.
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
}
그러면:
$ 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'.
주의: 따옴표 및/또는 따옴표를 이스케이프하거나 이중으로 묶어야 합니다.
$ 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'.
단순함수
이것은 BusyBox, Dash, 그리고 물론 Bash에서 테스트되었습니다.
stringContain() { [ -z "${2##*$1*}" ]; }
그러면 이제:
$ 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
또는 @Sjlver가 지적한 것처럼 제출된 문자열이 비어 있을 수 있는 경우 함수는 다음과 같습니다.
stringContain() { [ -z "${2##*$1*}" ] && [ -z "$1" -o -n "$2" ]; }
또는 에이드리언 귄터의 논평에서 제안된 것처럼, 피합니다.-o
스위치:
stringContain() { [ -z "${2##*$1*}" ] && { [ -z "$1" ] || [ -n "$2" ];};}
최종(단순) 기능:
그리고 테스트를 뒤집어서 테스트 속도를 높일 수 있습니다.
stringContain() { [ -z "$1" ] || { [ -z "${2##*$1*}" ] && [ -n "$2" ];};}
빈 문자열 포함:
$ if stringContain '' ''; then echo yes; else echo no; fi
yes
$ if stringContain 'o "M' ''; then echo yes; else echo no; fi
no
대/소문자 구분 없음(대/소문자 구분 없음!)
대소문자를 구분하지 않고 문자열을 테스트하는 경우 각 문자열을 소문자로 변환하기만 하면 됩니다.
stringContain() {
local _lc=${2,,}
[ -z "$1" ] || { [ -z "${_lc##*${1,,}*}" ] && [ -n "$2" ] ;} ;}
확인:
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
셸 스크립팅은 언어가 아니라 명령 모음에 가깝다는 점을 기억해야 합니다.은 이 이 본적으로언이가당 "생어능다"를 따라야 한다고 생각합니다.if
[
는또.[[
두 명령 모두 성공 또는 실패를 나타내는 종료 상태를 반환하는 명령입니다(다른 모든 명령과 마찬가지로).그런 이유로 제가 사용합니다.grep
그리고 그가 아닌[
지휘권
그냥 하기:
if grep -q foo <<<"$string"; then
echo "It's there"
fi
이제 당신이 생각하고 있는 것은if
다음 명령의 종료 상태를 테스트할 때(반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반-반
## 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
#...
그-q
옵션을 사용하면 grep은 아무것도 출력하지 않습니다. 왜냐하면 우리는 리턴 코드만 원합니다. <<<
. 는 셸이다 한 줄 입니다.<<
여기 문서(이것이 표준인지 바시즘인지 잘 모르겠습니다.)
허용된 답변이 최선이지만, 이를 위한 방법은 여러 가지가 있으므로 다음과 같은 다른 솔루션이 있습니다.
if [ "$string" != "${string/foo/}" ]; then
echo "It's there!"
fi
${var/search/replace}
이라$var
들예면를의 첫 search
로대된으로 replace
발견된다면 (은 변하지 $var
하려고 할 에는 .foo
아닌 것에 , , 아것도끈, 고바분고었, 히명럼그뀌이.foo
발견되었습니다.
따라서 이 질문에 대한 유용한 솔루션은 여러 가지가 있습니다. 하지만 어떤 솔루션이 가장 빠르고 리소스를 가장 적게 사용합니까?
이 프레임을 사용한 반복 테스트:
/usr/bin/time bash -c 'a=two;b=onetwothree; x=100000; while [ $x -gt 0 ]; do TEST ; x=$(($x-1)); done'
매번 테스트 교체:
[[ $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은 F에 있었습니다.Houri의 대답)
그리고 킥킥거리는 경우:
echo $b|grep -q $a 12.68 user 30.86 system 3:42.40 elapsed 19% CPU !ouch!
따라서 단순 대체 옵션은 연장된 테스트 또는 사례에서 예측 가능하게 승리합니다.그 케이스는 휴대용입니다.
100,000greps로 출력하는 것은 예측할 수 있는 고통입니다.불필요하게 외부 유틸리티를 사용하는 것에 대한 오래된 규칙은 사실입니다.
Bash 4개 이상의 예.참고: 따옴표를 사용하지 않으면 단어에 공백 등이 포함될 때 문제가 발생합니다.항상 바시, IMO로 인용합니다.
다음은 Bash 4+의 몇 가지 예입니다.
예 1, 문자열에서 '예'를 확인합니다(대소문자 구분 안 함).
if [[ "${str,,}" == *"yes"* ]] ;then
예 2, 문자열에서 '예'를 확인합니다(대소문자 구분 안 함).
if [[ "$(echo "$str" | tr '[:upper:]' '[:lower:]')" == *"yes"* ]] ;then
예 3, 문자열에서 '예'를 확인합니다(대소문자 구분).
if [[ "${str}" == *"yes"* ]] ;then
예 4, 문자열에서 '예'를 확인합니다(대소문자 구분).
if [[ "${str}" =~ "yes" ]] ;then
예 5, 정확한 일치(대소문자 구분):
if [[ "${str}" == "yes" ]] ;then
예 6, 정확한 일치(대소문자 구분 안 함):
if [[ "${str,,}" == "yes" ]] ;then
예 7, 정확한 일치:
if [ "$a" = "$b" ] ;then
예 8, 와일드카드 match.ext(대문자와 소문자를 구분하지 않음):
if echo "$a" | egrep -iq "\.(mp[3-4]|txt|css|jpg|png)" ; then
예 9: 대/소문자를 구분하는 문자열에 grep를 사용합니다.
if echo "SomeString" | grep -q "String"; then
예 10, 문자열 대소문자 구분 없이 grep 사용:
if echo "SomeString" | grep -iq "string"; then
예 11, 와일드카드가 있는 문자열 대소문자를 구분하지 않는 경우 grep를 사용합니다.
if echo "SomeString" | grep -iq "Some.*ing"; then
예 12, 이중 해시를 사용하여 비교합니다(변수가 비어 있으면 거짓 긍정 등이 발생할 수 있는 경우)(대소문자 구분).
if [[ ! ${str##*$substr*} ]] ;then #found
즐거운 시간 되세요.
이 기능은 다음과 같습니다.
if printf -- '%s' "$haystack" | egrep -q -- "$needle"
then
printf "Found needle in haystack"
fi
음성 테스트는 다음과 같습니다.
if ! printf -- '%s' "$haystack" | egrep -q -- "$needle"
then
echo "Did not find needle in haystack"
fi
저는 이 스타일이 좀 더 고전적이라고 생각합니다. Bash 쉘의 특징에 덜 의존합니다.
그--
POSIX 보호하는 데 됩니다. 를 들어 POSIX는 POSIX와 유사합니다.--abc
또는-a
.
참고: 엄격한 루프에서 이 코드는 내부 Bash 셸 기능을 사용하는 것보다 훨씬 느립니다. 하나 또는 두 개의 별도 프로세스가 생성되고 파이프를 통해 연결되기 때문입니다.
if echo "abcdefg" | grep -q "bcdef"; then
echo "String contains is true."
else
echo "String contains is not true."
fi
이것은 Marcus가 제공한 답변의 'case "$string"과 같이 POSIX를 준수하지만, 사례 설명 답변보다 읽기가 약간 더 쉽습니다.또한 이 작업은 사례 문을 사용하는 것보다 훨씬 느립니다.Paul이 지적했듯이, 그것을 반복해서 사용하지 마세요.
이거 어때:
text=" <tag>bmnmn</tag> "
if [[ "$text" =~ "<tag>" ]]; then
echo "matched"
else
echo "not matched"
fi
인정된 답은 맞지만 읽고 이해하기가 어렵습니다.
검색과 관련된 문제의 경우 항상 $haystack 관용구로 $needle을 사용해야 합니다.
제안된 편집 대기열이 가득 찼기 때문에 다음과 같이 게시합니다.
haystack='There are needles here.'
if [[ "$haystack" == *"needle"* ]]; then
echo "It's there!"
fi
[[ $string == *foo* ]] && echo "It's there" || echo "Couldn't find"
이 스택 오버플로 응답은 공백 및 대시 문자를 트랩할 수 있는 유일한 응답입니다.
# For null cmd arguments checking
to_check=' -t'
space_n_dash_chars=' -'
[[ $to_check == *"$space_n_dash_chars"* ]] && echo found
하나는:
[ $(expr $mystring : ".*${search}.*") -ne 0 ] && echo 'yes' || echo 'no'
정답(IMHO)을 제공하지 않고 POSIX/BusyBox 질문이 종료되었으므로 여기에 답변을 게시합니다.
가장 짧은 대답은 다음과 같습니다.
[ ${_string_##*$_substring_*} ] || echo Substring found!
또는
[ "${_string_##*$_substring_*}" ] || echo 'Substring found!'
이중 해시는 일부 셸에서 의무적으로 사용됩니다(ash
합니다.[ stringvalue ]
하위 문자열을 찾을 수 없는 경우.오류를 반환하지 않습니다.는 비어있고, 는 " 부문자이발결비는평있다니가됩고어과면되견분열다▁when니▁it부▁evaluates 평가합니다.[ ]
경우 에 오류 합니다.*
).
더 일반적인 가장 짧은 구문:
[ -z "${_string_##*$_substring_*}" ] && echo 'Substring found!'
또는
[ -n "${_string_##*$_substring_*}" ] || echo 'Substring found!'
또 다른 하나:
[ "${_string_##$_substring_}" != "$_string_" ] && echo 'Substring found!'
또는
[ "${_string_##$_substring_}" = "$_string_" ] || echo 'Substring found!'
단일 등호를 주의하십시오!
내 .bash_profile 파일 및 grep 사용 방법:
의 두 PATH가 된 bin
# .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
여기에 답변된 질문의 확장 POSIX sh에 다른 문자열이 포함되어 있는지 여부를 어떻게 알 수 있습니까?
이 솔루션은 특수 문자와 함께 작동합니다.
# 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*"
일반적인 Needle Haystack 예제는 변수를 사용하여 다음과 같습니다.
#!/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
grep -q
이 목적에 유용합니다.
를 사용해도 .awk
:
string="unix-bash 2389"
character="@"
printf '%s' "$string" | awk -vc="$character" '{ if (gsub(c, "")) { print "Found" } else { print "Not Found" } }'
출력:
찾을 수 없음
string="unix-bash 2389"
character="-"
printf '%s' "$string" | awk -vc="$character" '{ if (gsub(c, "")) { print "Found" } else { print "Not Found" } }'
출력:
찾았다
원본 출처: http://unstableme.blogspot.com/2008/06/bash-search-letter-in-string-awk.html
저는 sed 좋아해요.
substr="foo"
nonsub="$(echo "$string" | sed "s/$substr//")"
hassub=0 ; [ "$string" != "$nonsub" ] && hassub=1
편집, 논리:
문자열에서 하위 문자열 인스턴스를 제거하는 데 사용됩니다.
새 문자열이 이전 문자열과 다른 경우 하위 문자열이 존재합니다.
저는 이 기능이 꽤 자주 필요하다는 것을 알게 되었고, 그래서 저는 집에서 만든 쉘 기능을 사용하고 있습니다..bashrc
다음과 같이 쉽게 기억할 수 있는 이름을 사용하여 필요한 만큼 자주 재사용할 수 있습니다.
function stringinstring()
{
case "$2" in
*"$1"*)
return 0
;;
esac
return 1
}
여 부스▁if 여부를 테스트하는 $string1
(예: abc)가 포함되어 있습니다.$string2
(예: 123abcABC) 저는 그냥 뛰면 됩니다.stringinstring "$string1" "$string2"
그리고 반환 값을 확인합니다. 예를 들어,
stringinstring "$str1" "$str2" && echo YES || echo NO
case $string in (*foo*)
# Do stuff
esac
이는 https://stackoverflow.com/a/229585/11267590 과 동일한 답변입니다.하지만 심플한 스타일과 POSIX 준수.
정확한 단어 일치:
string='My long string'
exactSearch='long'
if grep -E -q "\b${exactSearch}\b" <<<${string} >/dev/null 2>&1
then
echo "It's there"
fi
오바쉬 해보세요.
Bash 4용 OO 스타일 문자열 라이브러리입니다.그것은 독일산 엄라우트를 지원합니다.그것은 Bash로 쓰여 있습니다.
할 수 있습니다: 많은사수있다니습용할기.-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
,그리고.-zfill
.
다음을 포함하는 예제를 확인하십시오.
[Desktop]$ String a testXccc
[Desktop]$ a.contains tX
true
[Desktop]$ a.contains XtX
false
oobash는 Sourceforge.net 에서 이용할 수 있습니다.
저는 이 기능을 사용합니다(하나의 종속성은 포함되지 않았지만 명백함).아래 표시된 테스트를 통과합니다.함수가 0보다 큰 값을 반환하면 문자열이 발견됩니다.대신 1이나 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
}
논리를 사용하여 보다 간결하게 할 수 있습니다.
#!/bin/bash
# NO MATCH EXAMPLE
string="test"
[[ "$string" == *"foo"* ]] && {
echo "YES"
}
# MATCH EXAMPLE
string="tefoost"
[[ "$string" == *"foo"* ]] && {
echo "YES"
}
msg="message"
function check {
echo $msg | egrep [abc] 1> /dev/null
if [ $? -ne 1 ];
then
echo "found"
else
echo "not found"
fi
}
check
이렇게 하면 a 또는 b 또는 c가 발생하는 것을 확인할 수 있습니다.
jq 포함:
string='My long string'
echo $string | jq -Rr 'select(contains("long"))|"It is there"'
jq에서 가장 어려운 것은 하나의 인용문을 인쇄하는 것입니다.
echo $string | jq --arg quote "'" -Rr 'select(contains("long"))|"It\($quote)s there"'
jq를 사용하여 상태 확인:
if jq -Re 'select(contains("long"))|halt' <<< $string; then
echo "It's there!"
fi
언급URL : https://stackoverflow.com/questions/229551/how-to-check-if-a-string-contains-a-substring-in-bash
'programing' 카테고리의 다른 글
JavaScript의 Zip 배열? (0) | 2023.05.16 |
---|---|
.NET 정수 대 Int16? (0) | 2023.05.16 |
가변 길이 어레이가 C++ 표준의 일부가 아닌 이유는 무엇입니까? (0) | 2023.05.16 |
Windows에서 Git:병합 도구는 어떻게 설정합니까? (0) | 2023.05.16 |
윈도우즈 cmd stdout 및 stderr을 단일 파일로 리디렉션하는 방법은 무엇입니까? (0) | 2023.05.16 |