我们也说说Android.mk(1) - 从函数说起

简介: 大家都习惯看从头,从构建目标讲起的makefile教程,导致每篇文档熟的都是前面的部分。很多教程也都是想办法能够观其大略,从整体上给大家一个思路。比如《深入理解Android内核设计思想》的第4章,比如《Android内核剖析》的第18章,比如《深入解析Android 5.0系统》的第2章。

我们也说说Android.mk(1)

从函数说起

大家都习惯看从头,从构建目标讲起的,导致每篇文档熟的都是前面的部分。很多教程也都是想办法能够观其大略,从整体上给大家一个思路。比如《深入理解Android内核设计思想》的第4章,比如《Android内核剖析》的第18章,比如《深入解析Android 5.0系统》的第2章。

于是我打算反其道而行之,先从调用函数开始讲。

最后一招:shell函数

我们最先把最后看家的绝招列出来吧,shell函数,可以用来执行shell命令。一切用之后讲到的函数解决不了的问题,都可以靠shell函数调用外部功能来解决。
不过,调用shell需要启动新进程,影响性能,只要有其它方法建议就不要使用它。

比如,我们想要列出当前目录下有哪些cpp文件,makefile可以这样写:

files := $(shell ls *.cpp)
all :
    @echo -n "The files is:"
    @echo $(files)
.PHONY : all

拼接字符串

在Makefile中用途相当广的就是拼接字符串,join函数就是干这事儿的。
格式:$(join 串1 , 串2)
也可以是多个串一起拼:$(join 串1 串2 串3, 串4 串5 串6)
多个串就是1和4拼,2和5拼。。。总之是逗号前的和逗号后的拼。

去空格

有事儿没事儿,去去空格总不是坏事儿。
$(strip 字符串)

文件路径操作函数

在实际的Makefile开发中,经常遇到要处理变量中存储的路径名。

先来个提纲:

  • 有一个不知道哪里来的文件名,想取它的路径,用dir
  • 有一个不知道哪里来的文件名,想只取它的文件名,用notdir
  • 如果只想要扩展名,判断是什么类型的,用suffix
  • 如果想不要扩展名,取出来拼个别的扩展名上去,比如xxx.c,只要xxx,将来拼个xxx.o或者就是xxx之类的,用basename
  • 想给文件添加个扩展名,用addsuffix
  • 想要把文件搞到另一个路径上去,拼个目录名上去,用addprefix

取一个文件的目录路径函数dir

例:

.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result := $(dir $(oatfile))
all2 :
    @echo -n "The result is: "
    @echo $(result)

输出:The result is: out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/

取文件名函数

例:

.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result_dir := $(dir $(oatfile))
result_notdir := $(notdir $(oatfile))
all2 :
    @echo "The result is: "
    @echo $(result_dir)
    @echo $(result_notdir)

输出:
The result is:
out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/
package.odex

取文件扩展名

例:

.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result_dir := $(dir $(oatfile))
result_notdir := $(notdir $(oatfile))
result_suffix := $(suffix $(oatfile))
all2 :
    @echo "The result is: "
    @echo $(result_dir)
    @echo $(result_notdir)
    @echo $(result_suffix)

输出:
The result is:
out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/
package.odex
.odex

取文件基本名

如果只用basename函数,是连路径都有的,只去除掉了扩展名的名字。不过我们刚学过notdir,一起用下就是了。

例:

.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result_suffix := $(suffix $(oatfile))
result_basename := $(basename $(oatfile))
result_basename2 := $(basename $(notdir $(oatfile)))
all2 :
    @echo "The result is: "
    @echo $(result_suffix)
    @echo $(result_basename)
    @echo $(result_basename2)

输出:
The result is:
.odex
out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package
package

给名字加后缀

使用addsuffix函数,$(addsuffix 文件名,扩展名)
我们举个例子说明:现在想把oat文件压缩成.tar.gz:

.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result_basename := $(basename $(oatfile))
compress_oat := tar cfvz $(addsuffix .tar.gz , $(result_basename)) $(oatfile)
all2 :
    @echo "The result is: "
    @echo $(compress_oat)

输出如下:
The result is:
tar cfvz out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.tar.gz out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex

给文件名加前缀

这个用于换路径, addprefix。一个前缀可以后边都加上。

例:

.PHONY : all3
oatfile2 := system.odex music.odex Contacts.odex
result_oatfile2 = $(addprefix /data/dalvik-cache/, $(oatfile2))
all3:
    @echo $(result_oatfile2)

输出:
/data/dalvik-cache/system.odex /data/dalvik-cache/music.odex /data/dalvik-cache/Contacts.odex

直接做字符串替换

如果我们不想拆开加去的这么麻烦,有一个简易的方法是直接做字符串替换。
subst函数的定义如下:$(subst 源串,目标串,要做替换的字符串)

例,我们把刚才将.odex扩展名换成.tag.gz扩展名,并拼成一个tar命令的makefile重写一下:

.PHONY : all4
oatfile_targz := $(subst $(suffix $(oatfile)),.tar.gz,$(oatfile))
compress_oat := tar cfvz $(oatfile_targz) $(oatfile)
all4 :
    @echo "The result is: "
    @echo $(compress_oat)

