如何判断一个字符串是否包含POSIX中的另一个字符串?

Mat*_*att 120 unix shell

我想编写一个Unix shell脚本,如果另一个字符串中有一个字符串,它将执行各种逻辑.例如,如果我在某个文件夹中,请分支.有人可以告诉我如何做到这一点?如果可能的话,我想让这不是特定于shell的(即不仅仅是bash),但如果没有别的方法可以做到这一点.

#!/usr/bin/env sh

if [ "$PWD" contains "String1" ]
then
    echo "String1 present"
elif [ "$PWD" contains "String2" ]
then
    echo "String2 present"
else
    echo "Else"
fi
Run Code Online (Sandbox Code Playgroud)

fja*_*rlq 145

这是另一个解决方案.这使用POSIX子串参数扩展,所以它适用于bash,dash,ksh ......

test "${string#*$word}" != "$string" && echo "$word found in $string"
Run Code Online (Sandbox Code Playgroud)

编辑:这是一个好主意,C.罗斯.这是一个带有一些例子的功能化版本:

# contains(string, substring)
#
# Returns 0 if the specified string contains the specified substring,
# otherwise returns 1.
contains() {
    string="$1"
    substring="$2"
    if test "${string#*$substring}" != "$string"
    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"
Run Code Online (Sandbox Code Playgroud)

  • 错误的是双括号不是 POSIX,这是问题的前提,@Pablo。 (7认同)
  • 如果子字符串包含反斜杠,这对我不起作用.像往常一样,`substring ="$(printf'%q'"$ 2")"`节省了一天. (3认同)
  • 聪明,也适用于[[]]. (2认同)
  • 这不适用于像`[]`这样的特殊字符。请参阅我的回答 /sf/answers/3814331741/。 (2认同)
  • 我们必须双引号匹配的“$word”。https://www.shellcheck.net/wiki/SC2295 (2认同)
  • 谢谢,@midnite。ShellCheck 很酷。我已经编辑了这篇文章,使其现在支持特殊字符。还要感谢 Alex Skrypnyk 和 Egor Tensin 指出了同样的错误。 (2认同)

Nor*_*sey 74

纯POSIX外壳:

#!/bin/sh
CURRENT_DIR=`pwd`

case "$CURRENT_DIR" in
  *String1*) echo "String1 present" ;;
  *String2*) echo "String2 present" ;;
  *)         echo "else" ;;
esac
Run Code Online (Sandbox Code Playgroud)

像ksh或bash这样的扩展shell具有花哨的匹配机制,但旧式的case功能非常强大.


Joh*_*and 37

可悲的是,我不知道有什么方法可以做到这一点.但是,使用bash(从版本3.0.0开始,这可能就是你所拥有的),你可以像这样使用=〜运算符:

#!/bin/bash
CURRENT_DIR=`pwd`

if [[ "$CURRENT_DIR" =~ "String1" ]]
then
 echo "String1 present"
elif [[ "$CURRENT_DIR" =~ "String2" ]]
then
 echo "String2 present"
else
 echo "Else"
fi
Run Code Online (Sandbox Code Playgroud)

作为额外的奖励(和/或警告,如果你的字符串中有任何有趣的字符),=〜如果你省略引号,则接受正则表达式作为正确的操作数.

  • 这是 Bash,而不是问题所问的 POSIX sh。 (6认同)
  • 不引用正则表达式,或者它*不会*一般工作.例如,尝试`[[test =〜"test.*"]]`vs.`[[test = ~test.*]]`. (3认同)

a.s*_*abh 26

#!/usr/bin/env sh

# Searches a subset string in a string:
# 1st arg:reference string
# 2nd arg:subset string to be matched

if echo "$1" | grep -q "$2"
then
    echo "$2 is in $1"
else 
    echo "$2 is not in $1"
fi
Run Code Online (Sandbox Code Playgroud)

  • 将 `grep -q "$2"` 更改为 `grep -q "$2" 2>/dev/null` 以避免不需要的输出。 (3认同)
  • 将 `grep -q "$2"` 更改为 `grep -q "$2" > /dev/null` 以避免出现不需要的输出。 (2认同)

TT4*_*M.C 15

以下是您问题的各种解决方案的链接.

这是我的最爱,因为它具有最人性化的感觉:

星形通配符方法

if [[ "$string" == *"$substring"* ]]; then
    return 1
fi
return 0
Run Code Online (Sandbox Code Playgroud)

  • `[[`不是POSIX (10认同)

hel*_*hod 14

case $(pwd) in
  *path) echo "ends with path";;
  path*) echo "starts with path";;
  *path*) echo "contains path";;
  *) echo "this is the default";;
esac
Run Code Online (Sandbox Code Playgroud)


bma*_*ies 6

bash regexps.或者有'expr':

 if expr "$link" : '/.*' > /dev/null; then
    PRG="$link"
  else
    PRG=`dirname "$PRG"`/"$link"
  fi
Run Code Online (Sandbox Code Playgroud)