- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
所以我运行 make lex
并生成了 lex.yy.c
文件,一切正常
然后我运行 make scanner
,它将一个名为 scanner.c
的源文件编译,它应该简单地运行 cc lex.yy.c scanner.c -o 扫描仪
,而是这样做:
lex -t scanner.l > scanner.c
cc lex.yy.c scanner.c -o scanner
为什么它决定运行 lex -t scanner.l
并将其输出到 scanner.c
有效地覆盖我的代码?这该死的想法,这让我发疯。
我的生成文件:
scanner: scanner.h scanner.c lex.yy.c
cc lex.yy.c scanner.c -o scanner
lex: scanner.l
lex scanner.l > lex.yy.c
clean:
rm lex.yy.c
rm scanner
出了什么问题?
最佳答案
Why does it decide to run lex -t scanner.l and output it to scanner.c effectively overwritting my code?
无论何时发生这种情况,您的构建目录中都会有一个 scanner.c
和一个scanner.l
比 scanner.c
更新
当你运行 make scanner
, 配方:
scanner: scanner.h scanner.c lex.yy.c
cc lex.yy.c scanner.c -o scanner
要求其先决条件 scanner.c
应更新。你没有提供这样做的配方,所以 make falls back在其内置食谱数据库中。
通过运行 make -p
检查这些内置配方你会发现:
%.c: %.l
# recipe to execute (built-in):
@$(RM) $@
$(LEX.l) $< > $@
此内置配方将制作 file.c
来自匹配file.l
通过运行:
rm file.c # Not echoed
lex -t file.l > file.c
Make 发现此食谱的模式规则 - %.c: %.l
- 满意通过 scanner.c
,因为 scanner.l
存在并且比 scanner.c
更近.所以它使用这个配方来制作scanner.c
最新:
lex -t scanner.l > scanner.c
从而破坏你的scanner.c
.
如果你不想让 make 永远应用这个内置配方,你可以明确地通过只写规则来取消它:
%.c: %.l
在您的 makefile 中没有任何配方。
您还可以通过传递 --no-builtin-rules
来禁用所有 内置食谱在制作命令行。
但是,只要您对 makefile 的行为有期望被内置食谱破坏,这强烈表明您的期望不了解从输入文件制作输出文件的常用方法使用您的 makefile 调用的工具。品牌 Catalogue of Built-In Rules :
%.<target-type>: %.<prereq-type>
<command>
...
体现制作 <target-type>
的规范方式来自 <prereq-type>
的文件用 make 归档,这样你就不必自己写这个食谱了你的生成文件。例如,能够胜任 GNU make 的 C 和 C++ 程序员不要写食谱来制作.o
来自 .c
的文件文件或 .cpp
文件,除了极端情况,因为他们知道 make 会自动执行通常是他们想要的方式。
Make 的内置配方 %.c: %.l
规则表达的规范方式制作 file.c
给出 file.l
.所以问问自己:如果你不想要scanner.c
从 scanner.l
开始制作, 如果你想要 lex.yy.c
到由 scanner.l
制成,对于您调用的文件是否必要或有用 scanner.l
被称为,当你也有一个相当独立的源文件称为 scanner.c
?
假设您像这个玩具示例一样利用了 make 的内置食谱:
词法分析器.l
%{
#include <stdio.h>
%}
%%
[ \t] ;
[0-9]+\.[0-9]+ { printf("Found a floating-point number: [%s]\n",yytext); }
[0-9]+ { printf("Found an integer: [%s]\n",yytext); }
[a-zA-Z0-9]+ { printf("Found a string: [%s]\n",yytext); }
%%
扫描仪.c
#include "scanner.h"
int main(void) {
yylex();
return 0;
}
扫描仪.h
#ifndef SCANNER_H
#define SCANNER_H
extern int yylex(void);
#endif
那么你的 makefile 可能只是:
生成文件
SRCS := scanner.c lexer.c
OBJS := $(SRCS:.c=.o)
LDLIBS := -lfl
.PHONY: all clean
all: scanner
scanner: $(OBJS)
$(LINK.o) -o $@ $^ $(LDLIBS)
scanner.o: scanner.h
clean:
rm -f scanner *.o
运行方式如下:
$ make
cc -c -o scanner.o scanner.c
lex -t lexer.l > lexer.c
cc -c -o lexer.o lexer.c
cc -o scanner scanner.o lexer.o -lfl
rm lexer.c
请注意,make 会运行所有这些命令:
cc -c -o scanner.o scanner.c
lex -t lexer.l > lexer.c
cc -c -o lexer.o lexer.c
制作lexer.c
, lexer.o
和 scanner.o
没有你写任何告诉它如何做的食谱。它还会自动注意到 lexer.c
是一个中间文件 - 一个生成的文件只需要存在即可从 lexer.l
获取至 lexer.o
- 所以它删除它在最后:
rm lexer.c
没有被告知。
并且这个扫描器运行如下:
$ ./scanner
hello
Found a string: [hello]
42
Found an integer: [42]
42.42
Found a floating-point number: [42.42]
^C
关于c - 为什么 makefile 坚持编译它不应该编译的东西?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46261532/
我正在尝试在Elasticsearch中返回的值中考虑地理位置的接近性。我希望近距离比某些字段(例如legal_name)重要,但比其他字段重要。 从文档看来,当前的方法是使用distance_fea
我是Elasticsearch的初学者,今天在进行“多与或”查询时遇到问题。 我有一个SQL查询,需要在Elastic中进行转换: WHERE host_id = 999 AND psh_pid =
智能指针应该/可以在函数中通过引用传递吗? 即: void foo(const std::weak_ptr& x) 最佳答案 当然你可以通过const&传递一个智能指针。 这样做也是有原因的: 如果接
我想执行与以下MYSQL查询等效的查询 SELECT http_user, http_req_method, dst dst_port count(*) as total FROM my_table
我用这两个查询进行测试 用must查询 { "size": 200, "from": 0, "query": { "bool": { "must": [ { "mat
我仍在研究 Pro Android 2 的简短服务示例(第 304 页)同样,服务示例由两个类组成:如下所示的 BackgroundService.java 和如下所示的 MainActivity.j
给定标记 like this : header really_wide_table..........................................
根据 shouldJS 上的文档网站我应该能够做到这一点: ''.should.be.empty(); ChaiJS网站没有使用 should 语法的示例,但它列出了 expect 并且上面的示例似乎
我在 Stack Overflow 上读到一些 C 函数是“过时的”或“应该避免”。你能给我一些这种功能的例子以及原因吗? 这些功能有哪些替代方案? 我们可以安全地使用它们 - 有什么好的做法吗? 最
在 C++11 中,可变参数模板允许使用任意数量的参数和省略号运算符 ... 调用函数。允许该可变参数函数对每个参数做一些事情,即使每个参数的事情不是一样的: template void dummy(
我在我从事的项目之一上将Shoulda与Test::Unit结合使用。我遇到的问题是我最近更改了此设置: class MyModel :update end 以前,我的(通过)测试看起来像这样: c
我该如何做 or使用 chai.should 进行测试? 例如就像是 total.should.equal(4).or.equal(5) 或者 total.should.equal.any(4,5)
如果您要将存储库 B 中的更改 merge 到存储库 A 中,是否应该 merge .hgtags 中的更改? 存储库 B 可能具有 A 中没有的标签 1.01、1.02、1.03。为什么要将这些 m
我正在尝试执行X AND(y OR z)的查询 我需要获得该代理为上市代理或卖方的所有已售属性(property)。 我只用 bool(boolean) 值就可以得到9324个结果。当我添加 bool
我要离开 this教程,尝试使用 Mocha、Supertest 和 Should.js 进行测试。 我有以下基本测试来通过 PUT 创建用户接受 header 中数据的端点。 describe('U
我正在尝试为 Web 应用程序编写一些 UI 测试,但有一些复杂的问题希望您能帮助我解决。 首先,该应用程序有两种模式。其中一种模式是“训练”,另一种是“现场”。在实时模式下,数据直接从我们的数据库中
我有一个规范: require 'spec_helper' # hmm... I need to include it here because if I include it inside desc
我正在尝试用这个测试我在 Rails 中的更新操作: context "on PUT to :update" do setup do @countdown = Factory(:count
我还没有找到合适的答案: onclick="..." 中是否应该转义 &(& 符号)? (或者就此而言,在每个 HTML 属性中?) 我已经尝试在 jsFiddle 和 W3C 的验证器上运行转义和非
import java.applet.*; import java.awt.*; import java.awt.event.*; public class Main extends Applet i
我是一名优秀的程序员,十分优秀!