Unix 简明教程

Unix / Linux - Quick Guide

Unix - Getting Started

What is Unix ?

Unix 操作系统是一组程序,充当计算机与用户之间的链接。

在计算机内部分配系统资源和协调所有详细信息的计算机程序称为 operating systemkernel

用户通过称为 shell 的程序与内核通信。shell 是一个命令行解释器;它翻译用户输入的命令,并将它们转换为内核可以理解的语言。

  1. Unix 最初由 AT&T 员工 Ken Thompson、Dennis Ritchie、Douglas McIlroy 和 Joe Ossanna 于 1969 年在贝尔实验室开发。

  2. 市场上有多种 Unix 变体。Solaris Unix、AIX、HP Unix 和 BSD 就是一些例子。Linux 也是一种自由提供的 Unix 分支。

  3. 多人可以同时使用 Unix 计算机;因此,Unix 被称为多用户系统。

  4. 用户还可以同时运行多个程序;因此,Unix 是一个多任务环境。

Unix Architecture

以下是 Unix 系统的基本框图−

unix architecture

将所有 Unix 版本联合起来的主要概念是以下四个基本概念−

  1. Kernel − 内核是操作系统的核心。它与硬件和大多数任务(如内存管理、任务调度和文件管理)进行交互。

  2. Shell - Shell 是处理您请求的实用程序。当您在终端中键入命令时,Shell 会解释命令并调用您想要的程序。Shell 对所有命令使用标准语法。C Shell、Bourne Shell 和 Korn Shell 是最著名的 Shell,可用于大多数 Unix 变体。

  3. Commands and Utilities − 有各种各样的命令和实用程序,您可以在日常活动中使用它们。 cpmvcatgrep 等是命令和实用程序的一些示例。有 250 多个标准命令,另外还有许多通过第三方软件提供的命令。所有命令都带有各种选项。

  4. Files and Directories - Unix 的所有数据都组织到文件中。然后,所有文件被组织到目录中。这些目录进一步组织到一个称为 filesystem 的树状结构中。

System Bootup

如果您有一台安装了 Unix 操作系统的计算机,那么您只需要打开系统即可使其启动。

一旦您打开系统,它将开始启动,最后提示您登录系统,这是一项登录系统并将其用于您的日常活动的活动。

Login Unix

当您首次连接到 Unix 系统时,通常会看到如下提示−

login:

To log in

  1. 准备好你的用户 ID(用户识别)和密码。如果您还没有,请联系您的系统管理员。

  2. 在登录提示符处输入您的用户 ID,然后按 ENTER 。你的用户 ID 为 case-sensitive ,所以一定要按照系统管理员的指示输入。

  3. 在密码提示符处输入您的密码,然后按 ENTER 。您的密码也区分大小写。

  4. 如果您提供了正确的用户 ID 和密码,那么您将被允许进入系统。阅读屏幕上显示的信息和消息,如下所示。

login : amrood
amrood's password:
Last login: Sun Jun 14 09:32:32 2009 from 62.61.164.73
$

将为您提供一个命令提示符(有时称为 $ 提示符),您可以在其中键入所有命令。例如,要查看日历,您需要按如下方式键入 cal 命令−

$ cal
     June 2009
Su Mo Tu We Th Fr Sa
    1  2  3  4  5  6
 7  8  9 10 11 12 13
14 15 16 17 18 19 20
21 22 23 24 25 26 27
28 29 30

$

Change Password

所有 Unix 系统都需要密码,以帮助确保您的文件和数据仍然是您的文件和数据,并且系统本身可以抵御黑客和破解者。以下是更改密码的步骤−

Step 1 − 首先,在命令提示符下键入密码,如下所示。

Step 2 - 输入您的旧密码,即您当前正在使用的密码。

Step 3 - 输入您的新密码。始终保持密码足够复杂,以便没有人能够猜到它。但确保你能记住它。

Step 4 - 您必须通过再次输入密码来验证密码。

$ passwd
Changing password for amrood
(current) Unix password:******
New UNIX password:*******
Retype new UNIX password:*******
passwd: all authentication tokens updated  successfully

$

Note − 我们在这里添加星号 (*) 只是为了表示您需要输入当前密码和新密码的位置,否则在您的系统中。当您键入时,它不会向您显示任何字符。

Listing Directories and Files

Unix 中的所有数据都组织到文件中。所有文件都组织到目录中。这些目录组织成一个称为文件系统的树形结构。

您可以使用 ls 命令列出目录中可用的所有文件或目录。以下是使用 ls 命令和 -l 选项的示例。

$ ls -l
total 19621
drwxrwxr-x  2 amrood amrood      4096 Dec 25 09:59 uml
-rw-rw-r--  1 amrood amrood      5341 Dec 25 08:38 uml.jpg
drwxr-xr-x  2 amrood amrood      4096 Feb 15  2006 univ
drwxr-xr-x  2 root   root        4096 Dec  9  2007 urlspedia
-rw-r--r--  1 root   root      276480 Dec  9  2007 urlspedia.tar
drwxr-xr-x  8 root   root        4096 Nov 25  2007 usr
-rwxr-xr-x  1 root   root        3192 Nov 25  2007 webthumb.php
-rw-rw-r--  1 amrood amrood     20480 Nov 25  2007 webthumb.tar
-rw-rw-r--  1 amrood amrood      5654 Aug  9  2007 yourfile.mid
-rw-rw-r--  1 amrood amrood    166255 Aug  9  2007 yourfile.swf

$

此处以 d…​.. 开头的条目表示目录。例如,uml、univ 和 urlspedia 是目录,其余条目是文件。

Who Are You?

当您登录到系统时,您可能想知道: Who am I

找出“您是谁”的最简单方法是输入 whoami 命令−

$ whoami
 amrood

$

在你的系统上尝试一下。此命令列出了与当前登录关联的账户名称。你也可以尝试 who am i 命令来获取自己的信息。

Who is Logged in?

有时你可能想知道同时登录到电脑的是谁。

有三个命令可以让你获取此信息,根据你了解其他用户的程度: userswhow

$ users
 amrood bablu qadir

$ who
amrood ttyp0 Oct 8 14:10 (limbo)
bablu  ttyp2 Oct 4 09:08 (calliope)
qadir  ttyp4 Oct 8 12:09 (dent)

$

在你的系统上尝试 w 命令以查看输出。这列出了登录到系统用户的关联信息。

Logging Out

当你结束会话时,你需要退出系统。这样做是为了确保其他人无法访问你的文件。

To log out

  1. 只需在命令提示符下键入 logout 命令,系统将清理所有内容并断开连接。

System Shutdown

通过命令行正确关闭 Unix 系统的最一致的方法是使用以下命令之一−

Sr.No.

Command & Description

1

halt 立即关闭系统

2

init 0 使用预定义脚本关闭系统,以便在关闭前同步并清理系统

3

init 6 通过完全关闭系统,然后重新启动来重启系统

4

poweroff 通过关闭电源来关闭系统

5

reboot Reboots the system

6

shutdown Shuts down the system

您通常需要成为超级用户或 root(Unix 系统上权限最高的用户帐户)才能关闭系统。然而,在一些独立的或个人拥有的 Unix 机箱上,管理用户,有时是普通用户可以这样做。

Unix - File Management

在本章中,我们将详细讨论 Unix 中的文件管理。Unix 中的所有数据都组织到文件中。所有文件都组织到目录中。这些目录组织成一个称为文件系统的树形结构。

当您使用 Unix 时,无论哪种方式,您都会花费大量时间使用文件。本教程将帮助您了解如何创建和删除文件、复制和重命名文件、创建指向它们的链接等。

在 Unix 中,有三种基本类型的文件−

  1. Ordinary Files − 普通文件是系统上的一个文件,其中包含数据、文本或程序指令。在本教程中,您将了解如何处理普通文件。

  2. Directories − 目录存储特殊文件和普通文件。对于熟悉 Windows 或 Mac OS 的用户,Unix 目录等同于文件夹。

  3. Special Files − 某些特殊文件提供对硬件的访问,例如硬盘驱动器、CD-ROM 驱动器、调制解调器和以太网适配器。其他特殊文件类似于别名或快捷方式,使您可以使用不同的名称访问单个文件。

Listing Files

若要列出存储在当前目录中的文件和目录,请使用以下命令 -

$ls

以下是要在命令中列出示例输出 −

$ls

bin        hosts  lib     res.03
ch07       hw1    pub     test_results
ch07.bak   hw2    res.01  users
docs       hw3    res.02  work

命令 ls 支持 -l 选项,它将帮助您获取有关已列出文件详细信息 −

$ls -l
total 1962188

drwxrwxr-x  2 amrood amrood      4096 Dec 25 09:59 uml
-rw-rw-r--  1 amrood amrood      5341 Dec 25 08:38 uml.jpg
drwxr-xr-x  2 amrood amrood      4096 Feb 15  2006 univ
drwxr-xr-x  2 root   root        4096 Dec  9  2007 urlspedia
-rw-r--r--  1 root   root      276480 Dec  9  2007 urlspedia.tar
drwxr-xr-x  8 root   root        4096 Nov 25  2007 usr
drwxr-xr-x  2    200    300      4096 Nov 25  2007 webthumb-1.01
-rwxr-xr-x  1 root   root        3192 Nov 25  2007 webthumb.php
-rw-rw-r--  1 amrood amrood     20480 Nov 25  2007 webthumb.tar
-rw-rw-r--  1 amrood amrood      5654 Aug  9  2007 yourfile.mid
-rw-rw-r--  1 amrood amrood    166255 Aug  9  2007 yourfile.swf
drwxr-xr-x 11 amrood amrood      4096 May 29  2007 zlib-1.2.3
$

以下是所有列出列的信息 −

  1. First Column − 表示文件类型和文件上的权限。以下是所有类型的文件说明。

  2. Second Column − 表示文件或目录占用的内存块数。

  3. Third Column - 表示文件所有者。这是创建此文件的 Unix 用户。

  4. Fourth Column - 表示所有者的组。每个 Unix 用户都会具有一个关联的组。

  5. Fifth Column − 以字节为单位代表文件大小。

  6. Sixth Column − 代表创建或最后修改此文件的时间与日期。

  7. Seventh Column − 代表文件或目录名称。

ls -l 列表示例中,文件行的开头是 d-l 。这些字符表示列出的文件类型。

Sr.No.

Prefix & Description

1

- 常规文件,例如:ASCII 文本文件、二进制可执行文件或硬盘链接。

2

b 块特殊文件。块输入/输出设备文件,如物理硬盘。

3

c 字符特殊文件。原始输入/输出设备文件,如物理硬盘。

4

d 包含其他文件和目录的列表的目录文件。

5

l 符号链接文件。任何常规文件上的链接。

6

p 命名管道。进程间通信的机制。

7

s 用于进程间通信的套接字。

Metacharacters

元字符在 Unix 中具有特殊含义。例如, * and ? are metacharacters. We use * 匹配 0 个或更多字符,一个问号 ( ? ) 匹配单个字符。

例如 −

$ls ch*.doc

显示开头字母为 ch 且结尾字母为 .doc 的所有文件 −

ch01-1.doc   ch010.doc  ch02.doc    ch03-2.doc
ch04-1.doc   ch040.doc  ch05.doc    ch06-2.doc
ch01-2.doc ch02-1.doc c

此处, * 用作匹配任何字符的元字符。如果你只想显示所有以 .doc 结尾的文件,则可以使用以下命令 −

$ls *.doc

Hidden Files

不可见文件是指其第一个字符是圆点或句点字符 (.) 的文件。Unix 程序(包括 shell)使用这些文件的大多数来存储配置信息。

隐藏文件的一些常见示例包括文件 −

  1. .profile - Bourne shell ( sh) 初始化脚本

  2. .kshrc - Korn shell ( ksh) 初始化脚本

  3. .cshrc - C shell ( csh) 初始化脚本

  4. .rhosts - 远程 shell 配置文件

若要列出不可见文件,请向 ls 指定 -a 选项 -

$ ls -a

.         .profile       docs     lib     test_results
..        .rhosts        hosts    pub     users
.emacs    bin            hw1      res.01  work
.exrc     ch07           hw2      res.02
.kshrc    ch07.bak       hw3      res.03
$
  1. @s2 - 此项表示当前的目录。

  2. @s3 - 此项表示父目录。

Creating Files

您可以使用 vi 编辑器在任何 Unix 系统上创建普通文件。您只需提供以下命令 -

$ vi filename

上述命令将使用给定文件名打开一个文件。现在,按 i 键进入编辑模式。进入编辑模式后,您便可以开始在文件中写入内容,如下面的程序所示 -

This is unix file....I created it for the first time.....
I'm going to save this content in this file.

完成该程序后,按照以下步骤操作:

  1. 按下@+s8+键退出编辑模式。

  2. 一起按下两个键@s9 键完全退出文件。

您现在将在当前目录中使用@+s10+创建文件。

$ vi filename
$

Editing Files

您可以使用@+s11+编辑器编辑现有文件。我们将简要讨论如何打开现有文件:

$ vi filename

一旦打开文件,您可以按@+s12+键进入编辑模式,然后您可以继续编辑文件。如果您想在文件内四处移动,那么首先您需要按@+s13+键退出编辑模式。此后,您可以使用以下键在文件中移动:

  1. @+s14+键向右侧移动。

  2. @+s15+键向左侧移动。

  3. @+s16+键在文件内向上移动。

  4. @+s17+键在文件内向下移动。

使用上述按键,你可以将光标置于任意位置以进行编辑。定位后,你可以使用 i 键进入编辑模式。完成后,在文件内按 Esc ,最后同时按 Shift + ZZ 两个键以完全退出该文件。

Display Content of a File

你可以使用 cat 命令查看文件的内容。以下是一个查看上述创建的文件内容的简单示例:

$ cat filename
This is unix file....I created it for the first time.....
I'm going to save this content in this file.
$

你可以使用 -b 选项和 cat 命令一起显示行号,如下所示:

$ cat -b filename
1   This is unix file....I created it for the first time.....
2   I'm going to save this content in this file.
$

Counting Words in a File

你可以使用 wc 命令获取文件包含的行数、单词数和字符数。以下是一个查看上述创建的文件信息的简单示例:

$ wc filename
2  19 103 filename
$

以下是四个列的详细信息:

  1. First Column - 表示文件中总行数。

  2. Second Column - 表示文件中总单词数。

  3. Third Column - 表示文件中的总字节数。这是实际的文件大小。

  4. Fourth Column - 表示文件名称。

你可以给出多个文件并一次获取这些文件的信息。以下是简单的语法:

$ wc filename1 filename2 filename3

Copying Files

要复制文件,请使用 cp 命令。该命令的基本语法如下:

$ cp source_file destination_file

以下是创建 filename 现有文件副本的示例。

$ cp filename copyfile
$

你现在会在当前目录中找到另一个文件 copyfile 。该文件将与原始文件 filename 完全相同。

Renaming Files

要更改文件名称,请使用 mv 命令。基本语法如下:

$ mv old_file new_file

以下程序将现有文件 filename 重命名为 newfile

$ mv filename newfile
$

mv 命令将现有文件完全移动到新文件中。在这种情况下,你只会在当前目录中找到 newfile

Deleting Files

要删除现有文件,请使用 rm 命令。基本语法如下:

$ rm filename

Caution − 文件可能包含有用的信息。在使用该 Delete 命令时,始终建议谨慎小心。最好将 -i 选项与 rm 命令一起使用。

以下示例显示了如何完全删除现有文件 filename

$ rm filename
$

你可以使用以下命令同时删除多个文件:

$ rm filename1 filename2 filename3
$

Standard Unix Streams

在正常情况下,每个 Unix 程序在启动时都会为它打开三个流(文件):

  1. stdin − 这称为标准输入,关联的文件描述符为 0。这也表示为 STDIN。Unix 程序将从 STDIN 读取默认输入。

  2. stdout − 这称为标准输出,关联的文件描述符为 1。这也表示为 STDOUT。Unix 程序将在 STDOUT 处写入默认输出。

  3. stderr − 这称为标准错误,关联的文件描述符为 2。这也表示为 STDERR。Unix 程序将在 STDERR 处写入所有错误消息。

Unix - Directory Management

在本章中,我们将详细讨论 Unix 中的目录管理。

目录是一个文件,其唯一任务是存储文件名称及相关信息。所有的文件(普通文件、特殊文件或目录)都包含在目录中。

Unix 使用层次结构来组织文件和目录。该结构通常称为目录树。树有一个根节点斜杠字符 ( / ),所有其他目录都包含在它下面。

Home Directory

第一次登录时所在的目录称为你的主目录。

你将在你的主目录和你创建的子目录中完成大部分工作,以整理你的文件。

你可以随时使用以下命令进入你的主目录:

$cd ~
$

此处 ~ 表示主目录。假设你必须进入其他用户的目录,请使用以下命令:

$cd ~username
$

要进入你的上一个目录,可以使用以下命令:

$cd -
$

Absolute/Relative Pathnames

目录按层次结构排列,根 (/) 在最上面。目录树中任何文件的位置都由其路径名来描述。

路径名的元素由 / 分隔。如果路径名是相对于根来描述的,则该路径名是绝对路径名,因此绝对路径名总以 / 开头。

以下是一些绝对文件名示例。

/etc/passwd
/users/sjones/chem/notes
/dev/rdsk/Os3

路径名也可以相对于您的当前工作目录。相对路径名永远不会从 / 开始。相对于用户 amrood 的主目录,一些路径名可能看起来像这样 -

chem/notes
personal/res

要随时确定您在文件系统层次结构中的位置,请输入命令 (cd) 以打印当前工作目录 -

$pwd
/user0/home/amrood

$

Listing Directories

要列出目录中的文件,可以使用以下语法 -

$ls dirname

以下是列出 (ls) /usr/local 目录中包含的所有文件的示例 -

$ls /usr/local

X11       bin          gimp       jikes       sbin
ace       doc          include    lib         share
atalk     etc          info       man         ami

Creating Directories

我们现在将了解如何创建目录。目录是通过以下命令创建的 -

$mkdir dirname

此处,directory 是您要创建的目录的绝对或相对路径名。例如,命令 -

$mkdir mydir
$

在当前目录中创建一个目录 mydir 。这是另一个示例 -

$mkdir /tmp/test-dir
$

此命令在目录 /tmp 中创建目录 test-dir 。如果 mkdir 命令成功创建了请求的目录,则不会产生任何输出。

如果您在命令行上给出了多个目录, mkdir 将创建每个目录。例如,-

$mkdir docs pub
$

在当前目录下创建目录 docs 和 pub。

Creating Parent Directories

我们现在将了解如何创建父目录。有时,当您要创建目录时,它的父目录或目录可能不存在。在这种情况下, mkdir 会发出如下错误消息 -

$mkdir /tmp/amrood/test
mkdir: Failed to make directory "/tmp/amrood/test";
No such file or directory
$

在这种情况下,可以将 -p 选项指定给 mkdir 命令。它会为您创建所有必要的目录。例如 -

$mkdir -p /tmp/amrood/test
$

以上命令创建所有必需的父目录。

Removing Directories

可以使用 rmdir 命令以如下方式删除目录 -

$rmdir dirname
$

Note - 要删除目录,请确保它为空,这意味着该目录中不应有任何文件或子目录。

您可以一次删除多个目录,如下所示 -

$rmdir dirname1 dirname2 dirname3
$

如果以上目录为空,则该命令将删除目录 dirname1、dirname2 和 dirname3。如果成功, rmdir 命令不会产生输出。

Changing Directories

除了更改到主目录之外,还可以使用 cd 命令执行更多操作。您可以通过指定有效的绝对或相对路径来使用它更改到任何目录。语法如下 -

$cd dirname
$

此处, dirname 是你希望更改为的目录名称。例如,命令 −

$cd /usr/local/bin
$

更改为目录 /usr/local/bin 。在这个目录中,你可以使用以下路径进行 cd 至目录 /usr/home/amrood

$cd ../../home/amrood
$

Renaming Directories

mv (move) 命令还可以用于重命名目录。语法如下 −

$mv olddir newdir
$

你可以将目录 mydir 重命名为 yourdir ,如下 −

$mv mydir yourdir
$

The directories . (dot) and .. (dot dot)

filename . (点)表示当前工作目录; filename .. (双点)表示高于当前工作目录一级,通常称为父目录。

如果我们输入命令来显示当前工作目录/文件的列表并使用 -a option 来列出所有文件和 -l option 提供长列表,我们将会收到以下结果。

$ls -la
drwxrwxr-x    4    teacher   class   2048  Jul 16 17.56 .
drwxr-xr-x    60   root              1536  Jul 13 14:18 ..
----------    1    teacher   class   4210  May 1 08:27 .profile
-rwxr-xr-x    1    teacher   class   1948  May 12 13:42 memo
$

Unix - File Permission / Access Modes

在本章中,我们将详细讨论 Unix 中的文件权限和访问模式。文件所有权是 Unix 的一个重要组成部分,它提供了一种安全的文件存储方法。Unix 中的每个文件都具有以下属性 −

  1. Owner permissions − 所有者的权限决定了文件所有者可以对文件执行哪些操作。

  2. Group permissions − 组的权限决定了一个用户(该用户属于该文件所属的组的成员)可以对文件执行哪些操作。

  3. Other (world) permissions − 其他用户的权限指示所有其他用户可以对文件执行哪些动作。

The Permission Indicators

使用 ls -l 命令时,它会显示与文件权限相关的各种信息,如下所示 −

$ls -l /home/amrood
-rwxr-xr--  1 amrood   users 1024  Nov 2 00:10  myfile
drwxr-xr--- 1 amrood   users 1024  Nov 2 00:10  mydir

这里,第一列表示不同的访问模式,即文件或目录具有的权限。

权限分为每组 3 个,组中的每个位置都表示一个特定权限,其顺序如下:读 (r)、写 (w)、执行 (x) −

  1. 前三个字符 (2-4) 表示文件所有者的权限。例如, -rwxr-xr-- 表示所有者具有读取 (r)、写入 (w) 和执行 (x) 权限。

  2. 第二组三个字符 (5-7) 由该文件所属组的权限组成。例如, -rwxr-xr-- 表示该组具有读取 (r) 和执行 (x) 权限,但没有写入权限。

  3. 最后三组三个字符 (8-10) 表示其他所有人的权限。例如, -rwxr-xr-- 表示只有 read (r) 权限。

File Access Modes

文件的权限是 Unix 系统安全的第一道防线。Unix 权限的基本构建块是这里描述的 readwriteexecute 权限 −

Read

授予读取权限,即查看文件内容的权限。

Write

授予修改或删除文件内容的权限。

Execute

具有执行权限的用户可以作为程序运行文件。

Directory Access Modes

目录访问模式的列出和组织方式与任何其他文件相同。需要注意以下几个区别 −

Read

访问目录表示用户可以读取内容。用户可以查看目录中的 filenames

Write

访问权限意味着用户可以向目录中添加或删除文件。

Execute

执行一个目录实际上没有什么意义,因此您可以将此视为遍历权限。

用户必须对 bin 目录具有 execute 权限才能执行 cdls 命令。

Changing Permissions

要更改文件或目录许可权,请使用 chmod (更改模式)命令。有两种使用 chmod 的方法——符号模式和绝对模式。

Using chmod in Symbolic Mode

初学者修改文件或目录权限最简单的方法是使用符号模式。有了符号权限,您可以使用下表中的运算符来添加、删除或指定您想要的权限组。

Sr.No.

Chmod operator & Description

1

+ 向文件或目录添加指定权限。

2

- 从文件或目录中移除指定权限。

3

= Sets the designated permission(s).

这里有一个使用 testfile 的示例。在 testfile 中运行 ls -1 显示文件的权限如下 −

$ls -l testfile
-rwxrwxr--  1 amrood   users 1024  Nov 2 00:10  testfile

然后对 testfile 运行前面的表中的每个 chmod 命令示例,然后执行 ls –l ,这样您就可以看到权限更改 −

$chmod o+wx testfile
$ls -l testfile
-rwxrwxrwx  1 amrood   users 1024  Nov 2 00:10  testfile
$chmod u-x testfile
$ls -l testfile
-rw-rwxrwx  1 amrood   users 1024  Nov 2 00:10  testfile
$chmod g = rx testfile
$ls -l testfile
-rw-r-xrwx  1 amrood   users 1024  Nov 2 00:10  testfile

以下是如何将这些命令组合在单行中 −

$chmod o+wx,u-x,g = rx testfile
$ls -l testfile
-rw-r-xrwx  1 amrood   users 1024  Nov 2 00:10  testfile

Using chmod with Absolute Permissions

使用 chmod 命令修改权限的第二种方法是使用数字来指定文件每组权限。

每项权限都分配有一个值,如下表所示,每组权限的总和都为该组提供了一个数字。

Number

Octal Permission Representation

Ref

0

No permission

---

1

Execute permission

--x

2

Write permission

-w-

3

执行和写入权限:1(执行)+ 2(写入)= 3

-wx

4

Read permission

r--

5

读取和执行权限:4(读取)+ 1(执行)= 5

r-x

6

读取和写入权限:4(读取)+ 2(写入)= 6

rw-

7

所有权限:4(读取)+ 2(写入)+ 1(执行)= 7

rwx

这里有一个使用 testfile 的示例。在 testfile 中运行 ls -1 显示文件的权限如下 −

$ls -l testfile
-rwxrwxr--  1 amrood   users 1024  Nov 2 00:10  testfile

然后对 testfile 运行前面的表中的每个 chmod 命令示例,然后执行 ls –l ,这样您就可以看到权限更改 −

$ chmod 755 testfile
$ls -l testfile
-rwxr-xr-x  1 amrood   users 1024  Nov 2 00:10  testfile
$chmod 743 testfile
$ls -l testfile
-rwxr---wx  1 amrood   users 1024  Nov 2 00:10  testfile
$chmod 043 testfile
$ls -l testfile
----r---wx  1 amrood   users 1024  Nov 2 00:10  testfile

Changing Owners and Groups

在 Unix 中创建帐户时,它会为每个用户分配一个 owner ID 和一个 group ID 。上述所有权限也都根据所有者和组进行分配。

有两个可用的命令来更改文件的所属者和组 −

  1. chown - chown 命令表示 "change owner" ,用于更改文件的所有者。

  2. chgrp - chgrp 命令表示 "change group" ,用于更改文件的所有组。

Changing Ownership

chown 命令更改文件的拥有权。其基本语法如下:

$ chown user filelist

用户的代码值可以是系统上的 name of a user 或系统上用户的 user id (uid)

以下示例将帮助你了解这个概念:

$ chown amrood testfile
$

将给定文件的所有者更改为用户 amrood

NOTE - 超级用户 root 拥有更改任何文件所有权的非受限能力,但普通用户只能更改由自己拥有的文件的所有权。

Changing Group Ownership

chgrp 命令更改文件的组所有权。其基本语法如下:

$ chgrp group filelist

组的代码值可以是系统上的 name of a group 或系统上的组的 the group ID (GID)

以下示例将帮助你了解这个概念:

$ chgrp special testfile
$

将给定文件的组更改为 special 组。

SUID and SGID File Permission

通常,一个命令在执行时都需要利用特殊权限才能完成其任务。

举例来说,当用 passwd 命令更改口令时,你的新口令会存储在文件 /etc/shadow 中。

由于安全原因,作为一个普通用户,你没有对这个文件的 readwrite 访问权,但是当你更改你的口令时,你需要有这个文件的写权限。这意味着 passwd 程序需要给你其他权限,才能写文件 /etc/shadow

通过一种称为 Set User ID (SUID)Set Group ID (SGID) 位的机制给程序额外权限。

当一个有 SUID 位的程序执行时,你将继承该程序所有者的权限。没有 SUID 位的程序将以启动该程序的用户的权限运行。

SGID 也是如此。通常,一个程序会以你的组权限执行,但是对这个程序,你的组将更改为程序的程序组所有者。

如果权限可用,SUID 和 SGID 位将显示为字母 "s" 。SUID "s" 位将位于所有者 execute 权限通常所在的权限位中。

例如,此命令

$ ls -l /usr/bin/passwd
-r-sr-xr-x  1   root   bin  19031 Feb 7 13:47  /usr/bin/passwd*
$

显示 SUID 位已设置,并且该命令归根拥有。在执行位置的大写字母 S (而非小写字母 s )表示未设置执行位。

如果为目录启用了粘滞位,则只有下列用户之一才能删除文件:

  1. 粘滞目录的所有者

  2. 要删除的文件的所有者

  3. The super user, root

要为任何目录设置 SUID 和 SGID 位,请尝试以下命令:

$ chmod ug+s dirname
$ ls -l
drwsr-sr-x 2 root root  4096 Jun 19 06:45 dirname
$

Unix - Environment

在本章中,我们将详细讨论 Unix 环境。一个重要的 Unix 概念是 environment ,它由环境变量定义。有些是由系统设置的,有些是你设置的,还有些是由 shell 或加载另一个程序的任何程序设置的。

变量是一个我们向其分配值的字符串。分配的值可以是数字、文本、文件名、设备或任何其他类型的数据。

例如,我们首先设置变量 TEST,然后使用 echo 命令访问其值 −

$TEST="Unix Programming"
$echo $TEST

产生以下结果。

Unix Programming

请注意,设置环境变量时不用 $ 标号,但访问它们时用的前缀是 $ 标号。这些变量保留其值,直到我们退出 shell。

登录系统时,shell 会经过称为 initialization 的阶段来设置环境。这通常是一个分两步的过程,涉及 shell 读取以下文件−

  1. /etc/profile

  2. profile

此过程如下−

  1. shell 检查文件 /etc/profile 是否存在。

  2. 如果存在,shell 将读取它。否则,将跳过此文件。不会显示错误消息。

  3. shell 检查文件 .profile 是否存在于您的主目录中。主目录是您登录后所在的目录。

  4. 如果文件存在,shell 会读取它;否则,shell 将跳过它。不会显示错误消息。

这两个文件都已读取完毕后,shell 会显示一个提示−

$

这是一个提示,您可以在此处输入命令,以让他们执行。

Note −此处详细说明的 shell 初始化过程适用于所有 Bourne 类型的 shell,但 bashksh 使用了一些另外的文件。

The .profile File

文件 /etc/profile 由 Unix 电脑的系统管理员维护,其中包含系统上的所有用户所需的 shell 初始化信息。

文件 .profile 受您控制。您可以向此文件添加任意多的 shell 自定义信息。您需要配置的最少信息包括−

  1. 您使用的终端类型。

  2. 可搜索命令的目录列表。

  3. 一些变量会影响终端的外观和触感。

你可以在家目录中查看 .profile 。使用 vi 编辑器将其打开,并查看为你的环境设置的所有变量。

Setting the Terminal Type

通常,你正在使用的终端类型会由 logingetty 程序自动配置。有时,自动配置过程会错误猜测你的终端。

如果你的终端设置错误,则命令的输出可能看起来很奇怪,或者你可能无法与 shell 正确交互。

为了确保这种情况不会发生,大多数用户以以下方式将他们的终端设置为最低公分母 −

$TERM=vt100
$

Setting the PATH

当你 فى موجه الأوامر تكتب أى أمر، يتعين على الشل العثور على الأمر قبل تنفيذه.

يوضح متغير PATH المواقع التى يجب أن يبحث فيها الشل عن الأوامر. وعادة ما يتم تعيين متغير المسار كما يلى −

$PATH=/bin:/usr/bin
$

هنا، كل مدخلة فردية مفصولة بواسطة حرف النقطتين (:) هى دلائل. إذا طلبت من الشل تنفيذ أمر ولم يتمكن من العثور عليه فى أى من الدلائل المحددة فى متغير المسار، ستظهر رسالة مشابهة لما يلى −

$hello
hello: not found
$

هناك متغيرات مثل PS1 و PS2 يتم مناقشتها فى القسم التالى.

PS1 and PS2 Variables

يتم تخزين الأحرف التى يعرضها الشل على أنه موجه للأوامر فى متغير PS1. يمكنك تغيير هذا المتغير إلى أى شئ تريده. وبمجرد تغييره، سيستخدمه الشل من تلك النقطة فصاعدًا.

على سبيل المثال، إذا أصدرت الأمر −

$PS1='=>'
=>
=>
=>

سيصبح موجه الأوامر الخاص بك ⇒. لتعيين قيمة PS1 بحيث تعرض دليل العمل، أصدر الأمر −

=>PS1="[\u@\h \w]\$"
[root@ip-72-167-112-17 /var/www/tutorialspoint/unix]$
[root@ip-72-167-112-17 /var/www/tutorialspoint/unix]$

ويتمخض عن هذا الأمر إظهار موجه المستخدم لاسم المستخدم واسم الجهاز (اسم المضيف) ودليل العمل.

يوجد عدد غير قليل من escape sequences التى يمكن استخدامها كحجج قيمة لـ PS1؛ حاول أن تقصر نفسك على الأكثر أهمية حتى لا يغمرك الموجه بالمعلومات.

Sr.No.

Escape Sequence & Description

1

\t الوقت الحالي، معبرًا عنه بصيغة الساعات:الدقائق:الثوانى

2

\d التاريخ الحالي، معبرًا عنه بصيغة يوم الأسبوع والشهر والتاريخ

3

\n Newline

4

\s Current shell environment

5

\W Working directory

6

\w المسار الكامل لدليل العمل

7

\u Current user’s username

8

\h اسم المضيف الخاص بالجهاز الحالي

9

\# 当前命令的命令序号。输入新命令时会增加

10

\$ 如果有效 UID 为 0(即,如果以 root 身份登录),则以 # 字符结束提示符;否则,使用 $ 符号

每次登录时,您都可以自己进行更改,或者可以在 .profile 文件中添加更改使其在 PS1 中自动进行。

发出不完整的命令时,外壳将显示一个次要提示符,并等待您完成命令并再次按下 Enter

默认的次要提示符是 > (大于号),但可以通过重新定义 PS2 外壳变量进行更改 −

以下是使用默认次要提示符的示例 −

$ echo "this is a
> test"
this is a
test
$

下面给出的示例使用自定义提示符重新定义了 PS2 −

$ PS2="secondary prompt->"
$ echo "this is a
secondary prompt->test"
this is a
test
$

Environment Variables

以下是重要环境变量的部分列表。这些变量按如下所述设置和访问 −

Sr.No.

Variable & Description

1

DISPLAY 包含 X11 程序默认应使用的显示符标识符。

2

HOME 指示当前用户的家目录:cd built-in 命令的默认参数。

3

IFS 指示在展开后解析程序用于分割单词的 Internal Field Separator

4

LANG LANG 扩展到默认的系统语言环境;可以利用 LC_ALL 覆盖此项。例如,如果其值为 pt_BR ,则语言将设置为(巴西)葡萄牙语,语言环境设置为巴西。

5

LD_LIBRARY_PATH 具有动态链接器的 Unix 系统,其中包含一个冒号分隔的目录列表,动态链接器在执行后构建进程映像之前在其中搜索共享对象,然后在其他任何目录中搜索。

6

PATH 指示命令的搜索路径。这是外壳寻找命令的一系列冒号分隔目录。

7

PWD 指示由 cd 命令设置的当前工作目录。

8

RANDOM 每次引用时,都会生成一个 0 到 32,767 之间的随机整数。

9

SHLVL 每次启动 bash 实例时递增一次。此变量可用于确定内置 exit 命令是否结束当前会话。

10

TERM 指代显示类型。

11

TZ 指代时区。它可以采用诸如 GMT、AST 等值。

12

UID 展开为在 shell 启动时初始化的当前用户的数字用户 ID。

以下是显示几个环境变量的示例 -

$ echo $HOME
/root
]$ echo $DISPLAY

$ echo $TERM
xterm
$ echo $PATH
/usr/local/bin:/bin:/usr/bin:/home/amrood/bin:/usr/local/bin
$

Unix Basic Utilities - Printing, Email

在本章中,我们将详细讨论 Unix 基本实用工具中的打印和电子邮件。到目前为止,我们已尝试了解 Unix 操作系统及其基本命令的本质。在本章中,我们将学习一些可在日常生活中使用的重要 Unix 实用工具。

Printing Files

在 Unix 系统上打印文件之前,你可能希望重新整理格式以调整页边距、突出显示一些单词等。大多数文件也可以在不重新整理格式的情况下打印,但打印的原始文件可能并不可取。

许多版本的 Unix 包含两个功能强大的文本格式化程序, nrofftroff

The pr Command

pr 命令在终端屏幕或打印机上对文件进行微调格式。例如,如果你有一个文件中的长名字列表,你可以将其格式化到屏幕分别成两列或更多列。

以下是 pr 命令的语法 −

pr option(s) filename(s)

pr 仅在屏幕或打印副本上更改文件的格式;它不会修改原始文件。下表列出了一些 pr 选项 −

Sr.No.

Option & Description

1

-k 生成 k 输出列

2

-d 输出双倍空格(并非对所有 pr 版本)

3

-h "header" 将下一项作为报告页眉

4

-t 消除了打印页眉和页眉/页脚

5

-l PAGE_LENGTH 将页面长度设置为 PAGE_LENGTH(66)行。文本默认行数为 56

6

-o MARGIN 使用 MARGIN(0)空格偏移每行

7

-w PAGE_WIDTH 将页面宽度设置为 PAGE_WIDTH(72)字符,仅适用于多文本列输出

在使用 pr 之前,以下为一个名为 food 的示例文件的内容。

$cat food
Sweet Tooth
Bangkok Wok
Mandalay
Afghani Cuisine
Isle of Java
Big Apple Deli
Sushi and Sashimi
Tio Pepe's Peppers
........
$

让我们使用 pr 命令创建一个带 Restaurants − 头部的两列报告

$pr -2 -h "Restaurants" food
Nov  7  9:58 1997  Restaurants   Page 1

Sweet Tooth              Isle of Java
Bangkok Wok              Big Apple Deli
Mandalay                 Sushi and Sashimi
Afghani Cuisine          Tio Pepe's Peppers
........
$

The lp and lpr Commands

命令 lplpr 会将文件打印在纸上,而不是屏幕上显示。一旦使用 pr 命令对格式进行设置,便可以使用以下任何命令将文件打印到连接到计算机的打印机上。

你的系统管理员很可能已在你的站点设置了默认的打印机。若要在默认打印机上打印名为 food 的文件,请像以下示例那样使用 lplpr 命令 −

$lp food
request id is laserp-525  (1 file)
$

lp 命令会显示一个 ID,可用于取消打印作业或查看作业状态。

  1. 如果你使用 lp 命令,则可以使用 -n*Num* 选项打印 Num 份。在 lpr 命令中,可使用 - Num 作相同操作。

  2. 如果与共享网络连接有多台打印机,则可以使用 lp 命令以及 -d*printer* 选项选择一台打印机,对于相同目的,你可以在 lpr 命令中使用 -P*printer* 选项。此处 printer 是打印机名称。

The lpstat and lpq Commands

lpstat 命令显示打印机队列中的内容:请求 ID、所有者、文件大小、发送打印作业的时间,以及请求的状态。

如果你想查看除自己以外的所有输出请求,请使用 lpstat -o 。请求按打印顺序显示 −

$lpstat -o
laserp-573  john  128865  Nov 7  11:27  on laserp
laserp-574  grace  82744  Nov 7  11:28
laserp-575  john   23347  Nov 7  11:35
$

lpq 提供的信息与 lpstat -o 略有不同 −

$lpq
laserp is ready and printing
Rank   Owner      Job  Files                  Total Size
active john       573  report.ps              128865 bytes
1st    grace      574  ch03.ps ch04.ps        82744 bytes
2nd    john       575  standard input         23347 bytes
$

此处第一行显示打印机状态。如果打印机已禁用或缺纸,你可能在此第一行看到不同的消息。

The cancel and lprm Commands

命令 cancel 可从 lp command 终结打印请求。 lprm 命令可终结所有 lpr requests 。你可以指定请求 ID(由 lp 或 lpq 显示)或打印机名称。

$cancel laserp-575
request "laserp-575" cancelled
$

要取消当前正在打印的任何请求,无论其 ID 如何,只需输入 cancel 和打印机名称 −

$cancel laserp
request "laserp-573" cancelled
$

lprm 命令会在活动作业属于你的情况下将其取消。否则,你可以将作业号作为参数给出,或使用 dash (-) 删除你的所有作业 −

$lprm 575
dfA575diamond dequeued
cfA575diamond dequeued
$

lprm 命令会告诉你实际从打印机队列中删除的文件名。

Sending Email

你使用终端邮件命令来发送和接收邮件。以下是发送电子邮件的语法 −

$mail [-s subject] [-c cc-addr] [-b bcc-addr] to-addr

以下是与邮件命令相关的部分重要选项 −s

Sr.No.

Option & Description

1

-s 在命令行中指定主题。

2

-c 将副本发送给用户列表。列表应该是由名称组成的、以逗号分隔的列表。

3

-b 将密件抄送给列表。列表必须为逗号分隔的名称列表。

以下是一个将测试消息发送给 admin@yahoo.com 的示例。

$mail -s "Test Message" admin@yahoo.com

然后,您需要在行首键入您的消息,后跟 "control-D" 。要停止,只需键入点 (.) ,如下所示 −

Hi,

This is a test
.
Cc:

您可以使用 redirect < operator 发送完整文件,如下所示 −

$mail -s "Report 05/06/07" admin@yahoo.com < demo.txt

要检查 Unix 系统中的传入电子邮件,只需键入电子邮件,如下所示 −

$mail
no email

Unix - Pipes and Filters

在本章中,我们将详细讨论 Unix 中的管道和过滤器。您可以将两个命令连接在一起,这样来自一个程序的输出就成为下一个程序的输入。以这种方式连接的两个或更多个命令形成一个管道。

要创建一个管道,请在两个命令之间的命令行上放置一个垂直线 ( | )。

当一个程序从另一个程序获取其输入时,它也会对该输入执行某些操作,并将结果写入标准输出。它被称为 filter

The grep Command

grep 命令搜索一个或多个文件中包含特定模式的行。语法如下 −

$grep pattern file(s)

名称 "grep" 来自 ed(Unix 行编辑器)命令 g/re/p ,它表示“全局搜索正则表达式,并打印包含该正则表达式的所有行”。

正则表达式不是普通文本(例如,一个单词)就是用来模式匹配的特殊字符。

grep 最简单的用法是查找由单个单词组成的模式。它可以在管道中使用,这样一来只有包含给定字符串的输入文件中的那些行才会发送到标准输出。如果你不给 grep 一个文件名来读取,它会读取其标准输入;所有过滤器程序都采用这种工作方式 −

$ls -l | grep "Aug"
-rw-rw-rw-   1 john  doc     11008 Aug  6 14:10 ch02
-rw-rw-rw-   1 john  doc      8515 Aug  6 15:30 ch07
-rw-rw-r--   1 john  doc      2488 Aug 15 10:51 intro
-rw-rw-r--   1 carol doc      1605 Aug 23 07:35 macros
$

有各种选项可以与 grep 命令一同使用 −

Sr.No.

Option & Description

1

-v 打印与模式不匹配的所有行。

2

-n 打印匹配的行及其行号。

3

-l 仅打印带有匹配行的文件(字母“l”)的名称

4

-c 仅打印匹配行的数量。

5

-i 匹配大写或小写。

现在,让我们使用一个正则表达式,告诉 grep 查找带有 "carol" 的行,然后是正则表达式中缩写为“.*”的零个或其他字符,然后是“Aug”。−

在这里,我们使用 -i 选项执行不区分大小写的搜索 −

$ls -l | grep -i "carol.*aug"
-rw-rw-r--   1 carol doc      1605 Aug 23 07:35 macros
$

The sort Command

sort 命令按字母或数字顺序排列文本行。以下示例对 food 文件中的行进行分类 −

$sort food
Afghani Cuisine
Bangkok Wok
Big Apple Deli
Isle of Java

Mandalay
Sushi and Sashimi
Sweet Tooth
Tio Pepe's Peppers
$

默认情况下, sort 命令按字母顺序排列文本行。有许多选项控制分类 −

Sr.No.

Description

1

-n 按数字分类(例如:10 将在 2 之后进行分类),忽略空格和制表符。

2

-r 反转分类顺序。

3

-f 将大写和小写一同分类。

4

+x 分类时忽略首个 x 字段。

可以将两个以上命令链接到管道中。以使用 grep 时先前的管道为例,我们可以进一步按大小对 8 月修改的文件进行分类。

以下管道包括命令 lsgrepsort

$ls -l | grep "Aug" | sort +4n
-rw-rw-r--  1 carol doc      1605 Aug 23 07:35 macros
-rw-rw-r--  1 john  doc      2488 Aug 15 10:51 intro
-rw-rw-rw-  1 john  doc      8515 Aug  6 15:30 ch07
-rw-rw-rw-  1 john  doc     11008 Aug  6 14:10 ch02
$

此管道按大小对 8 月修改的目录中所有文件进行分类,并将它们打印在终端屏幕上。排序选项 +4n 跳过 4 个字段(字段由空格分隔),然后按数字顺序对行进行排序。

The pg and more Commands

通常您可以将长输出在屏幕上压缩,但如果您通过更多或将 pg 命令用作过滤器来运行文本;当屏幕满载文本后,显示就停止。

假设您有一个长目录列表。要使读取排序的列表更容易,将输出通过 more 进行管道处理,如下所示 −

$ls -l | grep "Aug" | sort +4n | more
-rw-rw-r--  1 carol doc      1605 Aug 23 07:35 macros
-rw-rw-r--  1 john  doc      2488 Aug 15 10:51 intro
-rw-rw-rw-  1 john  doc      8515 Aug  6 15:30 ch07
-rw-rw-r--  1 john  doc     14827 Aug  9 12:40 ch03
	.
	.
	.
-rw-rw-rw-  1 john  doc     16867 Aug  6 15:56 ch05
--More--(74%)

当屏幕满载由文件大小顺序排序的行组成的文本时,屏幕将填满。屏幕底部是 more 提示符,您可以在其中键入命令以移动已排序的文本。

完成后,您可以使用 more 程序的讨论中列出的任何命令。

Unix - Processes Management

在本章中,我们将详细讨论 Unix 中的进程管理。当你在 Unix 系统中执行程序时,系统会为该程序创建一个特殊环境。此环境包含系统运行该程序所需的一切,就好像系统中没有其他程序在运行一样。

在 Unix 中发出命令时,它会创建一个新的进程或者启动一个新的进程。当你尝试 ls 命令以列出目录内容时,你就已经启动了一个进程。进程简单来说是正在运行的程序的一个实例。

操作系统通过一个五位数的 ID 号来跟踪进程,该 ID 号被称为 pidprocess ID 。系统中的每个进程都有一个唯一的 pid

由于用尽了所有可能的数字,下一个 pid 将滚动或重新开始,所以 pid 最终会重复。在任何时间点,系统中都无法存在两个 pid 相同的进程,因为 pid 是 Unix 用来跟踪每个进程的。

Starting a Process

启动进程(运行命令)时,有两种方法可以运行它 −

  1. Foreground Processes

  2. Background Processes

Foreground Processes

默认情况下,你启动的每个进程都在前台运行。它从键盘获取输入,并将输出发送到屏幕上。

你可以看到这是通过 ls 命令实现的。如果你希望列出你当前目录中的所有文件,可以使用以下命令 −

$ls ch*.doc

这将显示所有名称以 ch 开头并以 .doc 结尾的文件 −

ch01-1.doc   ch010.doc  ch02.doc    ch03-2.doc
ch04-1.doc   ch040.doc  ch05.doc    ch06-2.doc
ch01-2.doc   ch02-1.doc

进程在前台运行时,输出将定向到我的屏幕上,如果 ls 命令需要任何输入(实际上并不需要),它会从键盘上等待输入。

当程序在前台运行且耗时时,不能运行任何其他命令(启动任何其他进程),因为提示符在程序完成处理并退出之前都是不可用的。

Background Processes

后台进程运行时无需连接到你的键盘。如果后台进程需要任何键盘输入,它会等待。

在后台运行进程的优势在于,你可以运行其他命令;你无需等待它完成即可开始另一个命令!

启动后台进程最简单的方法是在命令末尾添加一个和号 ( & )。

$ls ch*.doc &

这会显示所有名称以 ch 开头并以 .doc 结尾的文件 −

ch01-1.doc   ch010.doc  ch02.doc    ch03-2.doc
ch04-1.doc   ch040.doc  ch05.doc    ch06-2.doc
ch01-2.doc   ch02-1.doc

在此,如果 ls 命令需要任何输入(实际上并不需要),它会进入停止状态,直到我们将其移至前台并从键盘提供数据为止。

第一行包含有关后台进程的信息 - 作业号和进程 ID。你需要了解作业号才能在后台和前台之间对其进行操作。

按 Enter 键,你将看到以下内容 −

[1]   +   Done                 ls ch*.doc &
$

第一行告诉我们, ls 命令的后台进程已成功完成。第二行提示输入另一个命令。

Listing Running Processes

运行 ps (进程状态)命令很容易看到您自己的进程,具体如下 −

$ps
PID       TTY      TIME        CMD
18358     ttyp3    00:00:00    sh
18361     ttyp3    00:01:31    abiword
18789     ttyp3    00:00:00    ps

One of the most commonly used flags for ps is the -f ( f for full) option, which provides more information as shown in the following example −

$ps -f
UID      PID  PPID C STIME    TTY   TIME CMD
amrood   6738 3662 0 10:23:03 pts/6 0:00 first_one
amrood   6739 3662 0 10:22:54 pts/6 0:00 second_one
amrood   3662 3657 0 08:10:53 pts/6 0:00 -ksh
amrood   6892 3662 4 10:51:50 pts/6 0:00 ps -f

Here is the description of all the fields displayed by ps -f command −

Sr.No.

Column & Description

1

UID 用户 ID 属于此进程(运行它的用户)

2

PID Process ID

3

PPID 父进程 ID(启动它的进程的 ID)

4

C CPU utilization of process

5

STIME Process start time

6

TTY 与进程关联的终端类型

7

TIME 进程占用的 CPU 时间

8

CMD 启动这个进程的命令

ps 命令一起使用的其他选项 −

Sr.No.

Option & Description

1

-a 显示所有用户的信息

2

-x 显示没有终端的进程的信息

3

-u 显示类似于 -f 选项的附加信息

4

-e Displays extended information

Stopping Processes

以多种不同的方式结束进程。通常,从基于控制台的命令发送 CTRL + C 键击(默认中断字符)将退出该命令。当进程在前景模式下运行时,此方法有效。

如果一个进程在后台运行,您应当使用 ps 命令获取其作业 ID。然后,您可以使用 kill 命令按如下方式中止该进程:

$ps -f
UID      PID  PPID C STIME    TTY   TIME CMD
amrood   6738 3662 0 10:23:03 pts/6 0:00 first_one
amrood   6739 3662 0 10:22:54 pts/6 0:00 second_one
amrood   3662 3657 0 08:10:53 pts/6 0:00 -ksh
amrood   6892 3662 4 10:51:50 pts/6 0:00 ps -f
$kill 6738
Terminated

此处, kill 命令中止了 first_one 进程。如果一个进程忽略常规中止命令,则您可以按如下方式使用 kill -9 再跟上进程 ID:

$kill -9 6738
Terminated

Parent and Child Processes

每个 Unix 进程都有两个分配给它的 ID 号:进程 ID (pid) 和父进程 ID (ppid)。系统中的每个用户进程都有一个父进程。

你运行的大多数命令都以 shell 作为它们的父进程。检查 ps -f 示例,其中此命令列出了进程 ID 和父进程 ID。

Zombie and Orphan Processes

通常,当子进程被杀死时,父进程将通过 SIGCHLD 信号进行更新。然后,父进程可以根据需要执行一些其他任务或重新启动一个新子进程。但是,有时父进程在子进程被杀死之前被杀死。在这种情况下,“所有进程的父进程”( init 进程)变为新的 PPID(父进程 ID)。在某些情况下,这些进程称为孤儿进程。

当进程被杀死时, ps 列表可能仍会显示处于 Z 状态的进程。这是一个僵尸进程或死进程。该进程已死亡且不再使用。这些进程与孤儿进程不同。它们已完成执行,但仍在进程表中找到条目。

Daemon Processes

守护程序是相关的背景进程,通常使用 root 权限以其他进程的服务请求来运行。

守护程序不控制本机终端。它无法打开 /dev/tty 。如果您执行 "ps -ef" ,并查看 tty 字段,则所有守护程序的 tty 都为 ?

准确地说,守护程序是在后台运行,通常等待能够处理的事情发生。例如,打印机守护程序将等待打印命令。

如果您有需要大量处理时间的程序,那么将它设为守护程序并在后台运行非常值得。

The top Command

top 命令是一个非常有用的工具,可以快速按各种条件对进程进行排序。

它是频繁更新并显示物理和虚拟内存、CPU 使用率、平均负载以及繁忙进程的信息的交互式诊断工具。

以下是运行 top 命令并查看不同进程的 CPU 使用情况统计信息的简单语法 −

$top

Job ID Versus Process ID

后台和暂停进程通常通过 job number (job ID) 进行控制。此号码不同于进程 ID,并且由于更短而使用。

此外,作业可能包含多个进程,这些进程可能会按顺序或同时并行运行。使用作业 ID 比跟踪各个进程更容易。

Unix - Network Communication Utilities

在本章中,我们将详细讨论 Unix 中的网络通信实用程序。在分布式环境中工作时,您需要与远程用户通信,还需要访问远程 Unix 计算机。

有多种 Unix 实用程序可以帮助用户在联网的分布式环境中进行计算。本章列举了几种实用程序。

The ping Utility

ping 命令向网络上可用的主机发送回显请求。使用此命令,您可以查看远程主机是否响应良好。

ping 命令对以下内容很有用 −

  1. 跟踪和隔离硬件和软件问题。

  2. 确定网络和各种外来主机的状态。

  3. 测试、测量和管理网络。

Syntax

以下是使用 ftp 命令的简单语法 −

$ping hostname or ip-address

上述命令在每秒之后开始打印响应。要退出命令,您可以按 CNTRL + C 键来终止命令。

Example

以下是一个检查网络上可用的主机是否可用的示例 −

$ping google.com
PING google.com (74.125.67.100) 56(84) bytes of data.
64 bytes from 74.125.67.100: icmp_seq = 1 ttl = 54 time = 39.4 ms
64 bytes from 74.125.67.100: icmp_seq = 2 ttl = 54 time = 39.9 ms
64 bytes from 74.125.67.100: icmp_seq = 3 ttl = 54 time = 39.3 ms
64 bytes from 74.125.67.100: icmp_seq = 4 ttl = 54 time = 39.1 ms
64 bytes from 74.125.67.100: icmp_seq = 5 ttl = 54 time = 38.8 ms
--- google.com ping statistics ---
22 packets transmitted, 22 received, 0% packet loss, time 21017ms
rtt min/avg/max/mdev = 38.867/39.334/39.900/0.396 ms
$

如果主机不存在,您将收到以下输出 −

$ping giiiiiigle.com
ping: unknown host giiiiigle.com
$

The ftp Utility

这里 ftp 代表 文件传输协议。此实用程序可帮助您将文件从一台计算机上载或下载到另一台计算机。

ftp 实用程序有自己的一组类 Unix 命令。这些命令可帮助您执行以下任务 −

  1. 连接并登录到远程主机。

  2. Navigate directories.

  3. List directory contents.

  4. Put and get files.

  5. asciiebcdicbinary 传输文件。

Syntax

以下是使用 ftp 命令的简单语法 −

$ftp hostname or ip-address

上述命令将提示您输入登录 ID 和密码。一旦通过身份验证,您便可以访问登录帐户的主目录,并能够执行多种命令。

下表列出了几个重要命令 −

Sr.No.

Command & Description

1

put filename 将 filename 从本地计算机上载到远程计算机。

2

get filename 将 filename 从远程计算机下载到本地计算机。

3

mput file list 从本地计算机将多个文件上载到远程计算机。

4

mget file list 从远程计算机将多个文件下载到本地计算机。

5

prompt off 关闭提示。默认情况下,您将收到提示,要求使用 mputmget 命令上载或下载文件。

6

prompt on 打开提示。

7

dir 列出远程计算机当前目录中的所有可用文件。

8

cd dirname 将目录更改为远程计算机上的 dirname。

9

lcd dirname 将目录更改为本地计算机上的 dirname。

10

quit 帮助从当前登录注销。

应该注意的是,所有文件都将下载或上传到当前目录或从当前目录下载或上传。如果您想要将文件上传到特定目录,则需要首先切换到该目录,然后再上传所需的文件。

Example

下面是显示一些命令的工作原理的示例 −

$ftp amrood.com
Connected to amrood.com.
220 amrood.com FTP server (Ver 4.9 Thu Sep 2 20:35:07 CDT 2009)
Name (amrood.com:amrood): amrood
331 Password required for amrood.
Password:
230 User amrood logged in.
ftp> dir
200 PORT command successful.
150 Opening data connection for /bin/ls.
total 1464
drwxr-sr-x   3 amrood   group       1024 Mar 11 20:04 Mail
drwxr-sr-x   2 amrood   group       1536 Mar  3 18:07 Misc
drwxr-sr-x   5 amrood   group        512 Dec  7 10:59 OldStuff
drwxr-sr-x   2 amrood   group       1024 Mar 11 15:24 bin
drwxr-sr-x   5 amrood   group       3072 Mar 13 16:10 mpl
-rw-r--r--   1 amrood   group     209671 Mar 15 10:57 myfile.out
drwxr-sr-x   3 amrood   group        512 Jan  5 13:32 public
drwxr-sr-x   3 amrood   group        512 Feb 10 10:17 pvm3
226 Transfer complete.
ftp> cd mpl
250 CWD command successful.
ftp> dir
200 PORT command successful.
150 Opening data connection for /bin/ls.
total 7320
-rw-r--r--   1 amrood   group       1630 Aug  8 1994  dboard.f
-rw-r-----   1 amrood   group       4340 Jul 17 1994  vttest.c
-rwxr-xr-x   1 amrood   group     525574 Feb 15 11:52 wave_shift
-rw-r--r--   1 amrood   group       1648 Aug  5 1994  wide.list
-rwxr-xr-x   1 amrood   group       4019 Feb 14 16:26 fix.c
226 Transfer complete.
ftp> get wave_shift
200 PORT command successful.
150 Opening data connection for wave_shift (525574 bytes).
226 Transfer complete.
528454 bytes received in 1.296 seconds (398.1 Kbytes/s)
ftp> quit
221 Goodbye.
$

The telnet Utility

有时候我们需要连接到远程 Unix 计算机并在该计算机上远程工作。 Telnet 是一个实用程序,允许某个位置的计算机用户建立连接、登录,然后在另一位置的计算机上工作。

使用 Telnet 登录后,您可以在远程连接的计算机上执行所有活动。以下是 Telnet 会话示例 −

C:>telnet amrood.com
Trying...
Connected to amrood.com.
Escape character is '^]'.

login: amrood
amrood's Password:
*****************************************************
*                                                   *
*                                                   *
*    WELCOME TO AMROOD.COM                          *
*                                                   *
*                                                   *
*****************************************************

Last unsuccessful login: Fri Mar  3 12:01:09 IST 2009
Last login: Wed Mar  8 18:33:27 IST 2009 on pts/10

   {  do your work }

$ logout
Connection closed.
C:>

The finger Utility

finger 命令显示关于指定主机上用户的信息。该主机可以是本地或远程主机。

由于安全原因,其他系统上的 Finger 命令可能被禁用。

以下是使用 finger 命令的简单语法 −

查看本地计算机上所有已登录的用户 −

$ finger
Login     Name       Tty      Idle  Login Time   Office
amrood               pts/0          Jun 25 08:03 (62.61.164.115)

获取本地计算机上特定可用用户的详细信息 −

$ finger amrood
Login: amrood                           Name: (null)
Directory: /home/amrood                 Shell: /bin/bash
On since Thu Jun 25 08:03 (MST) on pts/0 from 62.61.164.115
No mail.
No Plan.

查看远程计算机上所有已登录的用户 −

$ finger @avtar.com
Login     Name       Tty      Idle  Login Time   Office
amrood               pts/0          Jun 25 08:03 (62.61.164.115)

获取远程计算机上特定可用用户的详细信息 −

$ finger amrood@avtar.com
Login: amrood                           Name: (null)
Directory: /home/amrood                 Shell: /bin/bash
On since Thu Jun 25 08:03 (MST) on pts/0 from 62.61.164.115
No mail.
No Plan.

Unix - The vi Editor Tutorial

在本章中,我们将了解 vi 编辑器在 Unix 中如何工作。在 Unix 中有许多方法可以编辑文件。使用面向屏幕的文本编辑器 vi 编辑文件是最好的方法之一。此编辑器使您能够在与文件中其他行同上下文中编辑行。

现在还推出了一个名为 VIM 的 vi 编辑器的改进版本。在此,VIM 代表 *Vi IM*proved。

vi 通常被认为是 Unix 编辑器的实际标准,因为 −

  1. 它通常在 Unix 系统的所有版本中都可用。

  2. 它的实现方式在所有平台上都非常相似。

  3. 它需要的资源很少。

  4. 它比 edex 等其他编辑器更易于使用。

您可以使用 vi 编辑器编辑现有文件或从头开始创建新文件。您还可以使用此编辑器来仅读取文本文件。

Starting the vi Editor

下表列出了 vi 编辑器的基本命令 −

Sr.No.

Command & Description

1

vi filename 如果它尚不存在,则创建新文件,否则打开现有文件。

2

vi -R filename 以只读模式打开现有文件。

3

view filename 以只读模式打开现有文件。

以下是创建新文件 testfile 的示例(如果当前工作目录中尚不存在) −

$vi testfile

上述命令将生成以下输出 −

|
~
~
~
~
~
~
~
~
~
~
~
~
"testfile" [New File]

您会注意到光标后每一行上都有一个 tilde (~)。波浪线表示未使用的行。如果一行不以波浪线开头并且看起来是空白的,则那里有一个空格、制表符、换行符或其他不可查看的字符。

现在您有一个可供开始处理的打开文件。在进一步处理之前,让我们了解一些重要概念。

Operation Modes

使用 vi 编辑器时,我们通常会遇到以下两种模式 −

  1. Command mode - 此模式可以使用户执行管理任务,如保存文件、执行命令、移动光标、剪切(拖拽)和粘贴行或词,以及查找和替换。在此模式下,所键入的任何内容都将解释为命令。

  2. Insert mode - 此模式可以使用户在文件中插入文本。在此模式下键入的所有内容都将解释为输入并置于文件中。

vi 始终从 command mode 开始。要输入文本,您必须处于仅需键入 i 即可进入的插入模式。要退出插入模式,请按 Esc 键,该键将使您返回命令模式。

Hint - 如果您不确定自己处于哪种模式,请按两次 Esc 键;这将使您进入命令模式。您使用 vi 编辑器打开一个文件。首先键入一些字符,然后再进入命令模式以了解其差别。

Getting Out of vi

退出 vi 的命令为 :q 。进入命令模式后,键入冒号、“q”,然后按回车。如果文件以任何方式被修改,编辑器将警告您,并且不允许您退出。要忽略此消息,退出 vi 而不用保存的命令为 :q! 。这使您可以退出 vi,而不保存任何更改。

保存编辑器内容的命令为 :w 。您可以将以上命令与退出命令结合使用,或者使用 :wq 并回车。

最简单的方法是使用 ZZ 命令 save your changes and exit vi 。当您处于命令模式时,键入 ZZZZ 命令的操作方式与 :wq 命令相同。

如果想要指定/说明文件的任何特定名称,可以通过 :w 后指定该名称。例如,如果您想要将您正在处理的文件保存为以 filename2 为名的另一个文件名,您将键入 :w filename2 并回车。

Moving within a File

要在不影响文本的情况下在一个文件中四处移动,您必须处于命令模式(按两次 Esc)。下表列出了一些可供您逐个字符移动的命令 −

Sr.No.

Command & Description

1

k 光标向上移动一行

2

j 光标向下移动一行

3

h 光标向左移动一个字符位置

4

l 光标向右移动一个字符位置

在文件中移动需要考虑以下几点 −

  1. vi 区分大小写。使用命令时,您需要留意大小写。

  2. vi 中的大多数命令都可以通过您希望执行该操作的次数作为前缀。例如, 2j 将光标向下移动两行。

vi 中还有许多其他方法可以在文件中移动。请记住,您必须处于命令模式 ( press Esc twice )。下表列出了一些在文件中四处移动的命令 −

Control Commands

在表中给出了使用控制键执行相应功能的命令:

Editing Files

要编辑该文件,您需要进入插入模式。有许多方法可从命令模式进入插入模式:

Sr.No.

Command & Description

1

i 在当前光标位置之前插入文本

2

I 在当前行的开头插入文本

3

a 在当前光标位置之后插入文本

4

A 在当前行的末尾插入文本

5

o 在光标位置下方创建新行以进行文本输入

6

O 在光标位置上方创建新行以进行文本输入

Deleting Characters

这里列出了用于删除打开文件中的字符和行的重要命令:

Sr.No.

Command & Description

1

x 删除光标位置下的字符

2

X 删除光标位置之前的字符

3

dw 从当前光标位置一直删除到下一个单词

4

d^ 从当前光标位置一直删除到行的开头

5

d$ 从当前光标位置一直删除到行的末尾

6

D 从光标位置一直删除到当前行的末尾

7

dd 删除光标所在的行

如上所述,vi 中的大多数命令均可以有该操作发生的次数作为前置条件。例如, 2x 删除光标位置下的两个字符, 2dd 删除光标所在的两个行。

建议在继续之前练习这些命令。

Change Commands

您还可以更改字符、单词或 vi 中的行,而不删除它们。相关命令如下:

Sr.No.

Command & Description

1

cc 删除行内容,使您进入插入模式。

2

cw 将光标所在的单词从光标转换为单词的 w 小写结尾。

3

r 替换光标下的字符。在输入替换内容后,vi 会返回命令模式。

4

R 从当前光标下的字符开始覆盖多个字符。您必须使用 Esc 来停止覆盖。

5

s 使用您键入的字符替换当前字符。之后,您将保留在插入模式中。

6

S 删除光标所在的线并用新文本替换它。输入新文本后,vi 将保留在插入模式中。

Copy and Paste Commands

您可以从一个地方复制行或单词,然后使用以下命令将它们粘贴到另一个地方 −

Sr.No.

Command & Description

1

yy Copies the current line.

2

yw 从光标所处的 lowercase w 字符复制当前单词,直到单词末尾。

3

p 将复制的文本置于光标后。

4

P 将复制的文本置于光标前。

Advanced Commands

有一些高级命令可以简化日常编辑并更有效地使用 vi −

Word and Character Searching

vi 编辑器有两种搜索: stringcharacter 。对于字符串搜索,使用 /? 命令。当您启动这些命令时,刚刚键入的命令将显示在屏幕的最后一行,您可以在其中键入要查找的特定字符串。

这两个命令仅在搜索发生的方向上有所不同 −

  1. / 命令在文件中向前(向下)搜索。

  2. ? 命令在文件中向后(向上)搜索。

nN 命令分别在相同或相反的方向重复先前的搜索命令。一些字符具有特殊含义。这些字符之前必须加反斜杠 ( \ ) 才能作为搜索表达式的组成部分。

Sr.No.

Character &Description

1

^ 在行首搜索(在搜索表达式开始处使用)。

2

. Matches a single character.

3

匹配前一个字符的零个或多个。

4

$ 行尾(在搜索表达式的结尾处使用)。

5

