深入理解 source 和 sh、bash 的区别

1 引言

在日常使用 Linux 的过程中,脚本的执行是不可避免的需求之一,而 sourceshbash 等命令则是执行脚本的常用方式。尽管这些命令都能运行脚本,但它们之间的执行方式和效果却有着显著的区别。这些区别可能会影响到脚本的环境变量、工作目录、甚至当前 shell 的状态。因此,理解 sourceshbash 等命令在执行脚本时的差异,对于有效管理和维护系统环境、编写灵活的自动化脚本非常重要。

在本文中,我们将深入探讨 sourceshbash 的区别,包括它们的执行环境、变量传递、输出行为等方面。通过了解这些区别,你将能够更合理地选择适合的命令来执行脚本,从而在实际工作中更加高效地管理 Linux 系统环境。


2 执行环境与变量传递的差异

在 Linux 中使用 sourceshbash 等命令执行脚本时,执行环境的不同导致了变量和环境传递方式的差异。这些差异影响到脚本中的变量、环境设置等在当前 shell 中的作用。以下是它们的主要区别。

2.1 执行环境的差异

  • source 命令

    • source(或 .)是在当前 shell 环境中执行脚本内容。脚本中的所有变量定义、函数、环境变量修改等都会在当前 shell 中生效,并且在脚本执行完毕后依然保留在当前环境中。
    • 这种执行方式相当于在当前 shell 中逐行执行脚本内容,因此会直接影响当前 shell 的状态。
  • shbash 命令

    • 使用 sh script.shbash script.sh 执行脚本时,系统会启动一个新的子 shell 来运行脚本。在子 shell 中执行的所有变量和环境修改仅在子 shell 内生效,不会影响当前的父 shell。
    • 当子 shell 执行完毕后,它会关闭,所有在脚本中创建或修改的变量和环境都会随之销毁。

2.2 变量和环境的传递差异

由于执行环境的不同,sourceshbash 在变量和环境的传递方面也有差异:

  • source 命令

    • 在当前 shell 中执行脚本,脚本中的变量和环境设置会直接在当前 shell 中生效。这意味着使用 source 命令可以将脚本中的变量和环境配置保留在当前 shell 中,影响后续的操作。
    • 例如,通过 source 命令执行脚本后,定义的变量或改变的工作目录会保留在当前会话中。
  • shbash 命令

    • 在子 shell 中执行脚本,脚本中的变量和环境仅在子 shell 内部生效,无法传递回当前 shell。
    • 只有提前使用 export 设置为环境变量的值,才会从父 shell 传递给子 shell,普通变量无法传递。这意味着在子 shell 中执行的脚本对当前 shell 没有影响,适合独立运行的任务或一次性任务。

2.3 示例对比

假设有一个脚本 script.sh,内容如下:

# script.sh
VAR="Hello, World"
export ENV_VAR="Exported Variable"
cd /tmp
  • 使用 source 执行
    Pasted image 20241115110351

    在这种情况下,source 会将 VARENV_VAR 变量直接传递到当前 shell 中,并改变了当前目录。执行后,这些更改会保留在当前 shell 中,影响后续的操作。

  • 使用 bash 执行
    Pasted image 20241115110433

    使用 bash 启动子 shell 执行脚本,脚本中的 VARENV_VAR 变量及目录更改都仅在子 shell 中生效,不会影响当前的父 shell。执行结束后,变量在当前 shell 中不可访问,目录也未发生变化。

2.4 总结

  • source 在当前 shell 中执行脚本,直接影响当前 shell 的变量和环境设置。
  • shbash 在子 shell 中执行脚本,不会影响当前 shell,只有 export 的环境变量会传递到子 shell。

3 输出和影响范围的差异

在使用 sourceshbash 执行脚本时,输出行为和影响范围也存在差异。这种差异主要体现在脚本的输出显示、错误处理以及对当前 shell 的持久影响方面。

3.1 输出的显示

  • source 命令

    • source 命令在当前 shell 中直接执行脚本内容,因此脚本中的所有输出会立即在当前 shell 的终端窗口中显示,就像你手动输入这些命令一样。
    • 如果脚本包含 echoprintf 等输出命令,它们的输出会直接显示在当前 shell 中。并且因为是在当前 shell 中运行,所有输出和执行步骤都可以在当前 shell 直接看到和调试。
  • shbash 命令

    • 当使用 sh script.shbash script.sh 执行脚本时,输出同样会显示在当前终端窗口中,但因为是在子 shell 中执行,它与当前 shell 相对隔离。
    • 任何标准输出(stdout)和标准错误输出(stderr)都会在子 shell 中产生,但仍会显示在当前的终端上。这种方式的输出行为与 source 看起来类似,但在调试和环境影响上不同。

3.2 对当前 shell 的影响

  • source 命令

    • 如果脚本中包含 exit 命令,执行 source 会导致当前 shell 直接退出。因此在使用 source 时要谨慎处理 exit 等可能影响当前 shell 的指令。
  • shbash 命令

    • 如果脚本中包含 exit 命令,只会导致子 shell 退出,不会影响当前的父 shell。

3.3 错误处理的影响

  • source 命令

    • 由于 source 在当前 shell 中运行,脚本中的错误会直接影响当前 shell。如果脚本出错,可能导致当前 shell 产生意外行为,甚至需要手动干预才能恢复正常状态。
    • 脚本中的错误会直接在当前 shell 中显示出来,可以立即调试和修复。
  • shbash 命令

    • 使用 shbash 执行脚本时,错误仅在子 shell 中发生,不会对当前 shell 产生直接影响。
    • 如果脚本运行失败,子 shell 会处理错误并退出,但当前 shell 不会受到影响。这种方式适合执行不希望影响当前 shell 的任务,特别是那些可能失败或包含危险命令的脚本。

3.4 小结

  • source 会在当前 shell 中直接显示脚本输出,所有环境更改和错误直接影响当前 shell。
  • shbash 在子 shell 中执行,输出显示在当前终端,但对当前 shell 没有环境影响,错误也仅在子 shell 内部处理。

4 使用场景与总结

在实际工作中,选择 sourceshbash 来执行脚本应视具体需求而定。理解它们在执行环境、变量传递、输出和影响范围上的差异,有助于我们在不同场景中做出更合适的选择。

4.1 使用 source 的场景

当需要在当前 shell 中定义变量、加载环境设置、或执行会影响当前 shell 的配置脚本时,source 是更好的选择。比如,加载环境配置文件(如 source ~/.bashrc),或者定义会在当前会话中持续使用的变量和函数。因为 source 会直接在当前 shell 中执行脚本内容,所有环境变量和配置修改会对后续命令产生直接影响。

4.2 使用 shbash 的场景

当希望脚本执行的内容与当前 shell 相对隔离,或者不希望它修改当前环境时,shbash 更为合适。执行独立的自动化任务、批处理脚本,或可能出错而不希望影响当前 shell 的任务时,可以使用 sh script.shbash script.sh。这些命令会在子 shell 中运行脚本,确保当前 shell 的环境不会被改变,适合一次性任务或后台运行的脚本。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/917077.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

LogViewer NLog, Log4Net, Log4j 文本日志可视化

LogViewer 下载 示例&#xff1a;NLog文本日志可视化软件&#xff0c;并且能够实时监听输出最新的日志 nlog.config 通过udp方式传输给LogViewer (udp://ip:port) <?xml version"1.0" encoding"utf-8" ?> <nlog xmlns"http://www.nlog-…

探索 HTML 和 CSS 实现的蜡烛火焰

效果演示 这段代码是一个模拟蜡烛火焰的HTML和CSS代码。它创建了一个具有动态效果的蜡烛火焰动画&#xff0c;包括火焰的摆动、伸缩和光晕的闪烁。 HTML <div class"holder"><div class"candle"><div class"blinking-glow"&g…

vue3【实战】切换全屏【组件封装】FullScreen.vue

效果预览 原理解析 使用 vueUse 里的 useFullscreen() 实现 代码实现 技术方案 vue3 vite UnoCSS vueUse 组件封装 src/components/FullScreen.vue <template><component:is"tag"click"toggle":class"[!isFullscreen ? i-ep:full-sc…

GPT-5 要来了:抢先了解其创新突破

Microsoft 的工程师计划于 2024 年 11 月在 Azure 上部署 Orion (GPT-5)。虽然这一版本不会向公众开放&#xff0c;但其上线被视为人工智能领域的一个重要里程碑&#xff0c;并将产生深远的影响。 文章目录 GPT-5 真的要来了GPT-4 的局限性GPT-5 的创新突破与遗留挑战GPT-5 预期…

C++【深入项目-检测键盘】

神马是检测键盘&#xff0c;就是让编辑器可以检测键盘按下了什么按键&#xff0c;我们先科普复习检测键盘 。 检测键盘需要用到一些函数&#xff0c;请见下&#xff1a; ! KEY_DOWN( 80 ) 这个代码是检测按下键盘上P按键。那80是什么&#xff1f;原来是对应按键的&#xff0…

springboot的依赖实现原理:spring-boot-starter-parent解析

01 dependencyManagement的作用 在使用springboot时我们会在项目pom引入以下配置和依赖 <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.7.18</version> &l…

Thinkphp6视图介绍

一.MVC MVC 软件系统分为三个基本部分&#xff1a;模型&#xff08;Model&#xff09;、视图&#xff08;View&#xff09;和控制器&#xff08;Controller&#xff09; ThinkPHP6 是一个典型的 MVC 架构 控制器—控制器&#xff0c;用于将用户请求转发给相应的Model进行处理&a…

【idea】更换快捷键

因为个人习惯问题需要把快捷键替换一下。我喜欢用CTRLD删除一下&#xff0c;用CTRLY复制一样。恰好这两个快捷键需要互换一下。 打开file——>setting——>Keymap——>Edit Actions 找到CTRLY并且把它删除 找到CTRLD 并且把它删除 鼠标右键添加CTRLY 同样操作在Delet…

