Appendix K. 一个简单的.bashrc文件

~/.bashrc文件决定了交互shell的行为. 好好的了解这个文件, 将会使你更加了解Bash.

Emmanuel Rouat捐献了下边这个注释非常详细的.bashrc文件, 这个文件是为Linux系统编写的. 他希望读者能够给他一些回馈.

仔细的学习这个文件, 直到你可以自由重用其中的代码片断和函数, 并把它们用到你自己的.bashrc文件中, 甚至可以放到你的脚本中.


例子 K-1. .bashrc文件样本

  1 #===============================================================
  2 #
  3 # 个人的$HOME/.bashrc文件, 基于bash-2.05a(或更高版本)
  4 #
  5 # 最后更新日期: 星期2 4月15 20:32:34 CEST 2003
  6 #
  7 # 这个文件(一般情况下)被只会被交互式shell读取. 
  8 # 这里可以定义你的别名, 函数, 
  9 # 和其他的一些交互式特征, 比如你的提示符. 
 10 #
 11 # 这个文件(开始时)是为Solaris设计的, 
 12 # 但是基于Redhat的默认.bashrc文件
 13 # --> 为Linux修改. 
 14 # 你在这里看到的大部分代码都是从网上找来的
 15 # (即internet). 
 16 # 这个bashrc文件有点挤 - 
 17 # 记住, 它仅仅是个例子而已. 按照你自己的需求进行裁减. 
 18 #
 19 #
 20 #===============================================================
 21 
 22 # --> 注释由HOWTO的作者添加. 
 23 # --> 然后又被ER编辑了一下 :-)
 24 
 25 #--------------------------------------
 26 # 如果有源代码的全局定义, 请在此处定义.
 27 #--------------------------------------
 28 
 29 if [ -f /etc/bashrc ]; then
 30         . /etc/bashrc   # --> 读取/etc/bashrc, 如果存在的话. 
 31 fi
 32 
 33 #-------------------------------------------------------------
 34 # $DISPLAY的自动设置 (如果还没设置的话)
 35 # 这用于linux - 可能运行的结果不同.... 
 36 # 问题是不同的终端种类对于'who am i'来说, 
 37 # 将会给出不同的答案......
 38 # 我还没发现一种'通用'方法
 39 #-------------------------------------------------------------
 40 
 41 function get_xserver ()
 42 {
 43     case $TERM in
 44 	xterm )
 45             XSERVER=$(who am i | awk '{print $NF}' | tr -d ')''(' ) 
 46             # Ane-Pieter Wieringa建议使用下面这种方式:
 47             # I_AM=$(who am i)
 48             # SERVER=${I_AM#*(}
 49             # SERVER=${SERVER%*)}
 50 
 51             XSERVER=${XSERVER%%:*}
 52 	    ;;
 53 	aterm | rxvt)
 54  	# 找出一些运行在这里的代码.....
 55 	    ;;
 56     esac  
 57 }
 58 
 59 if [ -z ${DISPLAY:=""} ]; then
 60     get_xserver
 61     if [[ -z ${XSERVER}  || ${XSERVER} == $(hostname) || ${XSERVER} == "unix" ]]; then 
 62 	DISPLAY=":0.0"		# 在本地主机上显示
 63     else		
 64 	DISPLAY=${XSERVER}:0.0	# 在远端主机上显示
 65     fi
 66 fi
 67 
 68 export DISPLAY
 69 
 70 #----------
 71 # 一些设置
 72 #----------
 73 
 74 ulimit -S -c 0		# 不需要任何coredump
 75 set -o notify
 76 set -o noclobber
 77 set -o ignoreeof
 78 set -o nounset
 79 #set -o xtrace          # 对于调试来说非常有用
 80 
 81 # 使能选项:
 82 shopt -s cdspell
 83 shopt -s cdable_vars
 84 shopt -s checkhash
 85 shopt -s checkwinsize
 86 shopt -s mailwarn
 87 shopt -s sourcepath
 88 shopt -s no_empty_cmd_completion  # 仅限于bash>=2.04
 89 shopt -s cmdhist
 90 shopt -s histappend histreedit histverify
 91 shopt -s extglob	# 对于complete命令(按情况补全)来说是必要的
 92 
 93 # 禁用选项:
 94 shopt -u mailwarn
 95 unset MAILCHECK		# 当有邮件到达时, 我不希望我的shell提示我
 96 
 97 
 98 export TIMEFORMAT=$'\nreal %3R\tuser %3U\tsys %3S\tpcpu %P\n'
 99 export HISTIGNORE="&:bg:fg:ll:h"
