AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 559323
Accepted
John Jones
John Jones
Asked: 2019-12-29 13:06:29 +0800 CST2019-12-29 13:06:29 +0800 CST 2019-12-29 13:06:29 +0800 CST

如何区分 case 语句中的大写和小写字符?

  • 772

我正在尝试恢复我生锈的 shell 脚本编写技能,但我遇到了 case 语句的问题。我在下面的程序中的目标是评估用户提供的字符串是以大写字母还是小写字母开头:

# practicing case statements
echo "enter a string"
read yourstring
echo -e "your string is $yourstring\n"

case "$yourstring" in
    [A-Z]* )
       echo "your string begins with a Capital Letter"
       ;; 
    [a-z]* )
       echo "your string begins with a lowercase letter"
       ;; 
    *)     
       echo "your string did not begin with an English letter"
       ;;
esac

myvar=nope

case $myvar in
   N*)
     echo "begins with CAPITAL 'N'"
     ;;
   n*)
     echo "begins with lowercase 'n'"
     ;;
   *)
     echo "hahahaha"
     ;;
esac

当我输入一个以小写字母开头的字符串(例如,没有引号的“mystring”)时,case 语句将我的输入与第一个 case 匹配,并通知我该字符串以大写字母开头。我写了第二个 case 语句,看看我是否犯了一些明显的语法或逻辑错误(也许我仍然是),但我没有同样的问题。第二种情况结构正确地告诉我 $myvar 保存的字符串以小写字母开头。

我尝试使用引号将 $yourstring 括在 case 语句的第一行,并且我尝试过不使用引号。我阅读了“shopt”选项并确认“nocasematch”已关闭。(为了更好的衡量,我打开它并再次尝试,但我仍然没有从我的第一个 case 语句中得到正确的结果。)我还尝试使用 sh 和 bash 运行脚本,但输出是相同的。(我用“sh ./case1.sh”和“bash ./case1.sh”显式调用shell,因为我没有设置执行位。复制文件并在新文件上设置执行位并没有改变输出.)

虽然我不了解使用“-x”调试选项运行 shell 的所有输出,但输出显示了 shell 从第一个“case”行到按照第一个模式执行命令的过程。我将此解释为第一个模式与输入字符串匹配,但我不确定为什么。

当我切换前两个模式(和相应命令)的顺序时,case 语句对小写字母成功,但错误地将“MYSTRING”报告为以小写字母开头。由于检测到任何字母匹配首先出现的模式,我认为我有一个逻辑错误......但我不确定是什么。

