Chinaunix首页 | 论坛 | 博客
  • 博客访问: 214163
  • 博文数量: 87
  • 博客积分: 192
  • 博客等级: 入伍新兵
  • 技术积分: 455
  • 用 户 组: 普通用户
  • 注册时间: 2011-04-14 07:44
文章分类

全部博文(87)

文章存档

2013年(1)

2012年(86)

分类:

2012-08-09 13:31:37

原文地址:详细介绍Linux重定向 作者:zooyo

  Linux重定向是指修改原来默认的一些东西,对原来系统命令的默认执行方式进行改变,比如说简单的我不想看到在显示器的输出而是希望输出到某一文件中就可以通过Linux重定向来进行这项工作。

  Linux默认输入是键盘,输出是显示器。你可以用重定向来改变这些设置。比如用wc命令的时候本来是要手动输入一篇文字来计算字符数的,用了重定向后可以直接把一个已经写好的文件用‘<’指向这条命令,就直接可以统计这个文件的字符数等了。输出也是一样,你可以把屏幕输出重定向到一个文件里,再到文件里去看结果。重定向操作符可以用来将命令输入和输出数据流从默认位置重定向到其他位置,其输入或输出数据流的位置称为句柄;常见的句柄有三种,当然句柄可以自行扩展,一般的OS都提供类似的功能。句柄 句柄代号 句柄描述

  STDIN 0 键盘输入

  STDOUT 1 输出信息到提示符窗口

  STDERR 2 输出错误信息到提示符窗口

  默认的 < 重定向输入操作符是 0,而默认的 > 重定向输出操作符是 1。键入 < 或 > 操作符之后,必须指定数据的读写位置,可以是文件名或其他现有的句柄。

  要指定重定向到现有句柄,请使用与 & 字符,后面接要重定向的句柄号(即 &句柄号)。

  例如,下面的命令可以将句柄 2(即 STDERR)重定向到句柄 1(即 STDOUT):2>&1

  下表列出了可用于重定向输入和输出数据流的操作符:

  Linux重定向操作符 功能描述

  > 将命令输出写入文件或设备,而不是命令提示符或句柄

  < 从文件而不是从键盘或句柄读入命令输入

  >> 将命令输出添加到文件末尾而不删除文件中已有的信息

  >& 将一个句柄的输出写入到另一个句柄的输入中

  <& 从一个句柄读取输入并将其写入到另一个句柄输出中

  | 从一个命令中读取输出并将其写入另一个命令的输入中;也称为管道操作符

  现在我们回过头来看看上面的那条语句mysh > mylog.txt 2>&1就可明白:

  > mylog.txt意思是将标准输出重定向到mylog.txt,等价于mysh 1> mylog.txt;

  2 >& 1 意思是将错误输出重定向到句柄1标准输出;综合起来就是mysh命令执行过程中产生的标准输出和错误输出都会被重定向到mylog.txt中;

  重定向的功能十分强大,有兴趣的可以去尝试各种不同的组合,看看前后位置变下会有什么结果?

  某些时候我们可能并不希望记录什么标准输出或者是错误输出,那可以用mysh >null 2>null或者mysh >/dev/null 2>/dev/null;

  I/O重定向详解

  1、 基本概念(这是理解后面的知识的前提,请务必理解)

  a、 I/O重定向通常与 FD有关,shell的FD通常为10个,即 0~9;

  b、 常用FD有3个,为0(stdin,标准输入)、1(stdout,标准输出)、2(stderr,标准错误输出),默认与keyboard、monitor、monitor有关;

  c、 用 < 来改变读进的数据信道(stdin),使之从指定的档案读进;

  d、 用 > 来改变送出的数据信道(stdout, stderr),使之输出到指定的档案;

  e、 0 是 < 的默认值,因此 < 与 0<是一样的;同理,> 与 1> 是一样的;

  f、 在IO重定向 中,stdout 与 stderr 的管道会先准备好,才会从 stdin 读进资料;

  g、 管道“|”(pipe line):上一个命令的 stdout 接到下一个命令的 stdin;

  h、 tee 命令是在不影响原本 I/O 的情况下,将 stdout 复制一份到档案去;

  i、 bash(ksh)执行命令的过程:分析命令-变量求值-命令替代(``和$( ))-重定向-通配符展开-确定路径-执行命令;

  j、 ( ) 将 command group 置于 sub-shell 去执行,也称 nested sub-shell,它有一点非常重要的特性是:继承父shell的Standard input, output, and error plus any other open file descriptors。

  k、 exec 命令:常用来替代当前 shell 并重新启动一个 shell,换句话说,并没有启动子 shell。使用这一命令时任何现有环境都将会被清除。exec 在对文件描述符进行操作的时候,也只有在这时,exec 不会覆盖你当前的 shell 环境。

  2、 基本IO

  cmd > file 把 stdout 重定向到 file 文件中;

  cmd >> file 把 stdout 重定向到 file 文件中(追加);

  cmd 1> fiel 把 stdout 重定向到 file 文件中;

  cmd > file 2>&1 把 stdout 和 stderr 一起重定向到 file 文件中( &> 和 >& 功能一样);

  cmd 2> file 把 stderr 重定向到 file 文件中;

  cmd 2>> file 把 stderr 重定向到 file 文件中(追加);

  cmd >> file 2>&1 把 stderr 和 stderr 一起重定向到 file 文件中(追加) ( &>> 和 >>& 功能一样)

  cmd < file >file2 cmd 命令以 file 文件作为 stdin,以 file2 文件作为 stdout;

  cat <>file 以读写的方式打开 file;

  cmd < file cmd 命令以 file 文件作为 stdin;

  cmd << delimiter 从 stdin 中读入,直至遇到 delimiter 分界符。

  3、 进阶IO

  >&n 使用系统调用 dup (2) 复制文件描述符 n 并把结果用作标准输出;

  <&n 标准输入复制自文件描述符 n;

  <&- 关闭标准输入(键盘);

  >&- 关闭标准输出;

  n<&- 表示将 n 号输入关闭;

  n>&- 表示将 n 号输出关闭;

  上述所有形式都可以前导一个数字,此时建立的文件描述符由这个数字指定而不是缺省的 0 或 1。如:

  ... 2>file 运行一个命令并把错误输出(文件描述符 2)定向到 file。

  ... 2>&1 运行一个命令并把它的标准输出和输出合并。(严格的说是通过复制文件描述符 1 来建立文件描述符 2 ,但效果通常是合并了两个流。)

  我们对 2>&1详细说明一下 :2>&1 也就是 FD2=FD1 ,这里并不是说FD2 的值 等于FD1的值,因为 > 是改变送出的数据信道,也就是说把 FD2 的 “数据输出通道” 改为 FD1 的 “数据输出通道”。如果仅仅这样,这个改变好像没有什么作用,因为 FD2 的默认输出和 FD1的默认输出本来都是 monitor,一样的!但是,当 FD1 是其他文件,甚至是其他 FD 时,这个就具有特殊的用途了。请大家务必理解这一点。

      exec 1>outfilename   # 打开文件outfilename作为stdout。

  exec 2>errfilename   # 打开文件 errfilename作为 stderr。

  exec 0<&-               # 关闭 FD0。

  exec 1>&-               # 关闭 FD1。

  exec 5>&-               # 关闭 FD5。

