当前位置: 首页 > news >正文

shell脚本对编码和行尾符敏感吗

问题

我正在 macOS 上制作一个 NW.js 应用程序,并想通过双击图标在开发模式下运行该应用程序。在第一步中,我试图使我的 shell 脚本正常工作。

在 Windows 上使用 VS Code,我在项目的根目录下创建了一个 run-nw 文件,包含以下内容:

#!/bin/bashcd "src"
npm installcd ..
./tools/nwjs-sdk-v0.17.3-osx-x64/nwjs.app/Contents/MacOS/nwjs "src" &

但是我得到了这个输出:

$ sh ./run-nw: command not found  
: No such file or directory  
: command not found  
: No such file or directory  Usage: npm <command>where <command> is one of:  (snip commands list)(snip npm help)npm@3.10.3 /usr/local/lib/node_modules/npm  
: command not found  
: No such file or directory  
: command not found

有些事情我不明白。

  • 它似乎将空行作为命令。在我的编辑器(VS Code)中,我尝试将 \r\n 替换为 \n (以防 \r 产生问题),但它没有改变什么。
  • 它似乎没有找到文件夹(有或没有 dirname 指令),或者可能它不知道 cd 命令?
  • 它似乎不理解 npminstall 参数。
  • 真正让我感到奇怪的是,它仍然运行应用程序(如果我手动执行 npm install)……

由于无法正常工作,并且怀疑文件本身有什么奇怪的地方,我直接在 Mac 上创建了一个新的文件,这次使用了 vim。我输入了完全相同的指令,然后…现在它工作起来没有任何问题。

diff 对比两个文件的差异显示完全没有差异。

有什么区别?是什么导致第一个脚本无法运行?我怎样才能知道?


回答

是的。Bash 脚本对行结束很敏感,无论是在脚本本身还是在它处理的数据中。它们应该有 Unix 风格的行结束符,即每行以换行符结束(ASCII 中的十进制 10,十六进制 0A)。

对于 Windows 或 DOS 风格的行尾,每一行都以回车符和换行符结束。你可以在命令 cat -v yourfile 的输出中看到这个不可见的字符:

$ cat -v yourfile
#!/bin/bash^M
^M
cd "src"^M
npm install^M
^M
cd ..^M
./tools/nwjs-sdk-v0.17.3-osx-x64/nwjs.app/Contents/MacOS/nwjs "src" &^M

在这种情况下,回车(插入符号中的 ^M 或 C 转义符号中的 \r)不会被视为空白。Bash 将 shebang 之后的第一行(由一个回车字符组成)解释为要运行的命令/程序的名称。

  • 因为没有名为 ^M 的命令,所以它输出: command not found
  • 因为没有名为 "src"^M(或 src^M)的目录,所以它输出: no such file or directory
  • 它将 install^M 而不是 install 作为参数传递给 npm,这会导致 npm 报错。

解决方案

解决方案是将文件转换为使用 Unix 风格的行结尾(将回车符删除)。有很多方法可以做到这一点:

  1. 最简单的方法是使用 dos2unix 命令
dos2unix filename
  1. 使用 sed 命令处理
sed -i 's/\r$//g' filename
#or
sed 's/\r$//g' filename > newfile
  1. 使用 vim 处理
#vi filename
:set fileformat=unix 
:w 
  1. 使用 tr 命令删除回车符
cat filename | tr -d '\r' > newfile

参考

  • stackoverflow question 39527571
  • man sed

相关阅读:

  • 在shell程序里如何从文件中获取第n行
  • 用Bash变量进行sed替换
  • 如何用命令行将文本每两行合并为一行?
  • 如何使用bash脚本并行运行多个程序
  • 为什么要使用xargs命令
  • xargs命令用法实例
http://www.lryc.cn/news/365926.html

相关文章:

  • 神经网络----现有网络的下载和使用(vgg16)
  • Java 异常处理 - 自定义异常
  • Excel 交叉表的格转成列,行转成格
  • 【C++软件调试技术】什么是pdb文件?如何使用pdb文件?哪些工具需要使用pdb文件?
  • 如何搭建一台永久运行的个人服务器?
  • Ant Design+react 表单只读
  • 推荐系统三十六式学习笔记:原理篇.近邻推荐07|人以群分,你是什么人就看到什么世界
  • 要改进单例模式的实现以确保线程安全并优化性能,有几种常见的方法
  • k8s——Pod容器中的存储方式及PV、PVC
  • Java/Golang:活用interface,增加程序扩展性
  • SQL语句练习每日5题(四)
  • Java排序算法汇总篇,八种排序算法
  • R语言探索与分析20-北京市气温预测分析
  • 2024年安全现状报告
  • OV通配符ssl证书是什么
  • Selenium三种等待方式的使用!
  • websockets怎么工作的呢?
  • 栈 数组和链表实现
  • 如何备份和恢复华为手机?
  • 微波电路S参数测量实验方案
  • SpringTask Cron表达式
  • docker与docker-compose安装
  • 跨境反向海淘系统:业务流程解析与未来发展展望
  • Python语言字母:深度解析与魅力探索
  • 基于JSP技术的社区疫情防控管理信息系统
  • 区间预测 | Matlab实现QRBiTCN分位数回归双向时间卷积神经网络注意力机制时序区间预测
  • Spring MVC中,一个HTTP请求可能会被多个Handler处理
  • Vue3 时间格式化
  • SVN中trunk,branch,tag
  • React ahooks库和React Query库使用场景分析