输出如下:
The result is:
tar cfvz out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.tar.gz out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex

过滤函数

过滤函数有两种:

  • filter是符合条件的留下
  • filter-out是符合条件的去除掉

我们看一个例子:

all8:
    @echo $(filter-out default interpreter jit optimizing,xoc)
    @echo $(filter-out default interpreter jit optimizing,default)

输出结果:

$ make all8
xoc

filter-out default interpreter jit optimizing,default这一句,因为default在列表中,所以被过滤掉了,变成一个空串。
filter-out default interpreter jit optimizing,xoc:这句因为xoc不在过滤列表之中,所以留下了。

单词的处理

文件列表,参数列表等等可以看成对单词的处理

比如下面的例子,我们想要查MAKEFILE_LIST中的最后一个文件,可以这样写:

my-dir = $(call parent-dir,$(lastword $(MAKEFILE_LIST)))

针对单词处理,有下面的常用函数:

  • firstword:取第一个单词
  • lastword:取最后一个单词
  • words:统计一共有多少个单词
  • word:取第n个单词
  • wordlist:取单词的子集

word的取值是从1开始,不能取0。

我们看一个例子:

SETTINGS_ART_DST := out/target/product/6753_doov_l5_64_m/system/priv-app/Settings/oat/arm64/Settings.odex
.PHONY : all10
all10:
    @echo "Install: $@"
    $(eval SETTINGS_ART_DST_LIST := $(subst /, ,$(SETTINGS_ART_DST)))
    @echo $(words $(SETTINGS_ART_DST_LIST))
    @echo $(word 1,$(SETTINGS_ART_DST_LIST))
    @echo $(word 2,$(SETTINGS_ART_DST_LIST))
    @echo $(wordlist 5,10,$(SETTINGS_ART_DST_LIST))

输出如下:

$ make all10
Install: all10
10
out
target
system priv-app Settings oat arm64 Settings.odex
目录
相关文章
|
10月前
|
Java Android开发
Java、Android 新建类自动添加头部注释以及如何添加函数注释模板和快捷键
Java、Android 新建类自动添加头部注释以及如何添加函数注释模板和快捷键
|
12月前
|
Java 编译器 Android开发
Android C++系列:C++11函数特殊特性
在Python中函数有默认参数等,在C++11中我们发现C++也支持了默认参数;还有C++特有的内联函数、constexpr函数等知识都有不少细节,本文对这些知识做详细介绍。
91 0
|
12月前
|
存储 IDE 编译器
Android C++系列:函数返回值注意事项
函数返回值就是使用return语句终止正在执行的函数,看是很简单的问题有什么说的呢?因为越是简单的问题里面越是有一些不易发现的坑。
72 0
|
12月前
|
编译器 Android开发 C++
Android C++系列:数组在函数中注意事项
数组作为函数形参传递的是数组首元素的地址本来是很简单的知识点,但是在具体使用中还会有一些坑需要注意。
107 0
|
12月前
|
C语言 Android开发 C++
Android C++系列:函数知识知多少
函数可以理解为功能的封装,很基础的功能单元,但是因为它虽然看似简单,但是里面涉及了不少知识点和技巧,我们花一篇文章来整理。
95 0
|
12月前
|
网络协议 Unix Linux
Android C++ 系列:Linux Socket 编程(二)网络套接字函数
socket()打开一个网络通讯端口,如果成功的话,就像open()一样返回一个文件描 述符,应用程序可以像读写文件一样用read/write在网络上收发数据,如果socket()调 用出错则返回-1。对于IPv4,domain参数指定为AF_INET。对于TCP协议,type参数指定为 SOCK_STREAM,表示面向流的传输协议。如果是UDP协议,则type参数指定为SOCK_DGRAM,表 示面向数据报的传输协议。protocol参数的介绍从略,指定为0即可。
154 0
|
12月前
|
安全 Java 程序员
Android C++系列:C++最佳实践1虚函数
C++多态的核心技术基础就是虚函数,虚函数允许我们使用同样的基类指针调用同一个方法的不同实现版本。我们Android使用Java开发过程中,方法重写技术自动实现了多态,C++角度可能更繁琐一些,本文从Java程序员思维角度来阐述C++虚函数及开发过程一些准则。
64 0
|
12月前
|
网络协议 Linux Android开发
Android C++ 系列:Linux 常用函数和工具
如果times是非空指针,则存取时间和修改时间被设置为 times所指向的结构中的值。此 时,进程的有效用户ID必须等于该文件的所有者 ID,或者进程必须是一个超级用户进程。对 文件只具有写许可权是不够的
94 0
|
Android开发
Android 重写TextView的onDraw函数遇坑记
Android 重写TextView的onDraw函数遇坑记
157 0
Android 重写TextView的onDraw函数遇坑记
|
Java 编译器 API
Android使用NDK(从java调用本地函数'JNI')
Android使用NDK(从java调用本地函数'JNI')
190 0
Android使用NDK(从java调用本地函数'JNI')