Tiktok对接和内容发布申请流程

这段时间在搞AI生成视频&#xff0c;希望用户能一键发布到Tiktok&#xff0c;因此研究了一下Tiktok的开发者申请流程&#xff0c;发现好复杂&#xff0c;同时也发现Tiktok的开发也跟我一样&#xff0c;挺草台班子的 0、流程简述 废话不多说&#xff0c;Tiktok的开发者申请和…

[刷题]入门1.矩阵转置

博客主页&#xff1a;算法歌者本篇专栏&#xff1a;[刷题]您的支持&#xff0c;是我的创作动力。 文章目录 1、题目2、基础3、思路4、结果 1、题目 链接&#xff1a;洛谷-B2106-矩阵转置 2、基础 此题目主要考察二维数组的掌控能力。 3、思路 观察&#xff0c;可知&#…

ODC 如何精确呈现SQL耗时 | OceanBase 开发者工具解析

前言 在程序员或DBA的日常工作中&#xff0c;编写并执行SQL语句如同日常饮食中的一餐一饭&#xff0c;再寻常不过。然而&#xff0c;在使用命令行或黑屏客户端处理SQL时&#xff0c;常会遇到编写难、错误排查缓慢以及查询结果可读性不佳等难题&#xff0c;因此&#xff0c;图形…

大数据学习15之Scala集合与泛型

1. 概述 大部分编程语言都提供了数据结构对应的编程库&#xff0c;并称之为集合库(Collection Library)&#xff0c;Scala 也不例外&#xff0c;且它还拥有以下优点&#xff1a; 易用&#xff1a;灵活组合运用集合库提供的方法&#xff0c;可以解决大部分集合问题 简洁&#xf…

Linux网络——网络初识

目录 1. 认识协议 2. 协议的分层 3. OSI 七层模型 && TCP/IP 五层(四层)模型 4. 网络传输的基本流程 5. 以太网的通信原理 6. 数据的跨网络传播 7. 认识 IP 地址 ① IP 是什么 ② IP 与 MAC 的关系 ③ 为什么需要 IP 在谈及网络之前&#xff0c;我们要先对学…

数字IC后端低功耗设计实现案例分享(3个power domain,2个voltage domain)

下图所示为咱们社区T12nm A55低功耗实现项目。其实这个项目还可以根据产品的需求做一些改进。改进后项目实现的难度会大大增加。也希望通过今天的这个项目案例分享&#xff0c;帮助到今年IC秋招的同学。 芯片低功耗设计实现upf编写指南&#xff08;附低功耗项目案例&#xff0…

Ubuntu从入门到精通(一)系统安装

Ubuntu从入门到精通&#xff08;一&#xff09; 1 Ubuntu镜像选择 下载Ubuntu 20.04系统ISO镜像 安装 Ubuntu 20.04系统,就必须有 Ubuntu 20.04系统软件安装程序可以通过浏览器访问Ubuntu20.04的官方站点&#xff0c; 然后在导舰栏找划 Dowwnloads->Mirrors链接&#xff…

说说软件工程中的“协程”

在软件工程中&#xff0c;协程&#xff08;coroutine&#xff09;是一种程序运行的方式&#xff0c;可以理解成“协作的线程”或“协作的函数”。以下是对协程的详细解释&#xff1a; 一、协程的基本概念 定义&#xff1a;协程是一组序列化的子过程&#xff0c;用户能像指挥家…

【linux】进程等待与进程替换

&#x1f525;个人主页&#xff1a;Quitecoder &#x1f525;专栏&#xff1a;linux笔记仓 目录 01.进程等待系统调用获取子进程status常用宏使用示例 02.进程替换替换函数关键点解释&#xff1a;代码详细分析execvpe 函数的使用 01.进程等待 任何子进程&#xff0c;在退出的…

认证鉴权框架SpringSecurity-5--权限管理篇

上面两篇我们重点介绍了如何在代码上集成springSecurity&#xff0c;同时完成登录认证和token认证的过程。我们直到springSecurity处理能帮我们完成认证外&#xff0c;还可以帮助我们完成权限校验的工作&#xff0c;这篇我们来重点介绍下springSecurity是如何实现鉴权的。 一、…

RK3588开发板Android12-SDK更新通知

迅为RK3588开发板Android12 SDK升级至RK的android-12.1-mid-rkr14版本 内核版本&#xff1a;升级至 5.10.160 版本&#xff0c;提供更好兼容性和性能。 rkbin 版本&#xff1a;支持最新的 1.17 版本 bin 和 1.46 版本的 bl31。

stm32教程:OLED屏显示字母、汉字、图片工程讲解

早上好啊&#xff0c;大佬们&#xff0c;今天带来的是我们 stm32系列的第一个外设——OLED&#xff0c;相信大家对于OLED都不陌生了吧&#xff0c;这个可以说每一个项目里的必需品了&#xff0c;单片机离不开OLED就像西方离不开耶路撒冷。 在生活中&#xff0c;我们见到的OLED的…