shell中的基本操作备忘录
常用快捷键(默认使用 Emacs 键位) 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 CTRL+A # 移动到行首,同 <Home> CTRL+B # 向后移动,同 <Left> CTRL+C # 结束当前命令 CTRL+D # 删除光标前的字符,同 <Delete> ,或者没有内容时,退出会话 CTRL+E # 移动到行末,同 <End> CTRL+F # 向前移动,同 <Right> CTRL+G # 退出当前编辑(比如正在 CTRL+R 搜索历史时) CTRL+H # 删除光标左边的字符,同 <Backspace> CTRL+K # 删除光标位置到行末的内容 CTRL+L # 清屏并重新显示 CTRL+N # 移动到命令历史的下一行,同 <Down> CTRL+O # 类似回车,但是会显示下一行历史 CTRL+P # 移动到命令历史的上一行,同 <Up> CTRL+R # 历史命令反向搜索,使用 CTRL+G 退出搜索 CTRL+S # 历史命令正向搜索,使用 CTRL+G 退出搜索 CTRL+T # 交换前后两个字符 CTRL+U # 删除字符到行首 CTRL+V # 输入字符字面量,先按 CTRL+V 再按任意键 CTRL+W # 删除光标左边的一个单词 CTRL+X # 列出可能的补全 CTRL+Y # 粘贴前面 CTRL+u/k/w 删除过的内容 CTRL+Z # 暂停前台进程返回 bash,需要时可用 fg 将其切换回前台 CTRL+_ # 撤销(undo),有的终端将 CTRL+_ 映射为 CTRL+/ 或 CTRL+7 ALT+b # 向后(左边)移动一个单词 ALT+d # 删除光标后(右边)一个单词 ALT+f # 向前(右边)移动一个单词 ALT+t # 交换字符 ALT+BACKSPACE # 删除光标前面一个单词,类似 CTRL+W,但不影响剪贴板 CTRL+X CTRL+X # 连续按两次 CTRL+X,光标在当前位置和行首来回跳转 CTRL+X CTRL+E # 用你指定的编辑器,编辑当前命令
BASH 基本操作 1 2 3 4 5 6 7 8 9 10 11 exit env echo $SHELL bash which bash whereis bash whatis bash clear reset
目录操作 1 2 3 4 5 6 7 8 9 10 cd cd {dirname } pwd mkdir {dirname } mkdir -p {dirname } pushd {dirname } popd dirs -v cd - cd -{N}
文件操作 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 ls ls -l ls -1 ls -a ln -s {fn} {link } cp {src} {dest} rm {fn} mv {src} {dest} touch {fn} cat {fn} any_cmd > {fn} more {fn} less {fn} head {fn} tail {fn} tail -f {fn} nano {fn} vim {fn} diff {f1} {f2} wc {fn} chmod 644 {fn} chgrp group {fn} chown user1 {fn} file {fn} basename {fn} dirname {fn} grep {pat} {fn} grep -r {pat} . stat {fn}
用户管理 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 whoami who w users passwd finger {user} adduser {user} deluser {user} w su su - su {user} su -{user} id {user} id -u {user} id -g {user} write {user} last last {user} lastb lastlog sudo {command }
进程管理 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 ps ps ax ps aux ps auxww ps -u {user} ps axjf ps xjf -u {user} ps -eo pid,user,command ps aux | grep httpd ps --ppid {pid} pstree pstree {user} pstree -u pgrep {procname} kill {pid} kill -9 {pid} kill -KILL {pid} kill -l kill -l TERM killall {procname} pkill {procname} top top -u {user} any_command & jobs bg fg fg {job} trap cmd sig1 sig2 trap "" sig1 sig2 trap - sig1 sig2 nohup {command } nohup {command } & disown {PID|JID} wait
常用命令:SSH / 系统信息 / 网络 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 ssh user@host ssh -p {port} user@host ssh-copy-id user@host scp {fn} user@host:path scp user@host:path dest scp -P {port} ... uname -a man {help } man -k {keyword} info {help } uptime date cal vmstat vmstat 10 free df du du -h uname hostname showkey -a ping {host} ping -c N {host} traceroute {host} mtr {host} host {domain} whois {domain} dig {domain} route -n netstat -a netstat -an netstat -anp netstat -l netstat -t netstat -lntu netstat -lntup netstat -i netstat -rn ss -an ss -s wget {url} wget -qO- {url} curl -sL {url} sz {file} rz
变量操作 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 varname=value varname=value command echo $varname echo $$ echo $! echo $? export VARNAME=value array[0]=valA array[1]=valB array[2]=valC array=([0]=valA [1]=valB [2]=valC) array=(valA valB valC) ${array[i]} ${#array[@]} ${#array[i]} declare -a declare -f declare -F declare -i declare -r declare -x declare -p varname ${varname:-word} ${varname:=word} ${varname:?message} ${varname:+word} ${varname:offset:len} ${variable#pattern} ${variable##pattern} ${variable%pattern} ${variable%%pattern} ${variable/pattern/str} ${variable//pattern/str} ${#varname} *(patternlist) +(patternlist) ?(patternlist) @(patternlist) !(patternlist) array=($text ) IFS="/" array=($text ) text="${array[*]} " text=$(IFS=/; echo "${array[*]} " ) A=( foo bar "a b c" 42 ) B=("${A[@]:1:2} " ) C=("${A[@]:1} " ) echo "${B[@]} " echo "${B[1]} " echo "${C[@]} " echo "${C[@]: -2:2} " $(UNIX command ) varname=$(id -u user) num=$(expr 1 + 2) num=$(expr $num + 1) expr 2 \* \( 2 + 3 \) num=$((1 + 2 )) num=$(($num + 1 )) num=$((num + 1 )) num=$((1 + (2 + 3 ) * 2 ))
事件指示符 1 2 3 4 5 6 7 8 9 10 !! !^ !:n !:n-$ !$ !-n:$ !string !^string1^string2 ! !
函数 定义一个新函数 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 function myfunc () { {shell commands ...} } myfunc myfunc arg1 arg2 arg3 myfunc "$@ " myfunc "${array[@]} " shift unset -f myfunc declare -f }
条件判断(兼容 posix sh 的条件判断):man test 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 statement1 && statement2 statement1 || statement2 exp1 -a exp2 exp1 -o exp2 ( expression ) ! expression str1 = str2 str1 != str2 str1 < str2 str2 > str2 -n str1 -z str1 -a file -d file -e file -f file -r file -s file -w file -x file -N file -O file -G file file1 -nt file2 file1 -ot file2 num1 -eq num2 num1 -ne num2 num1 -lt num2 num1 -le num2 num1 -gt num2 num1 -ge num2
分支控制:if 和经典 test,兼容 posix sh 的条件判断语句 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 test {expression} [ expression ] test "abc" = "def" test "abc" != "def" test -a /tmp; echo $? [ -a /tmp ]; echo $? test cond && cmd1 [ cond ] && cmd1 [ cond ] && cmd1 || cmd2 if test -e /etc/passwd; then echo "alright it exists ... " else echo "it doesn't exist ... " fi if [ -e /etc/passwd ]; then echo "alright it exists ... " else echo "it doesn't exist ... " fi [ -e /etc/passwd ] && echo "alright it exists" || echo "it doesn't exist" if [ "$varname " = "foo" ]; then echo "this is foo" elif [ "$varname " = "bar" ]; then echo "this is bar" else echo "neither" fi if [ $x -gt 10 ] && [ $x -lt 20 ]; then echo "yes, between 10 and 20" fi [ $x -gt 10 ] && [ $x -lt 20 ] && echo "yes, between 10 and 20" if [ \( $x -gt 10 \) -a \( $x -lt 20 \) ]; then echo "yes, between 10 and 20" fi [ \( $x -gt 10 \) -a \( $x -lt 20 \) ] && echo "yes, between 10 and 20" [ -x /bin/ls ] && /bin/ls -l https://www.ibm.com/developerworks/library/l-bash-test/index.html
流程控制:while / for / case / until 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 while condition; do statementsdone i=1while [ $i -le 10 ]; do echo $i ; i=$(expr $i + 1)done for i in {1..10}; do echo $i done for name [in list]; do statementsdone for f in /home/*; do echo $f done for (( initialisation ; ending condition ; update )); do statementsdone for ((i = 0 ; i < 10 ; i++)); do echo $i ; done case expression in pattern1 ) statements ;; pattern2 ) statements ;; * ) otherwise ;;esac until condition; do statementsdone select name [in list]; do statements that can use $name done
命令处理 1 2 3 4 5 6 command ls builtin cd enable help {builtin_command} eval $script
输出/输入 重定向 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 cmd1 | cmd2 < file > file >> file >| file n>| file <> file n<> file n> file n< file n>& n<& n>&m n<&m &>file <&- >&- n>&- n<&- diff <(cmd1) <(cmd2)
文本处理 - cut 1 2 3 4 5 6 7 8 9 cut -c 1-16 cut -c 1-16 file cut -c3- cut -d':' -f5 cut -d';' -f2,10 cut -d' ' -f3-7 echo "hello" | cut -c1-3 echo "hello sir" | cut -d' ' -f2 ps | tr -s " " | cut -d " " -f 2,3,4
文本处理 - awk / sed 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 awk '{print $5}' file awk -F ',' '{print $5}' file awk '/str/ {print $2}' file awk -F ',' '{print $NF}' file awk '{s+=$1} END {print s}' file awk 'NR%3==1' file sed 's/find/replace/' file sed '10s/find/replace/' file sed '10,20s/find/replace/' file sed -r 's/regex/replace/g' file sed -i 's/find/replace/g' file sed -i '/find/i\newline' file sed -i '/find/a\newline' file sed '/line/s/find/replace/' file sed -e 's/f/r/' -e 's/f/r' file sed 's#find#replace#' file sed -i -r 's/^\s+//g' file sed '/^$/d' file sed -i 's/\s\+$//' file sed -n '2p' file sed -n '2,5p' file
排序 - sort 1 2 3 4 5 6 sort file sort -r file sort -n file sort -t: -k 3n /etc/passwd sort -u file sort -h file
1 2 3 4 5 6 7 source /path/to/z.sh z z foo z foo bar z -l foo z -r foo z -t foo
键盘绑定 1 2 3 4 5 6 7 8 9 bind '"\eh":"\C-b"' bind '"\el":"\C-f"' bind '"\ej":"\C-n"' bind '"\ek":"\C-p"' bind '"\eH":"\eb"' bind '"\eL":"\ef"' bind '"\eJ":"\C-a"' bind '"\eK":"\C-e"' bind '"\e;":"ls -l\n"'
网络管理:ip / ifconfig / nmap … 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 ip a ip a show eth1 ip a add 172.16.1.23/24 dev eth1 ip a del 172.16.1.23/24 dev eth1 ip link show dev eth0 ip link set eth1 up ip link set eth1 down ip link set eth1 address {mac} ip neighbour ip route ip route add 10.1.0.0/24 via 10.0.0.253 dev eth0 ip route del 10.1.0.0/24 ifconfig ifconfig -a ifconfig eth0 ifconfig eth0 up ifconfig eth0 down ifconfig eth0 192.168.120.56 ifconfig eth0 10.0.0.8 netmask 255.255.255.0 up ifconfig eth0 hw ether 00:aa:bb:cc:dd :ee nmap 10.0.0.12 nmap -p 1024-65535 10.0.0.12 nmap 10.0.0.0/24 nmap -O -sV 10.0.0.12
有趣的命令 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 man hier man test man ascii getconf LONG_BIT bind -P mount | column -t curl ip.cn disown -a && exit cat /etc/issue lsof -i port:80 showkey -a svn diff | view - mv filename.{old,new} time read cp file.txt{,.bak} sudo touch /forcefsck find ~ -mmin 60 -type f curl wttr.in/~beijing echo ${SSH_CLIENT%% *} echo $[RANDOM%X+1] bind -x '"\C-l":ls -l' find / -type f -size +5M chmod --reference f1 f2 curl -L cheat.sh
常用技巧 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 history | awk '{a[$2]++}END{for(i in a){print a[i] " " i}}' | sort -rn | head netstat -n | awk '/^tcp/ {++tt[$NF]} END {for (a in tt) print a, tt[a]}' sshfs name@server:/path/to/folder /path/to/mount/point ps aux | sort -nk +4 | tail while sleep 1;do tput sc;tput cup 0 $(($(tput cols)-29 ));date ;tput rc;done & wget -qO - "http://www.tarball.com/tarball.gz" | tar zxvf - python -c "import test.pystone;print(test.pystone.pystones())" dd if =/dev/zero of=/dev/null bs=1M count=32768 mount /path/to/file.iso /mnt/cdrom -oloop ssh -t hostA ssh hostB wget -r -l1 --no-parent -nH -nd -P/tmp -A".gif,.jpg" http://example.com/imagesmkdir -p work/{project1,project2}/{src,bin,bak} find . -type f -newermt "2010-01-01" ! -newermt "2010-06-01" lsof -P -i -n | cut -f 1 -d " " | uniq | tail -n +2 :w !sudo tee > /dev/null %source ~/github/profiles/my_bash_init.sh ssh -CqTnN -R 0.0.0.0:8443:192.168.1.2:443 user@202.115.8.1 ssh -CqTnN -L 0.0.0.0:8443:192.168.1.2:443 user@192.168.1.3 ssh -CqTnN -D localhost:1080 user@202.115.8.1 http://www.skywind.me/blog/archives/2021
有用的函数 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 function q-extract () { if [ -f $1 ] ; then case $1 in *.tar.bz2) tar -xvjf $1 ;; *.tar.gz) tar -xvzf $1 ;; *.tar.xz) tar -xvJf $1 ;; *.bz2) bunzip2 $1 ;; *.rar) rar x $1 ;; *.gz) gunzip $1 ;; *.tar) tar -xvf $1 ;; *.tbz2) tar -xvjf $1 ;; *.tgz) tar -xvzf $1 ;; *.zip) unzip $1 ;; *.Z) uncompress $1 ;; *.7z) 7z x $1 ;; *) echo "don't know how to extract '$1 '..." ;; esac else echo "'$1 ' is not a valid file!" fi }function q-compress () { if [ -n "$1 " ] ; then FILE=$1 case $FILE in *.tar) shift && tar -cf $FILE $* ;; *.tar.bz2) shift && tar -cjf $FILE $* ;; *.tar.xz) shift && tar -cJf $FILE $* ;; *.tar.gz) shift && tar -czf $FILE $* ;; *.tgz) shift && tar -czf $FILE $* ;; *.zip) shift && zip $FILE $* ;; *.rar) shift && rar $FILE $* ;; esac else echo "usage: q-compress <foo.tar.gz> ./foo ./bar" fi }function ccat () { local style="monokai" if [ $# -eq 0 ]; then pygmentize -P style=$style -P tabsize=4 -f terminal256 -g else for NAME in $@ ; do pygmentize -P style=$style -P tabsize=4 -f terminal256 -g "$NAME " done fi }
好玩的配置 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 export LESS_TERMCAP_mb=$'\E[1m\E[32m' export LESS_TERMCAP_mh=$'\E[2m' export LESS_TERMCAP_mr=$'\E[7m' export LESS_TERMCAP_md=$'\E[1m\E[36m' export LESS_TERMCAP_ZW="" export LESS_TERMCAP_us=$'\E[4m\E[1m\E[37m' export LESS_TERMCAP_me=$'\E(B\E[m' export LESS_TERMCAP_ue=$'\E[24m\E(B\E[m' export LESS_TERMCAP_ZO="" export LESS_TERMCAP_ZN="" export LESS_TERMCAP_se=$'\E[27m\E(B\E[m' export LESS_TERMCAP_ZV="" export LESS_TERMCAP_so=$'\E[1m\E[33m\E[44m' "\eh" : backward-char"\el" : forward-char"\ej" : next-history"\ek" : previous-history"\eH" : backward-word"\eL" : forward-word"\eJ" : beginning-of-line"\eK" : end-of-line
References https://github.com/skywind3000/awesome-cheatsheets/blob/master/languages/bash.sh https://github.com/Idnan/bash-guide http://www.linuxstall.com/linux-command-line-tips-that-every-linux-user-should-know/ https://ss64.com/bash/syntax-keyboard.html http://wiki.bash-hackers.org/commands/classictest https://www.ibm.com/developerworks/library/l-bash-test/index.html https://www.cyberciti.biz/faq/bash-loop-over-file/ https://linuxconfig.org/bash-scripting-tutorial https://github.com/LeCoupa/awesome-cheatsheets/blob/master/languages/bash.sh https://devhints.io/bash https://github.com/jlevy/the-art-of-command-line https://yq.aliyun.com/articles/68541