介绍

Linux越来越重要,对于开发者来说学习必要的bash技术必不可少,对运维来说更是如此。学习shell可以参考很多学习材料和图书,比如ChinaUnix论坛网中人大哥总结的《Shell十三问》的帖子,《ABS 指导》(《Advanced Bash-Scripting Guide》),《linux shell脚本攻略》等大家可以参考。本文虫虫给大家一些常用的shell技巧分享给大家,希望可以对大家有所帮助。

引导顺序

首先我们要说下bash脚本的启动,Linux中系统初始化都是按照一定顺序加载各个文件初始化脚本(Shell脚本)。脚本所在文件启动顺序很重要,下面这个图显示了Linux系统个初始化脚本和环境变量的加载顺序,包括了bash、sh和zsh等常见的shell。

bash的加载遵循上图,从上到下顺序执行加载,尤其要注意non-login、非交互(上面蓝色线)执行,它不会加载很多脚本,比如/etc/profile(总profile),/etc/bash.bashrc(总basrc),和个人的.profile, .bash_profie,.bash_login和.bashrc等。所以如果你脚本处于这样环境下(比如cron脚本),你就要在脚本中自我设置一些环境变量。这样就可以避免由于环境变量导致的一些莫名其妙的的问题,比如写的好脚本为什么登陆就可以执行,放到cron里面自动执行就不行。

输入/输出和重定向

我们知道Linux下一切皆文件,包括硬件设备,其中三个特殊的文件句斌,标准输入(stdin)、标准输出(stdout)和标准错误(stderr),其句柄号为三个整数0,1和2。

< :用于重定向改变标准输入(stdin)。

0< :0< 和

<< :输入一段文本,直到读到<

> :用于重定向改变标准输出(stdout)。

1> :1> 和>一样 ,> 默认的FD为1。

2> :改变标准错误(stderr)。

>> :重定向输出到文件,追加到文件末尾。

set -o noclobber :设置 > 不能覆盖已经存在的文件。

set +o noclobber :这支 > 可以覆盖已经存在的文件。

>| :设置set -o noclobber后,使用 >| 可以临时覆盖已经存在的文件。

/dev/null 和 /dev/zero

这两个文件是两个比较特殊设备。/dev/null,或称空设备,它会丢弃一切写入它的数据,但是读取它则会抛出错误。在shell中常用它来表示放弃执行的结果(或者错误),这样只关注于程序执行,比如在cron中的任务,输出和错误都没有意义。

command > /dev/null

可以将stdout、stderr都重定向输出到/dev/null:

command > /dev/null 2>&1 或者

command &> /dev/null

command >& /dev/null

/dev/zero 设备用于提供无限制的空null内供读取。当读取该设备(文件)时候,它会提供无限的空字符 null。我们可以从/dev/zero 读取任意数量的 null 字符。和 /dev/null 不同,/dev/zero主要用于作为null数据源供读取,当然/dev/zero也支持写入,可以用做数据黑洞,。

/dev/zero主要用途是提供字符流来初始化数据存储,也就是使用空字符覆盖目标数据。还可以用来一个特定大小的空白文件。

比如用空白文件覆盖分区,删除分区的数据可以用(慎用!):

dd if=/dev/zero of=/dev/

创建一个一个1M的文件,可以使用

dd if=/dev/zero of=cc count=1024 bs=1024

``和$()

反引号(Tab键上面的键)和$(),两者基本上类似,在shell中都表示执行命令。

echo`ls`

echo $(ls)

最常用的用法是将一些命令结果输出传递个其他命令作为参数。比如要查询当前目录和子目录下所有PHP中的包含"eval"函数(似疑木马),可以用下面命令:

grep eval `find -type f -name "*.php"`

两者主要的区别在于嵌套更简单。

对``来说要实现嵌套就要增加很多反斜杠来转义,比如:

echo `echo \`echo \\\`echo hello,Chongchong\!\\\`\``

而用$()就简介的多了:

echo $(echo $(echo $(echo hello,Chongchong\!)))

这是个很有意义的操作符,他有点类似于$(),而对括弧中命令的输出重用。但是在

比如我们要执行下面一些例子

grep keys file1> /tmp/a

grep keys file2> /tmp/b

diff /tmp/a /tmp/b

diff

引用

bash引用比较复杂,我们先从简单的开始。

首先,引号中的变量:

A="123"

echo "$A"

echo "$A"

这个结果很简单,双引号会解释变量,输出为其值123,而单引号不会解释,输出为$A。

下面三个echo 都会打印啥呢?

mkdir -p tmp

cd tmp

touch a

echo *

echo "*"

echo "*"

Bash中还有很多引用方式的快捷操作符,其中一些可以极大的提高我们的工作效率

!$