[ 开始一组匹配或不匹配表达式。

6

&lt; 该项内容放于带反斜杠转义符的表达式中,用于查找单词的结尾或开头。

7

&gt; 这有助于查看上面的“ &lt; ”字符说明。

字符搜索在单行内进行,用于找到命令后输入的字符。 fF 命令仅在当前行中搜索字符。 f 向前搜索, F 向后搜索,光标移动到找到的字符位置。

tT 命令仅在当前行中搜索字符,但对于 t ,光标移动到字符前面的位置,而 T 向后搜索整行直到字符的后面位置。

Set Commands

您可以使用以下 :set 命令来更改 vi 屏幕的外观。在命令模式后,键入 :set ,再输入下述任意命令。

Sr.No.

Command & Description

1

:set ic 搜索时忽略大小写

2

:set ai Sets autoindent

3

:set noai Unsets autoindent

4

:set nu 在左侧显示带行号的行

5

:set sw 设置软件制表符的宽度。例如,您可以使用此命令 — :set sw = 4 设置一个4的shift宽度

6

:set ws 如果设置了 wrapscan 并且在文件底部未找到该单词,系统将尝试在开头处搜索该单词

7

:set wm 如果该选项的值大于零,编辑器将自动“换行”。例如,若要将换行边距设置为两个字符,可键入以下内容: :set wm = 2

8

:set ro 将文件类型更改为“只读”

9

:set term Prints terminal type

10

:set bf 丢弃输入中的控制字符

Running Commands

vi 有能力在编辑器内运行命令。要运行命令,您只需进入命令模式并输入 :! 命令。

例如,如果您希望在尝试使用该文件名保存文件之前检查是否存在文件,您可以键入 :! ls ,您将在屏幕上看到 ls 的输出。

您可以按任意键(或命令的转义序列)返回到 vi 会话。

Replacing Text

替换命令 ( :s/ ) 使您可以快速替换文件中的单词或单词组。以下是替换文本的语法 -

:s/search/replace/g

g 表示全局。此命令的结果是光标行上的所有出现的单词均发生改变。

Important Points to Note

下列要点将让您使用 vi 更加成功 -

  1. 您必须在命令模式下才能使用命令。(随时按两次 Esc,以确保您在命令模式下。)

  2. 您必须小心使用命令。区分大小写。

  3. 您必须在插入模式下才能输入文本。

Unix - What is Shells?

Shell 为您提供了 Unix 系统的接口。它会从您处收集输入,并根据该输入执行程序。当一个程序完成执行时,它会显示该程序的输出。

Shell 是一个环境,我们可以在其中运行我们的命令、程序和 shell 脚本。Shell 有不同的类型,就像操作系统有不同的类型一样。每种类型的 shell 都有一组自己识别的命令和功能。

Shell Prompt

提示符 $ (称为 command prompt )是由 shell 发出的。在提示符显示时,您可以键入一个命令。

在您按下 Enter 后,Shell 会读取您的输入。它通过查看您的输入的第一个单词来确定您想要执行的命令。一个单词是一组不间断的字符。空格和制表符用于分隔单词。

以下是一个 date 命令的简单示例,该命令显示当前日期和时间 −

$date
Thu Jun 25 08:30:19 MST 2009

您可以使用环境教程中说明的环境变量 PS1 来自定义您的命令提示符。

Shell Types

在 Unix 中,主要有两种类型的 shell −

  1. Bourne shell − 如果您正在使用 Bourne 类型的 shell,则 $ 字符是默认提示符。

  2. C shell − 如果您正在使用 C 型 shell,则 % 字符是默认提示。

Bourne Shell 有以下子类别 −

  1. Bourne shell (sh)

  2. Korn shell (ksh)

  3. Bourne Again shell (bash)

  4. POSIX shell (sh)

各种 C 型 shell 如下 −

  1. C shell (csh)

  2. TENEX/TOPS C shell (tcsh)

最初的 Unix shell 是由斯蒂芬·R·伯恩在 20 世纪 70 年代中期编写,当时他在新泽西的 AT&T 贝尔实验室工作。

Bourne Shell 是第一个出现在 Unix 系统上的 shell,因此它被称为“shell”。

Bourne Shell 通常安装为 /bin/sh ,用于大多数版本的 Unix。出于此原因,对于编写可以在不同版本的 Unix 上使用的脚本,它是首选 shell。

在本章中,我们将介绍基于 Borne Shell 的大多数 Shell 概念。

Shell Scripts

shell 脚本的基本概念是命令列表,这些命令按执行顺序列出。一个好的 shell 脚本将有注释,在前面加 # 符号,描述步骤。

有条件测试,例如值 A 大于值 B,循环允许我们遍历海量数据、文件来读取和存储数据,以及读取和存储数据的变量,并且该脚本可能包含函数。

我们将在下一节中编写许多脚本。它将是一个简单的文本文件,其中我们将放置所有命令和大量其他所需结构,告诉 shell 环境做什么以及何时执行。

Shell 脚本和函数都经过解释。这意味着它们未被编译。

Example Script

假设我们创建一个 test.sh 脚本。请注意,所有脚本都应具有 .sh 扩展名。在向脚本添加任何其他内容之前,您需要提醒系统正在启动 shell 脚本。使用 shebang 构造完成此操作。例如 −

#!/bin/sh

这告诉系统后续命令将由 Bourne Shell 执行。它被称为 Shebang,因为 # 符号称为井号,而 ! 符号称为惊叹号。

要在包含这些命令的脚本中,您首先放置 Shebang 行,然后再添加命令 −

#!/bin/bash
pwd
ls

Shell Comments

您可以按如下方式将注释放入脚本中 −

#!/bin/bash

# Author : Zara Ali
# Copyright (c) Tutorialspoint.com
# Script follows here:
pwd
ls

保存上述内容并使脚本可执行 −

$chmod +x test.sh

Shell 脚本现在可以执行 −

$./test.sh

执行后,您会收到以下结果 −

/home/amrood
index.htm  unix-basic_utilities.htm  unix-directories.htm
test.sh    unix-communication.htm    unix-environment.htm

Note − 要执行当前目录中可用的程序,请使用 ./program_name

Extended Shell Scripts

Shell 脚本有几个必需的构造,告诉 shell 环境做什么以及何时执行它。当然,大多数脚本都比上述脚本更复杂。

Shell 毕竟是一种真正的编程语言,具有变量、控制结构等。无论脚本多复杂,它仍然只是一个按顺序执行的命令列表。

以下脚本使用 read 命令,该命令从键盘获取输入并将其分配为 PERSON 变量的值,最后将其打印到 STDOUT。

#!/bin/sh

# Author : Zara Ali
# Copyright (c) Tutorialspoint.com
# Script follows here:

echo "What is your name?"
read PERSON
echo "Hello, $PERSON"

以下是脚本的一个示例运行 −

$./test.sh
What is your name?
Zara Ali
Hello, Zara Ali
$

Unix - Using Shell Variables

在本章中,我们将学习如何在 Unix 中使用 Shell 变量。变量是一个字符字符串,我们为其分配一个值。分配的值可以是数字、文本、文件名、设备或任何其他类型的数据。

变量不过是实际数据的指针。shell 允许你创建、分配和删除变量。

Variable Names

变量的名称只能包含字母(a 到 z 或 A 到 Z)、数字(0 到 9)或下划线字符(_)。

根据惯例,Unix shell 变量的名称使用大写。

以下示例是有效的变量名称 −

_ALI
TOKEN_A
VAR_1
VAR_2

以下是无效变量名称的示例 −

2_VAR
-VARIABLE
VAR1-VAR2
VAR_A!

你无法使用 !*- 等其他字符的原因是,这些字符对 shell 具有特殊含义。

Defining Variables

变量的定义如下 −

variable_name=variable_value

例如 -

NAME="Zara Ali"

以上示例定义了变量 NAME 并为其分配值 "Zara Ali"。此类型的变量称为 scalar variables 。标量变量一次只能保存一个值。

Shell 允许你将任何值存储在变量中。例如 −

VAR1="Zara Ali"
VAR2=100

Accessing Values

要访问存储在变量中的值,请在其名称前加上美元符号 ( $ ) −

例如,以下脚本将访问定义的变量 NAME 的值并将其打印到 STDOUT −

#!/bin/sh

NAME="Zara Ali"
echo $NAME

以上脚本将生成以下值 −

Zara Ali

Read-only Variables

Shell 通过使用只读命令提供了一种将变量标记为只读的方法。变量标记为只读后,其值无法更改。

例如,以下脚本在尝试更改 NAME 的值时生成错误 −

#!/bin/sh

NAME="Zara Ali"
readonly NAME
NAME="Qadiri"

上述脚本会生成以下结果 -

/bin/sh: NAME: This variable is read only.

Unsetting Variables

取消或删除变量会指示 shell 从其跟踪的变量列表中删除该变量。取消变量后,你将无法访问变量中存储的值。

以下是使用 unset 命令取消已定义变量的语法 −

unset variable_name

以上命令取消已定义变量的值。以下是一个简单的示例,演示了该命令如何工作 −

#!/bin/sh

NAME="Zara Ali"
unset NAME
echo $NAME

以上示例不打印任何内容。你无法使用 unset 命令取消标记为 readonlyunset 变量。

Variable Types

当 shell 运行时,存在三种主要类型的变量 −

  1. Local Variables − 局部变量是当前 shell 实例中存在的一个变量。启动 shell 的程序无法访问它。它们是通过命令提示符设置的。

  2. Environment Variables − 子 shell 的所有进程都可以使用环境变量。某些程序需要环境变量才能正常运行。通常,shell 脚本仅定义其运行的程序所需的那些环境变量。

  3. Shell Variables − shell 变量是 shell 设置的特殊变量,shell 需要它才能正常运行。其中有些变量是环境变量,而另一些则是局部变量。

Unix - Special Variables

在本章中,我们将详细讨论 Unix 中的特殊变量。在以前的章节中,我们已经了解了在变量名中使用某些非字母数字字符时的注意事项。这是因为这些字符用于 Unix 特殊变量的名称中。这些变量是为特定函数保留的。

例如,字符 $ 表示当前 shell 的进程 ID 号或 PID −

$echo $$

以上命令会写入当前 shell 的 PID −

29949

下表显示了可以在 shell 脚本中使用的许多特殊变量 −

Sr.No.

Variable & Description

1

$0 当前脚本的文件名。

2

$n 这些变量对应于调用某个脚本时所带的参数。其中 n 是一个正的十进制数,对应于参数的位置(第一个参数是 $1,第二个参数是 $2,依此类推)。

3

$# 提供给脚本的参数数量。

4

$ *所有参数都使用双引号引起来。如果脚本接收两个参数,$ * 等于 $1 $2。

5

$@ 所有参数都单独用双引号引起来。如果脚本接收两个参数,$@ 等于 $1 $2。

6

$? 最后执行的命令的退出状态。

7

$$ 当前 shell 的进程号。对于 shell 脚本来说,这是它们正在执行之下的进程 ID。

8

$! 上一个后台命令的进程号。

Command-Line Arguments

命令行参数 $1、$2、$3、…​$9 是位置参数,其中 $0 指向实际命令、程序、shell 脚本或函数,$1、$2、$3、…​$9 作为指令的参数。

以下脚本使用与命令行相关的各种特殊变量 −

#!/bin/sh

echo "File Name: $0"
echo "First Parameter : $1"
echo "Second Parameter : $2"
echo "Quoted Values: $@"
echo "Quoted Values: $*"
echo "Total Number of Parameters : $#"

下面是上述脚本的一个运行示例——

$./test.sh Zara Ali
File Name : ./test.sh
First Parameter : Zara
Second Parameter : Ali
Quoted Values: Zara Ali
Quoted Values: Zara Ali
Total Number of Parameters : 2

Special Parameters $* and $@

有特殊的参数允许一次访问所有命令行参数。 $ * 和 $@ 二者扮演同样的角色,除非它们被包含在双引号中, ""

这两个参数都指定了命令行参数。但是,"$*" 特殊参数将整个列表作为一个参数,参数之间有空格,而 "$@" 特殊参数将整个列表分解为单独的参数。

我们可以编写如下所示的 shell 脚本以使用 $* 或 $@ 特殊参数处理任意数量的命令行参数——

#!/bin/sh

for TOKEN in $*
do
   echo $TOKEN
done

下面是上述脚本的一个运行示例——

$./test.sh Zara Ali 10 Years Old
Zara
Ali
10
Years
Old

Note ——此处 do…​done 是将涵盖在后续教程中的某种循环。

Exit Status

$? 变量表示上一个命令的退出状态。

退出状态是在命令执行完毕后返回的数值。按照规则,大多数命令如果成功会返回退出状态 0,如果失败会返回 1。

有些命令会出于特定原因返回其他退出状态。例如,有些命令会区分不同类型的错误,并会根据特定类型的错误而返回不同的退出值。

以下是成功命令的示例——

$./test.sh Zara Ali
File Name : ./test.sh
First Parameter : Zara
Second Parameter : Ali
Quoted Values: Zara Ali
Quoted Values: Zara Ali
Total Number of Parameters : 2
$echo $?
0
$

Unix - Using Shell Arrays

在本章中,我们将讨论如何在 Unix 中使用 shell 数组。shell 变量有足够的能力来保存单个值。这些变量称为标量变量。

Shell 支持一种不同的变量类型,称为 array variable 。这可以同时保存多个值。数组提供了一种对变量集合进行分组的方法。你可以使用一个保存所有其他变量的单一数组变量,而不必为每个必需的变量创建一个新名称。

为命名数组时,所有讨论过的 Shell 变量的命名规则均适用。

Defining Array Values

数组变量和标量变量之间的差异可以解释如下。

假设你要将各种学生的姓名表示为一组变量。每个单独变量都是一个标量变量,如下所示 –

NAME01="Zara"
NAME02="Qadir"
NAME03="Mahnaz"
NAME04="Ayan"
NAME05="Daisy"

我们可以使用一个单独的数组来存储所有上面提到的姓名。以下是创建数组变量的最简单方法。这有助于将其一个索引值分配给其一个索引。

array_name[index]=value

此处,array_name 是数组的名称,index 是你要设置的数组中该项的索引,而 value 是你要为该项设置的值。

例如,以下命令 –

NAME[0]="Zara"
NAME[1]="Qadir"
NAME[2]="Mahnaz"
NAME[3]="Ayan"
NAME[4]="Daisy"

如果你使用的是 ksh shell,则以下为数组初始化的语法 –

set -A array_name value1 value2 ... valuen

如果你使用的是 bash shell,则以下为数组初始化的语法 –

array_name=(value1 ... valuen)

Accessing Array Values

在你设置任何数组变量后,访问方式如下 –

${array_name[index]}

此处,array_name 是数组的名称,而 index 是要访问的值的索引。以下是一个了解这个概念的例子 –

#!/bin/sh

NAME[0]="Zara"
NAME[1]="Qadir"
NAME[2]="Mahnaz"
NAME[3]="Ayan"
NAME[4]="Daisy"
echo "First Index: ${NAME[0]}"
echo "Second Index: ${NAME[1]}"

以上示例将生成以下结果 –

$./test.sh
First Index: Zara
Second Index: Qadir

你可以使用以下其中一种方式访问数组中的所有项 –

${array_name[*]}
${array_name[@]}

此处 array_name 是您感兴趣的数组的名称。以下示例将帮助您理解该概念 -

#!/bin/sh

NAME[0]="Zara"
NAME[1]="Qadir"
NAME[2]="Mahnaz"
NAME[3]="Ayan"
NAME[4]="Daisy"
echo "First Method: ${NAME[*]}"
echo "Second Method: ${NAME[@]}"

以上示例将生成以下结果 –

$./test.sh
First Method: Zara Qadir Mahnaz Ayan Daisy
Second Method: Zara Qadir Mahnaz Ayan Daisy

Unix - Shell Basic Operators

每个 shell 都支持各种运算符。我们将在本章详细讨论 Bourne shell(默认 shell)。

我们现在将讨论以下运算符 -

  1. Arithmetic Operators

  2. Relational Operators

  3. Boolean Operators

  4. String Operators

  5. File Test Operators

Bourne shell 最初没有任何执行简单算术运算的机制,但它使用外部程序,要么是 awk ,要么是 expr

以下示例显示了如何添加两个数字 -

#!/bin/sh

val=`expr 2 + 2`
echo "Total value : $val"

上述脚本会生成以下结果 -

Total value : 4

在添加时需要考虑以下几点 -

  1. 运算符和表达式之间必须有空格。例如,2+2 不正确;应写成 2 + 2。

  2. 完整的表达式应包含在 ‘ ‘ 之间,称为反引号。

Arithmetic Operators

Bourne Shell 支持以下算术运算符。

假设变量 a 为 10,变量 b 为 20,那么 -

Operator

Description

Example

+ (Addition)

添加运算符两侧的值

expr $a + $b 将得到 30

- (Subtraction)

从左操作数中减去右操作数

expr $a - $b 将得到 -10

* (Multiplication)

乘以运算符两侧的值

expr $a \* $b 将得到 200

/ (Division)

将左操作数除以右操作数

expr $b / $a 将得到 2

% (Modulus)

将左手操作数除以右手操作数,并返回余数

expr $b % $a 将给出 0

= (Assignment)

将右操作符赋值给左操作符

a = $b 会将 b 的值赋值给 a

== (Equality)

对比两个数字,如果二者相同则返回 true。

[ $a == $b ] 会返回 false。

!= (Not Equality)

对比两个数字,如果二者不同则返回 true。

[ $a != $b ] 会返回 true。

了解到所有条件表达式都应位于带空格的方括号当中非常重要,例如 [ $a == $b ] 是正确的,而 [$a==$b] 是不正确的。

所有算术运算都使用长整数进行。

Relational Operators

Bourne Shell 支持以下特定于数值的关联运算符。这些运算符不适用于字符串值,除非其值为数值。

例如,以下运算符将用于检查 10 和 20 之间的关系以及 "10" 和 "20" 之间的关系,但不用于检查 "ten" 和 "twenty" 之间的关系。

假设变量 a 为 10,变量 b 为 20,那么 -

Operator

Description

Example

-eq

检查两个操作数的值是否相等;如果相等,则条件为 true。

[ $a -eq $b ] 不为 true。

-ne

检查两个操作数的值是否相等;如果值不相等,则条件为 true。

[ $a -ne $b ] 为 true。

-gt

检查左操作数的值是否大于右操作数的值;如果是,则条件变为 true。

[ $a -gt $b ] 不为 true。

-lt

检查左操作数的值是否小于右操作数的值;如果是,则条件变为 true。

[ $a -lt $b ] 为 true。

-ge

检查左操作数的值是否大于或等于右操作数的值;如果是,则条件变为 true。

[ $a -ge $b ] 为假。

-le

检查左操作数的值是否小于或等于右操作数的值;如果是,则条件变为 true。

[ $a -le $b ] 为真。

了解所有条件表达式都应放在方括号中且周围带有空格非常重要。例如, [ $a ⇐ $b ] 正确,而 [$a ⇐ $b] 错误。

Boolean Operators

Bourne Shell 支持以下布尔运算符。

假设变量 a 为 10,变量 b 为 20,那么 -

Operator

Description

Example

!

这是逻辑否定。这会将真条件反转为假,反之亦然。

[ ! false ] 为真。

-o

这是逻辑 OR 。如果其中一个操作数为真,则条件变为真。

[ $a -lt 20 -o $b -gt 100 ] 为真。

-a

这是逻辑 AND 。如果两个操作数都为真,则条件变为真,否则为假。

[ $a -lt 20 -a $b -gt 100 ] 为假。

String Operators

Bourne Shell 支持以下字符串运算符。

假设 a 变量保存了“abc”, b 变量保存了“efg”,则 −

Operator

Description

Example

=

检查两个操作数的值是否相等;如果相等,则条件为 true。

[ $a = $b ] 为假。

!=

检查两个操作数的值是否相等;如果值不相等,则条件变为真。

[ $a != $b ] 为真。

-z

检查给定的字符串操作数大小是否为零;如果长度为零,则返回真。

[ -z $a ] 为假。

-n

检查给定字符串操作数大小是否非零;如果长度不为零,则返回真。

[ -n $a ] 非假。

str

检查 str 是否不是空字符串;如果是空字符串,则返回假。

[ $a ] 非假。

File Test Operators

我们有几个运算符,可用于测试与 Unix 文件关联的各种属性。

假设变量 file 保存现有文件名“test”,其大小为 100 字节,对 − 的权限为 readwriteexecute

Operator

Description

Example

-b file

检查文件是否为块特殊文件;如果是,则条件变为真。

[ -b $file ] 为假。

-c file

检查文件是否为字符特殊文件;如果是,则条件变为真。

[ -c $file ] 为假。

-d file

检查文件是否为目录;如果是,则条件变为真。

[ -d $file ] 非真。

-f file

检查文件是否为普通文件,而不是目录或特殊文件;如果是,则条件变为真。

[ -f $file ] 为真。

-g file

检查文件是否设置了其 set 组 ID (SGID) 位;如果是,则条件变为真。

[ -g $file ] 为假。

-k file

检查文件是否设置了其粘滞位;如果是,则条件变为真。

[ -k $file ] 为假。

-p file

检查文件是否为有名管道;如果是,则条件变为真。

[ -p $file ] 为假。

-t file

检查文件描述符是否已打开并与终端关联;如果是,则条件变为真。

[ -t $file ] 为假。

-u file

检查文件是否设置了设置用户 ID (SUID) 位;如果是,则条件变为真。

[ -u $file ] 为假。

-r file

检查文件是否可读;如果是,则条件变为真。

[ -r $file ] 为真。

-w file

检查文件是否可写;如果是,则条件变为真。

[ -w $file ] 为真。

-x file

检查文件是否可执行;如果是,则条件变为真。

[ -x $file ] 为真。

-s file

检查文件的大小是否大于 0;如果是,则条件变为真。

[ -s $file ] 为真。

-e file

检查文件是否存在;即使文件是目录但存在,也是真的。

[ -e $file ] 为真。

C Shell Operators

以下链接将给您简要了解 C Shell 算符−

Korn Shell Operators

以下链接帮助您理解康恩壳算符−

Unix - Shell Decision Making

在本章中,我们将了解 Unix 中的 shell 决策制定。在编写 shell 脚本时,可能会遇到需要在给定的两条路径中选择一条路径的情况。因此,您需要使用条件语句,让您的程序可以做出正确的决策并执行正确操作。

Unix Shell 支持条件语句,用于根据不同的条件执行不同的操作。现在,我们将了解此处的两个决策制定语句 −

  1. The if…​else statement

  2. The case…​esac statement

The if…​else statements

if else 语句是有用的决策制定语句,可用于从给定的一组选项中选择一个选项。

Unix Shell 支持以下 if…else 语句形式 −

大多数 if 语句使用上一章中讨论的关系运算符来检查关系。

The case…​esac Statement

您可以使用多个 if…​elif 语句执行多路分支。但是,这并不总是一种最佳解决方案,尤其是当所有分支都依赖于单个变量的值时。

Unix Shell 支持 case…​esac 语句,该语句可以精确地处理这种情况,而且比重复的 if…​elif 语句更高效。

只有一种形式的 case…​esac 语句,此处已对其进行了详细描述 −

Unix shell 中的 case…​esac 语句与我们在其他编程语言中有的 switch…​case 语句非常类似,例如 CC++PERL 等。

Unix - Shell Loop Types

在本章节中,我们将讨论 Unix 中的 shell 循环。循环是一个功能强大的编程工具,使您能够重复执行一组命令。在本章中,我们将检查 shell 编程人员可用的以下类型的循环−

根据具体情况,您将使用不同的循环。例如, while 循环会在给定条件保持真时执行给定的命令; until 循环会在给定条件变为真时执行。

一旦您有了良好的编程习惯,您将获得专业知识,因此,开始根据具体情况使用适当的循环。此处, whilefor 循环在大多数其他编程语言中可用,如 CC++PERL 等。

Nesting Loops

所有循环都支持嵌套概念,这意味着您可以将一个循环放入另一个类似循环或不同循环中。根据您的要求,这种嵌套可以进行无限次。

以下是一个嵌套 while 循环的示例。其他循环可以类似地根据编程要求进行嵌套−

Nesting while Loops

可以将一个 while 循环用作另一个 while 循环的主体的一部分。

Syntax

while command1 ; # this is loop1, the outer loop
do
   Statement(s) to be executed if command1 is true

   while command2 ; # this is loop2, the inner loop
   do
      Statement(s) to be executed if command2 is true
   done

   Statement(s) to be executed if command1 is true
done

Example

这里有一个简单的循环嵌套示例。让我们在您用来计数到九的循环中添加另一个倒数循环−

#!/bin/sh

a=0
while [ "$a" -lt 10 ]    # this is loop1
do
   b="$a"
   while [ "$b" -ge 0 ]  # this is loop2
   do
      echo -n "$b "
      b=`expr $b - 1`
   done
   echo
   a=`expr $a + 1`
done

这将产生以下结果。重要的是要注意 echo -n 在此处如何工作的。此处 -n 选项让 echo 避免打印一个新行字符。

0
1 0
2 1 0
3 2 1 0
4 3 2 1 0
5 4 3 2 1 0
6 5 4 3 2 1 0
7 6 5 4 3 2 1 0
8 7 6 5 4 3 2 1 0
9 8 7 6 5 4 3 2 1 0

Unix - Shell Loop Control

在本章中,我们将讨论 Unix 中的 Shell 循环控制。迄今为止,您已了解如何创建循环并使用循环来完成不同任务。有时,您需要停止循环或跳过循环迭代。

在本章中,我们将学习用于控制 Shell 循环的以下两个语句 −

  1. The break statement

  2. The continue statement

The infinite Loop

所有循环都有一个有限的生命周期,并且根据循环的不同,当条件为假或真时,它们就会退出。

如果未满足所需条件,则循环可能无限期地继续。无限期执行并且不会终止的循环将无限次执行。因此,这样的循环称为无限循环。

Example

下面是一个简单的示例,它使用 while 循环来显示数字 0 到 9 −

#!/bin/sh

a=10

until [ $a -lt 10 ]
do
   echo $a
   a=`expr $a + 1`
done

此循环无限期地继续,因为 a 始终为 greater thanequal to 10 ,并且它永远不会小于 10。

The break Statement

break 语句用于在执行到 break 语句的所有代码行后终止整个循环的执行。然后,它向下执行到循环结束后面的代码。

Syntax

以下 break 语句用于退出循环 −

break

break 命令还可以使用此格式从嵌套循环中退出 −

break n

这里 n 指定了退出的 nth 循环。

Example

下面是一个简单的示例,它显示了当 a 变为 5 时循环如何终止 −

#!/bin/sh

a=0

while [ $a -lt 10 ]
do
   echo $a
   if [ $a -eq 5 ]
   then
      break
   fi
   a=`expr $a + 1`
done

执行后,您会收到以下结果 −

0
1
2
3
4
5

下面是嵌套 for 循环的一个简单示例。如果 var1 equals 2var2 equals 0

#!/bin/sh

for var1 in 1 2 3
do
   for var2 in 0 5
   do
      if [ $var1 -eq 2 -a $var2 -eq 0 ]
      then
         break 2
      else
         echo "$var1 $var2"
      fi
   done
done

执行后,您将收到以下结果。在内部循环中,您有一个 break 命令,参数为 2。这表示,如果满足条件,则您应该退出外部循环,并最终也退出内部循环。

1 0
1 5

The continue statement

continue 语句类似于 break 命令,只不过它导致退出循环的当前迭代,而不是整个循环。

出现错误但您希望尝试执行循环的下一个迭代时,此语句很有用。

Syntax

continue

与 break 语句一样,可以向 continue 命令提供一个整数参数,以便从嵌套循环中跳过命令。

continue n

这里 n 指定了继续执行的 nth 循环。

Example

以下循环使用 continue 语句,它从 continue 语句返回并开始处理下一条语句 −

#!/bin/sh

NUMS="1 2 3 4 5 6 7"

for NUM in $NUMS
do
   Q=`expr $NUM % 2`
   if [ $Q -eq 0 ]
   then
      echo "Number is an even number!!"
      continue
   fi
   echo "Found odd number"
done

执行后,您会收到以下结果 −

Found odd number
Number is an even number!!
Found odd number
Number is an even number!!
Found odd number
Number is an even number!!
Found odd number

Unix - Shell Substitution

What is Substitution?

当 Shell 遇到包含一个或多个特殊字符的表达式时,它会执行替换。

Example

此处,变量的打印值被替换为其值。同时, "\n" 被替换为换行符 −

#!/bin/sh

a=10
echo -e "Value of a is $a \n"

您会收到以下结果。在此处 -e 选项允许解释反斜杠转义符。

Value of a is 10

以下是没有 -e 选项的结果 −

Value of a is 10\n

本教程中用到了可用于 echo 命令的下列转义序列——

Sr.No.

Escape & Description

1

\\ backslash

2

\a alert (BEL)

3

\b backspace

4

*\c * suppress trailing newline

5

*\f * form feed

6

\n new line

7

\r carriage return

8

*\t * horizontal tab

9

*\v * vertical tab

您可以使用 -E 选项禁用对反斜杠转义符的解释(默认)。

您可以使用 -n 选项禁用插入换行符。

Command Substitution

命令替换是一种机制,通过该机制,Shell 执行一组给定的命令,然后用这些命令的输出替换命令。

Syntax

当给出命令时,执行命令替换 −

`command`

在执行命令替换时,请确保使用反引号,而不是单引号字符。

Example

命令替换通常用于将命令的输出分配给变量。以下每个示例都展示了命令替换 −

#!/bin/sh

DATE=`date`
echo "Date is $DATE"

USERS=`who | wc -l`
echo "Logged in user are $USERS"

UP=`date ; uptime`
echo "Uptime is $UP"

执行后,您会收到以下结果 −

Date is Thu Jul  2 03:59:57 MST 2009
Logged in user are 1
Uptime is Thu Jul  2 03:59:57 MST 2009
03:59:57 up 20 days, 14:03,  1 user,  load avg: 0.13, 0.07, 0.15

Variable Substitution

变量替换使 Shell 程序员能够根据变量的状态来操作变量的值。

以下是所有可能替换操作的表格 −

Sr.No.

Form & Description

1

${var} 替换 var 的值。

2

${var:-word} 如果 var 为空或未设置,那么 word 会替换 var 。var 的值不会改变。

3

${var:=word} 如果 var 为空或未设置,那么 var 会被设置为 word 的值。

4

${var:?message} 如果 var 为空或未设置,那么消息会被打印到标准错误中。这会检查变量是否正确设置。

5

${var:+word} 如果设置了 var,则将 word 替换为 var。var 的值不会发生改变。

Example

以下示例显示了上述替换的各种状态 −

#!/bin/sh

echo ${var:-"Variable is not set"}
echo "1 - Value of var is ${var}"

echo ${var:="Variable is not set"}
echo "2 - Value of var is ${var}"

unset var
echo ${var:+"This is default value"}
echo "3 - Value of var is $var"

var="Prefix"
echo ${var:+"This is default value"}
echo "4 - Value of var is $var"

echo ${var:?"Print this message"}
echo "5 - Value of var is ${var}"

执行后,您会收到以下结果 −

Variable is not set
1 - Value of var is
Variable is not set
2 - Value of var is Variable is not set

3 - Value of var is
This is default value
4 - Value of var is Prefix
Prefix
5 - Value of var is Prefix

Unix - Shell Quoting Mechanisms

在本章中,我们将详细讨论 Shell 引用机制。我们将从讨论元字符开始。

The Metacharacters

Unix Shell 提供了各种元字符,在任何 Shell 脚本中使用它们时都具有特殊含义,并且会导致某个单词终止,除非加引号。

例如, ? 与目录中列出的单个字符匹配,而 * 与多个字符匹配。以下是大多数 shell 特殊字符(也称为元字符)的列表 −

* ? [ ] ' " \ $ ; & ( ) | ^ < > new-line space tab

通过在字符前加上 \ 可以引用一个字符(即,使它代表自身)。

Example

以下示例显示了如何打印 *?

#!/bin/sh

echo Hello; Word

执行后,您会收到以下结果 −

Hello
./test.sh: line 2: Word: command not found

shell returned 127

我们现在尝试使用一个带引号的字符 −

#!/bin/sh

echo Hello\; Word

执行后,您会收到以下结果 −

Hello; Word

$ 符号是元字符之一,因此必须加上引号以避免 shell 对其进行特殊处理 −

#!/bin/sh

echo "I have \$1200"

执行后,您会收到以下结果 −

I have $1200

下表列出了四种引用形式 −

Sr.No.

Quoting & Description

1

Single quote 这些引号之间的所有特殊字符都会失去其特殊含义。

2

Double quote 这些引号之间的大多数特殊字符都会失去其特殊含义,但以下情况例外 −$`\$\'\"\\

3

Backslash 任何紧跟反斜槓之后的字符都将失去其特殊含义。

4

Back quote 嵌入反引号中的任何内容都将被视为命令并被执行。

The Single Quotes

考虑一个包含许多特殊 shell 字符的 echo 命令−

echo <-$1500.**>; (update?) [y|n]

在每个特殊字符前放置反斜杠既繁琐又增加了阅读的难度−

echo \<-\$1500.\*\*\>\; \(update\?\) \[y\|n\]

有一种简单的方法可以引用一大组字符。在字符串的开头和结尾处放置单引号(')−

echo '<-$1500.**>; (update?) [y|n]'

单引号中的字符就像在每个字符前加上反斜杠一样被引用。使用这种方法,echo 命令可以正常显示。

如果字符串中出现单引号需要输出,你不应该将整个字符串放在单引号中,而应在该字符前加上反斜杠(\),如下所示−

echo 'It\'s Shell Programming

The Double Quotes

尝试执行以下 shell 脚本。此 shell 脚本使用了单引号−

VAR=ZARA
echo '$VAR owes <-$1500.**>; [ as of (`date +%m/%d`) ]'

执行后,您会收到以下结果 −

$VAR owes <-$1500.**>; [ as of (`date +%m/%d`) ]

这不是预期要显示的内容。很明显,单引号可以防止变量替换。如果你希望替换变量值并让引号正常工作,那么你需要将命令放入双引号中,如下所示−

VAR=ZARA
echo "$VAR owes <-\$1500.**>; [ as of (`date +%m/%d`) ]"

执行后,您会收到以下结果 −

ZARA owes <-$1500.**>; [ as of (07/02) ]

双引号消除了除以下字符之外所有字符的特殊含义−

  1. $ for parameter substitution

  2. Backquotes for command substitution

  3. \$ 启用字面美元符号

  4. \` 启用字面反引号

  5. \" 启用嵌入双引号

  6. \\ 启用嵌入反斜杠

  7. 所有其他 \ 字符都是字面的(非特殊)

单引号中的字符就像在每个字符前加上反斜杠一样被引用。这有助于 echo 命令正确显示。

如果字符串中出现单引号需要输出,你不应该将整个字符串放在单引号中,而应在该字符前加上反斜杠(\),如下所示−

echo 'It\'s Shell Programming'

The Backquotes

将任何 Shell 命令放入 backquotes 中会执行该命令。

Syntax

以下是将任何 Shell command 放入反引号中的简单语法−

var=`command`

Example

在以下示例中, date 命令被执行,而产生的结果存储在 DATA 变量中。

DATE=`date`

echo "Current Date: $DATE"

执行后,您会收到以下结果 −

Current Date: Thu Jul  2 05:28:45 MST 2009

Unix - Shell Input/Output Redirections

在本章中,我们将详细讨论 Shell 输入/输出重定向。大多数 Unix 系统命令从你的终端获取输入并将产生的输出发回你的终端。一个命令通常从标准输入读取其输入,而默认情况下你的终端就是标准输入。类似地,一个命令通常将它的输出写入到标准输出,而默认情况下你的终端也是标准输出。

Output Redirection

本来打算用作标准输出的命令输出可以轻松地改发到文件。这种功能称为输出重定向。

如果符号 > file 附加到任何通常将它的输出写入到标准输出的命令,那么该命令的输出将被写入到 file 而不是你的终端。

查看以下 who 命令,它将命令的完整输出重定向到 users 文件。

$ who > users

请注意,没有输出出现在终端上。这是因为输出已被从默认的标准输出设备(终端)重定向到了指定的文件中。你可以查看 users 文件以获得完整内容 −

$ cat users
oko         tty01   Sep 12 07:30
ai          tty15   Sep 12 13:32
ruth        tty21   Sep 12 10:10
pat         tty24   Sep 12 13:07
steve       tty25   Sep 12 13:03
$

如果一个命令的输出被重定向到一个文件且该文件已经包含了一些数据,那么那些数据将被丢失。考虑以下示例 −

$ echo line 1 > users
$ cat users
line 1
$

您可以使用 >> 运算符将输出附加到现有文件中,如下所示:

$ echo line 2 >> users
$ cat users
line 1
line 2
$

Input Redirection

与将命令的输出重定向到文件一样,也可以将命令的输入重定向到文件。与 greater-than character > 用于输出重定向一样, less-than character < 用于重定向命令的输入。

通常从标准输入获取输入的命令可以以这种方式将它们的输入重定向到一个文件中。例如,要计算上面由用户生成的文本文件中的行数,您可以执行如下命令:

$ wc -l users
2 users
$

执行后,您将收到以下输出。您可以通过将 wc 命令的标准输入从文件用户重定向来计算文件中行的数量:

$ wc -l < users
2
$

请注意,wc 命令产生的两种形式的输出存在差异。在第一种情况下,文件 users 的名称用引号与行数列在一起;在第二种情况下,则没有。

在第一种情况下,wc 知道它正在从文件 users 读取输入。在第二种情况下,它只知道它正在从标准输入读取输入,因此它不显示文件名称。

Here Document

here document 用于将输入重定向到交互式 shell 脚本或程序。

我们可以在 shell 脚本中运行一个交互式程序,而无需用户操作,方法是为交互式程序或交互式 shell 脚本提供所需的输入。

here 文档的一般形式是:

command << delimiter
document
delimiter

这里 shell 将 << 运算符解释为一个指令,指在它找到包含指定分界符的行之前,读取输入。然后将包含分界符的行之前的全部输入行送入命令的标准输入。

分界符告诉 shell, here 文档已完成。如果没有分界符,shell 将继续永远读取输入。分界符必须是不包含空格或选项卡的单个单词。

以下是对命令 wc -l 的输入,用于计算总行数:

$wc -l << EOF
   This is a simple lookup program
	for good (and bad) restaurants
	in Cape Town.
EOF
3
$

您可以使用 here document 来使用您的脚本打印多行,如下所示:

#!/bin/sh

cat << EOF
This is a simple lookup program
for good (and bad) restaurants
in Cape Town.
EOF

执行后,您会收到以下结果 −

This is a simple lookup program
for good (and bad) restaurants
in Cape Town.

以下脚本运行了文本编辑器 vi 的一个会话,并将输入保存在文件 test.txt 中。

#!/bin/sh

filename=test.txt
vi $filename <<EndOfCommands
i
This file was created automatically from
a shell script
^[
ZZ
EndOfCommands

如果您使用 vim 作为 vi 运行此脚本,那么您可能会看到如下的输出:

$ sh test.sh
Vim: Warning: Input is not from a terminal
$

运行脚本后,您应该看到以下内容被添加到文件 test.txt 中:

$ cat test.txt
This file was created automatically from
a shell script
$

Discard the output

有时您需要执行命令,但是您不想在屏幕上显示输出。在这种情况下,您可以通过将输出重定向到文件 /dev/null 来丢弃输出:

$ command > /dev/null

这里 command 是您要执行的命令的名称。文件 /dev/null 是一个特殊文件,自动丢弃其所有输入。

要丢弃命令的输出及其错误输出,请使用标准重定向将 STDERR 重定向到 STDOUT

$ command > /dev/null 2>&1

此处 2 表示 STDERR ,而 1 表示 STDOUT 。您可以将消息重定向到 STDERR,方法是将 STDOUT 重定向到 STDERR,如下所示 −

$ echo message 1>&2

Redirection Commands

以下是可用于重定向的命令的完整列表 −

Sr.No.

Command & Description

1

pgm &gt; file 程序的输出重定向到文件

2

pgm &lt; file 程序 pgm 从文件读取其输入

3

pgm &gt;&gt; file 程序的输出附加到文件

4

n &gt; file 从描述符为 n 的流输出重定向到文件

5

n &gt;&gt; file 从描述符为 n 的流输出附加到文件

6

n &gt;&amp; m 将从流 n 中输出合并到流 m

7

n &lt;&amp; m 将从流 n 中输入合并到流 m

8

&lt;&lt; tag 标准输入从此处通过行首的下一个标记获得

9

*

请注意,文件描述符 0 通常是标准输入 (STDIN), 1 是标准输出 (STDOUT),而 2 是标准错误输出 (STDERR)。

Unix - Shell Functions

在本章中,我们将详细讨论 Shell 函数。使用函数可以将脚本的整体功能分解为更小的逻辑子部分,然后根据需要调用它们来执行各自的任务。

使用函数执行重复任务是创建 code reuse 的绝佳方式。这是现代面向对象编程原则的重要组成部分。

Shell 函数类似于其他编程语言中的子例程、过程和函数。

Creating Functions

要声明函数,只需使用以下语法:

function_name () {
   list of commands
}

函数的名称是 function_name ,您将在脚本中的其他地方使用它来调用此函数。函数名称后面必须紧跟括号,然后是一对大括号中包含的命令列表。

Example

以下示例显示了函数的用法:

#!/bin/sh

# Define your function here
Hello () {
   echo "Hello World"
}

# Invoke your function
Hello

执行后,您将收到以下输出:

$./test.sh
Hello World

Pass Parameters to a Function

您可以在调用函数时定义一个接受参数的函数。这些参数将由 $1$2 等表示。

以下是一个示例,其中我们传递两个参数 Zara 和 Ali,然后在函数中捕获并打印这些参数。

#!/bin/sh

# Define your function here
Hello () {
   echo "Hello World $1 $2"
}

# Invoke your function
Hello Zara Ali

执行后,您会收到以下结果 −

$./test.sh
Hello World Zara Ali

Returning Values from Functions

如果您从函数内部执行 exit 命令,其效果不仅会终止函数的执行,还会终止调用该函数的 shell 程序。

如果您只想终止函数的执行,则有方法退出已定义的函数。

根据具体情况,您可以使用 return 命令从函数返回任何值,其语法如下:

return code

此处 code 可以是您选择的内容,但显然您应该选择在整个脚本上下文中具有意义或有用的内容。

Example

以下函数返回一个值 10:

#!/bin/sh

# Define your function here
Hello () {
   echo "Hello World $1 $2"
   return 10
}

# Invoke your function
Hello Zara Ali

# Capture value returnd by last command
ret=$?

echo "Return value is $ret"

执行后,您会收到以下结果 −

$./test.sh
Hello World Zara Ali
Return value is 10

Nested Functions

函数一个更有趣的功能是它们可以调用它们自己以及其他函数。调用自身的函数称为 recursive function

以下示例演示两个函数的嵌套 –

#!/bin/sh

# Calling one function from another
number_one () {
   echo "This is the first function speaking..."
   number_two
}

number_two () {
   echo "This is now the second function speaking..."
}

# Calling function one.
number_one

执行后,您会收到以下结果 −

This is the first function speaking...
This is now the second function speaking...

Function Call from Prompt

可以在 .profile 内放置常用函数的定义。每当你登录时,这些定义都会可用,你可以在命令提示符中使用这些定义。

或者,你可以将定义分组放在一个文件中,称为 test.sh ,然后通过键入以下内容在当前 shell 中执行文件 –

$. test.sh

这会产生一个效果,就是由 test.sh 定义的函数被读入并定义到当前 shell,如下所示 –

$ number_one
This is the first function speaking...
This is now the second function speaking...
$

若要从 shell 中删除函数定义,请使用 unset 命令和 .f 选项。此命令还用于删除对 shell 的变量定义。

$ unset -f function_name

Unix - Shell Manpage Help

所有 Unix 命令都带有一些可选项和必选项。忘记这些命令的完整语法很常见。

因为没有人可能记住每一个 Unix 命令和它的所有选项,因此,我们拥有在线帮助来缓解这一点,并且这种在线帮助 منذ أن كان Unix 处于开发阶段起就可用了。

Unix 的下载版本 Help files 称为 man pages 。如果有一个命令名称,你不确定如何使用它,那么手册页可以帮你解决每一步。

Syntax

这是一个简单的命令,它可以帮助你获取任何 Unix 命令的详细信息,同时与系统一起使用:

$man command

Example

假设有一条命令要求您获得帮助;假设您想要了解 pwd ,那么您只需使用以下命令:

$man pwd

上述命令可帮助您获得 pwd 命令的完整信息。在您的命令提示符处自己尝试一下以获取更多详细信息。

您可以使用以下命令来获取 man 命令本身的完整信息:

$man man

Man Page Sections

手册页通常分为几节,一般根据手册页作者的偏好而有所不同。下表列出了一些常见部分:

Sr.No.

Section & Description

1

NAME Name of the command

2

SYNOPSIS 命令的一般用法参数

3

DESCRIPTION 描述该命令的作用

4

OPTIONS 描述该命令的所有参数或选项

5

SEE ALSO 列出与手册页中的命令直接相关或与其功能非常相似的其他命令

6

BUGS 解释已知的与该命令或其输出相关的任何问题或错误

7

EXAMPLES 常见的使用示例,让读者了解如何使用该命令

8

AUTHORS 手册页/命令的作者

总而言之,手册页是一项至关重要的资源,当您需要有关 Unix 系统中的命令或文件的信息时,它是一种首要的研究途径。

Useful Shell Commands

以下链接为您提供了最重要的和最常用的 Unix Shell 命令列表。

如果您不知道如何使用任何命令,请使用手册页来了解有关该命令的完整详细信息。

以下是 Unix Shell - Useful Commands 的列表:

Unix - Regular Expressions with SED

在本篇教程中,我们将详细讨论 Unix 中的 SED 正则表达式。

正则表达式是一个可以用来描述多个字符序列的字符串。包括 edsedawkgrep 等许多不同的 Unix 命令都使用了正则表达式,在一定程度上甚至包括 vi

此处 SED 代表 *s*tream *ed*itor。此流定向编辑器完全用于执行脚本。因此,你输入其中的所有内容都将经过处理并传递到 STDOUT,并且它不会改变输入文件。

Invoking sed

在正式开始之前,我们先确保有一份 /etc/passwd 文本文件的本地副本,以便进行练习 sed

如前文所述,通过管道向 sed 发送数据就可以调用它,如下所示——

$ cat /etc/passwd | sed
Usage: sed [OPTION]... {script-other-script} [input-file]...

  -n, --quiet, --silent
                 suppress automatic printing of pattern space
  -e script, --expression = script
...............................

cat 命令通过管道将 /etc/passwd 的内容转储到 sed ,再转储到 sed 的模式空间。模式空间是 sed 用于执行操作的内部工作缓冲区。

The sed General Syntax

以下是 sed 的通用语法——

/pattern/action

此处, pattern 是正则表达式,而 action 是下表中给出的一个命令。如果 pattern 被忽略,则对每一行执行 action ,如我们在上面看到的那样。

把模式包围起来的斜杠 (/)是必需的,因为它们用作分隔符。

Sr.No.

Range & Description

1

p Prints the line

2

d Deletes the line

3

s/pattern1/pattern2/ 用 pattern2 替换 pattern1 的首次出现

Deleting All Lines with sed

现在我们将了解如何使用 sed 删除所有行。再次调用 sed;但是,现在 sed 应该使用 editing command delete line ,该字符用字母 d - 表示

$ cat /etc/passwd | sed 'd'
$

不需要通过管道向 sed 发送文件即可调用 sed,可以指示 sed 从文件读取数据,如下面的示例所示。

以下命令与前一个示例中的命令完全相同,不需要 cat 命令:

$ sed -e 'd' /etc/passwd
$

The sed Addresses

sed 还支持地址。地址要么是文件中的特定位置,要么是应该应用特定编辑命令的范围。当 sed 遇到没有地址时,它会在文件中的每一行执行操作。

以下命令向您已在使用的 sed 命令添加了基本地址:

$ cat /etc/passwd | sed '1d' |more
daemon:x:1:1:daemon:/usr/sbin:/bin/sh
bin:x:2:2:bin:/bin:/bin/sh
sys:x:3:3:sys:/dev:/bin/sh
sync:x:4:65534:sync:/bin:/bin/sync
games:x:5:60:games:/usr/games:/bin/sh
man:x:6:12:man:/var/cache/man:/bin/sh
mail:x:8:8:mail:/var/mail:/bin/sh
news:x:9:9:news:/var/spool/news:/bin/sh
backup:x:34:34:backup:/var/backups:/bin/sh
$

请注意,该 delete edit 命令前添加了数字 1。这指示 sed 对文件的第 1 行执行编辑命令。在此示例中,sed 将删除 /etc/password 的第一行并打印文件的其余部分。

The sed Address Ranges

现在我们将了解如何使用 the sed address ranges 。那么,如果您想从文件中移除多行,该怎么办?您可以用 sed 指定一个地址范围,如下所示:

$ cat /etc/passwd | sed '1, 5d' |more
games:x:5:60:games:/usr/games:/bin/sh
man:x:6:12:man:/var/cache/man:/bin/sh
mail:x:8:8:mail:/var/mail:/bin/sh
news:x:9:9:news:/var/spool/news:/bin/sh
backup:x:34:34:backup:/var/backups:/bin/sh
$

上述命令将应用于从 1 到 5 开始的所有行。这会删除前五行。

尝试以下地址范围:

Sr.No.

Range & Description

1

'4,10d' 从第 4 行到第 10 行的各行会被删除

2

'10,4d' 只有第 10 行会被删除,因为 sed 不支持反向执行

3

'4,+5d' 该命令匹配文件中的第 4 行,删除该行,继续删除后续 5 行,然后停止删除并打印其余部分

4

'2,5!d' 该命令会删除除第 2 行到第 5 行以外的所有内容

5

'1~3d' 该命令会删除第 1 行,跳过后续三行,然后删除第 4 行。Sed 继续应用此模式,直到文件末尾。

6

'2~2d' 该命令告诉 sed 删除第 2 行,跳过后续一行,删除后续一行,重复此过程直到文件结尾

7

'4,10p' 从第 4 行到第 10 行的各行会被打印

8

'4,d' 该命令会生成语法错误

9

',10d' 该命令也会生成语法错误

Note − 在使用 p 操作时,您应该使用 -n 选项来避免重复打印行。查看以下两个命令之间的差异 −

$ cat /etc/passwd | sed -n '1,3p'
Check the above command without -n as follows −
$ cat /etc/passwd | sed '1,3p'

The Substitution Command

s 表示的替换命令将用您指定的任何其他字符串替换您指定的任何字符串。

为了用一个字符串替换另一个字符串,sed 需要有关第一个字符串结束位置和替换字符串开始位置的信息。为此,我们使用正斜杠 ( / ) 字符将这两个字符串作为书挡。

以下命令用字符串 amrood 替换一行中字符串 root 的首次出现。

$ cat /etc/passwd | sed 's/root/amrood/'
amrood:x:0:0:root user:/root:/bin/sh
daemon:x:1:1:daemon:/usr/sbin:/bin/sh
..........................

需要注意的是,sed 仅替换一行中的首次出现。如果字符串 root 在一行中出现多次,则仅替换第一个匹配项。

为了让 sed 执行全局替换,请按以下方式将字母 g 添加到命令的末尾 −

$ cat /etc/passwd | sed 's/root/amrood/g'
amrood:x:0:0:amrood user:/amrood:/bin/sh
daemon:x:1:1:daemon:/usr/sbin:/bin/sh
bin:x:2:2:bin:/bin:/bin/sh
sys:x:3:3:sys:/dev:/bin/sh
...........................

Substitution Flags

除了 g 标志外,还可以传递许多其他有用的标志,并且您一次可以指定多个标志。

Sr.No.

Flag & Description

1

g 替换所有匹配项,而不仅仅是第一个匹配项

2

NUMBER Replaces only NUMBERth match

3

p 如果进行了替换,则打印模式空间

4

w FILENAME 如果进行了替换,则将结果写入 FILENAME

5

I or i 以不区分大小写的方式匹配

6

M or m 除了特殊正则表达式字符 ^ 和 $ 的正常行为外,此标志还导致 ^ 在换行后匹配空字符串,而 $ 在换行前匹配空字符串

Using an Alternative String Separator

假设您必须对包含正斜杠字符的字符串执行替换。在这种情况下,您可以通过在 s 后提供指定字符来指定不同的分隔符。

$ cat /etc/passwd | sed 's:/root:/amrood:g'
amrood:x:0:0:amrood user:/amrood:/bin/sh
daemon:x:1:1:daemon:/usr/sbin:/bin/sh

在上面的示例中,我们使用了 : 作为 delimiter 而不是斜杠 /,因为我们试图搜索 /root 而不是简单的 root。

Replacing with Empty Space

使用空替换字符串从 /etc/passwd 文件中完全删除 root 字符串 −

$ cat /etc/passwd | sed 's/root//g'
:x:0:0::/:/bin/sh
daemon:x:1:1:daemon:/usr/sbin:/bin/sh

Address Substitution

如果您只想在第 10 行上用字符串 quiet 替换字符串 sh ,则可以按如下方式指定:

$ cat /etc/passwd | sed '10s/sh/quiet/g'
root:x:0:0:root user:/root:/bin/sh
daemon:x:1:1:daemon:/usr/sbin:/bin/sh
bin:x:2:2:bin:/bin:/bin/sh
sys:x:3:3:sys:/dev:/bin/sh
sync:x:4:65534:sync:/bin:/bin/sync
games:x:5:60:games:/usr/games:/bin/sh
man:x:6:12:man:/var/cache/man:/bin/sh
mail:x:8:8:mail:/var/mail:/bin/sh
news:x:9:9:news:/var/spool/news:/bin/sh
backup:x:34:34:backup:/var/backups:/bin/quiet

类似地,要进行地址范围替换,您可以执行以下操作:

$ cat /etc/passwd | sed '1,5s/sh/quiet/g'
root:x:0:0:root user:/root:/bin/quiet
daemon:x:1:1:daemon:/usr/sbin:/bin/quiet
bin:x:2:2:bin:/bin:/bin/quiet
sys:x:3:3:sys:/dev:/bin/quiet
sync:x:4:65534:sync:/bin:/bin/sync
games:x:5:60:games:/usr/games:/bin/sh
man:x:6:12:man:/var/cache/man:/bin/sh
mail:x:8:8:mail:/var/mail:/bin/sh
news:x:9:9:news:/var/spool/news:/bin/sh
backup:x:34:34:backup:/var/backups:/bin/sh

正如您从输出中看到的,前五行的字符串 sh 已更改为 quiet ,但其余行保持不变。

The Matching Command

您将使用 p 选项和 -n 选项来打印所有匹配行,如下所示:

$ cat testing | sed -n '/root/p'
root:x:0:0:root user:/root:/bin/sh
[root@ip-72-167-112-17 amrood]# vi testing
root:x:0:0:root user:/root:/bin/sh
daemon:x:1:1:daemon:/usr/sbin:/bin/sh
bin:x:2:2:bin:/bin:/bin/sh
sys:x:3:3:sys:/dev:/bin/sh
sync:x:4:65534:sync:/bin:/bin/sync
games:x:5:60:games:/usr/games:/bin/sh
man:x:6:12:man:/var/cache/man:/bin/sh
mail:x:8:8:mail:/var/mail:/bin/sh
news:x:9:9:news:/var/spool/news:/bin/sh
backup:x:34:34:backup:/var/backups:/bin/sh

Using Regular Expression

在匹配模式时,可以使用正则表达式,它提供更大的灵活性。

检查以下示例,它匹配以 daemon 开头的所有行,然后删除它们 −

$ cat testing | sed '/^daemon/d'
root:x:0:0:root user:/root:/bin/sh
bin:x:2:2:bin:/bin:/bin/sh
sys:x:3:3:sys:/dev:/bin/sh
sync:x:4:65534:sync:/bin:/bin/sync
games:x:5:60:games:/usr/games:/bin/sh
man:x:6:12:man:/var/cache/man:/bin/sh
mail:x:8:8:mail:/var/mail:/bin/sh
news:x:9:9:news:/var/spool/news:/bin/sh
backup:x:34:34:backup:/var/backups:/bin/sh

以下是删除所有以 sh 结尾的行的示例 −

$ cat testing | sed '/sh$/d'
sync:x:4:65534:sync:/bin:/bin/sync

下表列出了四个在正则表达式中非常有用的特殊字符。

Sr.No.

Character & Description

1

^ 匹配行首

2

$ 匹配行尾

3

. Matches any single character

4

匹配前一个字符的零次或多次出现

5

[chars] 匹配 chars 中给定的任何一个字符,其中 chars 是一个字符序列。可以使用 - 字符来表示字符范围。

Matching Characters

再看一些表达式,以演示 metacharacters 的用法。例如,以下模式 −

Sr.No.

Expression & Description

1

/a.c/ 匹配包含诸如 a+ca-cabcmatcha3c 等字符串的行

2

/a*c/ 匹配相同的字符串以及 aceyaccarctic 等字符串

3

/[tT]he/ 匹配字符串 Thethe

4

/^$/ Matches blank lines

5

/^. $/* 匹配整行无论内容是什么

6

/ */ 匹配一个或多个空格

7

/^$/ Matches blank lines

下表显示了一些常用字符集 −

Sr.No.

Set & Description

1

[a-z] 匹配单个小写字母

2

[A-Z] 匹配单个大写字母

3

[a-zA-Z] Matches a single letter

4

[0-9] Matches a single number

5

[a-zA-Z0-9] 匹配单个字母或数字

Character Class Keywords

某些特殊关键字通常可用于 regexps ,尤其是使用 regexps 的 GNU 实用程序。这些对于 sed 正则表达式非常有用,因为它们简化了操作并增强了可读性。

例如,字符 a through z 和字符 A through Z 构成包含关键字 [id=":alpha:"] 的此类字符。

使用字母字符类关键字,此命令只打印 /etc/syslog.conf 文件中以字母开头的行 −

$ cat /etc/syslog.conf | sed -n '/^[[:alpha:]]/p'
authpriv.*                         /var/log/secure
mail.*                             -/var/log/maillog
cron.*                             /var/log/cron
uucp,news.crit                     /var/log/spooler
local7.*                           /var/log/boot.log

下表是 GNU sed 中可用字符类关键字的完整列表。

Sr.No.

Character Class & Description

1

[id=":alnum:"] Alphanumeric [a-z A-Z 0-9]

2

[id=":alpha:"] Alphabetic [a-z A-Z]

3

[id=":blank:"] 空白字符(空格或制表符)

4

[id=":cntrl:"] Control characters

5

[id=":digit:"] Numbers [0-9]

6

[id=":graph:"] 任意可见字符(不包含空白)

7

[id=":lower:"] Lowercase letters [a-z]

8

[id=":print:"] Printable characters (non-control characters)

9

[id=":punct:"] Punctuation characters

10

[id=":space:"] Whitespace

11

[id=":upper:"] Uppercase letters [A-Z]

12

[id=":xdigit:"] 十六进制数字 [0-9 a-f A-F]

Aampersand Referencing

sed metacharacter & 表示匹配的模式的内容。例如,假设你有一个名为 phone.txt 的文件,其中包含许多电话号码,如下所示 −

5555551212
5555551213
5555551214
6665551215
6665551216
7775551217

你想将 area code (前三个数字)括在括号中以方便阅读。为此,你可以使用替换字符“&” −

$ sed -e 's/^[[:digit:]][[:digit:]][[:digit:]]/(&)/g' phone.txt
(555)5551212
(555)5551213
(555)5551214
(666)5551215

(666)5551216
(777)5551217

在模式部分,你匹配前 3 个数字,然后使用 & 来用 parentheses 包围替换这 3 个数字。

Using Multiple sed Commands

你可以使用以下方法在单个 sed 命令中使用多个 sed 命令 −

$ sed -e 'command1' -e 'command2' ... -e 'commandN' files

此处 command1commandN 为前面讨论过的类型 sed 命令。这些命令应用于给定文件中列表中的每一行。

使用相同机制,我们可以如下编写上面的电话号码示例 −

$ sed -e 's/^[[:digit:]]\{3\}/(&)/g'  \
   -e 's/)[[:digit:]]\{3\}/&-/g' phone.txt
(555)555-1212
(555)555-1213
(555)555-1214
(666)555-1215
(666)555-1216
(777)555-1217

Note − 在上述示例中,我们用 {3} 替换了重复字符类关键字 [id=":digit:"] 三次,这意味着前面正则表达式匹配了三次。我们还使用了 \ 来换行,并且在运行命令之前必须将其删除。

Back References

ampersand metacharacter 很有用,但更重要的是能够定义正则表达式中的特定区域。这些特殊区域可以用作替换字符串中的参考。通过定义正则表达式的特定部分,你可以使用特殊参考字符引用这些部分。

要执行 back references ,你必须先定义一个区域,然后引用该区域。要定义一个区域,你可以在每个感兴趣的区域周围插入 backslashed parentheses 。你用反斜杠包围的第一个区域由 \1 引用,第二个区域由 \2 引用,依此类推。

假设 phone.txt 具有以下文本 −

(555)555-1212
(555)555-1213
(555)555-1214
(666)555-1215
(666)555-1216
(777)555-1217

尝试以下命令 −

$ cat phone.txt | sed 's/\(.*)\)\(.*-\)\(.*$\)/Area \
   code: \1 Second: \2 Third: \3/'
Area code: (555) Second: 555- Third: 1212
Area code: (555) Second: 555- Third: 1213
Area code: (555) Second: 555- Third: 1214
Area code: (666) Second: 555- Third: 1215
Area code: (666) Second: 555- Third: 1216
Area code: (777) Second: 555- Third: 1217

Note − 在上述示例中,括号内的每个正则表达式都将由 \1\2 等进行反向引用。我们在此处使用 \ 来换行。在运行命令之前,应该将其删除。

Unix - File System Basics

文件系统是分区或磁盘上的文件的逻辑集合。分区是信息的容器,可以跨越整个硬盘,如果需要的话。

您的硬盘可以有各种分区,这些分区通常只包含一个文件系统,例如一个文件系统容纳 /file system 或另一个包含 /home file system

每个分区一个文件系统可以对不同的文件系统进行逻辑维护和管理。

在 Unix 中,所有内容都被认为是一个文件,包括物理设备,例如 DVD-ROM、USB 设备和软盘驱动器。

Directory Structure

Unix 使用分层文件系统结构,非常像一棵倒置的树,其中根 (/) 位于文件系统的底部,所有其他目录都从那里展开。

Unix 文件系统是一组文件和目录,具有以下属性:

  1. 它有一个根目录 ( / ),其中包含其他文件和目录。

  2. 每个文件或目录通过其名称、它所在的目录和一个唯一的标识符(通常称为 inode )进行唯一标识。

  3. 根据惯例,根目录具有 inode2 ,而 lost+found 目录具有 inode3 。inode 编号 01 不被使用。可以通过向 -i option 指定 ls command 来查看文件 inode 编号。

  4. 它自成一体。一个文件系统之间没有任何依赖关系。

目录有特定的用途,通常保存相同类型的信息,以便轻松地查找文件。下面是在 Unix 主要版本中存在的目录:

Sr.No.

Directory & Description

1

/ 这是根目录,其中应该只包含文件结构顶层需要的目录

2

/bin 可执行文件位于这里。这些文件对所有用户可用

3

/dev These are device drivers

4

/etc 管理员目录命令、配置文件、磁盘配置文件、有效用户列表、组、以太网、主机、发送关键消息的位置

5

/lib 包含共享库文件,有时也包含其他与内核相关文件。

6

/boot 包含引导系统所需文件。

7

/home 包含用户和其他账户的主目录。

8

/mnt 用于加载其他临时文件系统,如 cdromfloppy ,分别用于 CD-ROM 驱动器和 floppy diskette drive

9

/proc 包含 process number 标记为文件的所有进程或其他对系统来说是动态的信息。

10

/tmp 包含在系统启动之间使用的临时文件。

11

/usr 用于各种用途,可供多位用户使用。包括管理命令、共享文件、库文件以及其他文件。

12

/var 通常包含可变长度文件,如日志和打印文件,以及其他可能包含可变数量数据的任何类型文件。

13

/sbin 包含二进制(可执行)文件,通常用于系统管理,例如 fdiskifconfig 实用工具。

14

/kernel Contains kernel files

Navigating the File System

现在您已经了解了文件系统的基础知识,即可开始导航到所需文件。以下命令用于导航系统:

Sr.No.

Command & Description

1

cat filename Displays a filename

2

cd dirname 将你移动到已识别目录中。

3

cp file1 file2 将一个文件/目录复制到指定位置。

4

file filename 识别文件类型(二进制、文本等)。

5

find filename dir 在文件/目录中查找。

6

head filename 显示文件开头。

7

less filename 从末尾或开头浏览文件。

8

ls dirname 显示指定目录的内容。

9

mkdir dirname 创建指定目录。

10

more filename 从开头到末尾浏览文件。

11

mv file1 file2 移动文件/目录位置或重命名文件/目录

12

pwd 显示用户当前所在目录

13

rm filename Removes a file

14

rmdir dirname Removes a directory

15

tail filename 显示文件结尾

16

touch filename 创建空白文件或修改现有文件或其属性

17

whereis filename 显示文件位置

18

which filename 如果文件在 PATH 中显示文件位置

您可以使用 Manpage Help 查看此处提到的每个命令的完整语法。

The df Command

管理分区空间的第一种方法是使用 df (disk free) 命令。命令 df -k (disk free) 显示 disk space usage in kilobytes ,如下所示:

$df -k
Filesystem      1K-blocks      Used   Available Use% Mounted on
/dev/vzfs        10485760   7836644     2649116  75% /
/devices                0         0           0   0% /devices
$

某些目录,如 /devices 在 kbytes 中显示 0,使用的和可用的列以及容量的 0%。这些是特殊(或虚拟)文件系统,尽管它们驻留在 / 下的磁盘上,但本身不占用磁盘空间。

df -k 输出在所有 Unix 系统上通常都是相同的。以下是它通常包括的内容:

Sr.No.

Column & Description

1

Filesystem 物理文件系统名称

2

kbytes 存储介质上可用的总千字节空间

3

used 已用(由文件)的总千字节空间

4

avail 可用的总千字节空间

5

capacity 文件使用的总空间的百分比

6

Mounted on 文件系统安装位置

您可以使用 -h (human readable) option 以更容易理解的符号显示输出格式。

The du Command

du (disk usage) command 使您能够指定目录以显示特定目录的磁盘空间使用情况。

如果您想确定某个目录占用了多少空间,此命令很有用。以下命令显示每个目录消耗的块数。根据您的系统,单个块可能占用 512 字节或 1 千字节。

$du /etc
10     /etc/cron.d
126    /etc/default
6      /etc/dfs
...
$

-h 选项能让输出更易于理解 −

$du -h /etc
5k    /etc/cron.d
63k   /etc/default
3k    /etc/dfs
...
$

Mounting the File System

文件系统必须被挂载才能被系统使用。要查看系统当前挂载(可用于使用)的内容,请使用以下命令 −

$ mount
/dev/vzfs on / type reiserfs (rw,usrquota,grpquota)
proc on /proc type proc (rw,nodiratime)
devpts on /dev/pts type devpts (rw)
$

按照 Unix 公约, /mnt 目录是临时挂载(例如 CDROM 驱动器、远程网络驱动器和软盘驱动器)的位置。如果需要挂载文件系统,则可按以下语法使用 mount 命令 −

mount -t file_system_type device_to_mount directory_to_mount_to

例如,如果要将 CD-ROM 挂载到 /mnt/cdrom 目录,则可输入 −

$ mount -t iso9660 /dev/cdrom /mnt/cdrom

这假设您的 CD-ROM 设备名为 /dev/cdrom ,并且您希望将其挂载到 /mnt/cdrom 。有关更具体的信息,请参阅 mount 手册页,或在命令行中输入 mount -h 以获取帮助信息。

挂载后,您可以使用 cd 命令通过刚刚创建的挂载点来导航新可用的文件系统。

Unmounting the File System

要卸载(移除)系统中的文件系统,请使用 umount 命令识别挂载点或设备。

例如, to unmount cdrom ,使用以下命令 −

$ umount /dev/cdrom

mount command 使您能够访问自己的文件系统,但在大多数现代 Unix 系统上, automount function 使此进程对于用户来说不可见,无需介入。

User and Group Quotas

用户配额和组配额提供了限制单个用户或特定组内所有用户所用空间量的机制,可限制为管理员定义的值。

配额围绕两个限制运行,在空间量或磁盘块数开始超过管理员定义的限制时,允许用户采取一些措施 −

  1. Soft Limit − 如果用户超过已定义限制,则有一个宽限期,允许用户释放一些空间。

  2. Hard Limit − 无论宽限期如何,当达到硬性限制时,将无法分配更多文件或块。

有多个用于管理配额的命令 −

Sr.No.

Command & Description

1

quota 显示组用户的磁盘使用量和限制

2

edquota 这是一个配额编辑器。可以使用此命令编辑用户或组配额

3

quotacheck 扫描文件系统以了解磁盘使用情况,创建、检查和修复配额文件

4

setquota 这是一个命令行配额编辑器

5

quotaon 这会向系统通告应在某个或多个文件系统上启用磁盘配额

6

quotaoff 这会向系统宣布,应该为一个或多个文件系统禁用磁盘配额

7

repquota 这会打印指定文件系统的磁盘使用情况和配额摘要

您可以使用 Manpage Help 查看此处提到的每个命令的完整语法。

Unix - User Administration

在本章中,我们将详细讨论 Unix 中的用户管理。

Unix 系统上有三种类型的帐户 -

Root account

这也是 superuser ,将拥有对系统的完整和不受限制的控制。超级用户可以在没有任何限制的情况下运行任何命令。此用户应被假定为系统管理员。

System accounts

系统帐户是为特定组件的运行而必需的,例如邮件帐户和 sshd 帐户。这些帐户通常在系统上需要一些特定功能,对它们的任何修改都可能对系统产生不利影响。

User accounts

用户账户为用户及其用户组提供对系统的交互式访问权限。一般用户通常被分配到这些账户,并且通常对关键系统文件和目录的访问权限有限。

Unix 支持组账户的概念,其逻辑上分组了多个账户。每个账户都将成为另一个组账户的一部分。Unix 组在处理文件权限和进程管理中扮演着重要角色。

Managing Users and Groups

有四个主要的用户管理文件 −

  1. /etc/passwd − 保存用户账户和密码信息。此文件包含有关 Unix 系统上账户的大部分信息。

  2. /etc/shadow − 保存对应账户的加密密码。并非所有系统都支持此文件。

  3. /etc/group − 此文件包含每个账户的组信息。

  4. /etc/gshadow − 此文件包含安全组账户信息。

使用 cat 命令检查以上所有文件。

下表列出了大多数 Unix 系统上可用于创建和管理账户及组的命令 −

Sr.No.

Command & Description

1

useradd 将账户添加到系统

2

usermod Modifies account attributes

3

userdel 从系统中删除账户

4

groupadd 将组添加到系统

5

groupmod Modifies group attributes

6

groupdel 从系统中删除组

您可以使用 Manpage Help 查看此处提到的每个命令的完整语法。

Create a Group

我们现在了解如何创建组。为此,我们需要在创建任何账户之前创建组,否则,我们可以使用系统中的现有组。我们已将所有组列在 /etc/groups 文件中。

所有默认组都是特定于系统账户的组,不建议将其用于普通账户。因此,以下是如何创建新组账户的语法 −

 groupadd [-g gid [-o]] [-r] [-f] groupname

下表列出了参数 −

Sr.No.

Option & Description

1

-g GID 组 ID 的数值

2

-o 此选项允许添加具有非唯一 GID 的组

3

-r 此标志指示 groupadd 添加系统账户

4

*-f *如果指定的组已存在,则此选项仅退出并显示成功状态。如果指定 GID 已存在,则会选择其他(唯一)GID

5

groupname 要创建的实际组名

如果您未指定任何参数,则系统会使用默认值。

以下示例使用大多数管理员都极易接受的默认值创建了 developers 组。

$ groupadd developers

Modify a Group

要修改组,请使用 groupmod 语法 −

$ groupmod -n new_modified_group_name old_group_name

要将 developers_2 组名更改为 developer,请键入 −

$ groupmod -n developer developer_2

下面是将 financial GID 更改为 545 的方法 −

$ groupmod -g 545 developer

Delete a Group

我们现在将了解如何删除组。要删除现有组,您只需要 groupdel commandgroup name 。要删除 financial 组,命令为 −

$ groupdel developer

这仅删除组,而不删除与此组关联的文件。文件仍可由其所有者访问。

Create an Account

让我们看看如何在 Unix 系统上创建新帐户。以下是创建用户帐户的语法 −

useradd -d homedir -g groupname -m -s shell -u userid accountname

下表列出了参数 −

Sr.No.

Option & Description

1

-d homedir 为帐户指定主目录

2

-g groupname 为此帐户指定组帐户

3

-m 如果主目录不存在,则创建主目录

4

-s shell 为此帐户指定默认 shell

5

-u userid 您可以为此帐户指定用户 ID

6

accountname 要创建的实际帐户名

如果您未指定任何参数,则系统会使用默认值。 useradd 命令会修改 /etc/passwd/etc/shadow/etc/group 文件,并创建主目录。

以下示例创建了一个名为 mcmohd 的帐户,将其主目录设置为 /home/mcmohd ,并将组设置为 developers 。此用户已将 Korn Shell 分配给它。

$ useradd -d /home/mcmohd -g developers -s /bin/ksh mcmohd

在发出上述命令之前,请确保已使用 groupadd 命令创建了 developers 组。

创建帐号后,你可以使用 passwd 命令设置其密码,如下所示:

$ passwd mcmohd20
Changing password for user mcmohd20.
New UNIX password:
Retype new UNIX password:
passwd: all authentication tokens updated successfully.

当你键入 passwd accountname 时,它会给你一个更改密码选项,前提是你是超级用户。否则,你只能使用同一命令更改你的密码,但不指定你的帐号名称。

Modify an Account

usermod 命令让你能够从命令行更改现有帐号。它使用与 useradd 命令相同的参数,加上 -l 参数,它允许你更改帐号名称。

例如,要把帐号名称 mcmohd 更改为 mcmohd20 ,并相应更改主目录,你需要发出以下命令:

$ usermod -d /home/mcmohd20 -m -l mcmohd mcmohd20

Delete an Account

userdel 命令可以用来删除现有用户。如果不谨慎使用,这是一个非常危险的命令。

命令 .r 只有一个可用的参数或选项,用于移除此帐号的主目录和邮件文件。

例如,要删除帐号 mcmohd20,请发出以下命令:

$ userdel -r mcmohd20

如果你想保留主目录以备将来用,请省略 -r 选项。你可以在以后的需要时删除主目录。

Unix - System Performance

在本章中,我们将详细讨论 Unix 系统的系统性能。

我们将向你介绍几个可用来监控和管理 Unix 系统性能的免费工具。这些工具还提供了关于如何诊断和修复 Unix 环境中的性能问题的指南。

Unix 具有以下需要监控和调整的主要资源类型:

  1. CPU

  2. Memory

  3. Disk space

  4. Communications lines

  5. I/O Time

  6. Network Time

  7. Applications programs

Performance Components

下表列出了占用系统时间的五个主要组件:

Sr.No.

Component & Description

1

User State CPU CPU 在用户状态下运行用户程序所花费的实际时间。它包括执行库调用所花费的时间,但不包括代表其在内核中花费的时间

2

System State CPU * This is the amount of time the CPU spends in the system state on behalf of this program. All *I/O routines 需要内核服务。程序员可以通过阻塞 I/O 传输来影响此值

3

*I/O 时间和网络时间*这是用于移动数据和处理 I/O 请求所花费的时间

4

*虚拟内存性能*这包括上下文切换和交换

5

Application Program 运行其他程序所花费的时间 - 当系统没有处理该应用程序时,因为另一个应用程序当前具有 CPU

Performance Tools

Unix 提供了以下重要工具来测量和微调 Unix 系统性能:

Sr.No.

Command & Description

1

nice/renice 运行一个以修改后的调度优先级运行程序

2

netstat 打印网络连接、路由表、接口统计、化装连接和多播成员

3

time 帮助计算简单命令或提供资源使用情况

4

uptime 系统负载平均值

5

ps 报告当前进程的快照

6

vmstat Reports virtual memory statistics

7

gprof 显示调用图廓形数据

8

prof Facilitates Process Profiling

9

top Displays system tasks

您可以使用 Manpage Help 查看此处提到的每个命令的完整语法。

Unix - System Logging

在本章中,我们将详细讨论 Unix 中的系统日志记录。

Unix 系统拥有非常灵活而强大的日志记录系统,它让你能够记录你所能想象的几乎所有内容,然后操作日志来检索所需的信息。

许多版本的 Unix 提供了一个称为 syslog 的通用日志记录工具。需要记录信息的各个程序会将信息发送到 syslog。

Unix syslog 是一种可由主机配置的统一系统日志记录工具。该系统使用一个集中式系统日志记录进程,它运行程序 /etc/syslogd/etc/syslog

系统记录器的运行非常直接。程序会将日志条目发送至 syslogd,syslogd 会查阅配置文件 /etc/syslogd.conf/etc/syslog ,并在找到匹配时将日志消息写入所需的日志文件。

有四个基本的 syslog 术语你需要掌握 -

Sr.No.

Term & Description

1

Facility 用于描述提交日志消息的应用程序或进程的标识符。例如,mail、kernel 和 ftp。

2

Priority 消息重要性的指标。syslog 中将级别定义为指南,从调试信息到严重事件。

3

Selector 一种或多种工具和级别的组合。当传入事件匹配一个选择器时,会执行一个操作。

4

Action 匹配选择器的传入消息会怎么样 - 操作可以将消息写入日志文件、将消息回显至控制台或其他设备、向已登录的用户写入消息或将消息发送至另一个 syslog 服务器。

Syslog Facilities

我们现在将了解 syslog 工具。以下是选择器的可用工具。并非所有工具都在 Unix 的所有版本中都存在。

Facility

Description

1

auth 与请求名称和密码相关的活动(getty、su、login)

2

authpriv 与 auth 相同,但记录到仅由选定用户可以读取的文件中

3

console 用于捕获通常直接发送到系统控制台的消息

4

cron 来自 cron 系统计划程序的消息

5

daemon System daemon catch-all

6

ftp 与 ftp 守护进程相关消息

7

kern Kernel messages

8

local0.local7 每个站点定义的本地功能

9

lpr 来自行式打印系统的消息

10

mail 与邮件系统关联的消息

11

mark 用于在日志文件中生成时间戳的伪事件

12

news 与网络新闻协议 (nntp) 相关消息

13

ntp 与网络时间协议相关消息

14

user Regular user processes

15

uucp UUCP subsystem

Syslog Priorities

syslog 优先级汇总在以下表格中

Sr.No.

Priority & Description

1

emerg 紧急情况,例如迫在眉睫的系统崩溃,通常会广播给所有用户

2

alert 应立即更正的情况,例如损坏的系统数据库

3

crit 严重情况,例如硬件错误

4

err Ordinary error

5

Warning Warning

6

notice 并非错误的情况,但可能需要以特殊方式处理

7

info Informational message

8

debug 调试程序时使用消息

9

none 用于指定不记录消息的伪级别

设施和级别的组合使您能够明辨要记录的内容以及这些信息存放的位置。

随着每个程序尽职尽责地将自己的消息发送到系统日志,日志记录器将根据选择器中定义的级别,决定跟踪哪些内容,以及丢弃哪些内容。

当你指定一个级别,系统将跟踪该级别及更高级别下的所有信息。

The /etc/syslog.conf file

/etc/syslog.conf 文件控制日志消息记录的位置。一个典型的 syslog.conf 文件可能如下所示 −

*.err;kern.debug;auth.notice /dev/console
daemon,auth.notice           /var/log/messages
lpr.info                     /var/log/lpr.log
mail.*                       /var/log/mail.log
ftp.*                        /var/log/ftp.log
auth.*                       @prep.ai.mit.edu
auth.*                       root,amrood
netinfo.err                  /var/log/netinfo.log
install.*                    /var/log/install.log
*.emerg                      *
*.alert                      |program_name
mark.*                       /dev/console

该文件的每一个行包含两个部分 −

  1. 一个 message selector ,用来指定应该记录哪类消息。例如,从内核生成的所有错误消息或所有调试消息。

  2. 一个 action field ,用来表明应该如何处理该消息。例如,将其放入一个文件或将其发送到用户的终端。

以下是上述配置的值得关注的要点 −

  1. 日志选择器有两个部分: a facilitya priority 。例如,kern.debug 选择由内核(设备)生成的所有调试消息(优先级)。

  2. 日志选择器 kern.debug 选择所有优先级高于调试的优先级。

  3. 在设备或优先级的位置使用星号表示“全部”。例如, .debug 表示所有调试消息,而 kern. 表示由内核生成的所有消息。

  4. 你还可以使用逗号来指定多个设备。使用分号可以将两个或更多选择器组合在一起。

Logging Actions

操作字段指定五个操作中的一个 −

  1. 将日志消息记录到文件或设备。例如, /var/log/lpr.log/dev/console

  2. 向用户发送消息。你可以通过用逗号分隔多个用户名来指定多个用户名;例如,root、amrood。

  3. 向所有用户发送消息。在这种情况下,操作字段包含一个星号;例如,*。

  4. 使用 Unix 管道符号(|)将消息管道到程序。在这种情况下,Unix 管道符号后会指定程序。

  5. 将消息发送到其他主机上的 syslog。在这种情况下,操作字段包括一个主机名,前面加上一个 at 符号;例如,@tutorialspoint.com。

The logger Command

Unix 提供了 logger 命令,这是一个非常有用的命令,用于处理系统日志记录。 logger 命令将日志记录消息发送到 syslogd 守护程序,并因此触发系统日志记录。

这意味着我们可以随时从命令行中查看 syslogd 守护程序及其配置。logger 命令提供了一种从命令行向系统日志文件添加单行条目的方法。

命令的格式为 −

logger [-i] [-f file] [-p priority] [-t tag] [message]...

以下是参数的说明:

Sr.No.

Option & Description

1

-f filename 使用文件 filename 的内容作为要记录的消息。

2

-i 使用每一行的记录器进程记录进程 ID。

3

-p priority 使用指定优先级(指定的选定器条目)输入消息;消息优先级可以用数字指定,或者作为设施.priority 对指定。默认优先级为 user.notice。

4

-t tag 使用指定标签标记添加到日志的每一行。

5

message 字符串参数,它们的文本内容以指定顺序连接在一起,并用空格分隔。

你可以使用 Manpage Help 检查此命令的完整语法。

Log Rotation

日志文件有很快速增长和占用大量磁盘空间的趋势。为了启用日志轮换,大多数发行版会使用诸如 newsysloglogrotate 的工具。

这些工具应该使用 cron daemon 在频繁的时间间隔内被调用。查看 newsyslog 或 logrotate 的手册页以获得更多详细信息。

Important Log Locations

所有系统应用程序都在 /var/log 及其子目录中创建它们的日志文件。以下是一些重要的应用程序及其对应的日志目录 −

Application

Directory

httpd

/var/log/httpd

samba

/var/log/samba

cron

/var/log/

mail

/var/log/

mysql

/var/log/

Unix - Signals and Traps

在本章中,我们将详细讨论 Unix 中的信号和陷阱。

信号是发送到程序中的软件中断,用于指明重要事件已发生。事件可以从用户请求到非法内存访问错误。一些信号(例如中断信号)表明用户已要求程序执行不在通常控制流中的操作。

下表列出了你可能遇到的和希望在程序中使用的常见信号 -

Signal Name

Signal Number

Description

SIGHUP

1

在控制终端上检测到挂起或控制进程死亡

SIGINT

2

如果用户发送中断信号(Ctrl + C),则发出

SIGQUIT

3

如果用户发送退出信号(Ctrl + D),则发出

SIGFPE

8

如果尝试非法数学运算,则发出

SIGKILL

9

如果进程接收到此信号,它必须立即退出,并且不会执行任何清理操作

SIGALRM

14

闹钟信号(用于定时器)

SIGTERM

15

软件中止信号(默认情况下由 kill 发送)

List of Signals

有一种简单的方法可以列出系统支持的所有信号。只需发出 kill -l 命令,它将显示所有受支持的信号 -

$ kill -l
 1) SIGHUP       2) SIGINT       3) SIGQUIT      4) SIGILL
 5) SIGTRAP      6) SIGABRT      7) SIGBUS       8) SIGFPE
 9) SIGKILL     10) SIGUSR1     11) SIGSEGV     12) SIGUSR2
13) SIGPIPE     14) SIGALRM     15) SIGTERM     16) SIGSTKFLT
17) SIGCHLD     18) SIGCONT     19) SIGSTOP     20) SIGTSTP
21) SIGTTIN     22) SIGTTOU     23) SIGURG      24) SIGXCPU
25) SIGXFSZ     26) SIGVTALRM   27) SIGPROF     28) SIGWINCH
29) SIGIO       30) SIGPWR      31) SIGSYS      34) SIGRTMIN
35) SIGRTMIN+1  36) SIGRTMIN+2  37) SIGRTMIN+3  38) SIGRTMIN+4
39) SIGRTMIN+5  40) SIGRTMIN+6  41) SIGRTMIN+7  42) SIGRTMIN+8
43) SIGRTMIN+9  44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12
47) SIGRTMIN+13 48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14
51) SIGRTMAX-13 52) SIGRTMAX-12 53) SIGRTMAX-11 54) SIGRTMAX-10
55) SIGRTMAX-9  56) SIGRTMAX-8  57) SIGRTMAX-7  58) SIGRTMAX-6
59) SIGRTMAX-5  60) SIGRTMAX-4  61) SIGRTMAX-3  62) SIGRTMAX-2
63) SIGRTMAX-1  64) SIGRTMAX

信号的实际列表因 Solaris、HP-UX 和 Linux 而异。

Default Actions

每个信号都有与其关联的默认操作。信号的默认操作是脚本或程序在接收到信号时执行的操作。

一些可能的默认操作 -

  1. Terminate the process.

  2. Ignore the signal.

  3. 转储核心。这会创建一个名为 core 的文件,其中包含进程在收到信号时的内存映像。

  4. Stop the process.

  5. Continue a stopped process.

Sending Signals

有几种将信号传递给程序或脚本的方法。最常见的方法之一是让用户在脚本执行时键入 CONTROL-CINTERRUPT key

当您按下 Ctrl+C 键时,一条 SIGINT 被发送到脚本中,并且根据定义的默认操作脚本终止。

发出信号的另一种常用方法是使用 kill command ,其语法如下 −

$ kill -signal pid

其中 signal 是要发送的信号的号码或名称, pid 是要向其发送信号的进程 ID。例如 −

$ kill -1 1001

上述命令将 HUP 或挂起信号发送到正在使用 process ID 1001 运行的程序。要向同一进程发送终止信号,请使用以下命令 −

$ kill -9 1001

这将杀死正在使用 process ID 1001 运行的进程。

Trapping Signals

当你在终端执行 shell 程序时按下 Ctrl+C 或 Break 键时,通常该程序会立即终止,并且你的命令提示符会返回。这可能并不总是需要的。例如,你最终可能会留下一些未清理的临时文件。

捕获这些信号非常容易,并且陷阱命令具有以下语法:

$ trap commands signals

在这里,command 可以是任何有效的 Unix 命令,甚至可以是用户定义的函数,而 signal 可能是你想要捕获的任何数量信号的列表。

在 shell 脚本中,trap 有两种常见用法:

  1. Clean up temporary files

  2. Ignore signals

Cleaning Up Temporary Files

作为 trap 命令的示例,以下显示了当有人尝试从终端中止程序时,如何删除一些文件并随后退出:

$ trap "rm -f $WORKDIR/work1$$ $WORKDIR/dataout$$; exit" 2

从执行此陷阱的 shell 程序开始,如果程序收到信号号 2,将会自动删除这两个文件 work1dataout

因此,如果用户在此陷阱执行后中断程序的执行,你可以确保会清理这两个文件。紧随 rm 之后的 exit 命令是必需的,因为没有它,执行将在接收到信号时停止的位置继续进行。

信号号 1 为 hangup 生成。有人故意挂断电话线或电话线意外断开。

你可以通过向信号列表中添加信号号 1 来修改前面的 trap,以便在此情况下也删除两个指定的文件:

$ trap "rm $WORKDIR/work1$$ $WORKDIR/dataout$$; exit" 1 2

现在,如果电话挂断或按下 Ctrl+C 键,这些文件将被删除。

如果指定的陷阱命令包含不止一个命令,则必须用引号引起来。还要注意,shell 在执行 trap 命令和接收到其中一个列出的信号时扫描命令行。

因此,在前面的示例中, WORKDIR$$ 的值将在执行 trap 命令时进行替换。如果你希望在接收到信号 1 或 2 时执行此替换,可以将命令放在单引号中:

$ trap 'rm $WORKDIR/work1$$ $WORKDIR/dataout$$; exit' 1 2

Ignoring Signals

如果为 trap 列出的命令为空,则在接收到指定信号时将忽略该信号。例如,命令:

$ trap '' 2

这指定了要忽略中断信号。你在执行不想被中断的操作时可能希望忽略某些信号。你可以指定要忽略的多个信号,如下所示:

$ trap '' 1 2 3 15

请注意,必须指定第一个参数才能忽略信号,并且不等于写以下内容,后者有其自己的单独含义:

$ trap  2

如果你忽略信号,所有子 shell 也将忽略该信号。但是,如果你指定在接收到信号时执行某个操作,则所有子 shell 在接收到该信号时仍将执行默认操作。

Resetting Traps

在你更改了在接收到信号时执行的默认操作之后,如果你只省略第一个参数,你可以使用 trap 再次将其更改回来;因此:

$ trap 1 2

这将接收到信号 1 或 2 时要执行的动作重置为默认值。