string 스크립트 - Bash에서 문자열에 하위 문자열이 포함되어 있는지 확인하는 방법




특정 찾기 (18)

배쉬에 문자열이 있습니다.

string="My string"

다른 문자열이 포함되어 있는지 어떻게 테스트 할 수 있습니까?

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

어디 ?? 내 알 수없는 연산자입니다. echo와 grep 합니까?

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

그건 좀 어색해 보인다.


Answers

허용 된 답변이 가장 좋지만 한 가지 이상의 방법이 있으므로 다음과 같은 또 다른 해결책이 있습니다.

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

${var/search/replace}$var 이고, search 의 첫 번째 인스턴스는 replace 로 대체됩니다 ( $var 는 변경되지 않습니다). foo 를 아무것도 대체하려고 시도하지 않고 문자열이 변경된 경우 분명히 foo 가 발견되었습니다.


나는이 함수를 사용한다. 아래에 나와있는 테스트를 통과합니다. 함수가 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 "(" "'[email protected] (dev,web)'" | assert_eq 11
   str_instr ")" "'[email protected] (dev,web)'" | assert_eq 19
   str_instr "[" "'[email protected] [dev,web]'" | assert_eq 11
   str_instr "]" "'[email protected] [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
}

쉘 스크립팅은 언어와 명령 모음의 수가 적다는 것을 기억해야합니다. 본능적으로 당신은이 "언어"가 당신이 [ 또는 [[ . 두 명령 모두 성공 또는 실패를 나타내는 종료 상태를 반환하는 명령입니다 (다른 모든 명령과 동일). 그런 이유로 나는 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 만 리턴 코드를 원하므로 아무 것도 출력하지 않습니다. <<< 는 셸을 다음 단어로 확장하고 << here 문서의 한 줄짜리 버전 인 명령에 대한 입력으로 사용합니다 (표준이든 배쉬시이든간에).


이 스택 오버플로 대답 은 공간 및 대시 문자를 잡는 유일한 방법이었습니다.

# For null cmd arguments checking   
to_check=' -t'
space_n_dash_chars=' -'
[[ $to_check == *"$space_n_dash_chars"* ]] && echo found

호환 가능한 답변

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 , ksh 및 ash (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="[email protected]"
    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" ]; }

또는 Adrian Günter의 의견에 의해 제안 된 것처럼 -o switche를 피하십시오.

stringContain() { [ -z "${2##*$1*}" ] && { [ -z "$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

정규식 접근 방식을 선호하는 경우 :

string='My string';

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

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

바울이 자신의 성과 비교에서 언급 한 것처럼 :

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

이것은 Marcus가 제공 한 'case'$ string "in"답변과 같이 POSIX를 준수하지만 case 문 대답보다 약간 읽기 쉽습니다. 또한 Paul이 지적했듯이 case 문을 사용하는 것보다 훨씬 느릴 수 있습니다. 루프에서 사용하지 마십시오.


이것은 또한 작동합니다 :

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 편집증이며, --abc-a 와 같은 옵션과 비슷한 입력 문자열로부터 보호하는 데 사용됩니다.

참고 : 단단한 루프에서는이 코드가 하나의 (또는 두 개의) 별도 프로세스가 생성되어 파이프를 통해 연결되기 때문에 내부 Bash 쉘 기능을 사용하는 것보다 훨씬 느립니다.


내 .bash_profile 및 PATH에 2 bin 디렉토리가 포함되어있는 경우 어떻게 grep을 사용했는지, 추가하지 마십시오.

# .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

if 문을 사용하는 것은 확실하지 않지만 case 문과 비슷한 효과를 얻을 수 있습니다.

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

일치하는 단어 :

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

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

이 기능이 꽤 자주 필요하다는 것을 알았습니다. 그래서 .bashrc 에서 집에서 만든 셸 함수를 사용하고 있습니다. 이렇게하면 이름을 기억하기 쉽도록 필요할 때 자주 다시 사용할 수 있습니다.

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

$string1 (예 : abc )이 $string2 (예 : 123abcABC )에 포함되어 있는지 테스트하려면 stringinstring "$string1" "$string2" 를 실행하고 반환 값을 확인해야합니다.

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

하나는:

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

Bash4 + 예제. 참고 : 따옴표를 사용하지 않으면 단어에 공백이있을 때 문제가 발생할 것입니다. 항상 bash IMO에서 인용하십시오.

다음은 BASH4 +의 몇 가지 예입니다.

예제 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

즐겨.


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


이것은 어떤가요:

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

적절한 메서드를 사용하면됩니다 : String#split() .

String string = "004-034556";
String[] parts = string.split("-");
String part1 = parts[0]; // 004
String part2 = parts[1]; // 034556

정규 표현식 을 사용하므로 필요한 경우 특수 문자 를 이스케이프 처리해야합니다.

백 슬래시 \ , 캐럿 ^ , 달러 기호 $ , 마침표 또는 점과 같은 특수 의미 있는 12 개의 문자가 . , 수직 막대 또는 파이프 기호 | , 물음표 ? , 별표 또는 별 * , 더하기 기호 + , 여는 괄호 ( , 닫는 괄호 ) 및 여는 대괄호 [ , 여는 중괄호 { , 이러한 특수 문자는 종종 "메타 문자"라고합니다.

따라서, 예를 들어 마침표 / 점으로 분리하려는 경우 . split("\\.") 와 같은 개별 특수 문자를 이스케이프 처리하거나 문자 클래스 [] 를 사용하여 split("\\.") [] ) 같은 리터럴 문자를 나타낼 수 split("[.]") , Pattern#quote() 를 사용하여 split(Pattern.quote(".")) 처럼 전체 문자열을 이스케이프 할 수 있습니다.

String[] parts = string.split(Pattern.quote(".")); // Split on period.

문자열에 특정 문자가 포함되어 있는지 미리 테스트하려면 String#contains() .

if (string.contains("-")) {
    // Split it.
} else {
    throw new IllegalArgumentException("String " + string + " does not contain -");
}

여기에는 정규 표현식이 적용되지 않습니다. 이를 위해서는 String#matches() 대신 사용하십시오.

결과로 나오는 부분에 분할 문자를 유지하려면 긍정적 인 검색 방법 을 사용 하십시오 . 분할 문자가 왼편에 오게하고 싶다면, 패턴에 ?<= group을 접두사로 사용하여 긍정적 인 표정을 사용하십시오.

String string = "004-034556";
String[] parts = string.split("(?<=-)");
String part1 = parts[0]; // 004-
String part2 = parts[1]; // 034556

분할 문자가 오른쪽에 오도록하려면 패턴에 ?= group 접두사를 붙이면 긍정적 인 미리보기를 사용하십시오.

String string = "004-034556";
String[] parts = string.split("(?=-)");
String part1 = parts[0]; // 004
String part2 = parts[1]; // -034556

결과 파트의 수를 제한하려면 split() 메소드의 두 번째 인수로 원하는 숫자를 제공 할 수 있습니다.

String string = "004-034556-42";
String[] parts = string.split("-", 2);
String part1 = parts[0]; // 004
String part2 = parts[1]; // 034556-42




string bash substring