高级运用:

  1. exec 3<>test.sh;
  2. #打开test.sh可读写操作,与文件描述符3绑定
  3.  
  4. while read line<&3
  5.  do
  6.     echo $line;
  7. done
  8. #循环读取文件描述符3(读取的是test.sh内容)

  9. exec 3>&-
  10. exec 3<&-
  11. #关闭文件的,输入,输出绑定

  1. $ exec 6>&1
  2. #将标准输出与fd 6绑定

  3. $ ls /proc/self/fd/
  4. 0 1 2 3 6
  5. #出现文件描述符6

  6. $ exec 1>suc.txt
  7. #将接下来所有命令标准输出,绑定到suc.txt文件(输出到该文件)

  8. $ ls -al
  9. #执行命令,发现什么都不返回了,因为标准输出已经输出到suc.txt文件了

  10. $ exec 1>&6
  11. #恢复标准输出

  12. $ exec 6>&-
  13. #关闭fd 6描述符

  14. $ ls /proc/self/fd/
  15. 0 1 2 3

  1. $ exec >outfile
  2. #把标准输出重定向到outfile文件

  3. $ echo "hello"
  4. #打印hello,屏幕并没有输出

  5. $ exec >/dev/tty
  6. #再把标准输出定向到终端

  7. $ echo "hello"
  8. hello
  9. #再打印时屏幕恢复了输出

  10. $ cat outfile
  11. hello
  12. #先前打印的字符串也被定向到了outfile文件里

  1. $ exec 3>&1 1>outfile
  2. #把FD3输出定向到标准输出,标准输出定向到文件outfile

  3. $ echo "hello"
  4. #打印hello屏幕并没有输出,因为标准输出定向去了文件outfile

  5. $ exec 1>&3
  6. #把标准输出定向去FD3,因为开头FD3是定向的标准输出,这样的作用就是恢复了标准输出到屏幕.

  7. $ echo hello
  8. hello
  9. #再打印hello时,屏幕便有了输出.

  10. $ cat outfile
  11. hello
  12. #再查看outfile文件,第一次打印定向到了这里.

仔细体会上面的四个运用例子,想必很快就掌握了重定向和文件描述的用法了。

阅读(1218) | 评论(0) | 转发(0) |
0

上一篇:ASCII码表

下一篇:sed的模式匹配

给主人留下些什么吧!~~