Shell中空格引起的血案

2023-05-16

最近开始写点shell脚本,对linux命令还是比较熟悉的,但是shell脚本却没写过,没想,刚开始写,就郁闷重重。

各种语法错误!!!

最简单的自定义变量赋值,给我报个“未找到命令”的错误。我就郁闷了,代码如下:

client = "/home/works/client"

前后耗时3个小时,也没搞明白哪里出了错,最终知道问题所在是 空格 出了问题。写java写多了,不自觉就敲两个空格,于是喷血就出现了。

还有if的时候,一定得有空格!下面是让我豁然开朗的一篇文章,分享一下。

原文地址http://www.igigo.net/archives/152

空格,一个看不见的字符,很不起眼,也正由于不起眼,很多人经常忽略它,导致代码出错,却还找不着北。这里,我们来聊聊bash中空格的那点事。
先了解下bash中什么时候该用空格,什么时候不该用。


    1. 等号赋值两边不能有空格
    2. 命令与选项之间需要空格
    3. 管道两边空格可有可无
  

我们来看看常见的问题
1. 赋值时等号两边或者只有左边多了空格

1
2
3
4
5
6
7
8
9
10
11
12
igi@gentoo ~ $ var1 = test
bash : var1: command not found
igi@gentoo ~ $ echo ${var1:?error}
bash : var1: error
igi@gentoo ~ $ echo ${var1?error}
bash : var1: error
igi@gentoo ~ $ var2 = test
bash : var2: command not found
igi@gentoo ~ $ echo ${var2:?error}
bash : var2: error
igi@gentoo ~ $ echo ${var2?error}
bash : var2: error

这里我用了bash的变量扩展,${var1:?error}当var1为unset或null(未定义或空)时, 报指定错误; ${var1?error}当var1为unset时,报指定错误 。从执行结果来看,如果等号左边有空格,则变量名当成命令执行,结果报command not found,变量没有被赋值

2. 赋值时等号左边没有空格,右边有空格(这种情况有点特别,你会发现两种情况)

1
2
3
igi@gentoo ~ $ var= test
igi@gentoo ~ $ var= nocmd
bash : nocmd: command not found

同样是等号右边有空格,第一条命令没报错,而第二条报错了。
这是因为shell中有这么一种执行命令的方式: var=string command
命令command将得到变量var的值(至于在命令执行后,变量var的值是否保留下来,bash4中没有保留,但我在dash中发现时保留下来的,不同的shell对这个的处理不同), 由于test是个命令,而nocmd不是,所以报了command not found.

1
2
3
igi@gentoo ~ $ var=newtest eval echo \$var
newtest
igi@gentoo ~ $ echo $var

注意: 这里我使用了eval, 是想避免在第一次解析时$var被替换成空字符串, 不然就会出现下面的情况(下面是错误的测试方法,在echo还没执行时,$var已经被替换成空字符串)

1
2
3
igi@gentoo ~ $ var=newtest echo $var
 
igi@gentoo ~ $ echo $var

到这里,相信大家都明白了吧, 对于等号赋值,左右两边不可以有空格,虽然右边有空格不一定报错,但那绝对不是你想要的结果。

