- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
foobar
即使失败也可能会创建输出文件,因此在这种情况下我需要将其删除。
我可以做到这一点:
foo: bar baz
foobar $^ -o $@ || (rm -f $@ && exit 1)
但这不会传播 foobar
返回的相同退出代码(然后由 make
输出)。有没有办法在 Makefile 中而不是在 shell 中捕获错误?
最佳答案
万一DELETE_ON_ERROR
并没有削减它,你所看到的有点像 tearDown
, @After
,或finally
在 Java/JUnit 中,您可以执行以下操作:
.ONESHELL:
在单个 shell 中执行所有 shell 命令。trap
对于 EXIT
完成清理工作。errexit
确保 shell 在出现任何错误时退出。当我们这样做时,我们还可以设置 pipefail
马上。例如,假设您要启动一个 docker 容器,进行测试,无论如何停止 docker 容器,但获取测试结果。操作方法如下:
export SHELL:=/bin/bash
export SHELLOPTS:=$(if $(SHELLOPTS),$(SHELLOPTS):)pipefail:errexit
.ONESHELL:
.PHONY: test
test:
function tearDown {
docker stop test-image
}
trap tearDown EXIT
docker run --name test-image …
testStep1…
testStep2…
testStep3…
…
export SHELL
export 告诉 GNU make 使用 bash
作为 shell,其占用空间比默认的 sh
更重但功能更多。export SHELLOPTS
设置pipefail
和errexit
bash
的标志壳。
pipefail
确保管道的退出状态不是最后一个命令,而是最后一个非零退出状态。所以,false | true
将返回1
而不是0
.errexit
确保命令序列的退出状态不是最后一个命令,而是最后一个非零退出状态,并且后续命令不会被执行。所以,false ; true
将返回1
而不是0
和true
不会被执行。.ONESHELL:
告诉 GNU make 在单个 shell 中运行所有命令。这意味着,您的食谱现在实际上是一个 shell 脚本。 (需要 GNU make 3.82 或更高版本。)function tearDown { docker stop test-image }
定义了一个名为 tearDown
的 shell 函数。在此示例中,它将停止 docker 容器。trap tearDown EXIT
是此示例中所有内容中最关键的部分。它告诉为配方调用的 shell 运行 tearDown
退出时起作用,即无论命令成功还是失败。这类似于 finally
在 java 。不可能跨多个目标/测试重用。绝对不是@AfterClass
这样的/@AfterAll
或tearDown()
/@After
/@AfterEach
在 JUnit 中。
但是如果您需要的话,您可以这样做。比如说,您想在同一个 docker 容器上运行多个测试,并无论如何将其拆除。这类似于 @AfterClass
/@AfterAll
在 JUnit 中。那么它可能看起来像这样:
export SHELL:=/bin/bash
export SHELLOPTS:=$(if $(SHELLOPTS),$(SHELLOPTS):)pipefail:errexit
.ONESHELL:
.PHONY: start
start:
docker run --name test-image …
.PHONY: stop
stop:
docker stop test-image
.PHONY: test
test: start
function tearDown {
$(MAKE) stop
}
trap tearDown EXIT
$(MAKE) -k testImpl
.PHONY: testImpl
testImpl: testCase1 testCase2 testCase3
.PHONY: testCase1
testCase1:
…
.PHONY: testCase2
testCase2:
…
.PHONY: testCase3
testCase3:
…
现在将运行所有测试,即使第一个测试失败,也会在所有测试完成后进行清理,并在任何测试失败时报告错误。
免责声明:这需要 .ONESHELL
GNU make 的功能,在 GNU make 3.82 中引入。截至本次编辑,GNU make 的当前版本是 GNU make 4.2.1,Mac OS X 仍然附带 GNU make 3.81。
关于makefile - Makefile 中出现错误后如何清除?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28597794/
我有一个 Makefile,它针对特定目标调用另一个 Makefile。假设主 Makefile 包含 some_dir/some_target: cd some_dir && make so
这两个文件大多出现在开源项目中。 它们的用途是什么?它们如何工作? 最佳答案 Makefile.am 是程序员定义的文件,由 automake 使用来生成 Makefile.in 文件( .am 代表
我的源代码位于一堆子目录中,例如: src/widgets/apple.cpp src/widgets/knob.cpp src/tests/blend.cpp src/ui/flash.cpp 在项
这就是我所拥有的: SUBDIRS = src/lib/ResourceManager all: $(SUBDIRS) $(SUBDIRS): make install -C $@ 我正在尝试
我想写一个 Makefile 来执行来自两个不同数组的两个输入的命令例如 a = A B C b = 1 2 3 ./run A 1 ./run B 2 ./run C 3 我不知道怎么写,因为在Ma
在 GNU make 手册的早期部分之一,Section 3.7 , 有一个 makefile 配方的大纲 immediate : immediate ; deferred defer
是否存在将 gmake 的 GNU Makefile 转换为可用于 make (FreeBSD-make) 的 Makefile 的实用程序? 最佳答案 该实用程序称为开发人员(程序员,制作大师,..
所以我前段时间了解了什么是 Makefile,创建了一个模板 Makefile,我所做的就是为我正在执行的每个程序复制和更改相同的文件。我改了几次,但它仍然是一个非常粗糙的Makefile。我应该如何
我正在做一些 Makefile 重构,并试图找出最简洁的方法来实现一个 Makefile,它执行以下操作: 有一个变量列出了所有源文件(可以是 C 和 C++ 文件) 所有目标文件都在 OBJ_DIR
我正在尝试创建一个 Makefile,它将通过 tic 编译位于目录中的 terminfo 文件。 tic 还将它自动创建的 termcap 文件复制到系统或用户特定的目标文件夹。对于普通用户,如果
我想要类似的东西 BROKEN_THINGS = \ thing1 \ # thing1 is completely broken thing2 \ # thing2 is broken to
如果我的程序必须为不同的结果(主要是错误)返回不同的值(例如 0、1、2、3 等),则调用该程序的 makefile 将不得不停止执行其余的 makefile 命令。即使该命令产生错误(返回非零值),
我正在学习使用漂亮的 Linux 工具:make。还有一点我想了解的: 让我们看一下这个简单的例子: JADE = $(shell find pages/*.jade) HTML = $(JADE:.
假设您有一个包含两个伪目标“all”和“debug”的 Makefile。 'debug' 目标旨在构建与 'all' 相同的项目,除了一些不同的编译开关(例如 -ggdb)。由于目标使用不同的编译开
我有一个调用多个其他生成文件的生成文件。 我想将 -j 参数传递给其他 makefile 调用。 类似(make -j8): all: make -f libpng_linux.mk -j
我处理过的 Makefile 大部分都很复杂,并且隐藏了很多关系。我自己从来没有写过一个,想知道是否有人有一些关于编写易于阅读和可重用的 Makefile 的提示? 最佳答案 我通常使用这样的东西,在
嘿,我有一个简单的“主” Makefile,它只是调用其他 makefile。我正在尝试执行以下操作,以便以正确的顺序构建组件: LIB_A = folder_a LIB_B = folder_b L
生成文件: #there is a whitespace after "/my/path/to" FOO = "/my/path/to" BAR = "dir" INCLUDE_DIRS = $(FO
我正在学习 makefile,我知道如何创建一个简单的 makefile。我正在继续使用嵌套的 makefile。这是我的目录结构 /src ...makefile ...main.cpp ...fo
什么TEMP0_FILES下面计算到? SOURCE_FILES可以等于多个源文件。请告诉我以下语法 :.cpp=.o 的用途 SOURCE_FILES = main.cpp TEMP0_FILES
我是一名优秀的程序员,十分优秀!