so true

          心懷未來,開創未來!
          隨筆 - 160, 文章 - 0, 評論 - 40, 引用 - 0
          數據加載中……

          bash redirect re-study

          本質:先理解linux里的int dup2(int oldfd, int newfd)命令,dup2() makes newfd be the copy of oldfd。
          理解:重定向這個概念其實不是很好理解,如果用編程里面的指針來說明,那就很容易了。舉個例子,比如指針p1指向瓶子,指針p2指向抽屜,現在執行p1>&p2(相當于p1 = p2),這樣p1這個指針也指向抽屜了,也就是說我們把p1這個指針重定向到一個新的地方,這個地方是p2所指的地方,即抽屜;
              簡而言之,這個例子里重定向的含義是:把p1指向p2所指的地方(重定向就是指向的含義),而瓶子、抽屜這2個“實體”是不變的,重定向能調整的只是指針的指向。
              $ ls -l /proc/$$/fd
              lr-x------ 1 admin admin 64 2016-03-30 10:14 0 -> /dev/pts/413
              lrwx------ 1 admin admin 64 2016-03-30 10:14 1 -> /dev/pts/413
              lrwx------ 1 admin admin 64 2016-03-30 10:14 2 -> /dev/pts/413
              lrwx------ 1 admin admin 64 2016-03-30 10:14 255 -> /dev/pts/413
              我們說0代表stdin,1代表stdout,2代表stderr,這里stdin/stdout/stderr是實體(這里都是/dev/pts/413,即terminal這個設備),0/1/2只是一個fd,可以理解為指針或者標記;
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) 2>&1 >file1之所以file1里只會有1行,原因是:
              我們首先把2指向了stdout(stdout正是1所指的東西),這個時候1和2都指向了stdout;
              然后我們把1指向了file1,這個時候2依然還是指著stdout;
              然后I'm from stdout這句話寫到了1里面,其實也就是寫到了file1里面,I'm from stderr這句話寫到了2里面,其實也就是寫到了stdout里面。
          整體規則:
              1. TO (<|>|<>) FROM;
              2. (<|>)這兩個東西后面可以跟&fd/&-/&fd-,也可以跟一個file名字;例外>&file1表示stdout和stderr都重定向到file1里面;<>后面只可以跟file1,例如<>file1表示從file1里既讀入數據,也寫入數據;
              3. (<|>|<>)這三個東西前面不能有空格,后面跟的&不能有空格,除此之外無要求,例如:exec 1>& 2- #但2和-之間不能有空格;
          操作:
              1. 打開一個fd的方法:exec fd>&1  或者   exec fd<&0 或者 exec fd>file1 或者 exec fd<file1 或者 exec fd <>file1;
                 如果你不想指定fd(希望系統自動給你分配一個fd),那么有一種特殊的用法,例子是:
                   exec {NEW_STDOUT}>&1 #注意這個用法很特殊,你不能NEW_STDOUT=15; exec ${NEW_STDOUT}>&1
                   echo "hello" >&$NEW_STDOUT # echo "hello" >&${NEW_STDOUT}也可以
                   echo ${NEW_STDOUT} #從10開始分配
                   exec {NEW_STDOUT}>&-
              2. 關閉一個fd的方法:exec fd>&- 或者 exec fd<&-
              3. move fd的方法:new_fd>&old_fd-,move完之后,old_fd會被關掉;
          實戰:
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) 2>&1 >file1 #file1里將會有1行I'm from stdout,原因是stdout賦給了stderr,file1賦給了stdout,2句話里,打印到stdout的那句被寫入file1,打印到stderr被寫入到了stdout
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) >file1 2>&1 #file1里將會有2行,原因是file1賦給了stdout,stdout賦給了stderr(其實也就是file1賦給了stderr),這個時候,stdout和stderr都指向了file1
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) &> file1 #file1里將會有2行,file1前面的空格可以省略
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) >& file1 #file1里將會有2行,file1前面的空格可以省略
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) |& cat > file1 #file1里將會有2行,file1前面的空格可以省略
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) 2>&1 | cat > file1 #file1里將會有2行,file1前面的空格可以省略
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) >file1 | grep err #grep fail
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) 2>&1 >file1 | grep err #grep ok
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) 3>&1 1>&2 2>&3-| grep stderr #grep ok,這其實實現了stdout和stderr的交換,類似于通過c=a;a=b;b=c;來實現a和b的交換(先把a賦給c,緊接著找到b賦給a,然后用c再賦給b)
              (echo "I'm from stdout"; echo "I'm from stderr" 1>&2) 1>&2- | grep stderr #grep ok
              echo "hello world" | tee >(grep hello) >(grep world) >/dev/null #結合上tee的多路dispatch,功能會更強大
              > file1 #create file1 or truncate file1 to zero size
              <file1 cat #<file1這部分放到命令的前面或者后面都可以
              2>&1 ls not_found_file | grep found #grep ok
              cat < file1 > file2 #等同于cp file1 file2
          <>高級用法:
              1. 寫到一個文件中指定的地方:
                  echo 1234567890 > file1   # 寫字符串到"File".
                  exec 3<> file1            # 打開"File"并且給它分配fd 3.
                  read -n 4 <&3             # 只讀4個字符.
                  echo -n . >&3             # 寫一個小數點.
                  exec 3>&-                 # 關閉fd 3.
                  cat file1                 # ==> 1234.67890
              2. 用bash下載網頁
                  $ exec 3<>/dev/tcp/www.baidu.com/80
                  $ echo -e "GET / HTTP/1.0\r\n\r\n" >&3
                  $ cat <&3
                  $ ls -l /proc/$$/fd
                  total 0
                  lr-x------ 1 admin admin 64 2016-03-30 10:14 0 -> /dev/pts/413
                  lrwx------ 1 admin admin 64 2016-03-30 10:14 1 -> /dev/pts/413
                  lrwx------ 1 admin admin 64 2016-03-30 10:14 2 -> /dev/pts/413
                  lrwx------ 1 admin admin 64 2016-03-30 10:14 255 -> /dev/pts/413
                  lrwx------ 1 admin admin 64 2016-03-30 10:14 3 -> socket:[173770096]
                  $ exec 3>&- # exec 3<&-也可以關閉

          posted on 2016-03-30 12:10 so true 閱讀(258) 評論(0)  編輯  收藏 所屬分類: Linux

          主站蜘蛛池模板: 南丹县| 习水县| 中方县| 定兴县| 赤峰市| 云安县| 博湖县| 阜宁县| 新龙县| 东乡族自治县| 富川| 海淀区| 保康县| 揭西县| 通海县| 拉孜县| 嘉峪关市| 麦盖提县| 汉沽区| 新平| 西昌市| 陆丰市| 祁门县| 泰来县| 任丘市| 望都县| 重庆市| 永川市| 辽中县| 夏河县| 五莲县| 鄂托克前旗| 河北区| 溆浦县| 凤山市| 浏阳市| 马公市| 分宜县| 汉川市| 武隆县| 当雄县|