表示重复最后一个命令的最后一个参数。如果你正在处理一个文件,通过该操作符可以在命令后重新引入参数,从而节省大量重复输入。

grep xxx /long/path/to/some/file/or/other.txt

vim !$

!:1-$

该操作符,更有意思,它获取上一个命令的所有参数并将它们引入。所以:

grep isthere /long/path/to/some/file/or/other.txt

egrep !:1-$

fgrep !:1-$

!表示"查看上一个命令",:是一个分隔符,1表示"取第一个单词", -表示范围"直到",$表示"最后一个单词"。

我们可以用!*实现同样的目的。但是基于上面规则,我们就可以随意定制参数的范围了。比如如使用!:2-3。

:h

如果把该操作符放在一个文件名后面,它会删除文件夹以外的文件名称。比如:

grep XXX /long/path/to/some/file/or/other.txt

cd !$:h

通配符和正则表达式

通配符合正则表达式中都会用*,看起来很相似,但它们差异很大

请解释下面这个命令:

rename -n "s/(.*)/HEAD$1/" *

上面两个星号表达意思完全不同:

第一个由于用单引号括住,shell不会对其解释,作为参数都提交给rename命令。rename会对其解释 ,而s/(.*)/HEAD$1/是一个正则替换。在正则表达式中 *表示对前面字符的0个或者多个模式。.表示一个字符,所以.*表示匹配所有内容;()表示对括住内容进行引用,在替换后面部分用\1或者$1表示它。

第二个会被shell解释,*在shell中表示通配符,表示当前工作文件夹中所有文件的列表替换。

所以整个命令表示将文件下所有文件名都增加上HEAD 前缀。请尝试下面两个命令:

ls *

ls .*

第二个看起来更像是一个正则表达式,但是实际是什么样的呢?请你尝试对输出解释下。

