来源:公众号【编程珠玑】

作者:守望先生

在Linux中,我们执行内置命令时,直接输入命令名称即可,如:

$ mv a b #将a重命名为b

而在执行自己写好的程序时,却要带上./,例如:

$ hellohello: command not found$ ./hellohello world

这是为什么呢?它们有什么区别呢?

shell是如何运行程序的

在说明清楚问题之前,我们必须了解shell是如何运行程序的。首先我们必须要清楚的是,执行一条Linux命令,本质是在运行一个程序,如执行ls命令,它执行的是ls程序。那么在shell中输入一条命令,到底发生了什么?它会经历哪几个查找过程?

alias中查找

alias命令可用来设置命令别名,而单独输入alias可以查看到已设置的别名:

$ aliasalias 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 echoecho 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 lsls: /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通常可以执行两种程序,一种是二进制程序,一种是脚本程序。

而一旦发现要执行的程序文件是文本文件,且文本未指定解释程序,那么就会默认当成shell脚本来执行。例如,假设有test.txt内容如下:

echo -e "hello world"

赋予执行权限并执行:

$ chmod +x test.txt$ ./test.txthello world

当然了,我们通常会在shell脚本程序的来头带上下面这句:

#!/bin/bash

这是告诉shell,你要用bash程序来解释执行test.txt。作为一位调皮的开发者,如果开头改成下面这样呢?

#!/usr/bin/python

再次执行之后结果如下:

$ ./test.txt  File "./test.txt", line 2    echo -e "hello world"                        ^SyntaxError: invalid syntax

是的,它被当成python脚本来执行了,自然就会报错了。

那么如果是二进制程序呢?就会使用execl族函数去创建一个新的进程来运行新的程序了。

小结一下前面的内容,就是说,如果是文本程序,且开头没有指定解释程序,则按照shell脚本处理,如果指定了解释程序,则使用解释程序来解释运行;对于二进制程序,则直接创建新的进程即可。

来源:公众号【编程珠玑】
id:shouwangxiansheng

运行

前面我们也已经看到了运行方式,设置环境变量或者使用相对路径,绝对路径即可。不过对于shell脚本,你还可以像下面这样执行:

$ sh test.txt$ . test.txt  

即便test.txt没有执行权限,也能够正常执行。

什么?你说为什么txt也能执行?注意,Linux下的文件后缀不过是为了方便识别文件类型罢了,以.txt结尾,并不代表一定是文本。当然在这里它确实是,而且还是ASCII text executable:

$ file test.txttest.txt: Bourne-Again shell script, ASCII text executable$ file hellohello: 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目录下:

$ hellohello world$ whereis hellohello: /bin/hello

也就是说,如果你的程序安装在了PATH指定的路径,就需要配置PATH环境变量,在命令行输入就可以直接找到了。

设置PATH环境变量

那么如果想在指定的目录能够直接运行呢?很简单,那就是添加环境变量,例如将当前路径加入到PATH中:

$ PATH=$PATH:./   #这种方式只在当前shell有效,所有shell生效可修改/etc/profile文件$ hellohello world

设置别名

例如:

$ alias hello="/temp/hello"$ hellohello world

以上三种方法都可以达到目的。

执行顺序

那么假设我写了一个自己的printf程序,当执行printf的时候,到底执行的是哪一个呢?
实际上它的查找顺序可以可以通过type -a来查看:

