linux: 使用Makefile封装功能并传递参数

423 阅读2分钟

Makefile是一个强大的工具,用于自动化构建过程和任务管理。通过Makefile,我们可以定义和调用其他目标,并传递参数,实现功能的封装和复用。本文将介绍如何在Makefile中实现这一目标,并提供详细的示例代码。

cover (1).png

获取当天日期

首先,我们介绍如何在Makefile中获取当天的日期,并将其格式化为YYYYMMDD格式。例如,20240627。

# 定义一个变量,用于存储格式化的日期
DATE := $(shell date +%Y%m%d)

# 示例目标,展示如何使用这个日期变量
all: show_date

# 一个示例目标,输出格式化的日期
show_date:
	@echo 今天的日期是:$(DATE)

在这个示例中,DATE变量使用date命令获取当前日期,并格式化为YYYYMMDD格式。show_date目标使用@echo命令输出格式化的日期。

赋值符号

Makefile中有多种赋值符号,每种符号有不同的行为:

  • =:简单赋值,变量在使用时才计算其值。
  • :=:立即赋值,变量在定义时计算其值并固定。
  • ?=:条件赋值,只有当变量未定义时才赋值。
  • +=:追加赋值,将新值追加到现有值之后。

例如:

FOO = $(BAR)
BAR = hello

BAZ := $(BAR)
BAR = world

QUX ?= default
QUX ?= new_value

LIST = item1
LIST += item2

传递多个参数并遍历操作

我们希望一次传递多个host参数,并遍历操作这些hosts。以下是一个示例Makefile,展示如何实现这一需求:

# 定义一个变量,用于存储hosts
HOSTS := host1 host2 host3

# 示例目标,展示如何遍历操作这些hosts
all: iterate_hosts

# 目标,用于遍历hosts并执行操作
iterate_hosts:
	@for host in $(HOSTS); do \
		echo "Processing $$host"; \
		# 在这里添加对每个host的操作,例如ping
		ping -c 1 $$host; \
	done

在这个示例中,我们定义了一个HOSTS变量,包含多个host。iterate_hosts目标使用for循环遍历HOSTS变量中的每个host,并对每个host执行操作,例如使用ping命令检测连通性。

封装功能并传递参数

通过Makefile中的函数和目标参数化,我们可以封装功能并传递参数。以下是一个详细的示例:

# 默认的hosts变量
HOSTS := host1 host2 host3

# 主目标,调用封装的目标并传递参数
all: $(HOSTS)
	@echo "All hosts processed."

# 封装的目标,处理单个host
define PROCESS_HOST
$(1):
	@echo "Processing $(1)..."
	# 在这里添加对每个host的操作,例如ping
	ping -c 1 $(1)
endef

# 对每个host调用封装的目标
$(foreach host,$(HOSTS),$(eval $(call PROCESS_HOST,$(host))))

# 示例目标,展示如何手动调用带参数的目标
process_specific_host:
	$(MAKE) host1
	$(MAKE) host2

在这个示例中:

  1. 定义默认的HOSTS变量

    HOSTS := host1 host2 host3
    
  2. 定义主目标all

    all: $(HOSTS)
    	@echo "All hosts processed."
    
  3. 定义封装的目标PROCESS_HOST

    define PROCESS_HOST
    $(1):
    	@echo "Processing $(1)..."
    	ping -c 1 $(1)
    endef
    
  4. 使用foreacheval调用封装的目标

    $(foreach host,$(HOSTS),$(eval $(call PROCESS_HOST,$(host))))
    
  5. 定义示例目标process_specific_host

    process_specific_host:
    	$(MAKE) host1
    	$(MAKE) host2
    

执行Makefile

将上述内容保存为Makefile文件,然后在终端中运行:

yijie@heaven:~/snap$ make all
Processing host1...
# 在这里添加对每个host的操作,例如ping
echo ping -c 1 host1
ping -c 1 host1
Processing host2...
# 在这里添加对每个host的操作,例如ping
echo ping -c 1 host2
ping -c 1 host2
Processing host3...
# 在这里添加对每个host的操作,例如ping
echo ping -c 1 host3
ping -c 1 host3
All hosts processed.

这将遍历HOSTS变量中的每个host并执行指定的操作。也可以手动调用特定的host目标:

yijie@heaven:~/snap$ make all
Processing host1...
# 在这里添加对每个host的操作,例如ping
echo ping -c 1 host1
ping -c 1 host1
Processing host2...
# 在这里添加对每个host的操作,例如ping
echo ping -c 1 host2
ping -c 1 host2
Processing host3...
# 在这里添加对每个host的操作,例如ping
echo ping -c 1 host3
ping -c 1 host3
All hosts processed.

或者调用process_specific_host目标:

make process_specific_host
make host1
make[1]: Entering directory '/home/yijie/snap'
Processing host1...
# 在这里添加对每个host的操作,例如ping
echo ping -c 1 host1
ping -c 1 host1
make[1]: Leaving directory '/home/yijie/snap'
make host2
make[1]: Entering directory '/home/yijie/snap'
Processing host2...
# 在这里添加对每个host的操作,例如ping
echo ping -c 1 host2
ping -c 1 host2
make[1]: Leaving directory '/home/yijie/snap'

通过这种方式,我们可以在Makefile中封装功能并传递参数,提高Makefile的灵活性和可维护性。