我在 unix.com 上发现了“pludi”的一篇文章,其中建议“对小写和大写字符的测试是 [az] 和 [AZ]。这不再适用于某些语言环境和/或 Linux 发行版。” (见https://www.unix.com/shell-programming-and-scripting-128929-example-switch-case-bash.html)果然,用 [[:upper:]] 和 [[ :lower:]] 解决了这个问题。

我在 Fedora 31 上,我的语言环境输出如下:

LANG=en_US.UTF-8
LC_CTYPE="en_US.UTF-8" 
LC_NUMERIC="en_US.UTF-8" 
LC_TIME="en_US.UTF-8" 
LC_COLLATE="en_US.UTF-8" 
LC_MONETARY="en_US.UTF-8" 
LC_MESSAGES="en_US.UTF-8" 
LC_PAPER="en_US.UTF-8" 
LC_NAME="en_US.UTF-8" 
LC_ADDRESS="en_US.UTF-8" 
LC_TELEPHONE="en_US.UTF-8" 
LC_MEASUREMENT="en_US.UTF-8" 
LC_IDENTIFICATION="en_US.UTF-8" 
LC_ALL= 

我想知道我是否不了解字符范围,或者不了解模式匹配在 case 语句中的工作原理,或者底层的 shell 功能是否发生了变化(以及为什么?)。如果有人有耐心,我将不胜感激;我也很乐意阅读相关文档。谢谢!

shell-script case
  • 2 2 个回答
  • 2518 Views

2 个回答

  • Voted
  1. Best Answer
    roaima
    2019-12-29T15:41:11+08:002019-12-29T15:41:11+08:00

    一个简单的答案,毫无疑问其他人可以取代。

    字符集排序现在根据所使用的语言环境而有所不同。引入语言环境的概念是为了支持不同的民族和他们的不同语言。正如您从输出中看到的那样,locale现在解决了几个不同的领域 - 不仅仅是排序规则。

    在您的情况下,它是美国,并且出于排序和整理目的,字母表是 AaBbCc...Zz 或 A=a、B=b、C=c 等(我忘了哪个,而且我不在计算机上我可以验证另一个)。语言环境非常复杂,在某些语言环境中,就排序和整理而言,可能存在不可见的字符。相同的字符可以根据使用的语言环境进行不同的排序。

    如您所见,识别小写字符的正确方法是使用[[:lower:]]; 这将在必要时包括重音字符,甚至包括不同字母(希腊语、西里尔语等)中的小写字符。

    如果您想要经典排序,您可以通过设置恢复每个应用程序甚至每个命令LC_ALL=C。举一个人为的例子,

    grep some_pattern | LC_ALL=C sort | nl
    
    • 6
  2. user232326
    2019-12-29T16:10:02+08:002019-12-29T16:10:02+08:00

    字典顺序和 ASCII 顺序之间一直存在着一场持久战。
    需很长时间。

    从 Unicode 的角度来看,字符应该按照当地习俗按照字典顺序进行排序,因此 a A b B ... 表示美国字母(ASCII 字母)。这通常与 en_US.utf-8 语言环境中的 [a-zA-Z] 范围相匹配。国际化通常同意这一点。

    从程序员的角度来看,由于是 C 语言,[az] 应该只匹配从 97 到 122 的 ascii 字符作为一个字节值。[AZ] 也是如此。这通常会将字符的 C 语言定义匹配为一个字节。一些脚本编写者想要使用这个定义。

    这场战斗不时地从一种解释转移到另一种解释。
    有时 [az] 范围变成 only abcdefghijklmnopqrstuvwxyz。
    有时它会转移到aAbBcCdDeEfFgGhHiIjJkKlLmMnNoOpPqQrRsStTuUvVwWxXyYz.
    或者其他一些更复杂的列表。

    细节很复杂。历史悠久。战斗仍在激烈进行。


    因此,您可能会得到(测试字符串book):

    • bash 版本 2、3 和 4 的“您的字符串以大写字母开头”以及
    • bash 版本 5(和 1)的“您的字符串以小写字母开头”
    • 大多数 shell 会将其报告为“小写字母”。

    如果您测试字符串úber(在 en_US.UTF-8 中),您将得到:

    • ksh/ATT-sh 中的“小写”
    • dash、zsh、bash 5.0+ 或 [lm]ksh 中的“不是英文字母”。
    • bash 2,3 和 4 中的“大写字母”。

    以及字符串Úber。

    所以,结果是多种多样的。

    您还可以设置 LC_ALL=C 来强制解释a-z只有小写字母(并且A-Z只有大写字母)。这将冻结用于仅来自的排序规则C。如果语言环境更改,则不会更改。更健壮的脚本,但适应性较差的脚本。

    还有一个选项可以使用[[:lower:]],但同样,它保证是仅在 C 语言环境中的 ASCII 范围 az。它可能会在 POSIX 的未来版本(但尚未在 2020 年发布)中强制执行到所有语言环境。

    总而言之,确保没有外部决定(来自 Unix 规范的 shell 开发人员)会改变代码范围的唯一安全方法是:

    # practicing case statements
    echo "enter a string"
    read yourstring
    echo -e "your string is $yourstring\n"
    
    low='abcdefghijklmnopqrstuvwxyz'
    cap='ABCDEFGHIJKLMNOPQRSTUVWXYZ'
    
    case "$yourstring" in
        [$cap]* ) echo "your string begins with a Capital Letter"   ;; 
        [$low]* ) echo "your string begins with a lowercase letter" ;; 
        *)      echo "your string did not begin with an English letter" ;;
    esac
    
    • 2

相关问题

  • 在awk中的两行之间减去相同的列

  • 打印文件行及其长度的脚本[关闭]

  • 通过命令的标准输出以编程方式导出环境变量[重复]

  • 按分隔符拆分并连接字符串问题

  • MySQL Select with function IN () with bash array

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve