• LINUX基础实验报告


    实验一:主要是介绍Linux系统概况,无运行代码。

    实验二:Linux的基本操作

    重要知识点

    [Tab]

    使用Tab键来进行命令补全,Tab键一般键盘是在字母Q旁边,这个技巧给你带来的最大的好处就是当你忘记某个命令的全称时你可以只输入它的开头的一部分然后按下Tab键就可以得到提示或者帮助完成,当然不止补全命令,补全目录,补全命令参数都是没问题的。

    Ctrl+c键

    用来强行终止当前程序。

    一些其他常用快捷键

    按键

    作用

    Ctrl+d

    键盘输入结束或退出终端

    Ctrl+s

    暂定当前程序,暂停后按下任意键恢复运行

    Ctrl+z

    将当前程序放到后台运行,恢复到前台为命令fg

    Ctrl+a

    将光标移至输入行头,相当于Home键

    Ctrl+e

    将光标移至输入行末,相当于End键

    Ctrl+k

    删除从光标所在位置到行末

    Alt+Backspace

    向前删除一个单词

    Shift+PgUp

    将终端显示向上滚动

    Shift+PgDn

    将终端显示向下滚动

    通配符是一种特殊语句,主要有星号(*)和问号(?),用来对对字符串进行模糊匹配(比如文件名,参数名)。当查找文件夹时,可以使用它来代替一个或多个真正字符;当不知道真正字符或者懒得输入完整名字时,常常使用通配符代替一个或多个真正的字符。

    终端里面输入的通配符是由 Shell 处理的,不是由所涉及到命令语句处理的,它只会出现在命令的“参数值”里(它不用在 命令名称里, 命令不记得,那就用Tab补全)。当 Shell 在“参数值”中遇到了通配符时,Shell 会将其当作路径或文件名去在磁盘上搜寻可能的匹配:若符合要求的匹配存在,则进行代换(路径扩展);否则就将该通配符作为一个普通字符传递给“命令”,然后再由命令进行处理。总之,通配符 实际上就是一种 Shell 实现的路径扩展功能。在 通配符被处理后, Shell 会先完成该命令的重组,然后再继续处理重组后的命令,直至执行该命令。

    Shell 常用通配符:

    字符

    含义

    *

    匹配 0 或多个字符

    ?

    匹配任意一个字符

    [list]

    匹配 list 中的任意单一字符

    [!list]

    匹配 除list 中的任意单一字符以外的字符

    [c1-c2]

    匹配 c1-c2 中的任意单一字符 如:[0-9] [a-z]

    {string1,string2,...}

    匹配 sring1 或 string2 (或更多)其一字符串

    {c2..c2}

    匹配 c1-c2 中全部字符 如{1..10}

    在linux命令行中获取帮助

    使用man 命令

    通常情况下,man 手册里面的内容都是英文的,这就要求你有一定的英文基础。man 手册的内容很多,涉及了 Linux 使用过程中的方方面面,为了便于查找,是做了分册(分区段)处理的,在Research UNIX、BSD、OS X 和 Linux 中,手册通常被分为8个区段,安排如下:

    区段

    说明

    1

    一般命令

    2

    系统调用

    3

    库函数,涵盖了C标准函数库

    4

    特殊文件(通常是/dev中的设备)和驱动程序

    5

    文件格式和约定

    6

    游戏和屏保

    7

    杂项

    8

    系统管理命令和守护进程

    实验三:用户及文件权限管理

    重要知识点

    1.查看用户

    请打开终端,输入命令:

    who am i

    或者

    who mom likes

     

    who 命令其它常用参数

    参数

    说明

    -a

    打印能打印的全部

    -d

    打印死掉的进程

    -m

    同am i,mom likes

    -q

    打印当前登录用户数及用户名

    -u

    打印当前登录用户登录信息

    -r

    打印运行等级

    2.创建用户

    在命令行中输入

     sudo adduser +用户名

    3.用户组

    在 Linux 里面每个用户都有一个归属(用户组),用户组简单地理解就是一组用户的集合,它们共享一些资源和权限,同时拥有私有资源,就跟家的形式差不多,你的兄弟姐妹(不同的用户)属于同一个家(用户组),你们可以共同拥有这个家(共享资源),爸妈对待你们都一样(共享权限),你偶尔写写日记,其他人未经允许不能查看(私有资源和权限)。当然一个用户是可以属于多个用户组的,正如你既属于家庭,又属于学校或公司。

    查询自己属于哪个用户组:

    方法一:使用groups命令

    在命令行输入 groups+用户名

    方法二:查看/etc/group文件

    在命令行中输入

    cat /etc/group | sort

     

    4.删除用户

    在命令行中输入:sudo deluser 用户名 --remove-home

    5.查看文件权限

    在命令行中输入: ls -l

    6.显示除了 '.'(当前目录),'..' 上一级目录之外的所有包含隐藏文件(Linux 下以 '.' 开头的文件为隐藏文件)

    在命令行中输入: ls -A

     当然,你可以同时使用 '-A' 和 '-l' 参数:

    在命令行中输入: ls -Al

     

    7.查看某一个目录的完整属性,而不是显示目录里面的文件属性:

    在命令行中输入: ls -dl <目录名>

    8。显示所有文件大小,并以普通人类能看懂的方式呈现:

    在命令行中输入:ls -AsSh

    其中小 s 为显示文件大小,大 S 为按文件大小排序,若需要知道如何按其它方式排序,请使用“man”命令查询。

     

    9.变更文件所有者

    假设目前是 lilei 用户登录,新建一个文件,命名为 “iphone6”:

    $ touch iphone6

    可见文件所有者是 lilei :

    现在,使用以下命令变更文件所有者为 shiyanlou :

    $ cd /home/lilei

    $ ls iphone6

    $ sudo chown shiyanlou iphone6

    $ cp iphone6 /home/shiyanlou

    现在查看,发现 文件所有者成功修改为 shiyanlou 。

     
     
     
     
     
     
     

     

    10.修改文件权限

    • 方式一:二进制数字表示

    每个文件的三组权限(拥有者,所属用户组,其他用户,记住这个顺序是一定的)就对应这一个 "rwx",也就是一个 '7' ,所以如果我要将文件“iphone6”的权限改为只有我自己可以用那么就这样:

    在命令行中输入:chmod 700 iphone6

     

    • 方式二:加减赋值操作

    在命令行中输入:chmod go-rw iphone

    'g''o'还有'u',分别表示group,others,user,'+','-' 就分别表示增加和去掉相应的权限。

     

    实验截图:

     

    实验四:Linux 目录结构及文件基本操作

    重要知识点:

    1.Linux 目录结构

    在讲 Linux 目录结构之前,你首先要清楚一点东西,那就是 Linux 的目录与 Windows 的目录的区别,或许对于一般操作上的感受来说没有多大不同,但从它们的实现机制来说是完全不同的。

    一种不同是体现在目录与存储介质(磁盘,内存,DVD 等)的关系上,以往的 Windows 一直是以存储介质为主的,主要以盘符(C 盘,D 盘...)及分区的来实现文件管理,然后之下才是目录,目录就显得不是那么重要,除系统文件之外的用户文件放在任何地方任何目录也是没有多大关系。所以通常 Windows 在使用一段时间后,磁盘上面的文件目录会显得杂乱无章(少数善于整理的用户除外吧)。然而 UNIX/Linux 恰好相反,UNIX 是以目录为主的,Linux 也继承了这一优良特性。 Linux 是以树形目录结构的形式来构建整个系统的,可以理解为一个用户可操作系统的骨架。虽然本质上无论是目录结构还是操作系统内核都是存储在磁盘上的,但从逻辑上来说 Linux 的磁盘是“挂在”(挂载在)目录上的,每一个目录不仅能使用本地磁盘分区的文件系统,也可以使用网络上的文件系统。举例来说,可以利用网络文件系统(Network File System,NFS)服务器载入某特定目录等。

     

    2.目录路径

    路径

    有人可能不明白这路径是指什么,有什么用。顾名思义,路径就是你要去哪儿的路线嘛。如果你想进入某个具体的目录或者想获得某个目录的文件(目录本身也是文件)那就得用路径来找到了。

    使用 cd 命令可以切换目录,在 Linux 里面使用 . 表示当前目录,.. 表示上一级目录(**注意,还记得我们上一节介绍过的,以 . 开头的文件都是隐藏文件,所以这两个目录必然也是隐藏的,你可以使用 ls -a 命令查看隐藏文件), - 表示上一次所在目录,~ 通常表示当前用户的"home"目录。使用 pwd 命令可以获取当前所在路径(绝对路径)。

    进入上一级目录:

    $ cd ..

    进入你的“home”目录:

    $ cd ~ 

    使用 pwd 获取当前路径:

    $ pwd

    绝对路径

    关于绝对路径,简单地说就是以根"/"目录为起点的完整路径,以你所要到的目录为终点,表现形式如:/usr/local/bin,表示根目录下的 usr 目录中的 local 目录中的 bin 目录。

    相对路径

    相对路径,也就是相对于你当前的目录的路径,相对路径是以当前目录 . 为起点,以你所要到的目录为终点,表现形式如: usr/local/bin (这里假设你当前目录为根目录)。你可能注意到,我们表示相对路径实际并没有加上表示当前目录的那个 . ,而是直接以目录名开头,因为这个 usr 目录为 / 目录下的子目录,是可以省略这个 . 的(以后会讲到一个类似不能省略的情况);如果是当前目录的上一级目录,则需要使用 .. ,比如你当前目录为“home”目录,根目录就应该表示为 ../../ ,表示上一级目录("home"目录)的上一级目录("/"目录)。

     

    3.Linux 文件的基本操作

    新建空白文件

    在命令行中输入:touch 用户名

     

    新建目录

    在命令行中输入:mkdir 目录名 或者 mkdir -p 多级目录路径

    复制文件

    在命令行中输入:cp  文件名 目标路径

    复制目录

    在命令行中输入:cp -r 目录名

     

    删除文件

    在命令行中输入:rm 文件名 或者强制删除 rm -f 文件名

     

    删除目录

    在命令行中输入:rm -f 目录名

     

    移动文件

    在命令行中输入:mv 文件名 目标目录

     

    重命名文件

    在命令行中输入:mv 旧文件名 新文件名

     

    批量重命名

    使用rename命令

     

    查看文件

    在命令行中输入:cat 文件名

     

    nl命令

    -b : 指定添加行号的方式,主要有两种:

        -b a:表示无论是否为空行,同样列出行号("cat -n"就是这种方式)

        -b t:只列出非空行的编号并列出(默认为这种方式)

    -n : 设置行号的样式,主要有三种:

        -n ln:在行号字段最左端显示

        -n rn:在行号字段最右边显示,且不加 0

        -n rz:在行号字段最右边显示,且加 0

    -w : 行号字段占用的位数(默认为 6 位)

     

    使用more和less命令分页查看文件

    如果说上面的 cat 是用来快速查看一个文件内容的,那么这个more和less就是天生用来"阅读"一个文件的内容的,比如说"man"手册内部就是使用的 less 来显示内容。其中more命令比较简单,只能向一个方向滚动,而"less"为基于"more"和"vi"(一个强大的编辑器,我们有单独的课程来让你学习)开发,功能更强大。

    在命令行输入more 或者less 文件名

     

    使用head和tail命令查看文件

    这两个命令那些性子比较急的人应该会比较喜欢,因为它们一个是只查看的头几行(默认为10行,不足10行则显示全部)和尾几行。还是拿 passwd 文件举例,比如当我们想要查看最近新增加的用户,那么我们可以查看这个/etc/passwd文件,不过我们前面也看到了,这个文件里面一大堆乱糟糟的东西,看起来实在费神啊。这里想到系统新增加一个用户,应该会将用户的信息添加到passwd文件的最后,那么这时候我们就可以使用tail命令了

     

    查看文件类型

    我们通常使用file命令可以查看文件的类型:$ file 目标文件

     

    实验截图:

     

    实验五:环境变量与文件查找

    重要知识点:

    1.变量

    要解释环境变量,得先明白变量是什么,准确的说应该是 Shell 变量,所谓变量就是计算机中用于记录一个值(不一定是数值,也可以是字符或字符串)的符号,而这些符号将用于不同的运算处理中。通常变量与值是一对一的关系,可以通过表达式读取它的值赋值给其它变量,也可以直接指定数值赋值给任意变量。为了便于运算和处理,大部分的编程语言会区分变量的类型,用于分别记录数值、字符或者字符串等等数据类型。Shell 中的变量也基本如此,有不同类型(但不用专门指定类型名),可以参与运算,有作用域限定。

    变量的作用域即变量的有效范围(比如一个函数中、一个源文件中或者全局范围),在该范围内只能有一个同名变量。一旦离开则该变量无效,如同不存在这个变量一般。

    在 Shell 中如何创建一个变量,如何给变量赋值和如何读取变量的值呢?这部分内容会在bash 脚本编程这门课中详细介绍,这里我简单举例说明一下:

    使用declare命令创建一个变量名为 tmp 的变量:

    $ declare tmp

    其实也可以不用 declare 预声明一个变量,直接即用即创建,这里只是告诉你 declare 的作用,这在创建其它指定类型的变量(如数组)时会用到。

    使用=号赋值运算符为变量 tmp 赋值为 shiyanlou:

    $ tmp=shiyanlou

    读取变量的值,使用echo命令和$符号($符号用于表示引用一个变量的值,初学者经常会忘记输入):

    $ echo $tmp

     

    2.环境变量

    简单理解了变量的概念,就很好解释环境变量了,环境变量就是作用域比自定义变量要大,如Shell 的环境变量作用于自身和它的子进程。在所有的 UNIX 和类 UNIX 系统中,每个进程都有其各自的环境变量设置,且默认情况下,当一个进程被创建时,处理创建过程中明确指定的话,它将继承其父进程的绝大部分环境设置。Shell 程序也作为一个进程运行在操作系统之上,而我们在 Shell中运行的大部分命令都将以 Shell 的子进程的方式运行。

     

    通常我们会涉及到的环境变量有三种:

    • 当前 Shell 进程私有用户自定义变量,如上面我们创建的 temp 变量,只在当前 Shell 中有效。
    • Shell 本身内建的变量。
    • 从自定义变量导出的环境变量。

    也有三个与上述三种环境变量相关的命令,set,env,export。这三个命令很相似,都可以用于打印相关环境变量,区别在于涉及的是不同范围的环境变量,详见下表:

    命令

    说明

    set

    显示当前 Shell 所有环境变量,包括其内建环境变量(与 Shell 外观等相关),用户自定义变量及导出的环境变量

    env

    显示与当前用户相关的环境变量,还可以让命令在指定环境中运行

    export

    显示从 Shell 中导出成环境变量的变量,也能通过它将自定义变量导出为环境变量

    3.命令的查找路径与顺序

    你可能很早之前就有疑问,我们在 Shell 中输入一个命令,Shell 是怎么知道在哪去找到这个命令然后执行的呢?这是通过环境变量PATH来进行搜索的,熟悉 Windows 的用户可能知道 Windows 中的也是有这么一个 PATH 环境变量。这个PATH里面就保存了Shell中执行的命令的搜索路径。

    查看PATH环境变量的内容:

    $ echo $PATH

    默认情况下你会看到如下输出:

    /usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games

    如果你还记得我们在 Linux 目录结构那一节的内容,你就应该知道上面这些目录下放的是哪一类文件了。通常这一类目录下放的都是可执行文件,当我们在 Shell 中执行一个命令时,系统就会按照 PATH 中设定的路径按照顺序依次到目录中去查找,如果存在同名的命令,则执行先找到的那个。

    下面我们将练习创建一个最简单的可执行 Shell 脚本和一个使用 C 语言创建的"hello world"程序,如果这两部分内容你之前没有学习过,那么你可以进行一个入门学习: C 语言入门教程 高级 Bash 脚本编程指南 Linux Shell Scripting Tutorial (LSST) v2.0

    创建一个 Shell 脚本文件:

    $ vim hello_shell.sh

    在脚本中添加如下内容,保存并退出(注意不要省掉第一行,这不是注释,论坛有用户反应会有语法错误,就是因为没有了第一行):

    #!/bin/zsh

     

    for ((i=0; i<10; i++));do

        echo "hello shell"

    done

     

    exit 0

    为文件添加可执行权限:

    $ chmod 755 hello_shell.sh

    执行脚本

    $ ./hello_shell.sh

    创建一个 C 语言"hello world"程序:

    $ vim hello_world.c

    #include <stdio.h>

     

    int main(void)

    {

        printf("hello world!\n");

        return 0;

    }

    使用 gcc 生成可执行文件:

    $ gcc -o hello_world hello_world.c

    gcc 生成二进制文件默认具有可执行权限,不需要修改

    在 shiyanlou 家目录创建一个mybin目录,并将上述 hello_shell.sh 和 hello_world 文件移动到其中:

    $ mkdir mybin

    $ mv hello_shell.sh hello_world mybin/

    现在你可以在mybin目录中分别运行你刚刚创建的两个程序:

    $ cd mybin

    $ ./hello_shell.sh

    $ ./hello_world

    4.添加自定义路径到“PATH”环境变量

    在前面我们应该注意到PATH里面的路径是以:作为分割符,所以我们可以这样添加自定义路径:

    $ PATH=$PATH:/home/shiyanlou/mybin

    注意这里一定要使用绝对路径

    现在你就可以在其他任意目录执行那两个命令了。你可能会意识到这样还并没有很好的解决问题,因为我给 PATH 环境变量追加了一个路径,它也只是在当前 Shell 有效,我一旦退出终端,再打开就会发现又失效了。有没有方法让添加的环境变量全局有效?或者每次启动 Shell 时自动执行上面添加自定义路径到 PATH 的命令?下面我们就来说说后一种方式——让它自动执行。

    在每个用户的 home 目录中有一个 Shell 每次启动时会默认执行一个配置脚本,以初始化环境,包括添加一些用户自定义环境变量等等。zsh 的配置文件是.zshrc,相应 Bash 的配置文件为.bashrc。它们在etc下还都有一个或多个全局的配置文件,不过我们一般只修改用户目录下的配置文件。

    我们可以简单的使用下面命令直接添加内容到.zshrc中:

    $ echo "PATH=$PATH:/home/shiyanlou/mybin" >> .zshrc

    上述命令中>>表示将标准输出以追加的方式重定向到一个文件中,注意前面用到的>是以覆盖的方式重定向到一个文件中,使用的时候一定要注意分辨。在指定文件不存在的情况下都会创建新的文件。

    5.修改和删除已有变量

    变量修改

    变量的修改有以下几种方式:

    变量设置方式

    说明

    ${变量名#匹配字串}

    从头向后开始匹配,删除符合匹配字串的最短数据

    ${变量名##匹配字串}

    从头向后开始匹配,删除符合匹配字串的最长数据

    ${变量名%匹配字串}

    从尾向前开始匹配,删除符合匹配字串的最短数据

    ${变量名%%匹配字串}

    从尾向前开始匹配,删除符合匹配字串的最长数据

    ${变量名/旧的字串/新的字串}

    将符合旧字串的第一个字串替换为新的字串

    ${变量名//旧的字串/新的字串}

    将符合旧字串的全部字串替换为新的字串

    比如要修改我们前面添加到 PATH 的环境变量。为了避免操作失误导致命令找不到,我们先将 PATH 赋值给一个新的自定义变量 path:

    $ path=$PATH

    $ echo $path

    $ path=${path%/home/shiyanlou/mybin}

    # 或使用通配符,*表示任意多个任意字符

    $ path=${path%*/mybin}

    变量删除

    可以使用unset命令删除一个环境变量:

    $ unset temp

    6.如何让环境变量立即生效

    在上面我们在 Shell 中修改了一个配置脚本文件之后(比如 zsh 的配置文件 home 目录下的.zshrc),每次都要退出终端重新打开甚至重启主机之后其才能生效,很是麻烦,我们可以使用source命令来让其立即生效,如:

    $ source .zshrc

    source命令还有一个别名就是.,注意与表示当前路径的那个点区分开,虽然形式一样,但作用和使用方式一样,上面的命令如果替换成.的方式就该是

    $ . ./.zshrc

    注意第一个点后面有一个空格,而且后面的文件必须指定完整的绝对或相对路径名,source 则不需要。

     

    二、搜索文件

    与搜索相关的命令常用的有如下几个whereis,which,find,locate。

    • whereis简单快速

    $whereis who

     

    你会看到它找到了三个路径,两个可执行文件路径和一个 man 在线帮助文件所在路径,这个搜索很快,因为它并没有从硬盘中依次查找,而是直接从数据库中查询。whereis只能搜索二进制文件(-b),man帮助文件(-m)和源代码文件(-s)。如果想要获得更全面的搜索结果可以使用locate命令。

    • locate快而全

    通过"/var/lib/mlocate/mlocate.db"数据库查找,不过这个数据库也不是实时更新的,系统会使用定时任务每天自动执行updatedb命令更新一次,所以有时候你刚添加的文件,它可能会找不到,需要手动执行一次updatedb命令(在我们的环境中必须先执行一次该命令)。它可以用来查找指定目录下的不同文件类型,如查找 /etc 下所有以 sh 开头的文件:

    $ locate /etc/sh

    注意,它不只是在 etc 目录下查找并会自动递归子目录进行查找

    查找 /usr/share/ 下所有 jpg 文件:

    $ locate /usr/share/\*.jpg

    注意要添加*号前面的反斜杠转义,否则会无法找到

    如果想只统计数目可以加上-c参数,-i参数可以忽略大小写进行查找,whereis 的-b,-m,-s同样可以是使用。

    • which小而精

    which本身是 Shell 内建的一个命令,我们通常使用which来确定是否安装了某个指定的软件,因为它只从PATH环境变量指定的路径中去搜索命令:

    $ which man

    • find精而细

    find应该是这几个命令中最强大的了,它不但可以通过文件类型、文件名进行查找而且可以根据文件的属性(如文件的时间戳,文件的权限等)进行搜索。find命令强大到,要把它将明白至少需要单独好几节课程才行,我们这里只介绍一些常用的内容。

    在指定目录下搜索指定文件名的文件:

    $ find /etc/ -name interfaces

    注意 find 命令的路径是作为第一个参数的,基本命令格式为 find [path] [option] [action]

    与时间相关的命令参数:

    参数

    说明

    -atime

    最后访问时间

    -ctime

    创建时间

    -mtime

    最后修改时间

    下面以-mtime参数举例:

    • -mtime n: n 为数字,表示为在n天之前的”一天之内“修改过的文件
    • -mtime +n: 列出在n天之前(不包含n天本身)被修改过的文件
    • -mtime -n: 列出在n天之前(包含n天本身)被修改过的文件
    • newer file: file为一个已存在的文件,列出比file还要新的文件名

    列出 home 目录中,当天(24 小时之内)有改动的文件:

    $ find ~ -mtime 0

    列出用户家目录下比Code文件夹新的文件:

    $ find ~ -newer /home/shiyanlou/Code

     

    实验截图:

     

    实验六:文件打包与压缩

    重要知识点:

    一、文件打包和解压缩

    在讲 Linux 上的解压缩工具之前,有必要先了解以下常见常用的压缩包文件格式。在 Windows 上我们最常见的不外乎这三种*.zip,*.rar,*.7z后缀的压缩文件,而在 Linux 上面常见常用的除了以上这三种外,还有*.gz,*.xz,*.bz2,*.tar,*.tar.gz,*.tar.xz,*tar.bz2,简单介绍如下:

    文件后缀名

    说明

    *.zip

    zip程序打包压缩的文件

    *.rar

    rar程序压缩的文件

    *.7z

    7zip程序压缩的文件

    *.tar

    tar程序打包,未压缩的文件

    *.gz

    gzip程序(GNU zip)压缩的文件

    *.xz

    xz程序压缩的文件

    *.bz2

    bzip2程序压缩的文件

    *.tar.gz

    tar打包,gzip程序压缩的文件

    *.tar.xz

    tar打包,xz程序压缩的文件

    *tar.bz2

    tar打包,bzip2程序压缩的文件

    *.tar.7z

    tar打包,7z程序压缩的文件

    讲了这么多种压缩文件,这么多个命令,不过我们一般只需要掌握几个命令即可,包括zip,rar,tar。下面会依次介绍这几个命令及对应的解压命令。

    1.zip压缩打包程序

    • 使用zip打包文件夹:

    $ zip -r -q -o shiyanlou.zip /home/shiyanlou

    $ du -h shiyanlou.zip

    $ file shiyanlou.zip

     

    上面命令将 shiyanlou 的 home 目录打包成一个文件,并查看了打包后文件的大小和类型。第一行命令中,-r参数表示递归打包包含子目录的全部内容,-q参数表示为安静模式,即不向屏幕输出信息,-o,表示输出文件,需在其后紧跟打包输出文件名。后面使用du命令查看打包后文件的大小(后面会具体说明该命令)。

    • 设置压缩级别为9和1(9最大,1最小),重新打包:

    $ zip -r -9 -q -o shiyanlou_9.zip /home/shiyanlou -x ~/*.zip

    $ zip -r -1 -q -o shiyanlou_1.zip /home/shiyanlou -x ~/*.zip

    这里添加了一个参数用于设置压缩级别-[1-9],1表示最快压缩但体积大,9表示体积最小但耗时最久。最后那个-x是为了排除我们上一次创建的 zip 文件,否则又会被打包进这一次的压缩文件中,注意:这里只能使用绝对路径,否则不起作用

    我们再用du命令分别查看默认压缩级别、最低、最高压缩级别及未压缩的文件的大小:

    $ du -h -d 0 *.zip ~ | sort

    通过man 手册可知:

    • h, --human-readable(顾名思义,你可以试试不加的情况)
    • d, --max-depth(所查看文件的深度)

    这样一目了然,你可以看到默认压缩级别应该是最高的,效果很明显,不过你在环境中操作之后看到的大小可能跟图上的有些不同,因为在你使用过程中,会随时还生成一些缓存文件在当前用户的家目录中,这对于我们学习命令使用来说,是无关紧要的,可以忽略这些不同。

    • 创建加密zip包

    使用-e参数可以创建加密压缩包:

    $ zip -r -e -o shiyanlou_encryption.zip /home/shiyanlou

    注意: 关于zip命令,因为 Windows 系统与 Linux/Unix 在文本文件格式上的一些兼容问题,比如换行符(为不可见字符),在 Windows 为 CR+LF(Carriage-Return+Line-Feed:回车加换行),而在 Linux/Unix 上为 LF(换行),所以如果在不加处理的情况下,在 Linux 上编辑的文本,在 Windows 系统上打开可能看起来是没有换行的。如果你想让你在 Linux 创建的 zip 压缩文件在 Windows 上解压后没有任何问题,那么你还需要对命令做一些修改:

    $ zip -r -l -o shiyanlou.zip /home/shiyanlou

    需要加上-l参数将LF转换为CR+LF来达到以上目的。

    2.使用unzip命令解压缩zip文件

    将shiyanlou.zip解压到当前目录:

    $ unzip shiyanlou.zip

    使用安静模式,将文件解压到指定目录:

    $ unzip -q shiyanlou.zip -d ziptest

    上述指定目录不存在,将会自动创建。如果你不想解压只想查看压缩包的内容你可以使用-l参数:

    $ unzip -l shiyanlou.zip

    注意: 使用unzip解压文件时我们同样应该注意兼容问题,不过这里我们关心的不再是上面的问题,而是中文编码的问题,通常 Windows 系统上面创建的压缩文件,如果有有包含中文的文档或以中文作为文件名的文件时默认会采用 GBK 或其它编码,而 Linux 上面默认使用的是 UTF-8 编码,如果不加任何处理,直接解压的话可能会出现中文乱码的问题(有时候它会自动帮你处理),为了解决这个问题,我们可以在解压时指定编码类型。

    使用-O(英文字母,大写o)参数指定编码类型:

    unzip -O GBK 中文压缩文件.zip

    3.rar打包压缩命令

    rar也是 Windows 上常用的一种压缩文件格式,在 Linux 上可以使用rar和unrar工具分别创建和解压 rar 压缩包。

    • 安装rar和unrar工具:

    $ sudo apt-get update

    $ sudo apt-get install rar unrar

    • 从指定文件或目录创建压缩包或添加文件到压缩包:

    $ rm *.zip

    $ rar a shiyanlou.rar .

    上面的命令使用a参数添加一个目录~到一个归档文件中,如果该文件不存在就会自动创建。

    注意:rar 的命令参数没有-,如果加上会报错。

    • 从指定压缩包文件中删除某个文件:

    $ rar d shiyanlou.rar .zshrc

    • 查看不解压文件:

    $ rar l shiyanlou.rar

    • 使用unrar解压rar文件

    全路径解压:

    $ unrar x shiyanlou.rar

    去掉路径解压:

    $ mkdir tmp

    $ unrar e shiyanlou.rar tmp/

    rar命令参数非常多,上面只涉及了一些基本操作

    4.tar打包工具

    在 Linux 上面更常用的是tar工具,tar 原本只是一个打包工具,只是同时还是实现了对 7z,gzip,xz,bzip2 等工具的支持,这些压缩工具本身只能实现对文件或目录(单独压缩目录中的文件)的压缩,没有实现对文件的打包压缩,所以我们也无需再单独去学习其他几个工具,tar 的解压和压缩都是同一个命令,只需参数不同,使用比较方便。

    下面先掌握tar命令一些基本的使用方式,即不进行压缩只是进行打包(创建归档文件)和解包的操作。

    • 创建一个 tar 包:

    $ tar -cf shiyanlou.tar ~

    • 解包一个文件(-x参数)到指定路径的已存在目录(-C参数):

    $ mkdir tardir

    $ tar -xf shiyanlou.tar -C tardir

    • 只查看不解包文件-t参数:

    $ tar -tf shiyanlou.tar

    • 保留文件属性和跟随链接(符号链接或软链接),有时候我们使用tar备份文件当你在其他主机还原时希望保留文件的属性(-p参数)和备份链接指向的源文件而不是链接本身(-h参数):

    $ tar -cphf etc.tar /etc

    对于创建不同的压缩格式的文件,对于tar来说是相当简单的,需要的只是换一个参数,这里我们就以使用gzip工具创建*.tar.gz文件为例来说明。

    • 我们只需要在创建 tar 文件的基础上添加-z参数,使用gzip来压缩文件:

    $ tar -czf shiyanlou.tar.gz ~

    • 解压*.tar.gz文件:

    $ tar -xzf shiyanlou.tar.gz

    压缩文件格式

    参数

    *.tar.gz

    -z

    *.tar.xz

    -J

    *tar.bz2

    -j

    实验截图:

     实验七:文件系统操作与磁盘管理

    重要知识点:

    一、简单文件系统操作

    1.查看磁盘和目录的容量

    使用 df 命令查看磁盘的容量

    $ df

    使用 du 命令查看目录的容量

    这个命令前面其实已经用了很多次了:

    # 默认同样以 blocks 的大小展示

    $ du

    # 加上`-h`参数,以更易读的方式展示

    $ du -h

    -d参数指定查看目录的深度

    # 只查看1级目录的信息

    $ du -h -d 0 ~

    # 查看2级

    $ du -h -d 1 ~

    二、简单的磁盘管理

    下面涉及的命令具有一定的危险性,操作不当可能会丢失你的个人数据,初学者建议在虚拟环境中进行操作

    通常情况下,这一小节应该直接将如何挂载卸载磁盘,如何格式化磁盘,如何分区,但如你所见,我们的环境中没东西给你挂,也没东西给你格和分,所以首先我们会先创建一个虚拟磁盘来进行后续的练习操作

    1.创建虚拟磁盘

    dd 命令简介(部分说明来自dd (Unix) wiki))

    dd命令用于转换和复制文件,不过它的复制不同于cp。之前提到过关于 Linux 的很重要的一点,一切即文件,在 Linux 上,硬件的设备驱动(如硬盘)和特殊设备文件(如/dev/zero和/dev/random)都像普通文件一样,只要在各自的驱动程序中实现了对应的功能,dd 也可以读取自和/或写入到这些文件。这样,dd也可以用在备份硬件的引导扇区、获取一定数量的随机数据或者空数据等任务中。dd程序也可以在复制时处理数据,例如转换字节序、或在 ASCII 与 EBCDIC 编码间互换。

    dd的命令行语句与其他的 Linux 程序不同,因为它的命令行选项格式为选项=值,而不是更标准的--选项 值或-选项=值。dd默认从标准输入中读取,并写入到标准输出中,但可以用选项if(input file,输入文件)和of(output file,输出文件)改变。

    我们先来试试用dd命令从标准输入读入用户输入到标准输出或者一个文件:

    # 输出到文件

    $ dd of=test bs=10 count=1 # 或者 dd if=/dev/stdin of=test bs=10 count=1

    # 输出到标准输出

    $ dd if=/dev/stdin of=/dev/stdout bs=10 count=1

    前面说到dd在拷贝的同时还可以实现数据转换,那下面就举一个简单的例子:将输出的英文字符转换为大写再写入文件:

    $ dd if=/dev/stdin of=test bs=10 count=1 conv=ucase

    使用 dd 命令创建虚拟镜像文件

    通过上面一小节,你应该掌握了dd的基本使用,下面就来使用dd命令来完成创建虚拟磁盘的第一步。

    从/dev/zero设备创建一个容量为 256M 的空文件:

    $ dd if=/dev/zero of=virtual.img bs=1M count=256

    $ du -h virtual.img

    然后我们要将这个文件格式化(写入文件系统),这里我们要学到一个(准确的说是一组)新的命令来完成这个需求。

    使用 mkfs 命令格式化磁盘(我们这里是自己创建的虚拟磁盘镜像)

    你可以在命令行输入 mkfs 然后按下Tab键,你可以看到很多个以 mkfs 为前缀的命令,这些不同的后缀其实就是表示着不同的文件系统,可以用 mkfs 格式化成的文件系统:

    我们可以简单的使用下面的命令来将我们的虚拟磁盘镜像格式化为ext4文件系统:

    $ mkfs.ext4 virtual.img

    使用 mount 命令挂载磁盘到目录树

    用户在 Linux/UNIX 的机器上打开一个文件以前,包含该文件的文件系统必须先进行挂载的动作,此时用户要对该文件系统执行 mount 的指令以进行挂载。通常是使用在 USB 或其他可移除存储设备上,而根目录则需要始终保持挂载的状态。又因为 Linux/UNIX 文件系统可以对应一个文件而不一定要是硬件设备,所以可以挂载一个包含文件系统的文件到目录树。

    Linux/UNIX 命令行的 mount 指令是告诉操作系统,对应的文件系统已经准备好,可以使用了,而该文件系统会对应到一个特定的点(称为挂载点)。挂载好的文件、目录、设备以及特殊文件即可提供用户使用。

    我们先来使用mount来查看下主机已经挂载的文件系统:

    $ sudo mount

    那么我们如何挂载真正的磁盘到目录树呢,mount命令的一般格式如下:

    mount [options] [source] [directory]

    一些常用操作:

    mount [-o [操作选项]] [-t 文件系统类型] [-w|--rw|--ro] [文件系统源] [挂载点]

    我们现在直接来挂载我们创建的虚拟磁盘镜像到/mnt目录:

    $ mount -o loop -t ext4 virtual.img /mnt

    # 也可以省略挂载类型,很多时候 mount 会自动识别

     

    # 以只读方式挂载

    $ mount -o loop --ro virtual.img /mnt

    # 或者mount -o loop,ro virtual.img /mnt

    使用 umount 命令卸载已挂载磁盘

    # 命令格式 sudo umount 已挂载设备名或者挂载点,如:

    $ sudo umount /mnt

     

    # 查看硬盘分区表信息

    $ sudo fdisk -l

     

    # 进入磁盘分区模式

    $ sudo fdisk virtual.img

    实验截图:

    问题:

    已安装了程序但却 command not found

     

     实验八:命令执行顺序控制与管道

    重要知识点:

    一、命令执行顺序的控制

    1.顺序执行多条命令

    通常情况下,我们每次只能在终端输入一条命令,按下回车执行,执行完成后,我们再输入第二条命令,然后再按回车执行…… 你可能会遇到如下使用场景:我需要使用apt-get安装一个软件,然后安装完成后立即运行安装的软件(或命令工具),又恰巧你的主机才更换的软件源还没有更新软件列表(比如之前我们的环境中,每次重新开始实验就得sudo apt-get update,现在已经没有这个问题了),那么你可能会有如下一系列操作:

    $ sudo apt-get update

    # 等待——————————然后输入下面的命令

    $ sudo apt-get install some-tool

    # 等待——————————然后输入下面的命令

    $ some-tool

    这时你可能就会想要是我可以一次性输入完,让它自己去一次执行各命令就好了,这就是我们这一小节要解决的问题。

    简单的顺序执行你可以使用;来完成,比如上述操作你可以:

    $ sudo apt-get update;sudo apt-get install some-tool;some-tool

    # 让它自己运行

    2.有选择的执行命令

    关于上面的操作,不知你有没有思考过一个问题,如果我们在让它自动顺序执行命令时,前面的命令执行不成功,而后面的命令又依赖与上一条命令的结果,那么就会造成花了时间,最终却得到一个错误的结果,而且有时候直观的看你还无法判断结果是否正确。那么我们需要能够有选择性的来执行命令,比如上一条命令执行成功才继续下一条,或者不成功又该做出其它什么处理,比如我们使用which来查找是否安装某个命令,如果找到就执行该命令,否则什么也不做(虽然这个操作没有什么实际意义,但可帮你更好的理解一些概念):

    $ which cowsay>/dev/null && cowsay -f head-in ohch~

    二、管道

    管道是什么,管道是一种通信机制,通常用于进程间的通信(也可通过socket进行网络通信),它表现出来的形式就是将前面每一个进程的输出(stdout)直接作为下一个进程的输入(stdin)。

    管道又分为匿名管道和具名管道(这里将不会讨论在源程序中使用系统调用创建并使用管道的情况,它与命令行的管道在内核中实际都是采用相同的机制)。我们在使用一些过滤程序时经常会用到的就是匿名管道,在命令行中由|分隔符表示,|在前面的内容中我们已经多次使用到了。具名管道简单的说就是有名字的管道,通常只会在源程序中用到具名管道。下面我们就将通过一些常用的可以使用管道的"过滤程序"来帮助你熟练管道的使用。

    1.试用

    先试用一下管道,比如查看/etc目录下有哪些文件和目录,使用ls命令来查看:

    $ ls -al /etc

    有太多内容,屏幕不能完全显示,这时候可以使用滚动条或快捷键滚动窗口来查看。不过这时候可以使用管道:

    $ ls -al /etc | less

    通过管道将前一个命令(ls)的输出作为下一个命令(less)的输入,然后就可以一行一行地看。

    2.cut 命令,打印每一行的某一字段

    打印/etc/passwd文件中以:为分隔符的第1个字段和第6个字段分别表示用户名和其家目录:

    $ cut /etc/passwd -d ':' -f 1,6

    打印/etc/passwd文件中每一行的前N个字符:

    # 前五个(包含第五个)

    $ cut /etc/passwd -c -5

    # 前五个之后的(包含第五个)

    $ cut /etc/passwd -c 5-

    # 第五个

    $ cut /etc/passwd -c 5

    # 2到5之间的(包含第五个)

    $ cut /etc/passwd -c 2-5

    3.grep 命令,在文本中或 stdin 中查找匹配字符串

    grep命令是很强大的,也是相当常用的一个命令,它结合正则表达式可以实现很复杂却很高效的匹配和查找,不过在学习正则表达式之前,这里介绍它简单的使用,而关于正则表达式后面将会有单独一小节介绍到时会再继续学习grep命令和其他一些命令。

    grep命令的一般形式为:

    grep [命令选项]... 用于匹配的表达式 [文件]...

    还是先体验一下,我们搜索/home/shiyanlou目录下所有包含"shiyanlou"的所有文本文件,并显示出现在文本中的行号:

    $ grep -rnI "shiyanlou" ~

     

    # 查看环境变量中以"yanlou"结尾的字符串

    $ export | grep ".*yanlou$"

    4. wc 命令,简单小巧的计数工具

    wc 命令用于统计并输出一个文件中行、单词和字节的数目,比如输出/etc/passwd文件的统计信息:

    $ wc /etc/passwd

    分别只输出行数、单词数、字节数、字符数和输入文本中最长一行的字节数:

    # 行数

    $ wc -l /etc/passwd

    # 单词数

    $ wc -w /etc/passwd

    # 字节数

    $ wc -c /etc/passwd

    # 字符数

    $ wc -m /etc/passwd

    # 最长行字节数

    $ wc -L /etc/passwd

    注意:对于西文字符来说,一个字符就是一个字节,但对于中文字符一个汉字是大于2个字节的,具体数目是由字符编码决定的

     

    5.sort 排序命令

    这个命令前面我们也是用过多次,功能很简单就是将输入按照一定方式排序,然后再输出,它支持的排序有按字典排序,数字排序,按月份排序,随机排序,反转排序,指定特定字段进行排序等等。

    默认为字典排序:

    $ cat /etc/passswd | sort

    反转排序:

    $ cat /etc/passwd | sort -r

    按特定字段排序:

    $ cat /etc/passwd | sort -t':' -k 3

    上面的-t参数用于指定字段的分隔符,这里是以":"作为分隔符;-k 字段号用于指定对哪一个字段进行排序。这里/etc/passwd文件的第三个字段为数字,默认情况下是一字典序排序的,如果要按照数字排序就要加上-n参数:

    $ cat /etc/passwd | sort -t':' -k 3 -n

    6. uniq 去重命令

    uniq命令可以用于过滤或者输出重复行。

    • 过滤重复行

    我们可以使用history命令查看最近执行过的命令(实际为读取${SHELL}_history文件,如我们环境中的~/.zsh_history文件),不过你可能只想查看使用了那个命令而不需要知道具体干了什么,那么你可能就会要想去掉命令后面的参数然后去掉重复的命令:

    $ history | cut -c 8- | cut -d ' ' -f 1 | uniq

    然后经过层层过滤,你会发现确是只输出了执行的命令那一列,不过去重效果好像不明显,仔细看你会发现它趋势去重了,只是不那么明显,之所以不明显是因为uniq命令只能去连续重复的行,不是全文去重,所以要达到预期效果,我们先排序:

    $ history | cut -c 8- | cut -d ' ' -f 1 | sort | uniq

    # 或者$ history | cut -c 8- | cut -d ' ' -f 1 | sort -u

    这就是 Linux/UNIX 哲学吸引人的地方,大繁至简,一个命令只干一件事却能干到最好。

    • 输出重复行

    # 输出重复过的行(重复的只输出一个)及重复次数

    $ history | cut -c 8- | cut -d ' ' -f 1 | sort | uniq -dc

    # 输出所有重复的行

    $ history | cut -c 8- | cut -d ' ' -f 1 | sort | uniq -D

    文本处理命令还有很多,下一节将继续介绍一些常用的文本处理的命令。

     

    实验截图:

     

     实验九:简单的文本处理

     

    文本处理命令

    1.tr 命令

    tr 命令可以用来删除一段文本信息中的某些文字。或者将其进行转换。

    使用方式:

    tr [option]...SET1 [SET2]

    常用的选项有:

    选项

    说明

    -d

    删除和set1匹配的字符,注意不是全词匹配也不是按字符顺序匹配

    -s

    去除set1指定的在输入文本中连续并重复的字符

    操作举例:

    # 删除 "hello shiyanlou" 中所有的'o','l','h'

    $ echo 'hello shiyanlou' | tr -d 'olh'

    # 将"hello" 中的ll,去重为一个l

    $ echo 'hello' | tr -s 'l'

    # 将输入文本,全部转换为大写或小写输出

    $ cat /etc/passwd | tr '[:lower:]' '[:upper:]'

    # 上面的'[:lower:]' '[:upper:]'你也可以简单的写作'[a-z]' '[A-Z]',当然反过来将大写变小写也是可以的

    2.col 命令

    col 命令可以将Tab换成对等数量的空格建,或反转这个操作。

    使用方式:

    col [option]

    常用的选项有:

    选项

    说明

    -x

    将Tab转换为空格

    -h

    将空格转换为Tab(默认选项)

    操作举例:

    # 查看 /etc/protocols 中的不可见字符,可以看到很多 ^I ,这其实就是 Tab 转义成可见字符的符号

    $ cat -A /etc/protocols

    # 使用 col -x 将 /etc/protocols 中的 Tab 转换为空格,然后再使用 cat 查看,你发现 ^I 不见了

    $ cat /etc/protocols | col -x | cat -A

    3.join命令

    学过数据库的用户对这个应该不会陌生,这个命令就是用于将两个文件中包含相同内容的那一行合并在一起。

    使用方式:

    join [option]... file1 file2

    常用的选项有:

    选项

    说明

    -t

    指定分隔符,默认为空格

    -i

    忽略大小写的差异

    -1

    指明第一个文件要用哪个字段来对比,,默认对比第一个字段

    -2

    指明第二个文件要用哪个字段来对比,,默认对比第一个字段

    操作举例:

    # 创建两个文件

    $ echo '1 hello' > file1

    $ echo '1 shiyanlou' > file2

    $ join file1 file2

    # 将/etc/passwd与/etc/shadow两个文件合并,指定以':'作为分隔符

    $ sudo join -t':' /etc/passwd /etc/shadow

    # 将/etc/passwd与/etc/group两个文件合并,指定以':'作为分隔符, 分别比对第4和第3个字段

    $ sudo join -t':' -1 4 /etc/passwd -2 3 /etc/group

    4.paste命令

    paste这个命令与join 命令类似,它是在不对比数据的情况下,简单地将多个文件合并一起,以Tab隔开。

    使用方式:

    paste [option] file...

    常用的选项有:

    选项

    说明

    -d

    指定合并的分隔符,默认为Tab

    -s

    不合并到一行,每个文件为一行

    操作举例:

    $ echo hello > file1

    $ echo shiyanlou > file2

    $ echo www.shiyanlou.com > file3

    $ paste -d ':' file1 file2 file3

    $ paste -s file1 file2 file3

    实验截图:

     

     

    问题:

     

     实验十:数据流重定向

    重要知识点

    一、数据流重定向

    下面我们简单的回顾一下我们前面经常用到的两个重定向操作:

    $ echo 'hello shiyanlou' > redirect

    $ echo 'www.shiyanlou.com' >> redirect

    $ cat redirect

    当然前面没有用到的<和<<操作也是没有问题的,如你理解的一样,它们的区别在于重定向的方向不一致而已,>表示是从左到右,<右到左。

    1.简单的重定向

    在更多了解 Linux 的重定向之前,我们需要先知道一些基本的东西,前面我们已经提到过 Linux 默认提供了三个特殊设备,用于终端的显示和输出,分别为stdin(标准输入,对应于你在终端的输入),stdout(标准输出,对应于终端的输出),stderr(标准错误输出,对应于终端的输出)。

    文件描述符

    设备文件

    说明

    0

    /dev/stdin

    标准输入

    1

    /dev/stdout

    标准输出

    2

    /dev/stderr

    标准错误

    文件描述符:文件描述符在形式上是一个非负整数。实际上,它是一个索引值,指向内核为每一个进程所维护的该进程打开文件的记录表。当程序打开一个现有文件或者创建一个新文件时,内核向进程返回一个文件描述符。在程序设计中,一些涉及底层的程序编写往往会围绕着文件描述符展开。但是文件描述符这一概念往往只适用于 UNIX、Linux 这样的操作系统。

    另外还有一个符号-,它可以同时作为前一个命令的。

    我们可以这样使用这些文件描述符:

    默认使用终端的标准输入作为命令的输入和标准输出作为命令的输出

    $ cat        

    (按Ctrl+C退出)

    将cat的连续输出(heredoc方式)重定向到一个文件

    $ mkdir Documents

    $ cat > Documents/test.c\~ <<EOF

    #include <stdio.h>

     

    int main()

    {

        printf("hello world\n");

        return 0;

    }

     

    EOF

    将一个文件作为命令的输入,标准输出作为命令的输出

    $ cat Documents/test.c\~

    将echo命令通过管道传过来的数据作为cat命令的输入,将标准输出作为命令的输出

    $ echo 'hi' | cat

    将echo命令的输出从默认的标准输出重定向到一个普通文件

    $ echo 'hello shiyanlou' > redirect

    $ cat redirect

    2.标准错误重定向

    重定向标准输出到文件,这是一个很实用的操作,另一个很实用的操作是将标准错误重定向,标准输出和标准错误都被指向伪终端的屏幕显示,所以我们经常看到的一个命令的输出通常是同时包含了标准输出和标准错误的结果的。比如下面的操作:

    # 使用cat 命令同时读取两个文件,其中一个存在,另一个不存在

    $ cat Documents/test.c\~ hello.c

    # 你可以看到除了正确输出了前一个文件的内容,还在末尾出现了一条错误信息

    # 下面我们将输出重定向到一个文件,根据我们前面的经验,这里将在看不到任何输出了

    $ cat Documents/test.c\~ hello.c > somefile

    3.使用tee命令同时重定向到多个文件

    经常你可能还有这样的需求,除了将需要将输出重定向到文件之外也需要将信息打印在终端,那么你可以使用tee命令来实现:

    $ echo 'hello shiyanlou' | tee hello

    4.永久重定向

    你应该可以看出我们前面的重定向操作都只是临时性的,即只对当前命令有效,那如何做到“永久”有效呢,比如在一个脚本中,你需要某一部分的命令的输出全部进行重定向,难道要让你在每个命令上面加上临时重定向的操作嘛,当然不需要,我们可以使用exec命令实现“永久”重定向。exec命令的作用是使用指定的命令替换当前的 Shell,及使用一个进程替换当前进程,或者指定新的重定向:

    # 先开启一个子 Shell

    $ zsh

    # 使用exec替换当前进程的重定向,将标准输出重定向到一个文件

    $ exec 1>somefile

    # 后面你执行的命令的输出都将被重定向到文件中,直到你退出当前子shell,或取消exec的重定向(后面将告诉你怎么做)

    $ ls

    $ exit

    $ cat somefile

     

    5.创建输出文件描述符

    默认在 Shell 中可以有9个打开的文件描述符,上面我们使用了也是它默认提供的0,1,2号文件描述符,另外我们还可以使用3-8的文件描述符,只是它们默认没有打开而已,你可以使用下面命令查看当前 Shell 进程中打开的文件描述符:

    $ cd /dev/fd/;ls -Al

    同样使用exec命令可以创建新的文件描述符:

    $ zsh

    $ exec 3>somefile

    # 先进入目录,再查看,否则你可能不能得到正确的结果,然后再回到上一次的目录

    $ cd /dev/fd/;ls -Al;cd -

    # 注意下面的命令>与&之间不应该有空格,如果有空格则会出错

    $ echo "this is test" >&3

    $ cat somefile

    $ exit

    6.关闭文件描述符

    如上面我们打开的3号文件描述符,可以使用如下操作将它关闭:

    $ exec 3>&-

    $ cd /dev/fd;ls -Al;cd -

    7.完全屏蔽命令的输出

    在 Linux 中有一个被成为“黑洞”的设备文件,所以导入它的数据都将被“吞噬”。

    在类 UNIX 系统中,/dev/null,或称空设备,是一个特殊的设备文件,它通常被用于丢弃不需要的输出流,或作为用于输入流的空文件,这些操作通常由重定向完成。读取它则会立即得到一个EOF。

    我们可以利用设个/dev/null屏蔽命令的输出:

    $ cat Documents/test.c\~ nefile 1>/dev/null 2>&1

    向上面这样的操作将使你得不到任何输出结果。

    8.使用 xargs 分割参数列表

    xargs 是一条 UNIX 和类 UNIX 操作系统的常用命令。它的作用是将参数列表转换成小块分段传递给其他命令,以避免参数列表过长的问题。

    这个命令在有些时候十分有用,特别是当用来处理产生大量输出结果的命令如 find,locate 和 grep 的结果,详细用法请参看 man 文档。

    $ cut -d: -f1 < /etc/passwd | sort | xargs echo

    上面这个命令用于将/etc/passwd文件按:分割取第一个字段排序后,使用echo命令生成一个列表。

     

    实验截图:

     

    实验十一:正则表达式基础

    重要知识点:

    一、正则表达式

    什么是正则表达式呢?

    正则表达式,又称正规表示式、正规表示法、正规表达式、规则表达式、常规表示法(英语:Regular Expression,在代码中常简写为 regex、regexp 或 RE),计算机科学的一个概念。正则表达式使用单个字符串来描述、匹配一系列符合某个句法规则的字符串。在很多文本编辑器里,正则表达式通常被用来检索、替换那些符合某个模式的文本。

    许多程序设计语言都支持利用正则表达式进行字符串操作。例如,在 Perl 中就内建了一个功能强大的正则表达式引擎。正则表达式这个概念最初是由 UNIX 中的工具软件(例如sed和grep)普及开的。正则表达式通常缩写成“regex”,单数有 regexp、regex,复数有 regexps、regexes、regexen。

    简单的说形式和功能上正则表达式和我们前面讲的通配符很像,不过它们之间又有很大差别,特别在于一些特殊的匹配字符的含义上,希望初学者注意不要将两者弄混淆。

     

    1. 举例

     

    假设我们有这样一个文本文件,包含"shiyanlou",和"shilouyan"这两个字符串,同样一个表达式:

    shi*

    如果这作为一个正则表达式,它将只能匹配 shi,而如果不是作为正则表达式*作为一个通配符,则将同时匹配这两个字符串。这是为什么呢?因为在正则表达式中*表示匹配前面的子表达式(这里就是它前面一个字符)零次或多次,比如它可以匹配"sh","shii","shish","shiishi"等等,而作为通配符表示匹配通配符后面任意多个任意字符,所以它可以匹配"shiyanlou",和"shilouyan"两个字符。

    体验完了,下面就来开始正式学习正则表达式吧。

    2.基本语法:

    一个正则表达式通常被称为一个模式(pattern),为用来描述或者匹配一系列符合某个句法规则的字符串。

    选择

    |竖直分隔符表示选择,例如"boy|girl"可以匹配"boy"或者"girl"

    数量限定

    数量限定除了我们举例用的*,还有+加号,?问号,.点号,如果在一个模式中不加数量限定符则表示出现一次且仅出现一次:

    • +表示前面的字符必须出现至少一次(1次或多次),例如,"goo+gle",可以匹配"gooogle","goooogle"等;
    • ?表示前面的字符最多出现一次(0次或1次),例如,"colou?r",可以匹配"color"或者"colour";
    • *星号代表前面的字符可以不出现,也可以出现一次或者多次(0次、或1次、或多次),例如,“0*42”可以匹配42、042、0042、00042等。

    范围和优先级

    ()圆括号可以用来定义模式字符串的范围和优先级,这可以简单的理解为是否将括号内的模式串作为一个整体。例如,"gr(a|e)y"等价于"gray|grey",(这里体现了优先级,竖直分隔符用于选择a或者e而不是gra和ey),"(grand)?father"匹配father和grandfather(这里体验了范围,?将圆括号内容作为一个整体匹配)。

    语法(部分)

    正则表达式有多种不同的风格,下面列举一些常用的作为 PCRE 子集的适用于perl和python编程语言及grep或egrep的正则表达式匹配规则:(由于markdown表格解析的问题,下面的竖直分隔符用全角字符代替,实际使用时请换回半角字符)

    PCRE(Perl Compatible Regular Expressions中文含义:perl语言兼容正则表达式)是一个用 C 语言编写的正则表达式函数库,由菲利普.海泽(Philip Hazel)编写。PCRE是一个轻量级的函数库,比Boost 之类的正则表达式库小得多。PCRE 十分易用,同时功能也很强大,性能超过了 POSIX 正则表达式库和一些经典的正则表达式库。

    字符

    描述

    \

    将下一个字符标记为一个特殊字符、或一个原义字符。例如,“n”匹配字符“n”。“\n”匹配一个换行符。序列“\\”匹配“\”而“\(”则匹配“(”。

    ^

    匹配输入字符串的开始位置。

    $

    匹配输入字符串的结束位置。

    {n}

    n是一个非负整数。匹配确定的n。例如,“o{2}”不能匹配“Bob”中的“o”,但是能匹配“food”中的两个o。

    {n,}

    n是一个非负整数。至少匹配n。例如,“o{2,}”不能匹配“Bob”中的“o”,但能匹配“foooood”中的所有o。“o{1,}”等价于“o+”。“o{0,}”则等价于“o*”。

    {n,m}

    m和n均为非负整数,其中n<=m。最少匹配n次且最多匹配m次。例如,“o{1,3}”将匹配“fooooood”中的前三个o。“o{0,1}”等价于“o?”。请注意在逗号和两个数之间不能有空格。

    *

    匹配前面的子表达式零次或多次。例如,zo*能匹配“z”、“zo”以及“zoo”。*等价于{0,}。

    +

    匹配前面的子表达式一次或多次。例如,“zo+”能匹配“zo”以及“zoo”,但不能匹配“z”。+等价于{1,}。

    ?

    匹配前面的子表达式零次或一次。例如,“do(es)?”可以匹配“do”或“does”中的“do”。?等价于{0,1}。

    ?

    当该字符紧跟在任何一个其他限制符(*,+,?,{n},{n,},{n,m})后面时,匹配模式是非贪婪的。非贪婪模式尽可能少的匹配所搜索的字符串,而默认的贪婪模式则尽可能多的匹配所搜索的字符串。例如,对于字符串“oooo”,“o+?”将匹配单个“o”,而“o+”将匹配所有“o”。

    .

    匹配除“\n”之外的任何单个字符。要匹配包括“\n”在内的任何字符,请使用像“(.|\n)”的模式。

    (pattern)

    匹配pattern并获取这一匹配的子字符串。该子字符串用于向后引用。要匹配圆括号字符,请使用“\(”或“\)”。

    x|y

    匹配xy。例如,“z|food”能匹配“z”或“food”。“(z|f)ood”则匹配“zood”或“food”。

    [xyz]

    字符集合(character class)。匹配所包含的任意一个字符。例如,“[abc]”可以匹配“plain”中的“a”。其中特殊字符仅有反斜线\保持特殊含义,用于转义字符。其它特殊字符如星号、加号、各种括号等均作为普通字符。脱字符^如果出现在首位则表示负值字符集合;如果出现在字符串中间就仅作为普通字符。连字符 - 如果出现在字符串中间表示字符范围描述;如果如果出现在首位则仅作为普通字符。

    [^xyz]

    排除型(negate)字符集合。匹配未列出的任意字符。例如,“[^abc]”可以匹配“plain”中的“plin”。

    [a-z]

    字符范围。匹配指定范围内的任意字符。例如,“[a-z]”可以匹配“a”到“z”范围内的任意小写字母字符。

    [^a-z]

    排除型的字符范围。匹配任何不在指定范围内的任意字符。例如,“[^a-z]”可以匹配任何不在“a”到“z”范围内的任意字符。

    优先级

    优先级为从上到下从左到右,依次降低:

    运算符

    说明

    \

    转义符

    (), (?:), (?=), []

    括号和中括号

    *、+、?、{n}、{n,}、{n,m}

    限定符

    ^、$、\任何元字符

    定位点和序列

     选择

    更多正则表达式的内容可以参考以下链接:

    • 正则表达式wiki
    • 几种正则表达式引擎的语法差异
    • 各语言各平台对正则表达式的支持

    二、grep模式匹配命令

    上面空谈了那么多正则表达式的内容也并没有提及具体该如何使用它,实在枯燥,如果说正则表达式是一门武功话,那它也只能算得上一些口诀招式罢了,要把它真正练起来还得需要一些兵器在手才行,这里我们要介绍的grep命令以及后面要讲的sed,awk这些就该算作是这样的兵器了。

    1.基本操作

    grep命令用于打印输出文本中匹配的模式串,它使用正则表达式作为模式匹配的条件。grep支持三种正则表达式引擎,分别用三个参数指定:

    参数

    说明

    -E

    POSIX扩展正则表达式,ERE

    -G

    POSIX基本正则表达式,BRE

    -P

    Perl正则表达式,PCRE

    不过在你没学过perl语言的大多数情况下你将只会使用到ERE和BRE,所以我们接下来的内容都不会讨论到PCRE中特有的一些正则表达式语法(它们之间大部分内容是存在交集的,所以你不用担心会遗漏多少重要内容)

    在通过grep命令使用正则表达式之前,先介绍一下它的常用参数:

    参数

    说明

    -b

    将二进制文件作为文本来进行匹配

    -c

    统计以模式匹配的数目

    -i

    忽略大小写

    -n

    显示匹配文本所在行的行号

    -v

    反选,输出不匹配行的内容

    -r

    递归匹配查找

    -A n

    n为正整数,表示after的意思,除了列出匹配行之外,还列出后面的n行

    -B n

    n为正整数,表示before的意思,除了列出匹配行之外,还列出前面的n行

    --color=auto

    将输出中的匹配项设置为自动颜色显示

    注:在大多数发行版中是默认设置了grep的颜色的,你可以通过参数指定或修改GREP_COLOR环境变量。

     

    2.使用正则表达式

    使用基本正则表达式,BRE

    • 位置

    查找/etc/group文件中以"shiyanlou"为开头的行

    $ grep 'shiyanlou' /etc/group

    $ grep '^shiyanlou' /etc/group

    • 数量

    # 将匹配以'z'开头以'o'结尾的所有字符串

    $ echo 'zero\nzo\nzoo' | grep 'z.*o'

    # 将匹配以'z'开头以'o'结尾,中间包含一个任意字符的字符串

    $ echo 'zero\nzo\nzoo' | grep 'z.o'

    # 将匹配以'z'开头,以任意多个'o'结尾的字符串

    $ echo 'zero\nzo\nzoo' | grep 'zo*'

    注意:其中\n为换行符

     

    • 选择

    # grep默认是区分大小写的,这里将匹配所有的小写字母

    $ echo '1234\nabcd' | grep '[a-z]'

    # 将匹配所有的数字

    $ echo '1234\nabcd' | grep '[0-9]'

    # 将匹配所有的数字

    $ echo '1234\nabcd' | grep '[[:digit:]]'

    # 将匹配所有的小写字母

    $ echo '1234\nabcd' | grep '[[:lower:]]'

    # 将匹配所有的大写字母

    $ echo '1234\nabcd' | grep '[[:upper:]]'

    # 将匹配所有的字母和数字,包括0-9,a-z,A-Z

    $ echo '1234\nabcd' | grep '[[:alnum:]]'

    # 将匹配所有的字母

    $ echo '1234\nabcd' | grep '[[:alpha:]]'

     

    下面包含完整的特殊符号及说明:

    特殊符号

    说明

    [:alnum:]

    代表英文大小写字节及数字,亦即 0-9, A-Z, a-z

    [:alpha:]

    代表任何英文大小写字节,亦即 A-Z, a-z

    [:blank:]

    代表空白键与 [Tab] 按键两者

    [:cntrl:]

    代表键盘上面的控制按键,亦即包括 CR, LF, Tab, Del.. 等等

    [:digit:]

    代表数字而已,亦即 0-9

    [:graph:]

    除了空白字节 (空白键与 [Tab] 按键) 外的其他所有按键

    [:lower:]

    代表小写字节,亦即 a-z

    [:print:]

    代表任何可以被列印出来的字节

    [:punct:]

    代表标点符号 (punctuation symbol),亦即:" ' ? ! ; : # $...

    [:upper:]

    代表大写字节,亦即 A-Z

    [:space:]

    任何会产生空白的字节,包括空白键, [Tab], CR 等等

    [:xdigit:]

    代表 16 进位的数字类型,因此包括: 0-9, A-F, a-f 的数字与字节

    注意:之所以要使用特殊符号,是因为上面的[a-z]不是在所有情况下都管用,这还与主机当前的语系有关,即设置在LANG环境变量的值,zh_CN.UTF-8的话[a-z],即为所有小写字母,其它语系可能是大小写交替的如,"a A b B...z Z",[a-z]中就可能包含大写字母。所以在使用[a-z]时请确保当前语系的影响,使用[:lower:]则不会有这个问题。

    # 排除字符

    $ echo 'geek|good' | grep '[^o]'

    注意:当^放到中括号内为排除字符,否则表示行首。

    使用扩展正则表达式,ERE

    要通过grep使用扩展正则表达式需要加上-E参数,或使用egrep。

    • 数量

    # 只匹配"zo"

    $ echo 'zero\nzo\nzoo' | grep -E 'zo{1}'

    # 匹配以"zo"开头的所有单词

    $ echo 'zero\nzo\nzoo' | grep -E 'zo{1,}'

    注意:推荐掌握{n,m}即可,+,?,*,这几个不太直观,且容易弄混淆。

    • 选择

    # 匹配"www.shiyanlou.com"和"www.google.com"

    $ echo 'www.shiyanlou.com\nwww.baidu.com\nwww.google.com' | grep -E 'www\.(shiyanlou|google)\.com'

    # 或者匹配不包含"baidu"的内容

    $ echo 'www.shiyanlou.com\nwww.baidu.com\nwww.google.com' | grep -Ev 'www\.baidu\.com'

    注意:因为.号有特殊含义,所以需要转义。

    关于正则表达式和grep命令的内容就介绍这么多,下面会介绍两个更强大的工具sed和awk,但同样也正是因为这两个工具的强大,我们的内容无法包含它们的全部,这里将只对基本内容作介绍。

    三、sed 流编辑器

    sed工具在 man 手册里面的全名为"sed - stream editor for filtering and transforming text ",意即,用于过滤和转换文本的流编辑器。

    在 Linux/UNIX 的世界里敢称为编辑器的工具,大都非等闲之辈,比如前面的"vi/vim(编辑器之神)","emacs(神的编辑器)","gedit"这些个编辑器。sed与上述的最大不同之处大于它是一个非交互式的编辑器,下面我们就开始介绍sed这个编辑器。

    sed常用参数介绍

    sed 命令基本格式:

    sed [参数]... [执行命令] [输入文件]...

    # 形如:

    $ sed -i '1s/sad/happy/' test # 表示将test文件中第一行的"sad"替换为"happy"

    参数

    说明

    -n

    安静模式,只打印受影响的行,默认打印输入数据的全部内容

    -e

    用于在脚本中添加多个执行命令一次执行,在命令行中执行多个命令通常不需要加该参数

    -f filename

    指定执行filename文件中的命令

    -r

    使用扩展正则表达式,默认为标准正则表达式

    -i

    将直接修改输入文件内容,而不是打印到标准输出设备

    sed编辑器的执行命令(这里”执行“解释为名词)

    sed执行命令格式:

    [n1][,n2]command

    [n1][~step]command

    # 其中一些命令可以在后面加上作用范围,形如:

    $ sed -i 's/sad/happy/g' test # g表示全局范围

    $ sed -i 's/sad/happy/4' test # 4表示指定行中的第四个匹配字符串

    其中n1,n2表示输入内容的行号,它们之间为,逗号则表示从n1到n2行,如果为~波浪号则表示从n1开始以step为步进的所有行;command为执行动作,下面为一些常用动作指令:

    命令

    说明

    s

    行内替换

    c

    整行替换

    a

    插入到指定行的后面

    i

    插入到指定行的前面

    p

    打印指定行,通常与-n参数配合使用

    d

    删除指定行

    sed操作举例

    我们先找一个用于练习的文本文件:

    $ cp /etc/passwd ~

    打印指定行

    # 打印2-5行

    $ nl passwd | sed -n '2,5p'

    # 打印奇数行

    $ nl passwd | sed -n '1~2p'

    行内替换

    # 将输入文本中"shiyanlou" 全局替换为"hehe",并只打印替换的那一行,注意这里不能省略最后的"p"命令

    $ sed -n 's/shiyanlou/hehe/gp' passwd

    注意: 行内替换可以结合正则表达式使用。

    行间替换

    $ nl passwd | grep "shiyanlou"

    # 删除第21行

    $ sed -n '21c\www.shiyanlou.com' passwd

     

    四、awk文本处理语言

    看到上面的标题,你可能会感到惊异,难道我们这里要学习的是一门“语言”么,确切的说,我们是要在这里学习awk文本处理语言,只是我们并不会在这里学习到比较完整的关于awk的内容,还是因为前面的原因,它太强大了,它的应用无处不在,我们无法在这里以简短的文字描述面面俱到,如果你有目标成为一个linux系统管理员,确实想学好awk,你一不用担心,实验楼会在之后陆续上线linux系统管理员的学习路径,里面会有单独的关于正则表达式,awk,sed等相关课程,敬请期待吧。下面的内容,我们就作为一个关于awk的入门体验章节吧,其中会介绍一些awk的常用操作。

    1.awk介绍

    AWK是一种优良的文本处理工具,Linux及Unix环境中现有的功能最强大的数据处理引擎之一.其名称得自于它的创始人Alfred Aho(阿尔佛雷德·艾侯)、Peter Jay Weinberger(彼得·温伯格)和Brian Wilson Kernighan(布莱恩·柯林汉)姓氏的首个字母.AWK程序设计语言,三位创建者已将它正式定义为“样式扫描和处理语言”。它允许您创建简短的程序,这些程序读取输入文件、为数据排序、处理数据、对输入执行计算以及生成报表,还有无数其他的功能。最简单地说,AWK是一种用于处理文本的编程语言工具。

    在大多数linux发行版上面,实际我们使用的是gawk(GNU awk,awk的GNU版本),在我们的环境中ubuntu上,默认提供的是mawk,不过我们通常可以直接使用awk命令(awk语言的解释器),因为系统已经为我们创建好了awk指向mawk的符号链接。

    $ ll /usr/bin/awk

    nawk: 在 20 世纪 80 年代中期,对 awk语言进行了更新,并不同程度地使用一种称为 nawk(new awk) 的增强版本对其进行了替换。许多系统中仍然存在着旧的awk 解释器,但通常将其安装为 oawk (old awk) 命令,而 nawk 解释器则安装为主要的 awk 命令,也可以使用 nawk 命令。Dr. Kernighan 仍然在对 nawk 进行维护,与 gawk 一样,它也是开放源代码的,并且可以免费获得; gawk: 是 GNU Project 的awk解释器的开放源代码实现。尽管早期的 GAWK 发行版是旧的 AWK 的替代程序,但不断地对其进行了更新,以包含 NAWK 的特性; mawk 也是awk编程语言的一种解释器,mawk遵循 POSIX 1003.2 (草案 11.3)定义的 AWK 语言,包含了一些没有在AWK 手册中提到的特色,同时 mawk 提供一小部分扩展,另外据说mawk是实现最快的awk

    2.awk的一些基础概念

    awk所有的操作都是基于pattern(模式)—action(动作)对来完成的,如下面的形式:

    $ pattern {action}

    你可以看到就如同很多编程语言一样,它将所有的动作操作用一对{}花括号包围起来。其中pattern通常是是表示用于匹配输入的文本的“关系式”或“正则表达式”,action则是表示匹配后将执行的动作。在一个完整awk操作中,这两者可以只有其中一个,如果没有pattern则默认匹配输入的全部文本,如果没有action则默认为打印匹配内容到屏幕。

    awk处理文本的方式,是将文本分割成一些“字段”,然后再对这些字段进行处理,默认情况下,awk以空格作为一个字段的分割符,不过这不是固定了,你可以任意指定分隔符,下面将告诉你如何做到这一点。

    3.awk命令基本格式

    awk [-F fs] [-v var=value] [-f prog-file | 'program text'] [file...]

    其中-F参数用于预先指定前面提到的字段分隔符(还有其他指定字段的方式) ,-v用于预先为awk程序指定变量,-f参数用于指定awk命令要执行的程序文件,或者在不加-f参数的情况下直接将程序语句放在这里,最后为awk需要处理的文本输入,且可以同时输入多个文本文件。现在我们还是直接来具体体验一下吧。

    4.awk操作体验

    先用vim新建一个文本文档

    $ vim test

    包含如下内容:

    I like linux

    www.shiyanlou.com

    • 使用awk将文本内容打印到终端

    # "quote>" 不用输入

    $ awk '{

    > print

    > }' test

    # 或者写到一行

    $ awk '{print}' test

    • 将test的第一行的每个字段单独显示为一行

    $ awk '{

    > if(NR==1){

    > print $1 "\n" $2 "\n" $3

    > } else {

    > print}

    > }' test

     

    # 或者

    $ awk '{

    > if(NR==1){

    > OFS="\n"

    > print $1, $2, $3

    > } else {

    > print}

    > }' test

    6.awk常用的内置变量

    变量名

    说明

    FILENAME

    当前输入文件名,若有多个文件,则只表示第一个。如果输入是来自标准输入,则为空字符串

    $0

    当前记录的内容

    $N

    N表示字段号,最大值为NF变量的值

    FS

    字段分隔符,由正则表达式表示,默认为" "空格

    RS

    输入记录分隔符,默认为"\n",即一行为一个记录

    NF

    当前记录字段数

    NR

    已经读入的记录数

    FNR

    当前输入文件的记录数,请注意它与NR的区别

    OFS

    输出字段分隔符,默认为" "空格

    ORS

    输出记录分隔符,默认为"\n"

    实验截图:

     

     

     实验十二:Linux 下软件安装

    一、Linux 上的软件安装

    通常 Linux 上的软件安装主要有三种方式:

    • 在线安装
    • 从磁盘安装deb软件包
    • 从二进制软件包安装
    • 从源代码编译安装

    这几种安装方式各有优劣,而大多数软件包会采用多种方式发布软件,所以我们常常需要全部掌握这几种软件安装方式,以便适应各种环境。下面将介绍前三种安装方式,从源码编译安装你将在 Linux 程序设计中学习到。

    二、在线安装

    试想一下,平时我们在使用 Windows 的时候,想要安装一个软件,我们需要在网上去下载对应软件的安装包,接着安装的时候就是不断的去点击下一步,这些流程想必大家已经经历的无数回了,但是在 Linux 下,一个命令加回车,等待一下,软件就安装好了,这就是方便的在线安装软件的方式。在学习这种安装方式之前有一点需要说明的是,在不同的linux发行版上面在线安装方式会有一些差异包括使用的命令及它们的包管理工具,因为我们的开发环境是基于ubuntu的,所以这里我们涉及的在线安装方式将只适用于ubuntu发行版,或其它基于ubuntu的发行版如国内的ubuntukylin(优麒麟)ubuntu又是基于debian的发行版,它使用的是debian的包管理工具dpkg,所以一些操作也适用与debian。而在其它一些采用其它包管理工具的发行版如redhatcentos,fedora等将不适用(redhatcentos使用rpm)

    2. apt 包管理工具介绍

    APT是Advance Packaging Tool(高级包装工具)的缩写,是Debian及其派生发行版的软件包管理器,APT可以自动下载,配置,安装二进制或者源代码格式的软件包,因此简化了Unix系统上管理软件的过程。APT最早被设计成dpkg的前端,用来处理deb格式的软件包。现在经过APT-RPM组织修改,APT已经可以安装在支持RPM的系统管理RPM包。这个包管理器包含以 apt- 开头的的多个工具,如 apt-get apt-cache apt-cdrom 等,在Debian系列的发行版中使用。

    当你在执行安装操作时,首先apt-get 工具会在本地的一个数据库中搜索关于 w3m 软件的相关信息,并根据这些信息在相关的服务器上下载软件安装,这里大家可能会一个疑问:既然是在线安装软件,为啥会在本地的数据库中搜索?要解释这个问题就得提到几个名词了:

    • 软件源镜像服务器
    • 软件源

    我们需要定期从服务器上下载一个软件包列表,使用 sudo apt-get update 命令来保持本地的软件包列表是最新的(有时你也需要手动执行这个操作,比如更换了软件源),而这个表里会有软件依赖信息的记录,对于软件依赖,我举个例子:我们安装 w3m 软件的时候,而这个软件需要libgc1c2 这个软件包才能正常工作,这个时候 apt-get 在安装软件的时候会一并替我们安装了,以保证 w3m 能正常的工作。

    3.apt-get

    apt-get使用各用于处理apt包的公用程序集,我们可以用它来在线安装、卸载和升级软件包等,下面列出一些apt-get包含的常用的一些工具:

    工具

    说明

    install

    其后加上软件包名,用于安装一个软件包

    update

    从软件源镜像服务器上下载/更新用于更新本地软件源的软件包列表

    upgrade

    升级本地可更新的全部软件包,但存在依赖问题时将不会升级,通常会在更新之前执行一次update

    dist-upgrade

    解决依赖关系并升级(存在一定危险性)

    remove

    移除已安装的软件包,包括与被移除软件包有依赖关系的软件包,但不包含软件包的配置文件

    autoremove

    移除之前被其他软件包依赖,但现在不再被使用的软件包

    purge

    与remove相同,但会完全移除软件包,包含其配置文件

    clean

    移除下载到本地的已经安装的软件包,默认保存在/var/cache/apt/archives/

    autoclean

    移除已安装的软件的旧版本软件包

    下面是一些apt-get常用的参数:

    参数

    说明

    -y

    自动回应是否安装软件包的选项,在一些自动化安装脚本中使用这个参数将十分有用

    -s

    模拟安装

    -q

    静默安装方式,指定多个q或者-q=#,#表示数字,用于设定静默级别,这在你不想要在安装软件包时屏幕输出过多时很有用

    -f

    修复损坏的依赖关系

    -d

    只下载不安装

    --reinstall

    重新安装已经安装但可能存在问题的软件包

    --install-suggests

    同时安装APT给出的建议安装的软件包

    4.安装软件包

    关于安装,如前面演示的一样你只需要执行apt-get install <软件包名>即可,除了这一点,你还应该掌握的是如何重新安装软件包。 很多时候我们需要重新安装一个软件包,比如你的系统被破坏,或者一些错误的配置导致软件无法正常工作。

    你可以使用如下方式重新安装:

    $ sudo apt-get --reinstall install w3m

    另一个你需要掌握的是,如何在不知道软件包完整名的时候进行安装。通常我们是使用Tab键补全软件包名,后面会介绍更好的方法来搜索软件包。有时候你需要同时安装多个软件包,你还可以使用正则表达式匹配软件包名进行批量安装。

    5.软件升级

    # 更新软件源

    $ sudo apt-get update

    # 升级没有依赖问题的软件包

    $ sudo apt-get upgrade

    # 升级并解决依赖关系

    $ sudo apt-get dist-upgrade

    6.卸载软件

    如果你现在觉得 w3m 这个软件不合自己的胃口,或者是找到了更好的,你需要卸载它,那么简单!同样是一个命令加回车 sudo apt-get remove w3m ,系统会有一个确认的操作,之后这个软件便“滚蛋了”。

    或者,你可以执行

    # 不保留配置文件的移除

    $ sudo apt-get purge w3m

    # 或者 sudo apt-get --purge remove

    # 移除不再需要的被依赖的软件包

    $ sudo apt-get autoremove

    7.软件搜索

    当自己刚知道了一个软件,想下载使用,需要确认软件仓库里面有没有,就需要用到搜索功能了,命令如下:

    sudo apt-cache search softname1 softname2 softname3……

    三、使用 dpkg 从本地磁盘安装 deb 软件包

    1.dpkg 介绍

    dpkg 是 Debian 软件包管理器的基础,它被伊恩·默多克创建于 1993 年。dpkg 与 RPM 十分相似,同样被用于安装、卸载和供给和 .deb 软件包相关的信息。

    dpkg 本身是一个底层的工具。上层的工具,像是 APT,被用于从远程获取软件包以及处理复杂的软件包关系。"dpkg"是"Debian Package"的简写。

    我们经常可以在网络上简单以deb形式打包的软件包,就需要使用dpkg命令来安装。

    dpkg常用参数介绍:

    参数

    说明

    -i

    安装指定deb包

    -R

    后面加上目录名,用于安装该目录下的所有deb安装包

    -r

    remove,移除某个已安装的软件包

    -I

    显示deb包文件的信息

    -s

    显示已安装软件的信息

    -S

    搜索已安装的软件包

    -L

    显示已安装软件包的目录信息

    2.使用dpkg安装deb软件包

    我们先使用apt-get加上-d参数只下载不安装,下载emacs编辑器的deb包,下载完成后,我们可以查看/var/cache/apt/archives/目录下的内容,如下图:

     

    然后我们将第一个deb拷贝到home目录下,并使用dpkg安装

    $ cp /var/cache/apt/archives/emacs24_24.3+1-4ubuntu1_amd64.deb ~

    # 安装之前参看deb包的信息

    $ sudo dpkg -I emacs24_24.3+1-4ubuntu1_amd64.deb

    如你所见,这个包还额外依赖了一些软件包,这意味着,如果主机目前没有这些被依赖的软件包,直接使用dpkg安装可能会存在一些问题,因为dpkg并不能为你解决依赖关系。

    # 使用dpkg安装

    $ sudo dpkg -i emacs24_24.3+1-4ubuntu1_amd64.deb

    跟前面预料的一样,这里你可能出现了一些错误:

     

    我们将如何解决这个错误了,这就要用到apt-get了,使用它的-f参数了,修复依赖关系的安装

    $ sudo apt-get -f install

    没有任何错误,这样我们就安装成功了,然后你可以运行emacs程序

    3.查看已安装软件包的安装目录

    如果你依然在纠结到底linux将软件安装到了什么地方,那么很幸运你将可以通过dpkg找到答案

    使用dpkg -L查看deb包目录信息

    $ sudo dpkg -L emacs

     

    dpkg还有一些其他的参数,这里将作为练习题由你自己来学习

    四、从二进制包安装

    二进制包的安装比较简单,我们需要做的只是将从网络上下载的二进制包解压后放到合适的目录,然后将包含可执行的主程序文件的目录添加进PATH环境变量即可,如果你不知道该放到什么位置,请重新复习第四节关于 Linux 目录结构的内容。

     

    实验总结

    因为之前为准备计算机等级考试,因此做实验的时间比较仓促,因此仅仅只是大致了解了linux的基本操作和把实验代码都运行了一遍,之间出现了一些问题已经做了记录,之后将会抽空将问题解决。

    本次实验重点学习命令

     find

    功能说明:查找文件或目录。

    语  法:find [目录...][-amin <分钟>][-anewer <参考文件或目录>][-atime <24小时数>][-cmin <分钟>][-cnewer <参考文件或目录>][-ctime <24小时数>][-daystart][-depyh][-empty][-exec <执行指令>][-false][-fls <列表文件>][-follow][-fprint <列表文件>][-fprint0 <列表文件>][-fprintf <列表文件><输出格式>][-fstype <文件系统类型>][-gid <群组识别码>][-group <群组名称>][-help][-ilname <范本样式>][-iname <范本样式>][-inum <inode编号>][-ipath <范本样式>][-iregex <范本样式>][-links <连接数目>][-lname <范本样式>][-ls][-maxdepth <目录层级>][-mindepth <目录层级>][-mmin <分钟>][-mount]
    [-mtime <24小时数>][-name <范本样式>][-newer <参考文件或目录>][-nogroup][noleaf] [-nouser][-ok <执行指令>][-path <范本样式>][-perm <权限数值>][-print][-print0][-printf <输出格式>][-prune][-regex <范本样式>][-size <文件大小>][-true][-type <文件类型>][-uid <用户识别码>][-used <日数>][-user <拥有者名称>][-version][-xdev][-xtype <文件类型>]

    补充说明:find指令用于查找符合条件的文件。任何位于参数之前的字符串都将被视为欲查找的目录。

    参  数:
     -amin<分钟>  查找在指定时间曾被存取过的文件或目录,单位以分钟计算。
     -anewer<参考文件或目录>  查找其存取时间较指定文件或目录的存取时间更接近现在的文件或目录。
     -atime<24小时数>  查找在指定时间曾被存取过的文件或目录,单位以24小时计算。
     -cmin<分钟>  查找在指定时间之时被更改的文件或目录。
     -cnewer<参考文件或目录>  查找其更改时间较指定文件或目录的更改时间更接近现在的文件或目录。
     -ctime<24小时数>  查找在指定时间之时被更改的文件或目录,单位以24小时计算。
     -daystart  从本日开始计算时间。
     -depth  从指定目录下最深层的子目录开始查找。
     -expty  寻找文件大小为0 Byte的文件,或目录下没有任何子目录或文件的空目录。
     -exec<执行指令>  假设find指令的回传值为True,就执行该指令。
     -false  将find指令的回传值皆设为False。
     -fls<列表文件>  此参数的效果和指定"-ls"参数类似,但会把结果保存为指定的列表文件。
     -follow  排除符号连接。
     -fprint<列表文件>  此参数的效果和指定"-print"参数类似,但会把结果保存成指定的列表文件。
     -fprint0<列表文件>  此参数的效果和指定"-print0"参数类似,但会把结果保存成指定的列表文件。
     -fprintf<列表文件><输出格式>  此参数的效果和指定"-printf"参数类似,但会把结果保存成指定的列表文件。
     -fstype<文件系统类型>  只寻找该文件系统类型下的文件或目录。
     -gid<群组识别码>  查找符合指定之群组识别码的文件或目录。
     -group<群组名称>  查找符合指定之群组名称的文件或目录。
     -help或--help  在线帮助。
     -ilname<范本样式>  此参数的效果和指定"-lname"参数类似,但忽略字符大小写的差别。
     -iname<范本样式>  此参数的效果和指定"-name"参数类似,但忽略字符大小写的差别。
     -inum<inode编号>  查找符合指定的inode编号的文件或目录。
     -ipath<范本样式>  此参数的效果和指定"-ipath"参数类似,但忽略字符大小写的差别。
     -iregex<范本样式>  此参数的效果和指定"-regexe"参数类似,但忽略字符大小写的差别。
     -links<连接数目>  查找符合指定的硬连接数目的文件或目录。
     -iname<范本样式>  指定字符串作为寻找符号连接的范本样式。
     -ls  假设find指令的回传值为True,就将文件或目录名称列出到标准输出。
     -maxdepth<目录层级>  设置最大目录层级。
     -mindepth<目录层级>  设置最小目录层级。
     -mmin<分钟>  查找在指定时间曾被更改过的文件或目录,单位以分钟计算。
     -mount  此参数的效果和指定"-xdev"相同。
     -mtime<24小时数>  查找在指定时间曾被更改过的文件或目录,单位以24小时计算。
     -name<范本样式>  指定字符串作为寻找文件或目录的范本样式。
     -newer<参考文件或目录>  查找其更改时间较指定文件或目录的更改时间更接近现在的文件或目录。
     -nogroup  找出不属于本地主机群组识别码的文件或目录。
     -noleaf  不去考虑目录至少需拥有两个硬连接存在。
     -nouser  找出不属于本地主机用户识别码的文件或目录。
     -ok<执行指令>  此参数的效果和指定"-exec"参数类似,但在执行指令之前会先询问用户,若回答"y"或"Y",则放弃执行指令。
     -path<范本样式>  指定字符串作为寻找目录的范本样式。
     -perm<权限数值>  查找符合指定的权限数值的文件或目录。
     -print  假设find指令的回传值为True,就将文件或目录名称列出到标准输出。格式为每列一个名称,每个名称之前皆有"./"字符串。
     -print0  假设find指令的回传值为True,就将文件或目录名称列出到标准输出。格式为全部的名称皆在同一行。
     -printf<输出格式>  假设find指令的回传值为True,就将文件或目录名称列出到标准输出。格式可以自行指定。
     -prune  不寻找字符串作为寻找文件或目录的范本样式。
     -regex<范本样式>  指定字符串作为寻找文件或目录的范本样式。
     -size<文件大小>  查找符合指定的文件大小的文件。
     -true  将find指令的回传值皆设为True。
     -typ<文件类型>  只寻找符合指定的文件类型的文件。
     -uid<用户识别码>  查找符合指定的用户识别码的文件或目录。
     -used<日数>  查找文件或目录被更改之后在指定时间曾被存取过的文件或目录,单位以日计算。
     -user<拥有者名称>  查找符合指定的拥有者名称的文件或目录。
     -version或--version  显示版本信息。
     -xdev  将范围局限在先行的文件系统中。
     -xtype<文件类型>  此参数的效果和指定"-type"参数类似,差别在于它针对符号连接检查。

    locate

    功能说明:查找文件。

    语  法:locate [-d <数据库文件>][--help][--version][范本样式...]

    补充说明:locate指令用于查找符合条件的文件,它会去保存文件与目录名称的数据库内,查找合乎范本样式条件的文件或目录。

    参  数:
     -d<数据库文件>或--database=<数据库文件> 设置locate指令使用的数据库。locate指令预设的数据库位于/var/lib/slocate目录里,文件名为slocate.db,您可使用这个参数另行指定。
     --help  在线帮助。
     --version  显示版本信息。

     grep

    功能说明:查找文件里符合条件的字符串。

    语  法:grep [-abcEFGhHilLnqrsvVwxy][-A<显示列数>][-B<显示列数>][-C<显示列数>][-d<进行动作>][-e<范本样式>][-f<范本文件>][--help][范本样式][文件或目录...]

    补充说明:grep指令用于查找内容包含指定的范本样式的文件,如果发现某文件的内容符合所指定的范本样式,预设grep指令会把含有范本样式的那一列显示出来。若不指定任何文件名称,或是所给予的文件名为“-”,则grep指令会从标准输入设备读取数据。

    参  数:
      -a或--text   不要忽略二进制的数据。
      -A<显示列数>或--after-context=<显示列数>   除了显示符合范本样式的那一列之外,并显示该列之后的内容。
      -b或--byte-offset   在显示符合范本样式的那一列之前,标示出该列第一个字符的位编号。
      -B<显示列数>或--before-context=<显示列数>   除了显示符合范本样式的那一列之外,并显示该列之前的内容。
      -c或--count   计算符合范本样式的列数。
      -C<显示列数>或--context=<显示列数>或-<显示列数>   除了显示符合范本样式的那一列之外,并显示该列之前后的内容。
      -d<进行动作>或--directories=<进行动作>   当指定要查找的是目录而非文件时,必须使用这项参数,否则grep指令将回报信息并停止动作。
      -e<范本样式>或--regexp=<范本样式>   指定字符串做为查找文件内容的范本样式。
      -E或--extended-regexp   将范本样式为延伸的普通表示法来使用。
      -f<范本文件>或--file=<范本文件>   指定范本文件,其内容含有一个或多个范本样式,让grep查找符合范本条件的文件内容,格式为每列一个范本样式。
      -F或--fixed-regexp   将范本样式视为固定字符串的列表。
      -G或--basic-regexp   将范本样式视为普通的表示法来使用。
      -h或--no-filename   在显示符合范本样式的那一列之前,不标示该列所属的文件名称。
      -H或--with-filename   在显示符合范本样式的那一列之前,表示该列所属的文件名称。
      -i或--ignore-case   忽略字符大小写的差别。
      -l或--file-with-matches   列出文件内容符合指定的范本样式的文件名称。
      -L或--files-without-match   列出文件内容不符合指定的范本样式的文件名称。
      -n或--line-number   在显示符合范本样式的那一列之前,标示出该列的列数编号。
      -q或--quiet或--silent   不显示任何信息。
      -r或--recursive   此参数的效果和指定“-d recurse”参数相同。
      -s或--no-messages   不显示错误信息。
      -v或--revert-match   反转查找。
      -V或--version   显示版本信息。
      -w或--word-regexp   只显示全字符合的列。
      -x或--line-regexp   只显示全列符合的列。
      -y   此参数的效果和指定“-i”参数相同。
      --help   在线帮助。
    man

    在 Linux 环境中,如果你遇到困难,可以使用man 命令,它是Manual page的缩写。

    whereis

    功能说明:查找文件。

    语  法:whereis [-bfmsu][-B <目录>...][-M <目录>...][-S <目录>...][文件...]

    补充说明:whereis指令会在特定目录中查找符合条件的文件。这些文件的烈性应属于原始代码,二进制文件,或是帮助文件。

    参  数:
     -b  只查找二进制文件。
     -B<目录>  只在设置的目录下查找二进制文件。
     -f  不显示文件名前的路径名称。
     -m  只查找说明文件。
     -M<目录>  只在设置的目录下查找说明文件。
     -s  只查找原始代码文件。
     -S<目录>  只在设置的目录下查找原始代码文件。
     -u  查找不包含指定类型的文件。

    which

    功能说明:查找文件。

    语  法:which [文件...]

    补充说明:which指令会在环境变量$PATH设置的目录里查找符合条件的文件。

    参  数:
     -n<文件名长度>  指定文件名长度,指定的长度必须大于或等于所有文件中最长的文件名。
     -p<文件名长度>  与-n参数相同,但此处的<文件名长度>包括了文件的路径。
     -w  指定输出时栏位的宽度。
     -V  显示版本信息。

    apt-get

    在线获取软件,并安装。

  • 相关阅读:
    机器学习(01)——机器学习简介
    Harbor本地镜像库安装与使用
    大数据高可用集群环境安装与配置(10)——安装Kafka高可用集群
    大数据高可用集群环境安装与配置(09)——安装Spark高可用集群
    大数据高可用集群环境安装与配置(08)——安装Ganglia监控集群
    大数据高可用集群环境安装与配置(07)——安装HBase高可用集群
    大数据高可用集群环境安装与配置(06)——安装Hadoop高可用集群
    大数据高可用集群环境安装与配置(05)——安装zookeeper集群
    大数据高可用集群环境安装与配置(04)——安装JAVA运行环境
    大数据高可用集群环境安装与配置(03)——设置SSH免密登录
  • 原文地址:https://www.cnblogs.com/Ntime/p/4823436.html
Copyright © 2020-2023  润新知