条件语句,[和[[

请尝试执行一下语句会输出什么?

if grep XXX /dev/null

then

echo Chongchong

else

echo lo

fi

grep的返回代码使得像这样的代码更直观地作为其作为退出状态解释时候带来的副作用。

下面两个语句的输出呢?

if [ $(grep XXX /dev/null) = "" ]

then

echo -n Chongchong

else

echo -n lo

fi

if [[ $(grep XXX /dev/null) = "" ]]

then

echo -n Chongchong

else

echo -n lo

fi

[是测试的原始形式,然后[[引入了更灵活和直观的方式。在上面的第一个if块中,if语句因为$(grep XXX /dev/null)被评估为空,导致这种比较:[=""],这会抛出错误,针对这种情况就需要使用[[]]来避免这种异常,还有一个常用的技巧是使用:

if [x$(grep XXX /dev/null)="x"]

所以,如果命令没有返回任何内容,它仍然可以正常运行,而不会抛出异常,中断执行。

退出状态码

我们知道每次shell执行,推出是都会给shell一个状态码。

如果命令执行正常,则会返回给shell状态码0。如果不成功,则会得到一个非零代码,用来表具体错误的种类,其中 1表示"一般错误",还有其他更多信息,可以在程序中具体定义。状态码,可以通过$?查看。但有些命令会设置有特殊设置比如grep,请尝试下面的命令:

grep XXX /dev/null

echo $?

grep中使用退出代码来指示它是否匹配。

set 语句

Bash中也提供了可配置的选项,可以即时设置环境变量等。

set -e

如果任何命令返回非零退出状态,则从脚本退出。

set -X

会输出在运行时运行的命令。

所以脚本可能会使用这样的开头:

#!/bin/bash

set -e

set -x

grep XXX /dev/null

echo $?

getopts

如果要写一个较大的程序,涉及的输入比较复杂,则需要使用getopts来简化你的工作,用他来帮你做输入和参数处理的工作,比如下面一个例子:

getopts后面跟的是参数列表字符串,每个字母表示一个选项,带:选项表示选项还会有值,比如上面例子中对应的-j /home/soft/java 和-m /home/soft/maven 。而getopts字符串中没有跟随:的字母就是开关型选项,不需要指定值,等同于true/false,只要带上了这个参数就是true。

getopts是shell内部命令,可以在shell脚本中直接使用,同时shell也支持一个外部命令getopt有很多发行版本也都自带。详细可以搜索参考具体文档,本文不在赘述。

总结:

抛砖引玉,本文中列出了shell中一些常见的技巧和容易混淆的知识点,希望能对大家有所帮助,如果大家有更好的想法技巧也可以一起分享出来,一起参考学习。

linux命令行 基础,Linux命令行基础,关于Bash需要知道的一些常识相关推荐

  1. linux 复制代码 多出了很多空格_最基础Linux(一)——简单命令行操作

    Linux终端的命令很多,但作为零基础的入门者而言,一开始了解一下几个操作就可以撑过好一阵子了,一下接触太多可能反而会比较头疼.而等回过神来想学更多的命令的时候,自己琢磨的能力也差不多了. 将介绍的命 ...

  2. Linux基础--访问命令行

    命令行概念 命令行是基于文本的界面,可用于向计算机系统输入指令.Linux命令行由shell程序提供. 红帽默认shell是bash,普通用户启动shell时默认提示符的结尾是$,超级用户root是# ...

  3. linux命令行处理图片基础,linux下命令行图片处理工具ImageMagick

    linux下命令行图片处理工具ImageMagick (2012-05-25 21:21:40) 标签: linux imagemagick 命令行图片处理 it ImageMagick的主要命令如下 ...

  4. linux命令行 正则,Linux shell基础(四)正则表达式与grep命令 beta

    一.正则表达式 之前学习find命令时,就已经接触过一些简单的正则,那么我们现在来学习一些复杂的正则.首先,我们还是来复习什么是正则表达式,看这个教程先,一定要多看几遍,至少知道正则大概长什么样子.( ...

  5. Linux基础常用命令

    ps -ef用于查看全格式的全部进程,其中"ps"是在Linux中是查看进程的命令,"-e "参数代表显示所有进程,"-f"参数代表全格式. ...

  6. Linux 学习笔记之超详细基础linux命令 Part 3

    Linux学习笔记之超详细基础linux命令 by:授客 QQ:1033553122 ---------------------------------接Part 2----------------- ...

  7. linux sed给空文件首行插入_Sed命令高级功能,学好了工作不愁

    原创:小姐姐味道(微信公众号ID:xjjdog),欢迎分享,转载请保留出处. sed命令有两个空间,一个叫pattern space,一个叫hold space.这两个空间能够证明人类的脑瓜容量是非常 ...

  8. Linux配置最基础的命令

    Linux配置最基础的命令 Linux设置最基础的命令 切换用户 查看当前主机的完整名称 临时设置主机名 永久设置主机名 查看当前系统版本信息 查看当前内核版本 临时修改网卡IP 永久修改网卡IP 重 ...

  9. Linux:分享50个实用的基础命令,欢迎收藏!

    今天小编给大家分享50个实用的基础命令,欢迎收藏! 1.存放用户账号的文件在哪里? /etc/passwd 2.如何删除一个非空的目录? rm -rf 目录名 3.查看当前的工作目录用什么命令? pw ...

  10. linux基础:7、基础命令介绍(2)

    目录相关命令) mkdir 作用:make directories 创建目录 语法:mkdir [选项] 目录名称 选项: -p 级联创建目录,如果目标目录已存在不会报错 ============== ...

最新文章

  1. 卷积神经网络学习笔记与心得(4)池化
  2. 皮一皮:顶尖黑客技术,10秒教学,不会你打我!
  3. CodeForces 392C Yet Another Number Sequence 矩阵快速幂
  4. 4进程原语:fork()函数,getpid()函数和getppid()函数,getuid()函数,getgid()函数,vfork()
  5. 5、mysql中的库操作
  6. linux查看本机所有预设的系统变量,如何设置与查看Linux系统中的环境变量?
  7. 高速通道-冗余物理专线接入-健康检查配置
  8. python列表生成器语法_python列表生产式和生成器
  9. 趋势 | AI技能排行榜:TensorFlow热度飙升,Python最火
  10. 三星发布8nm芯片Exynos 850
  11. java setmessage_Java Message.setTitle方法代码示例
  12. 米筐量化不支持c语言_量化初级之量化选股
  13. 亲密关系--【承担情绪】-如何让对方的情绪软着陆
  14. 6to5 – 让你即刻体验 ECMAScript 6 编程
  15. Win7 便签设置字体方法
  16. 线性规划问题的目标函数灵敏度分析
  17. 编程之道(英汉对照)[转载]
  18. 吴伯凡-认知方法论-认知中的信道与噪音
  19. 新颖的_基于web的毕业设计题目50例
  20. 电脑xls图标未正常显示

热门文章

  1. 内部类-----Java
  2. matlab建立的发动机的模型,奇瑞使用基于模型的设计实现发动机管理系统软件的自主开发...
  3. mysql删除员工_数据库删除职工信息
  4. php做游戏登录服务器,游戏登陆服务器php简单实现
  5. html怎么设计自己的网页,求一份自己设计的简单网页 HTML格式
  6. android 自定义进度条_第一百八十九回:Android中自定义ProgressBar三
  7. 没有bug队——加贝——Python 练习实例 3,4
  8. js判断对象是否为空对象_js对象
  9. 【测试方法篇】效率测试
  10. 洛谷 P2384 最短路题解