3. 命令和选项之间必须有空格
这个似乎大家都明白,为何我还这么罗嗦呢?说到这里,不得不提一下一个非常特别的命令: [ 命令(你没看错,是[ ), 也就是test命令(当然bash中,这是个内置命令,但在这里不影响
我们的理解)。或许你会觉得[命令眼熟,没错,我保证你见过它,来看看下面的例子

1
2
3
4
5
igi@gentoo ~ $ if [ "abc" = "abc" ]; then echo ‘they are the same’; fi
they are the same
igi@gentoo ~ $ type -a [
[ is a shell builtin
[ is /usr/bin/ [

想起来了吧?[命令经常用到if判断中,当然也有人喜欢这么写

1
2
3
4
5
igi@gentoo ~ $ [ "abc" = "cba" ] || echo ‘they are not the same’
they are not the same
igi@gentoo ~ $ type -a [
[ is a shell builtin
[ is /usr/bin/ [

[ 命令正名叫test命令,它们两者几乎一样,为什么不是完全一样?来看看这个

1
2
3
4
5
6
igi@gentoo ~ $ [ "abc" = "cba"
bash : [: missing `]‘
igi@gentoo ~ $ [ "abc" = "cba" ]
igi@gentoo ~ $ test "abc" = "cba" ]
bash : test : too many arguments
igi@gentoo ~ $ test "abc" = "cba"

清晰了吧,用[命令时,你必须给它个尾巴], 用test命令时,就不能加个尾巴。尾巴]是[最后一个参数,不可缺少的参数, 代表[命令的结束

扯了这么多,那到底这个和空格有毛关系?说这些,是先让大家明白: [在shell中是个命令,它左右必须有空格!]是[的最后不可缺少的参数,它两边也需要空格(虽然有些命令的参数能连一起,例如ps, 但[命令不行,它的参数之间必须有空格)。让我们看看关于[常见的错误

a. if 与 [ 之间缺少空格

1
2
3
4
5
6
7
8
igi@gentoo ~ $ if [ "$HOME" = "/home/igi" ]; then echo 'equal' ; fi
bash : syntax error near unexpected token ` then '
igi@gentoo ~ $ if [ "$HOME" = "/home/igi" ]; then echo 'equal' ; fi
bash : syntax error near unexpected token ` then '
igi@gentoo ~ $ if [ "$HOME" = "/home/igi" ]; then echo 'equal' ; fi
bash : syntax error near unexpected token ` then '
igi@gentoo ~ $ if [ "$HOME" = "/home/igi" ]; then echo 'equal' ; fi
bash : syntax error near unexpected token ` then '

语法分析错误,很明显,if[ 对于bash来说,不知道是什么鬼东西

b. [与后面的参数之间缺少空格

1
2
3
4
igi@gentoo ~ $ if [ "$HOME" = "/home/igi" ]; then echo 'equal' ; fi
bash : [ /home/igi : No such file or directory
igi@gentoo ~ $ if [ "$HOME" = "/home/igi" ]; then echo 'equal' ; fi
bash : [ /home/igi : No such file or directory

["$HOME" 对于bash来说,也不知道是什么鬼东西

c. [ ] 之间的参数之间缺少空格

1
2
3
4
igi@gentoo ~ $ if [ "abc" = "abc" ]; then echo 'equal' ; fi
equal
igi@gentoo ~ $ if [ "abc" = "cba" ]; then echo 'equal' ; fi
equal

第一条命令似乎是对的(实际上是正巧而已),看看第二条命令"abc" 和 "cba"明显不同,但却判断为相同。这是因为参数之间缺少了空格,被[命令认为内部是个值而已。看看下面的命令,你就会释然

1
2
3
4
5
6
igi@gentoo ~ $ if [ 0 ]; then echo 'equal' ; fi
equal
igi@gentoo ~ $ if [ "1" ]; then echo 'equal' ; fi
equal
igi@gentoo ~ $ if [ "" ]; then echo 'equal' ; fi
igi@gentoo ~ $ if [ ]; then echo 'equal' ; fi

在[ ] 内部,如果只有一个值(那些因为缺少了空格而连一起的也算),不是空字符串就为真。所以在[ ] 之间的参数,也要两边有空格,而不能堆一起

d. 参数和尾巴]之间缺少空格
这个就不罗嗦了,尾巴]也是[命令的参数,如同上面所讲,参数之间必须有空格

扯了这么多[命令与空格的事,但有些时候,缺了空格却能正确运行, 当然这只是你好运, 一起来看看

1
2
3
4
5
igi@gentoo ~ $ var= ' abc'
igi@gentoo ~ $ if [$var = "abc" ]; then echo 'equal' ; fi
equal
igi@gentoo ~ $ if [ "$var" = "abc" ]; then echo 'equal' ; fi
bash : [ abc: command not found

之前Bash引号那点事提到过,双引号包围起来的是一个整体,而没双引号的时候,字符串前后的空格或制表符都被切开。如果恰巧你遇到了或者你故意要丢弃字符串前后的空格或制表符,那也不是不可能, 但非常不建议你这么写,你的代码将是非常脆弱的。

或者你该加的空格都加了,但还是报错,这也可能和缺少双引号有关。这样的情况很普遍,最后再看看

1
2
3
4
5
6
7
8
9
igi@gentoo ~ $ var= ''
igi@gentoo ~ $ if [ "$var" = "abc" ]; then echo 'equal' ; fi
igi@gentoo ~ $ if [ $var = "abc" ]; then echo 'equal' ; fi
bash : [: =: unary operator expected
igi@gentoo ~ $ dvar= 'a b c'
igi@gentoo ~ $ if [ $dvar = "a b c" ]; then echo 'equal' ; fi
bash : [: too many arguments
igi@gentoo ~ $ if [ "$dvar" = "a b c" ]; then echo 'equal' ; fi
equal

我再罗嗦一次,不要轻易省略双引号。很清楚了吧?如果你还不明白,


本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Shell中空格引起的血案 的相关文章

随机推荐

  • 大锤老湿教您如何配置TP-Link路由器组建wifi上网

    TP Link路由器设置教程 大家好 xff0c 今天由大锤老湿教大家如何设置使用最广的TP Link路由器 一般家庭都希望能上wifi 那么首先看看我们如何将新买回的或者由于故障已经恢复成重置出厂状态的路由器 xff0c 如何经过重新设置
  • 【ESP01S】使用串口调试助手,发送AT指令收回的是乱码/重复一遍AT指令发回的问题

    调试帮助 span class token punctuation span 技术交流Q xff1a span class token number 1083091092 span xff08 备注CSDN xff09 一 问题描述 在使用
  • 刷leetcode使用python还是c++?

    我身边80 的程序员朋友在刷题的时候会选择Java xff0c 很少有人用C 43 43 来刷题 这两门语言各有特点 xff1a C 43 43 xff1a 从C语言发展过来的一门语言 xff0c 继承了灵活 xff08 可以潜入任何现代的
  • VINS-Mono代码精简版代码详解-后端非线性优化(三)

    非线性优化部分代码解析 之前已经对VINS Mono的初始化部分进行了介绍 xff0c 下面结合代码和公式介绍其非线性优化部分 本文部分参考 https blog csdn net u012871872 article details 78
  • Ubuntu IO占用过多导致文件读取变慢的原因查找方法

    问题描述 xff1a 多用户服务器 xff0c ubuntu系统 xff0c 突然点开文件夹 xff0c 发现变慢 查看方法 xff1a step1 xff1a 进入管理员用户 step2 xff1a 运行iostat x 1 在显示的结果
  • ROS Docker

    Docker 常用指令 docker pull osrf ros galactic desktop 从网络上下载镜像 docker images 查看已加载镜像列表 window docker界面 xff1a 命令行结果 xff1a doc
  • Win10C盘文件夹内容详解(持续更新,欢迎留言)

    本文参考以下博客 Roaming和Local的区别 C Users 用户名 AppData 1 Local和Roaming之间的区别 xff1a Local 比较大 xff0c 非漫游应用数据 Roaming 一般是漫游应用数据 2 Roa
  • STM32运行FreeRTOS

    使用ARM Keil 的 Keil uVision IDE xff0c 在 STM32上运行 FreeRTOS 内核 物料清单 软件 在创建新项目之前 xff0c 我们必须安装软件包 下面是打印屏幕 xff0c 其中包含如何执行此操作的步骤
  • ESP32实践FreeRTOS

    将部分代码作为应用程序中的任务独立执行可以简化大型复杂问题的设计 当有多个 CPU 时 xff0c 任务支持还允许选定的功能并行运行 本文将调查 Arduino 框架对 ESP32 系列设备的 FreeRTOS 任务支持 除了少数例外 xf
  • 黑马程序员—5—Java基础:多态学习笔记和学习心得体会

    lt ahref 61 34 http www itheima com 34 target 61 34 blank 34 gt android 培训 lt a gt lt ahref 61 34 http www itheima com 3
  • 图像去噪算法简介

    一 xff0c 背景 随着各种数字仪器和数码产品的普及 xff0c 图像和视频已成为人类活动中最常用的信息载体 xff0c 它们包含着物体的大量信息 xff0c 成为人们获取外界原始信息的主要途径 然而在图像的获取 传输和存贮过程中常常会受
  • Android 7 Nougat 源码目录结构

    code style margin 0px auto font family none padding 0px color inherit background color transparent art Android Runtime x
  • 【无人驾驶规划】BOSS无人车规划算法

    无人驾驶规划 BOSS无人车规划算法 1 boss运动规划结构2 轨迹生成2 1 状态约束2 2 车辆模型2 3 控制参数化2 4 初始化轨迹2 5 轨迹优化 3 on road模式规划3 1 路径生成3 2 轨迹生成3 3 轨迹速度配置3
  • 这也太全面了 阿里王牌级“Docker全线笔记”,Github已标星80k+,我太爱

    写在开头 司汤达说过 xff1a 一个人只要强烈地坚持不懈地追求 xff0c 他就能达到目的 Docker的创始人Solomon Hykes就是以这样的精神 xff0c 在docker即将坚持不下去的时候 xff0c 选择的不是放弃 xff
  • 如何在keil5中新建.c和.h文件?

    有两种方法 xff1a 方法1 在keil5内部添加两个文件分别为 c和 h文件 xff0c 可以保存在一个新建的文件夹里 xff08 前提是此文件夹是在keil5内部保存时新建的文件夹 xff0c 而不是在keil5软件外自己新建的文件夹
  • CMake(十二):构建类型

    本章和下一章涉及两个密切相关的主题 构建类型 在某些IDE工具中也称为构建配置或构建方案 是一种高级控件 xff0c 它选择不同的编译器和链接器行为集 构建类型的操作是本章的主题 xff0c 而下一章将介绍控制编译器和链接器选项的更具体细节
  • CMake:构建、链接静态库和动态库

    CMake 构建 链接静态库和动态库 导言一 多目录多文件CMake构建方式1 项目结构2 message h3 message cpp4 hello world cpp5 CMakeLists txt6 构建及编译 二 静态库和动态库简介
  • msckf_mono构建运行方法

    背景 博主是在读Davide Scaramuzza投稿到ICRA 2018的VIO综述文章 A Benchmark Comparison of Monocular Visual Odometry Algorithms for Flying
  • IMU相关技术资料整理

    关于IMU噪声参数 xff1a IMU噪声参数模型的参考文档 xff1a https github com ethz asl kalibr wiki IMU Noise Model针对消费级IMU器件的噪声参数进行适度的不确定性放大 xff
  • Shell中空格引起的血案

    最近开始写点shell脚本 xff0c 对linux命令还是比较熟悉的 xff0c 但是shell脚本却没写过 xff0c 没想 xff0c 刚开始写 xff0c 就郁闷重重 各种语法错误 xff01 xff01 xff01 最简单的自定义