100 export HOSTFILE=$HOME/.hosts	# 将远端主机的列表放入~/.hosts
101 
102 
103 
104 #-----------------------
105 # 问候, 问侯报文等等...
106 #-----------------------
107 
108 # 先定义一些颜色:
109 red='\e[0;31m'
110 RED='\e[1;31m'
111 blue='\e[0;34m'
112 BLUE='\e[1;34m'
113 cyan='\e[0;36m'
114 CYAN='\e[1;36m'
115 NC='\e[0m'              # 没有颜色
116 # --> 很好. 与使用"ansi.sys"的DOS效果相同. 
117 
118 # 在黑色背景下看起来非常好.....
119 echo -e "${CYAN}This is BASH ${RED}${BASH_VERSION%.*}${CYAN} - DISPLAY on ${RED}$DISPLAY${NC}\n"
120 date
121 if [ -x /usr/games/fortune ]; then
122     /usr/games/fortune -s     # 让我们的每天充满乐趣.... :-)
123 fi
124 
125 function _exit()	# 在退出shell时运行的函数
126 {
127     echo -e "${RED}Hasta la vista, baby${NC}"
128 }
129 trap _exit EXIT
130 
131 #---------------
132 # Shell提示符
133 #---------------
134 
135 if [[ "${DISPLAY#$HOST}" != ":0.0" &&  "${DISPLAY}" != ":0" ]]; then  
136     HILIT=${red}   # 远端主机: 提示符为红
137 else
138     HILIT=${cyan}  # 本地主机: 提示符为青色
139 fi
140 
141 #  --> 下面提示符函数中\W和\w的替换实例, 
142 #+ --> 用来获得完整路径名的显示. 
143 
144 function fastprompt()
145 {
146     unset PROMPT_COMMAND
147     case $TERM in
148         *term | rxvt )
149             PS1="${HILIT}[\h]$NC \W > \[\033]0;\${TERM} [\u@\h] \w\007\]" ;;
150 	linux )
151 	    PS1="${HILIT}[\h]$NC \W > " ;;
152         *)
153             PS1="[\h] \W > " ;;
154     esac
155 }
156 
157 function powerprompt()
158 {
159     _powerprompt()
160     {
161         LOAD=$(uptime|sed -e "s/.*: \([^,]*\).*/\1/" -e "s/ //g")
162     }
163 
164     PROMPT_COMMAND=_powerprompt
165     case $TERM in
166         *term | rxvt  )
167             PS1="${HILIT}[\A \$LOAD]$NC\n[\h \#] \W > \[\033]0;\${TERM} [\u@\h] \w\007\]" ;;
168         linux )
169             PS1="${HILIT}[\A - \$LOAD]$NC\n[\h \#] \w > " ;;
170         * )
171             PS1="[\A - \$LOAD]\n[\h \#] \w > " ;;
172     esac
173 }
174 
175 powerprompt     # 这是默认提示符 - 可能比较慢
176                 # 如果很慢的话, 可以使用fastprompt来代替....
177 
178 #===============================================================
179 #
180 # 别名和函数
181 #
182 # 事实上, 这里定义的一些函数非常大
183 # (比如'lowercase'), 但是我的机器是512M内存, 所以 .....
184 # 如果你想让这个文件小一点, 
185 # 可以将这些函数放到脚本中. 
186 #
187 # 其中的许多函数来自于bash-2.04
188 # 中的例子. 
189 #
190 #===============================================================
191 
192 #-------------------
193 # 个人的别名
194 #-------------------
195 
196 alias rm='rm -i'
197 alias cp='cp -i'
198 alias mv='mv -i'
199 # -> 防止偶然的文件误操作. 
200 alias mkdir='mkdir -p'
201 
202 alias h='history'
203 alias j='jobs -l'
204 alias r='rlogin'
205 alias which='type -all'
206 alias ..='cd ..'
207 alias path='echo -e ${PATH//:/\\n}'
208 alias print='/usr/bin/lp -o nobanner -d $LPDEST'   # 假设LPDEST被定义
209 alias pjet='enscript -h -G -fCourier9 -d $LPDEST'  # 使用enscript的漂亮的打印
210 alias background='xv -root -quit -max -rmode 5'    # 将一张图片作为背景
211 alias du='du -kh'
212 alias df='df -kTh'
213 
214 # 'ls'家族 (假定使用GNU ls)
215 alias la='ls -Al'               # 显示隐藏文件
216 alias ls='ls -hF --color'	# 为识别的文件类型添加颜色
217 alias lx='ls -lXB'              # 按扩展名排序
218 alias lk='ls -lSr'              # 按尺寸排序
219 alias lc='ls -lcr'		# 按修改时间排序
220 alias lu='ls -lur'		# 按访问时间排序
221 alias lr='ls -lR'               # 递归ls
222 alias lt='ls -ltr'              # 按日期排序
223 alias lm='ls -al |more'         # 管道给'more'
224 alias tree='tree -Csu'		# 'ls'的另一种好方法
225 
226 # 裁减'less'
227 alias more='less'
228 export PAGER=less
229 export LESSCHARSET='latin1'
230 export LESSOPEN='|/usr/bin/lesspipe.sh %s 2>&-' # 如果lesspipe.sh存在, 就用这个
231 export LESS='-i -N -w  -z-4 -g -e -M -X -F -R -P%t?f%f \
232 :stdin .?pb%pb\%:?lbLine %lb:?bbByte %bb:-...'
233 
234 # 拼写错误 - 纯粹个人喜好 :-)
235 alias xs='cd'
236 alias vf='cd'
237 alias moer='more'
238 alias moew='more'
239 alias kk='ll'
240 
241 #----------------
242 # 一些有趣东西
243 #----------------
244 
245 function xtitle ()
246 {
247     case "$TERM" in
248         *term | rxvt)
249             echo -n -e "\033]0;$*\007" ;;
250         *)  
251 	    ;;
252     esac
253 }
254 
255 # 别名...
256 alias top='xtitle Processes on $HOST && top'
257 alias make='xtitle Making $(basename $PWD) ; make'
258 alias ncftp="xtitle ncFTP ; ncftp"
259 
260 # .. 和函数
261 function man ()
262 {
263     for i ; do
264 	xtitle The $(basename $1|tr -d .[:digit:]) manual
265 	command man -F -a "$i"
266     done
267 }
268 
269 function ll(){ ls -l "$@"| egrep "^d" ; ls -lXB "$@" 2>&-| egrep -v "^d|total "; }
270 function te()  # xemacs/gnuserv的包装器
271 {
272     if [ "$(gnuclient -batch -eval t 2>&-)" == "t" ]; then
273         gnuclient -q "$@";
274     else
275         ( xemacs "$@" &);
276     fi
277 }
278 
279 #---------------------------
280 # 与文件和字符串相关的函数:
281 #---------------------------
282 
283 # 使用名字模式来查找文件:
284 function ff() { find . -type f -iname '*'$*'*' -ls ; }
285 # 使用pattern $1和Execute $2来查找文件: 
286 function fe() { find . -type f -iname '*'$1'*' -exec "${2:-file}" {} \;  ; }
287 # 在一系列文件中找到模式, 并高亮
288 function fstr()
289 {
290     OPTIND=1
291     local case=""
292     local usage="fstr: find string in files.
293 Usage: fstr [-i] \"pattern\" [\"filename pattern\"] "
294     while getopts :it opt
295     do
296         case "$opt" in
297         i) case="-i " ;;
298         *) echo "$usage"; return;;
299         esac
300     done
301     shift $(( $OPTIND - 1 ))
302     if [ "$#" -lt 1 ]; then
303         echo "$usage"
304         return;
305     fi
306     local SMSO=$(tput smso)
307     local RMSO=$(tput rmso)
308     find . -type f -name "${2:-*}" -print0 | xargs -0 grep -sn ${case} "$1" 2>&- | \
309 sed "s/$1/${SMSO}\0${RMSO}/gI" | more
310 }
311 
312 function cuttail() # 在文件中切掉n行, 默认为10行
313 {
314     nlines=${2:-10}
315     sed -n -e :a -e "1,${nlines}!{P;N;D;};N;ba" $1
316 }
317 
318 function lowercase()  # 将文件名转换为小写
319 {
320     for file ; do
321         filename=${file##*/}
322         case "$filename" in
323         */*) dirname==${file%/*} ;;
324         *) dirname=.;;
325         esac
326         nf=$(echo $filename | tr A-Z a-z)
327         newname="${dirname}/${nf}"
328         if [ "$nf" != "$filename" ]; then
329             mv "$file" "$newname"
330             echo "lowercase: $file --> $newname"
331         else
332             echo "lowercase: $file not changed."
333         fi
334     done
335 }
336 
337 function swap()         # 交换两个文件名
338 {
339     local TMPFILE=tmp.$$
340     mv "$1" $TMPFILE
341     mv "$2" "$1"
342     mv $TMPFILE "$2"
343 }
344 
345 
346 #----------------------
347 # 进程/系统相关的函数:
348 #----------------------
349 
350 function my_ps() { ps $@ -u $USER -o pid,%cpu,%mem,bsdtime,command ; }
351 function pp() { my_ps f | awk '!/awk/ && $0~var' var=${1:-".*"} ; }
352 
353 # 这个函数与linux上的'killall'基本一致
354 # 但是与Solaris上的却不相同
355 function killps()   # 按进程名进行kill
356 {
357     local pid pname sig="-TERM"   # 默认signal
358     if [ "$#" -lt 1 ] || [ "$#" -gt 2 ]; then
359         echo "Usage: killps [-SIGNAL] pattern"
360         return;
361     fi
362     if [ $# = 2 ]; then sig=$1 ; fi
363     for pid in $(my_ps| awk '!/awk/ && $0~pat { print $1 }' pat=${!#} ) ; do
364         pname=$(my_ps | awk '$1~var { print $5 }' var=$pid )
365         if ask "Kill process $pid <$pname> with signal $sig?"
366             then kill $sig $pid
367         fi
368     done
369 }
370 
371 function my_ip() # 获得IP地址
372 {
373     MY_IP=$(/sbin/ifconfig ppp0 | awk '/inet/ { print $2 } ' | sed -e s/addr://)
374     MY_ISP=$(/sbin/ifconfig ppp0 | awk '/P-t-P/ { print $3 } ' | sed -e s/P-t-P://)
375 }
376 
377 function ii()   # 获得当前主机相关的信息
378 {
379     echo -e "\nYou are logged on ${RED}$HOST"
380     echo -e "\nAdditionnal information:$NC " ; uname -a
381     echo -e "\n${RED}Users logged on:$NC " ; w -h
382     echo -e "\n${RED}Current date :$NC " ; date
383     echo -e "\n${RED}Machine stats :$NC " ; uptime
384     echo -e "\n${RED}Memory stats :$NC " ; free
385     my_ip 2>&- ;
386     echo -e "\n${RED}Local IP Address :$NC" ; echo ${MY_IP:-"Not connected"}
387     echo -e "\n${RED}ISP Address :$NC" ; echo ${MY_ISP:-"Not connected"}
388     echo
389 }
390 
391 # 杂项工具:
392 
393 function repeat()       # 重复n次的命令
394 {
395     local i max
396     max=$1; shift;
397     for ((i=1; i <= max ; i++)); do  # --> C风格的语法
398         eval "$@";
399     done
400 }
401 
402 function ask()
403 {
404     echo -n "$@" '[y/n] ' ; read ans
405     case "$ans" in
406         y*|Y*) return 0 ;;
407         *) return 1 ;;
408     esac
409 }
410 
411 #=========================================================================
412 #
413 # 按情况补全, complete命令 - BASH-2.04及其后续版本
414 # 大部分摘自bash 2.05文档
415 # 和Ian McDonalds的'Bash completion'软件包(http://www.caliban.org/bash/index.shtml#completion)
416 # 某些特征可能需要使用bash-2.05a
417 #
418 #=========================================================================
419 
420 if [ "${BASH_VERSION%.*}" \< "2.05" ]; then
421     echo "You will need to upgrade to version 2.05 for programmable completion"
422     return
423 fi
424 
425 shopt -s extglob        # 必须的
426 set +o nounset          # 否则某些自动补全将会失败
427 
428 complete -A hostname   rsh rcp telnet rlogin r ftp ping disk
429 complete -A export     printenv
430 complete -A variable   export local readonly unset
431 complete -A enabled    builtin
432 complete -A alias      alias unalias
433 complete -A function   function
434 complete -A user       su mail finger
435 
436 complete -A helptopic  help     # 通常与内建命令一样
437 complete -A shopt      shopt
438 complete -A stopped -P '%' bg
439 complete -A job -P '%'     fg jobs disown
440 
441 complete -A directory  mkdir rmdir
442 complete -A directory   -o default cd
443 
444 # 压缩
445 complete -f -o default -X '*.+(zip|ZIP)'  zip
446 complete -f -o default -X '!*.+(zip|ZIP)' unzip
447 complete -f -o default -X '*.+(z|Z)'      compress
448 complete -f -o default -X '!*.+(z|Z)'     uncompress
449 complete -f -o default -X '*.+(gz|GZ)'    gzip
450 complete -f -o default -X '!*.+(gz|GZ)'   gunzip
451 complete -f -o default -X '*.+(bz2|BZ2)'  bzip2
452 complete -f -o default -X '!*.+(bz2|BZ2)' bunzip2
453 # Postscript,pdf,dvi.....(译者: 打印格式相关)
454 complete -f -o default -X '!*.ps'  gs ghostview ps2pdf ps2ascii
455 complete -f -o default -X '!*.dvi' dvips dvipdf xdvi dviselect dvitype
456 complete -f -o default -X '!*.pdf' acroread pdf2ps
457 complete -f -o default -X '!*.+(pdf|ps)' gv
458 complete -f -o default -X '!*.texi*' makeinfo texi2dvi texi2html texi2pdf
459 complete -f -o default -X '!*.tex' tex latex slitex
460 complete -f -o default -X '!*.lyx' lyx
461 complete -f -o default -X '!*.+(htm*|HTM*)' lynx html2ps
462 # 多媒体
463 complete -f -o default -X '!*.+(jp*g|gif|xpm|png|bmp)' xv gimp
464 complete -f -o default -X '!*.+(mp3|MP3)' mpg123 mpg321
465 complete -f -o default -X '!*.+(ogg|OGG)' ogg123
466 
467 
468 
469 complete -f -o default -X '!*.pl'  perl perl5
470 
471 # 这是一个'通用的'补全函数 - 当命令具有一个所谓的"长选项"模式it works when commands have
472 # 的时候, 它就会工作, 比如: 'ls --all' 代替 'ls -a'
473 
474 _get_longopts () 
475 { 
476     $1 --help | sed  -e '/--/!d' -e 's/.*--\([^[:space:].,]*\).*/--\1/'| \
477 grep ^"$2" |sort -u ;
478 }
479 
480 _longopts_func ()
481 {
482     case "${2:-*}" in
483 	-*)	;;
484 	*)	return ;;
485     esac
486 
487     case "$1" in
488 	\~*)	eval cmd="$1" ;;
489 	*)	cmd="$1" ;;
490     esac
491     COMPREPLY=( $(_get_longopts ${1} ${2} ) )
492 }
493 complete  -o default -F _longopts_func configure bash
494 complete  -o default -F _longopts_func wget id info a2ps ls recode
495 
496 
497 _make_targets ()
498 {
499     local mdef makef gcmd cur prev i
500 
501     COMPREPLY=()
502     cur=${COMP_WORDS[COMP_CWORD]}
503     prev=${COMP_WORDS[COMP_CWORD-1]}
504 
505     # 如果之前的参数为-f, 那就返回可能的补全文件名. 
506     # 我们可以让它更智能一些, 并且返回匹配的
507     # `makefile Makefile *.mk', 不管存在与否
508     case "$prev" in
509         -*f)    COMPREPLY=( $(compgen -f $cur ) ); return 0;;
510     esac
511 
512     # 如果我们需要一个选项, 那就返回可能的posix选项
513     case "$cur" in
514         -)      COMPREPLY=(-e -f -i -k -n -p -q -r -S -s -t); return 0;;
515     esac
516 
517     # 前尝试`makefile'再尝试`Makefile'
518     if [ -f makefile ]; then
519         mdef=makefile
520     elif [ -f Makefile ]; then
521         mdef=Makefile
522     else
523         mdef=*.mk               # 局部约定
524     fi
525 
526     # 在我们扫描目标文件之前, 察看makefile文件名是否
527     # 使用-f指定
528     for (( i=0; i < ${#COMP_WORDS[@]}; i++ )); do
529         if [[ ${COMP_WORDS[i]} == -*f ]]; then
530             eval makef=${COMP_WORDS[i+1]}       # eval for tilde expansion(波浪号扩展)
531             break
532         fi
533     done
534 
535         [ -z "$makef" ] && makef=$mdef
536 
537     # 如果我们有特别偏爱的补全单词, 
538     # 那么可以限制的补全这个单词
539     if [ -n "$2" ]; then gcmd='grep "^$2"' ; else gcmd=cat ; fi
540 
541     # 如果我们不想使用*.mk, 我们可以使用
542     # 或者使用test -f $makef或者使用输入重定向
543     COMPREPLY=( $(cat $makef 2>/dev/null | awk 'BEGIN {FS=":"} /^[^.#   ][^=]*:/ {print $1}' | tr -s ' ' '\012' | sort -u | eval $gcmd ) )
544 }
545 
546 complete -F _make_targets -X '+($*|*.[cho])' make gmake pmake
547 
548 
549 # cvs(1) 补全
550 _cvs ()
551 {
552     local cur prev
553     COMPREPLY=()
554     cur=${COMP_WORDS[COMP_CWORD]}
555     prev=${COMP_WORDS[COMP_CWORD-1]}
556 
557     if [ $COMP_CWORD -eq 1 ] || [ "${prev:0:1}" = "-" ]; then
558         COMPREPLY=( $( compgen -W 'add admin checkout commit diff \
559         export history import log rdiff release remove rtag status \
560         tag update' $cur ))
561     else
562         COMPREPLY=( $( compgen -f $cur ))
563     fi
564     return 0
565 }
566 complete -F _cvs cvs
567 
568 _killall ()
569 {
570     local cur prev
571     COMPREPLY=()
572     cur=${COMP_WORDS[COMP_CWORD]}
573 
574     # 获得进程列表(第一个sed表达式处理
575     # swap out出去的进程, 第二个
576     # 获得进程的basename)
577     COMPREPLY=( $( /usr/bin/ps -u $USER -o comm  | \
578         sed -e '1,1d' -e 's#[]\[]##g' -e 's#^.*/##'| \
579         awk '{if ($0 ~ /^'$cur'/) print $0}' ))
580 
581     return 0
582 }
583 
584 complete -F _killall killall killps
585 
586 
587 # 一个元命令补全函数, 用于sudo(8)这种命令, 
588 # 需要先对这个命令进行补全, 然后需要按照这个命令自己的补全定义进行补全
589 #  - 当前并不是非常可靠(比如 mount和umount命令
590 # 就不能很好的工作), 但还是很有用的 - 作者, Ian McDonald, 我修改了一下. 
591 
592 _my_command()
593 {
594     local cur func cline cspec
595     
596     COMPREPLY=()
597     cur=${COMP_WORDS[COMP_CWORD]}
598 
599     if [ $COMP_CWORD = 1 ]; then
600 	COMPREPLY=( $( compgen -c $cur ) )
601     elif complete -p ${COMP_WORDS[1]} &>/dev/null; then
602 	cspec=$( complete -p ${COMP_WORDS[1]} )
603 	if [ "${cspec%%-F *}" != "${cspec}" ]; then
604 	    # complete -F <function>
605 	    #
606 	    # COMP_CWORD和COMP_WORDS()不是只读的,
607 	    # 所以我们可以在传递到补全例程之前, 
608 	    # 设置它们
609 	
610 	    # 设置当前的标志号减1
611 	    COMP_CWORD=$(( $COMP_CWORD - 1 ))
612 	    # 获得函数名
613 	    func=${cspec#*-F }
614 	    func=${func%% *}
615 	    # 获得去掉第一个命令后的命令行
616 	    cline="${COMP_LINE#$1 }"
617 	    # 分离当前命令, 传递给数组
618 		COMP_WORDS=( $cline )
619 	    $func $cline
620 	elif [ "${cspec#*-[abcdefgjkvu]}" != "" ]; then
621 	    # complete -[abcdefgjkvu]
622 	    #func=$( echo $cspec | sed -e 's/^.*\(-[abcdefgjkvu]\).*$/\1/' )
623 	    func=$( echo $cspec | sed -e 's/^complete//' -e 's/[^ ]*$//' )
624 	    COMPREPLY=( $( eval compgen $func $cur ) )
625 	elif [ "${cspec#*-A}" != "$cspec" ]; then
626 	    # complete -A <type>
627 	    func=${cspec#*-A }
628 	func=${func%% *}
629 	COMPREPLY=( $( compgen -A $func $cur ) )
630 	fi
631     else
632 	COMPREPLY=( $( compgen -f $cur ) )
633     fi
634 }
635 
636 
637 complete -o default -F _my_command nohup exec eval trace truss strace sotruss gdb
638 complete -o default -F _my_command command type which man nice
639 
640 # 本地变量:
641 # mode:shell-script
642 # sh-shell:bash
643 # End: