string - bash字串包含 - shell script字串比對




字符串包含在Bash中 (14)

兼容答案

由於已經有很多使用Bash特定功能的答案,所以在busybox等功能較差的shell中有一種方法:

[ -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所問,這裡是一個相同的演示版本,在相同的shell下測試:

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" ]; }

或者如AdrianGünter的評論所暗示的那樣,避免-o

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

我在Bash中有一個字符串:

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

這看起來有點笨拙。


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 : http://unstableme.blogspot.com/2008/06/bash-search-letter-in-string-awk.html


你應該記住,shell腳本不是一門語言,更多的是一組命令。 本能地,你認為這個“語言”要求你跟隨一個if [或一個[[ 。 這兩者都只是返回退出狀態的命令,指示成功或失敗(就像其他命令一樣)。 出於這個原因,我會使用grep ,而不是[命令。

做就是了:

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

現在,您正在考慮是否正在測試後續命令的退出狀態(以分號結尾)。 為什麼不重新考慮你正在測試的字符串的來源?

## 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不輸出任何內容,因為我們只需要返回代碼。 <<<使shell擴展下一個單詞,並將其用作命令的輸入,這是<< here文檔的單行版本(我不確定這是標準還是雙擊)。


如果你喜歡正則表達式的方法:

string='My string';

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

我不確定使用if語句,但是您可以通過case語句獲得類似的效果:

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

我使用這個函數(不包括一個依賴項,但顯而易見)。 它通過了下面的測試。 如果該函數返回一個大於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
}

我發現需要這種功能的頻率非常高,所以我在我的.bashrc使用了一個自製的shell函數,這樣我就可以根據需要隨時重新使用它,並帶有一個易於記憶的名稱:

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

為了測試$string1 (比如說abc )是否包含在$string2 (比如123abcABC )中,我只需要運行stringinstring stringinstring "$string1" "$string2"並檢查返回值,例如

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

我的.bash_profile和我如何使用grep如果PATH包含我的2個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

正如Paul在他的表現比較中提到的:

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

這是符合POSIX標準的,就像Marcus提供的'case'$ string'in'答案一樣,但比case statement的答案稍微容易閱讀。 另外請注意,這將比使用case語句慢得多,正如Paul指出的,不要在循環中使用它。


準確的單詞匹配:

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

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

試試oobash它是一個用於bash 4的面向對象的字符串庫。它支持德語變音符號。 它是用bash寫的。 許多函數可用: -base64Decode-base64Encode-base64Decode-base64Encode-capitalize-charAt-concat-count-endsWith-equals-equalsIgnoreCase-reverse-hashCode-indexOf-isAlnum-isAlpha-isAlpha-isAscii-isDigit-isEmpty-isHexDigit-isLowerCase-isSpace-isPrintable-isUpperCase-isVisible-lastIndexOf-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上找到


這也適用於:

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 shell的特性。

--參數是純粹的POSIX偏執狂,用於防止類似於選項的輸入字符串,如--abc-a

注意:在嚴格的循環中,這個代碼將比使用內部Bash shell特性慢得多,因為一個(或兩個)單獨的進程將通過管道創建並連接。


這個怎麼樣:

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

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




substring