我對alias的重新認識:通過alias讓rm更安全


bash&shell系列文章:http://www.cnblogs.com/f-ck-need-u/p/7048359.html


rm的悲劇總是發生在不經意之間,所以無論是在shell腳本中還是交互式bash環境下,執行rm命令時總應該三思三思再三思。也因此,很多人想盡辦法防止文件誤刪除,方法也各種各樣。

1.1 alias一般用法

1.默認rm是"rm -i"的別名,ll就是"ls -l"的別名。可以自定義別名來代替某些命令配合某些選項,也可以定義別名組合多個命令。例如:

[root@xuexi ~]# alias ls='ls -lA'

這樣在列出目錄時將同時列出隱藏文件。

2.使用不帶參數的alias將列出當前shell環境下所有的已定義的別名。

3.另外需要說明的是,當別名和命令同名時,將優先執行別名(否則別名就沒有意義了),這可以從which的結果中看出:

[root@xuexi ~]# which mv
alias mv='mv -i'
        /bin/mv

如果定義的命名名稱和原始命令同名(例如定義的別名 ls='ls -l' ),此時如果想要明確使用原始命令,可以刪除別名或者使用絕對路徑或者使用轉義符來還原命令。

4.alias命令是臨時定義別名,要定義長久生效的別名就將別名定義語句寫入/etc/profile或~/.bash_profile或~/.bashrc,第一個對所有用戶有效,后面兩個對對應用戶有效。修改后記得使用source來重新調取這些配置文件。

5.使用unalias可以臨時取消別名。

1.2 alias的缺陷

別名這東西定義和使用起來有點模糊,以下面這個別名命令為例,在有的shell腳本的書籍上使用了這樣的定義,但卻是錯誤的,原因稍后說明。

alias rmm='cp $@ ~/backup;rm $@'

該別名的目的是刪除文件時先備份到一個目錄下,然后再刪除。按照man bash里的說明,別名rmm只是第一個cp命令的別名,分號后的rm不是別名的一部分,而是緊跟在別名后的下一行命令。當執行別名rmm時,首先讀取別名到分號位置處,然后進行別名擴展,執行完別名命令后,再執行分號后的rm命令。

之所以說上面的命令是錯誤的命令,問題出在cp的參數"$@",該變量本表示提供的所有參數,但由於cp命令后使用分號分隔並定義了另一個命令,這使得執行別名命令時,參數無法傳遞到cp命令上,而只能傳遞到最后一個命令rm上,也就是說cp后的"$@"是空值。所以該別名等價於:

alias rmm='cp ~/backup;rm $@'

是否真的如此,使用echo測試一番即可。

[root@xuexi ~]# alias rmm='echo cp $@ ~/backup;echo rm $@'
[root@xuexi ~]# rmm /etc/fstab /etc/hosts
cp /root/backup
rm /etc/fstab /etc/hosts

從上面的結果中看到cp后的"$@"根本就沒有進行擴展,而是空值。

那如果別名定義語句中沒有使用分號或其他方法定義額外的命令,而是只有一個命令呢?別名一定就能正確工作嗎?非也。以下面的例子為例:

[root@xuexi ~]# alias rmm='echo mv -f $@ ~/backup'

[root@xuexi ~]# rmm /etc/fstab /etc/hosts
mv -f /root/backup /etc/fstab /etc/hosts

發現問題了嗎?"$@"是擴展在"~/backup"目錄之后的,也就是說下面mv的別名想要替代rm,是無法正常工作的:

alias rm='mv -f $@ ~/backup'

之所以無法正常工作,是因為~/backup也是"$@"的一部分,且是"$@"中最前面的參數。執行下面的命令就知道了:

[root@xuexi ~]# echo mv -f "$@" ~/backup /etc/fstab /etc/hosts
mv -f /root/backup /etc/fstab /etc/hosts

從上面的分析可以知道,alias是有其缺陷的,它只適合進行簡單的命令和參數替換、補全,想要實現復雜的命令替代有點難度。因此man bash中建議盡量使用函數來取代別名(For almost every purpose, aliases are superseded by shell functions)。

1.3 別名的最佳實現

毫無疑問,寫個shell腳本比別名安全、完整多了,這是替代別名的一種方法。而我個人的建議是,在別名的定義語句中使用函數來克服別名的缺陷。

例如,為了讓rm安全執行,使用以下兩種方法定義別名:

alias rm='copy1(){ /bin/cp -a $@ ~/backup;rm $@; };copy1 $@'
alias rm='move1(){ /bin/mv -f $@ ~/backup; };move1 $@'

因為執行別名時的參數只能傳遞給最后一個命令即copy1或move1函數,但"$@"代表的參數可以傳遞給函數,讓函數中的"$@"得到正確的擴展,於是整個別名都能合理且正確地執行。

或者直接定義一個shell function替代rm。例如向/etc/profile.d/rm.sh文件中寫入:

function rm(){ [ -d ~/rmbackup ] || mkdir ~/rmbackup;/bin/mv -f $@ ~/rmbackup; }
chmod +x /etc/profile.d/rm.sh
source /etc/profile.d/rm.sh

如此,執行rm命令時,便會執行此處定義的rm函數,使得rm變得更安全。但注意,這樣的函數默認無法直接在腳本中使用,除非使用 export -f function_name 導出函數,使其可以被子shell繼承。所以,可在/etc/profile.d/rm.sh文件的尾部加上導出語句:

function rm(){ [ -d ~/rmbackup ] || mkdir ~/rmbackup;/bin/mv -f $@ ~/rmbackup; }
export -f rm

如果function名和命令名相同,則默認優先執行function,除非使用command明確指定。例如上面定義了rm函數,如果想執行rm命令,除了使用/bin/rm,還可以如下操作:

command rm a.txt

如果是在shell腳本里涉及到rm命令,那么更建議在每次rm之前先cd到那個目錄下,然后再rm相對路徑,這樣至少能保證不出現符號"/"。當然,更重要的是腳本習慣一些編寫腳本的規范,印在骨子里那種,就算想出問題也難。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM