为什么执行自己的程序要在前面加./
点击上方“视学算法”,选择“设为星标”
来源:公众号【编程珠玑】
作者:守望先生
ID:shouwangxiansheng
前言
在Linux中,我们执行内置命令时,直接输入命令名称即可,如:
$ mv a b #将a重命名为b
而在执行自己写好的程序时,却要带上./,例如:
$ hello
hello: command not found
$ ./hello
hello world
这是为什么呢?它们有什么区别呢?
shell是如何运行程序的
在说明清楚问题之前,我们必须了解shell是如何运行程序的。首先我们必须要清楚的是,执行一条Linux命令,本质是在运行一个程序,如执行ls命令,它执行的是ls程序。那么在shell中输入一条命令,到底发生了什么?它会经历哪几个查找过程?
alias中查找
alias命令可用来设置命令别名,而单独输入alias可以查看到已设置的别名:
$ alias
alias egrep='egrep --color=auto'
alias fgrep='fgrep --color=auto'
alias grep='grep --color=auto'
alias l='ls -CF'
alias la='ls -A'
alias ll='ls -alF'
alias ls='ls --color=auto'
如果这里没有找到你执行的命令,那么就会接下去查找。如果找到了,那么就会执行下去。
内置命令中查找
不同的shell包含一些不同的内置命令,通常不需要shell到磁盘中去搜索。通过help命令可以看到有哪些内置命令:
$ help
通过type 命令可以查看命令类型:
$ type echo
echo is a shell builtin
如果是内置命令,则会直接执行,否则继续查找。
PATH中查找
以ls为例,在shell输入ls时,首先它会从PATH环境变量中查找,PATH内容是什么呢,我们看看:
$ echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games
所以它会在这些路径下去寻找ls程序,按照路径找到的第一个ls程序就会被执行。使用whereis也能确定ls的位置:
$ whereis ls
ls: /bin/ls /usr/share/man/man1/ls.1.g
既然它是在bin目录下,那么我把ls从bin目录下移走是不是就找不到了呢?是的。
$ mv /bin/ls /temp/ls_bak #测试完后记得改回来奥
现在再来执行ls命令看看:
$ ls
The program 'ls' is currently not installed. You can install it by typing:
apt install coreutils
没错,它会提示你没有安装这个程序或者命令没有找到。
所以你现在明白为什么你第一次安装jdk或者python的时候要设置环境变量了吧?不设置的话行不行?
行。这个时候你就需要指定路径了。怎么指定路径?无非就是那么几种,相对路径,绝对路径等等。
$ cd /temp
$ ./ls_bak
或者:
$ /temp/ls_bak
是不是发现和运行自己的普通程序方式没什么差别呢?
到这里,如果还没有找到你要执行的命令,那么就会报错。
确定解释程序
在找到程序之后呢,需要确定解释程序。什么意思呢?
而一旦发现要执行的程序文件是文本文件,且文本未指定解释程序,那么就会默认当成shell脚本来执行。例如,假设有test.txt内容如下:
echo -e "hello world"
赋予执行权限并执行:
$ chmod +x test.txt
$ ./test.txt
hello world
当然了,我们通常会在shell脚本程序的来头带上下面这句:
#!/bin/bash
这是告诉shell,你要用bash程序来解释执行test.txt。作为一位调皮的开发者,如果开头改成下面这样呢?
#!/usr/bin/python
再次执行之后结果如下:
$ ./test.txtFile "./test.txt", line 2echo -e "hello world"^
SyntaxError: invalid syntax
是的,它被当成python脚本来执行了,自然就会报错了。
那么如果是二进制程序呢?就会使用execl族函数去创建一个新的进程来运行新的程序了。
小结一下前面的内容,就是说,如果是文本程序,且开头没有指定解释程序,则按照shell脚本处理,如果指定了解释程序,则使用解释程序来解释运行;对于二进制程序,则直接创建新的进程即可。
运行
前面我们也已经看到了运行方式,设置环境变量或者使用相对路径,绝对路径即可。不过对于shell脚本,你还可以像下面这样执行:
$ sh test.txt
$ . test.txt
即便test.txt没有执行权限,也能够正常执行。
什么?你说为什么txt也能执行?注意,Linux下的文件后缀不过是为了方便识别文件类型罢了,以.txt结尾,并不代表一定是文本。当然在这里它确实是,而且还是ASCII text executable:
$ file test.txt
test.txt: Bourne-Again shell script, ASCII text executable
$ file hello
hello: ELF 64-bit LSB executable, x86-64, version 1 (SYSV), dynamically linked, interpreter /lib64/l, for GNU/Linux 2.6.32, BuildID[sha1]=8ae48f0f84912dec98511581c876aa042824efdb, not stripped
扩展一下
那么如果让我们自己的程序也能够像Linux内置命令一样输入即可被识别呢?
将程序放到PATH路径下
第一种方法就是将我们自己的程序放到PATH中的路径中去,这样在shell输入hello时,也能找到,例如我们将其放在/bin目录下:
$ hello
hello world
$ whereis hello
hello: /bin/hello
也就是说,如果你的程序安装在了PATH指定的路径,就需要配置PATH环境变量,在命令行输入就可以直接找到了。
设置PATH环境变量
那么如果想在指定的目录能够直接运行呢?很简单,那就是添加环境变量,例如将当前路径加入到PATH中:
$ PATH=$PATH:./ #这种方式只在当前shell有效,所有shell生效可修改/etc/profile文件
$ hello
hello world
设置别名
例如:
$ alias hello="/temp/hello"
$ hello
hello world
以上三种方法都可以达到目的。
执行顺序
那么假设我写了一个自己的printf程序,当执行printf的时候,到底执行的是哪一个呢?
$ type -a printf
printf is aliased to `printf "hello\n"'
printf is a shell builtin
printf is /usr/bin/printf
printf is ./printf
这里就可以很清楚地看到查找顺序了。也就是说,如果你输入printf,它执行的是:
$ printf
hello
而如果删除别名:
unalias printf
它执行的将会是内置命令printf。
总结
说到这里,想必标题的问题以及下面的问题你都清楚了:
安装Python或者Jdk程序为什么要设置PATH环境变量?如果不设置,该如何运行?
除了./方式运行自己的程序还有什么方式?
如果让自己的程序能够像内置命令一样被识别?
如何查看文件类型?
执行一条命令,如何确定是哪里的命令被执行
本文涉及命令:
mv 移动/重命名
file 查看文件信息
whereis 查看命令或者手册位置
type 查看命令类别
朕已阅
为什么执行自己的程序要在前面加./相关推荐
- @echo off是什么意思_为什么执行自己的程序要在前面加./
作者:守望,Linux应用开发者,目前在公众号[编程珠玑] 分享Linux/C/C++/数据结构与算法/工具等原创技术文章和学习资源. 前言 在Linux中,我们执行内置命令时,直接输入命令名称即可, ...
- python print(f)执行将报错_Linux中为什么执行自己的程序要在前面加./
来源:公众号[编程珠玑] 作者:守望先生 在Linux中,我们执行内置命令时,直接输入命令名称即可,如: $ mv a b #将a重命名为b 而在执行自己写好的程序时,却要带上./,例如: $ hel ...
- “adb不是内部或外部命令,也不是可执行的应用程序”错误原因及解决方法
用SQLite时,可能会出现这样的错误. 原因可能是环境变量PATH没有配置或配置不正确.应该把adb.exe 所在目录加入到PATH环境变量.例如:C:\Program Files\android- ...
- TensorFlow2020:如何使用Tensorflow.js执行计算机视觉应用程序?
本文转载自公众号"读芯术"(ID:AI_Discovery). 很多人都能运行操作计算机视觉应用程序.是的,学习并执行它并不难,现在有很多库可以用来执行如此强大的计算机视觉应用程序 ...
- mysql 安装问题一:由于找不到MSVCR120.dll,无法继续执行代码.重新安装程序可能会解决此问题。
mysql 安装问题一:由于找不到MSVCR120.dll,无法继续执行代码.重新安装程序可能会解决此问题. 参考文章: (1)mysql 安装问题一:由于找不到MSVCR120.dll,无法继续执行 ...
- Windows 技术篇-任务管理器查看执行进程的程序所在位置方法,查看执行进程的命令行
很多时候有多个进程的名字是一样的,但是想判断这些进程都是由那些程序起来的. 比如: 我一台服务器搭建了多个环境,每个环境对应一个进程,但是只想杀掉某个进程,但是他们的进程名都是一样的,我判断不了,这时 ...
- 执行了的程序,才是你的程序.
工作的人都知道一个公司的核心文化就是执行力,今天我们在来谈谈执行力的重要性.我见过很多人买了<乔布斯告诉你如何成功>.<卡耐基黄金50年>,一般都是看看他们有哪些成功秘诀,个人 ...
- 【Arduino】按键按下执行不同模式程序
程序可直接复制到 Arduino IDE 运行,不需要额外添加其他库文件. 按键原理图如下: 原理图所接按键与示例程序不一致,原理图演示的为 D2 引脚,程序为 A0 引脚. 电阻 R1 的阻值,示例 ...
- python可以用来编写计算机网络程序吗_不必熟悉python或R编程语言,6步执行计算机视觉应用程序...
全文共1260字,预计学习时长7分钟 图源:morish 很多人都能运行操作计算机视觉应用程序.是的,学习并执行它并不难,现在有很多库可以用来执行如此强大的计算机视觉应用程序. 你有没有关注最近有没有 ...
最新文章
- ibatis源码浅析- 初探
- C++中std::function和std::bind
- 解决Extjs中textarea不支持keyup事件的问题
- 由获取子元素的方法find和children所获
- sql between 效率高吗_整个SQL语句的执行效率都靠它了...
- 回溯(backtrack)
- Hive窗口函数保姆级教程
- 当当图书分类html,基于httpclient与jsoup的抓取当当图书页面数据简单Demo
- 如何在Microsoft Excel电子表格之间交叉引用单元格
- 椰树集团引爆互联网背后的“流量秘密”
- 复盘世预赛:国足5-0马尔代夫,吴曦成国足世预赛福星!三大前锋齐进球
- 东风快递地址分拣程序
- python医院排队系统_叫号系统排队系统挂号系统实现(JAVA队列)
- Java Web基础性知识
- mysql 查看time zone_MySQL查看和修改时区time_zone
- windows7英文版变成中文版教程
- 对话屏幕Dynpro(se51)
- julia语言和python_后起之秀!Julia胜于Python的5个优势
- 怎样在64位安卓系统中使用32位SO库 2
- 图像平移 ——MATLAB实现
热门文章
- 【Python】SyntaxError: unexpected EOF while parsing
- 【CTF】实验吧 Fair-Play
- 高级程序员到底高级在哪里?
- 智能驾驶L2的黄金时代,打磨地图是关键
- 32岁程序员,补偿N+2:“谢谢裁我,让我翻倍!” 网友:榜样!
- 毕业就拿阿里offer,你和他比差在哪?
- 真正的博士是如何参加AAAI, ICML, ICLR等AI顶会的?
- Python文本预处理:步骤、使用工具及示例
- AI一分钟 | 万达网科裁员95%高达5000余人,被爆下一步将转型AI; 英伟达放话了:研究人员放心用,不更新驱动就没啥事儿
- AI一分钟 | 今天,百度又多了一个好基友华为,还互赠了信物;腾讯AI Lab“肢体动作追踪”技术造出了个“AI 尬舞机”