$ type -a printfprintf is aliased to `printf "hello\n"'printf is a shell builtinprintf is /usr/bin/printfprintf is ./printf

这里就可以很清楚地看到查找顺序了。也就是说,如果你输入printf,它执行的是:

$ printfhello

而如果删除别名:

unalias printf

它执行的将会是内置命令printf。
以此类推。

总结

说到这里,想必标题的问题以及下面的问题你都清楚了:

  • 安装Python或者Jdk程序为什么要设置PATH环境变量?如果不设置,该如何运行?

  • 除了./方式运行自己的程序还有什么方式?

  • 如果让自己的程序能够像内置命令一样被识别?

  • 如何查看文件类型?

  • 执行一条命令,如何确定是哪里的命令被执行

本文涉及命令:

  • mv 移动/重命名

  • file 查看文件信息

  • whereis 查看命令或者手册位置

  • type 查看命令类别

推荐阅读

leetcode 刷500道题,笔试/面试稳吗?谈谈算法的学习

【吐血整理】那些让你起飞的计算机基础知识:学什么,怎么学?

历经两个月,我的秋招之路结束了!

算法数据结构中有哪些奇技淫巧?

python print(f)执行将报错_Linux中为什么执行自己的程序要在前面加./相关推荐

  1. linux 查看硬盘报错_linux中挂载硬盘报错(you must specify the filesystem type)

    linux中挂载硬盘报错提示 you must specify the filesystem type ,意思是告诉我们必须要指定一个系统分区了,由于自己能看懂几句英文,所以解决起来也方便了,下面一起 ...

  2. linux下安装python3报错_Linux中安装python3

    (一)Linux中安装python3(含pip3自动安装) 1.首先安装依赖包 yum -y groupinstall "Development tools" yum -y ins ...

  3. linux句柄数不足的java报错_linux中文件句柄数问题

    问题描述: 有时候业务比较繁忙时,就会出现如下问题 too many open files:顾名思义即打开过多文件数.不过这里的files不单是文件的意思,也包括打开的通讯链接(比如socket),正 ...

  4. Ubuntu执行脚本报错-bash: ./send.py: /usr/bin/python: bad interpreter: Permission denied

    在Ubuntu系统终端中,使用命令行如下命令行运行*.py文件: 执行脚本报错 root@ubuntu:/usr/bin/python# chmod a+x send.py root@ubuntu:/ ...

  5. python之异常处理方式,报错怎么办,别慌

    文章目录 异常是什么 处理异常 异常的传播 异常捕获try ... except ... 异常捕获之锁定异常类型 Exception 异常名 else finally 完整异常捕获语法: 报错锦集参考 ...

  6. python 3.8.0 编译报错 Could not import runpy module 解决方案

    环境 CentOS 7 Gcc 4.8.5 Python 3.8.0 相关报错 make build_all CFLAGS_NODIST=" -fprofile-use -fprofile- ...

  7. windows下 python 使用 pip 安装TA-Lib报错的原因及解决方法

    windows下 python 使用 pip 安装TA-Lib报错的原因及解决方法 问题环境 问题描述 问题原因 & 解决办法 问题环境 Windows 10 x64 .python 3.8. ...

  8. 安装oracle 19c rac报错:2节点执行root.sh asm实例启动失败

    安装oracle 19c rac报错:2节点执行root.sh asm实例启动失败 背景 解决过程 查看lmon trc 查看mos 真的是网络的问题 haip 禁用haip 安装好的环境禁用haip ...

  9. Python——CSV读取大文件报错_csv.Error: field larger than field limit

    Python--CSV读取大文件报错_csv.Error: field larger than field limit 使用Python读取较大的csv文件时,可能出现大字段,导致超过字段默认限制,从 ...

最新文章

  1. python画三维几何图-Python常见几何图形绘制
  2. Android---手动创建线程与GUI线程同步(一)
  3. 有向图强连通分量SCC(全网最好理解)
  4. 关闭VS警告#pragma warning(disable:4996)
  5. WebView退出时停止视频播放
  6. python笔记(五) - 获取对象的引用
  7. How to install JDK on Linux
  8. Flutter 列表踩坑
  9. 解决纵向滚屏导致的轮播图异常
  10. 【论文笔记】Depth Map Prediction from a Single Image using a Multi-Scale Deep Network
  11. 关于google拼音输入法的坑爹问题-IE浏览器浏览网页蓝屏等问题
  12. mysql有mdf文件和ldf文件吗_mdf与ldf文件格式
  13. STM32CubeMX全部版本下载地址
  14. 【算法】搜索算法—盲目搜索和启发式搜索
  15. 与活体检测技术结合的人脸识别技术
  16. mybatis嵌套查询
  17. leetcode 1658
  18. ab间奇数的和c语言,小学五年级奥数奇数与偶数(AB)试题
  19. 姿态传感器DMS05检测出现异常
  20. 小狼毫 Rime 输入方案 设置

热门文章

  1. 小程序开发 宽度100%_这是您作为开发人员可以实现100%年度目标的方式
  2. Java 环形缓冲器(Ring Buffer)
  3. linux容器安卓下载,Docker 1.7.0 发布下载,Linux 容器引擎
  4. css3 下边框缓缓划过_CSS3 框大小
  5. Python3 装饰器解析
  6. Pycharm使用秘籍
  7. python如何操作excel 基础代码
  8. Python 3.8 已发布,现在是切换至新版本的好时机吗?
  9. 动态HTML处理和机器图像识别
  10. LeetCode题 - 1 两数之和