第零章、必读系列
- 学习算法和刷题的框架思维
- 学习数据结构和算法读什么书
- 动态规划解题框架
- 动态规划答疑篇
- 回溯算法解题框架
- 为了学会二分查找,我写了首诗
- 滑动窗口解题框架
- 双指针技巧解题框架
- Linux 的进程、线程、文件描述符是什么
- Git / SQL / 正则表达式的在线练习平台
- 动态规划设计:最长递增子序列
第一章、动态规划系列
- 编辑距离
- 经典动态规划问题:高楼扔鸡蛋
- 经典动态规划问题:高楼扔鸡蛋(进阶)
- 动态规划之子序列问题解题模板
- 动态规划之博弈问题
- 贪心算法之区间调度问题
- 动态规划之KMP字符匹配算法
- 团灭 LeetCode 股票买卖问题
- 团灭 LeetCode 打家劫舍问题
- 动态规划之四键键盘
- 动态规划之正则表达
- 最长公共子序列
第二章、数据结构系列
第三章、算法思维系列
- 算法学习之路
- 回溯算法团灭排列、组合、子集问题
- twoSum 问题的核心思想
- 常用的位操作
- 拆解复杂问题:实现计算器
- 烧饼排序
- 前缀和技巧
- 字符串乘法
- FloodFill 算法详解及应用
- 区间调度之区间合并问题
- 区间调度之区间交集问题
- 信封嵌套问题
- 几个反直觉的概率问题
- 洗牌算法
- 递归详解
第四章、高频面试系列
- 如何高效寻找素数
- 如何运用二分查找算法
- 如何高效解决接雨水问题
- 如何去除有序数组的重复元素
- 如何寻找最长回文子串
- 如何 k 个一组反转链表
- 如何判定括号合法性
- 如何寻找消失的元素
- 如何寻找缺失和重复的元素
- 如何判断回文链表
- 如何在无限序列中随机抽取元素
- 如何调度考生的座位
- Union-Find 算法详解
- Union-Find 算法应用
- 一行代码就能解决的算法题
- 二分查找高效判定子序列
第五章、计算机技术
关于 Linux shell 你必须知道的
我个人很喜欢使用 Linux 系统,虽然说 Windows 的图形化界面做的确实比 Linux 好,但是对脚本的支持太差了。一开始有点不习惯命令行操作,但是熟悉了之后反而发现移动鼠标点点点才是浪费时间的罪魁祸首。。。
那么对于 Linux 命令行,本文不是介绍某些命令的用法,而是说明一些简单却特别容易让人迷惑的细节问题。
1、标准输入和命令参数的区别。
2、在后台运行命令在退出终端后也全部退出了。
3、单引号和双引号表示字符串的区别。
4、有的命令和sudo
一起用就 command not found。
一、标准输入和参数的区别
这个问题一定是最容易让人迷惑的,具体来说,就是搞不清什么时候用管道符|
和文件重定向>
,<
,什么时候用变量$
。
比如说,我现在有个自动连接宽带的 shell 脚本connect.sh
,存在我的家目录:
$ where connect.sh
/home/fdl/bin/connect.sh
如果我想删除这个脚本,而且想少敲几次键盘,应该怎么操作呢?我曾经这样尝试过:
$ where connect.sh | rm
实际上,这样操作是错误的,正确的做法应该是这样的:
$ rm $(where connect.sh)
前者试图将where
的结果连接到rm
的标准输入,后者试图将结果作为命令行参数传入。
标准输入就是编程语言中诸如scanf
或者readline
这种命令;而参数是指程序的main
函数传入的args
字符数组。
前文「Linux文件描述符」说过,管道符和重定向符是将数据作为程序的标准输入,而$(cmd)
是读取cmd
命令输出的数据作为参数。
用刚才的例子说,rm
命令源代码中肯定不接受标准输入,而是接收命令行参数,删除相应的文件。作为对比,cat
命令是既接受标准输入,又接受命令行参数:
$ cat filename
...file text...
$ cat < filename
...file text...
$ echo 'hello world' | cat
hello world
如果命令能够让终端阻塞,说明该命令接收标准输入,反之就是不接受,比如你只运行cat
命令不加任何参数,终端就会阻塞,等待你输入字符串并回显相同的字符串。
二、后台运行程序
比如说你远程登录到服务器上,运行一个 Django web 程序:
$ python manager.py runserver 0.0.0.0
Listening on 0.0.0.0:8080...
现在你可以通过服务器的 IP 地址测试 Django 服务,但是终端此时就阻塞了,你输入什么都不响应,除非输入 Ctrl-C 或者 Ctrl-/ 终止 python 进程。
可以在命令之后加一个&
符号,这样命令行不会阻塞,可以响应你后续输入的命令,但是如果你退出服务器的登录,就不能访问该网页了。
如果你想在退出服务器之后仍然能够访问 web 服务,应该这样写命令 (cmd &)
:
$ (python manager.py runserver 0.0.0.0 &)
Listening on 0.0.0.0:8080...
$ logout
底层原理是这样的:
每一个命令行终端都是一个 shell 进程,你在这个终端里执行的程序实际上都是这个 shell 进程分出来的子进程。正常情况下,shell 进程会阻塞,等待子进程退出才重新接收你输入的新的命令。加上&
号,只是让 shell 进程不再阻塞,可以继续响应你的新命令。但是无论如何,你如果关掉了这个 shell 命令行端口,依附于它的所有子进程都会退出。
而(cmd &)
这样运行命令,则是将cmd
命令挂到一个systemd
系统守护进程名下,认systemd
做爸爸,这样当你退出当前终端时,对于刚才的cmd
命令就完全没有影响了。
类似的,还有一种后台运行常用的做法是这样:
$ nohub some_cmd &
nohub
命令也是类似的原理,不过通过我的测试,还是(cmd &)
这种形式更加稳定。
三、单引号和双引号的区别
不同的 shell 行为会有细微区别,但有一点是确定的,对于$
,(
,)
这几个符号,单引号包围的字符串不会做任何转义,双引号包围的字符串会转义。
shell 的行为可以测试,使用set -x
命令,会开启 shell 的命令回显,你可以通过回显观察 shell 到底在执行什么命令:
可见 echo $(cmd)
和 echo "$(cmd)"
,结果差不多,但是仍然有区别。注意观察,双引号转义完成的结果会自动增加单引号,而前者不会。
也就是说,如果 $
读取出的参数字符串包含空格,应该用双引号括起来,否则就会出错。
四、sudo 找不到命令
有时候我们普通用户可以用的命令,用sudo
加权限之后却报错 command not found:
$ connect.sh
network-manager: Permission denied
$ sudo connect.sh
sudo: command not found
原因在于,connect.sh
这个脚本仅存在于该用户的环境变量中:
$ where connect.sh
/home/fdl/bin/connect.sh
当使用sudo
时,系统认为是 root 用户在执行命令,所以会去搜索 root 用户的环境变量,而这个脚本在 root 的环境变量目录中当然是找不到的。
解决方法是使用脚本文件的路径,而不是仅仅通过脚本名称:
$ sudo /home/fdl/bin/connect.sh
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论