- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
平台:Windows
查询:http://gnuwin32.sourceforge.net/packages/grep.htm
python :2.7.2
用于执行命令的 Windows 命令提示符。
我正在文件中搜索以下模式 "2345$"
。文件内容如下:
abcd 2345
2345
abcd 2345$
grep "2345$"文件.txt
grep 成功返回 2 行(第一行和第二行)。
当我尝试通过 python 运行上述命令时,我没有看到任何输出。Python代码片段如下:
temp = open('file.txt', "r+")
grep_cmd = []
grep_cmd.extend([grep, '"2345$"' ,temp.name])
print grep_cmd
p = subprocess.Popen(grep_cmd,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE)
stdoutdata = p.communicate()[0]
print stdoutdata
如果我有
grep_cmd.extend([grep, '2345$' ,temp.name])
在我的 python 脚本中,我得到了正确的答案。
问题是为什么grep命令用"
grep_cmd.extend([grep, '"2345$"' ,temp.name])
从 python 执行失败。 python不应该执行吗命令原样。
谢谢古奇。
最佳答案
不要在您的模式周围加上双引号。只需要在命令行上引用 shell 元字符。从 python 调用程序时,不需要这个。
您也不需要自己打开文件 - grep 会这样做:
grep_cmd.extend([grep, '2345$', 'file.txt'])
要了解不需要双引号并导致命令失败的原因,您需要了解双引号的用途及其处理方式。
Shell 使用双引号来防止对某些 Shell 元字符进行特殊处理。 Shell 元字符是 Shell 专门处理的那些字符,不会按字面意思传递给它执行的程序。最常用的 shell 元字符是“空格”。 shell 在空间边界上拆分命令以构建参数向量来执行程序。如果要在参数中包含空格,则必须以某种方式将其引用(单引号或双引号、反斜杠等)。另一种是美元符号($),用于表示变量扩展。
当您在不涉及 shell 的情况下执行程序时,所有这些关于引用和 shell 元字符的规则都不相关。在 python 中,您自己构建参数向量,因此相关的引用规则是 python 引用规则(例如,要在双引号字符串中包含双引号,在双引号前加上反斜杠 - 反斜杠不会出现在最后字符串)。完成构造后,参数向量的每个元素中的字符是将传递给您正在执行的程序的文字字符。
Grep 不会将双引号视为特殊字符,因此如果 grep 在其搜索模式中获得双引号,它将尝试从其输入中匹配双引号。
我原来的答案对 shell=True
的引用是不正确的 - 首先我没有注意到你最初指定了 shell=True
,其次我是从这个角度来看的Unix/Linux 实现,而不是 Windows。
Python 子进程模块页面有关于 shell=True
和 Windows 的说法:
On Windows: the Popen class uses CreateProcess() to execute the child child program, which operates on strings. If args is a sequence, it will be converted to a string in a manner described in Converting an argument sequence to a string on Windows.
关于在 Windows 上将参数序列转换为字符串的链接部分对我来说没有意义。首先,字符串是一个序列,列表也是,但是“常用参数”部分对参数是这样说的:
args is required for all calls and should be a string, or a sequence of program arguments. Providing a sequence of arguments is generally preferred, as it allows the module to take care of any required escaping and quoting of arguments (e.g. to permit spaces in file names).
这与 Python 文档中描述的转换过程相矛盾,鉴于您观察到的行为,我会说文档是错误的,并且只应用于参数字符串,而不是参数向量。我自己无法验证这一点,因为我没有 Windows 或 Python 的源代码。
我怀疑如果你像这样调用 subprocess.Popen
:
p = subprocess.Popen(grep + ' "2345$" file.txt', stdout=..., shell_True)
您可能会发现双引号作为记录的参数转换的一部分被去除了。
关于python - 从 python 调用的 grep 命令,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9542414/
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 关闭 9 年前。 Improve this
我有一系列 SQL 命令,我想在大约 40 个不同的表上运行。必须有一种方法可以在不编写 40 条不同命令的情况下执行此操作... 我在 SQL Server 中运行它。所有表都有不同的名称,我要操作
我习惯在 PHP 中使用命令“mysql_insert_id()”来返回插入到我的数据库中的最后一行的 id。 在 C# 中的 SQLite 中是否有等效的命令? 谢谢! -阿德娜 最佳答案 选择 l
试图找出一种方法来回填 ds 分区 Hive 表的分区。 我知道如何从 CLI 运行 Hive 命令,例如 $HIVE_HOME/bin/hive -e 'select a.col from tab1
我有 .bat 文件。看起来像下一个 ....many commands1 ftp -i -s:copy.txt ...many commands2 copy.txt 包含下一个命令 open ...
基本上我想输入 show 并检查是否有 show 命令或别名已定义并触发它,如果未定义则触发 git show 。 例如 rm 应该执行 rm 但 checkout 应该执行 git checkout
我公司的主数据库是 iSeries 机器,我已经非常习惯使用 DB2 命令和结构。我现在正在尝试做一个小项目,更新一个包含超过 300 万条记录的表。我想出一种比较和“清理”数据的更快方法是使用 My
我想在带有 Node 的终端中制作一个简单的按钮板,并“blessed”用于连接或运行不同的命令。 ----------------------------------------------- _
我们有一个 selenium IDE 脚本,正在转换为 python webdriver。以下命令未转换: [openWindow | http://mywebsite.com/index.php |
我正在学习这个关于从 GIT HUB 下载和安装 Web 文件的在线教程。我进入主题:启动我们的静态网站,系统提示我输入命令以下载和安装 Web 文件。但是,当我输入命令 yarn install 时
我在 shell 脚本中使用 elif 命令时遇到问题,就像在 fortran 中一样。 我有 100 家公司的员工名单。我想屏蔽那些员工少于 500 人的公司。我的脚本是 rm -f categor
我有一些 Linux 命令可以生成 token 。我在 Linux 机器上使用操作系统库形式的 Python 自动化了这些命令。它工作正常。 但是,当我在 Windows 中尝试相同的代码时,它没有返
本文分享自华为云社区《Git你有可能不知道交互式暂存》,作者:龙哥手记。 本节中的几个交互式 Git 命令可以帮助你将文件的特定部分组合成提交。 当你在修改了大量文件后,希望这些改动能拆分为若干提交而
我想知道如何使用 IN 比较语法来做到这一点。 当前的 SQL 查询是: select * from employee where (employeeName = 'AJAY' and month(e
我在这个位置安装了 Hadoop /usr/local/hadoop$ 现在我想列出 dfs 中的文件。我使用的命令是: hduser@ubuntu:/usr/local/hadoop$ bin/ha
是否有一个单一的 docker 命令可用于清除所有内容?如果正在运行,请停止所有容器、删除所有图像、删除所有卷...等。 最佳答案 我认为没有一个命令可以做到这一点。您首先需要停止所有容器使用 $ d
我基本上是在 clojure/nrepl 模式中寻找与 C-u C-x C-e 或 C-c C-p 等效的 Scheme。 我想要一个 C-x C-e 将输出打印到缓冲区,而不是仅仅在 repl 中。
我可以在 vim 中使用 pudb(一个 ncurses Python 调试器),因为,例如,:!python %在实际的终端窗口中运行。我更喜欢使用 gvim,但 gvim 运行 :!python
我正在尝试编写一个 FFMPEG 命令: 取为 输入 一个视频 input.mp4 和一个图像 pic.jpg 作为 输出 将 input.mp4 拆分为 20 秒的视频,按顺序重命名;对于每个分割视
我想转储视频每帧的比特率。我正在尝试使用 -vstats 获取此信息命令。当我运行此命令时 - ffmpeg -i input.mp4 -vstats 它显示至少应该定义一个文件。 如果有人能建议我任
我是一名优秀的程序员,十分优秀!