Android中aidl如何import文件

如果你做Android开发,那就应该会知道aidl工具的,但是会使用aidl命令行工具的人有吗有吗有吗,如果有的话你们为神马不出来写写怎么用!!!!

用aidl命令行的如果不出意外肯定会遇见类似这样的错误

$ aidl IRemoteServiceCallback.aidl 
IRemoteServiceCallback.aidl:19: couldn't find import for class android.location.Location

遇到错误,自然要先看下aidl的帮助是怎么写的

$ aidl
usage: aidl OPTIONS INPUT [OUTPUT]
       aidl --preprocess OUTPUT INPUT...

OPTIONS:
   -I<DIR>    search path for import statements.
   -d<FILE>   generate dependency file.
   -p<FILE>   file created by --preprocess to import.
   -o<FOLDER> base output folder for generated files.
   -b         fail when trying to compile a parcelable.

看到如此简洁不明了的介绍你有何感想?

多写一句会死啊!!!我讨厌这种假人,以为开发Android的人理解能力都很强吗,不仅help此般不负责任,tutorial也是如此,只有一句

Add the .aidl file to your makefile - (the ADT Plugin for Eclipse manages this for you). Android includes the compiler, called AIDL, in the tools/ directory.

其实这个介绍更废物,它不告诉你是哪个tools,我想当然的以为是sdk下的tools文件夹,怎么都找不到aidl的所在

难道所有的人都在用Eclipse开发Android吗,我去google了一下,问这个问题的大有人在,Google为什么也不修正一下文档。

google在批判apple时说

如果 Google 不行动起来,那我们都将面临一个残酷的未来:一个人、一家公司、一款设备、一个运营商将是我们唯一的选择。

而现在google做的呢,它虽然没有强迫我们只用Eclipse,但它实际上就是这么做的,如同Adobe非强制的要求你用Flash来开发ActionScript一样

牢骚完了,让我们来一步步解决这个猪血问题


当文档不能帮你的时候,看Android代码吧。如何获取代码,Google的文档写的很清楚,其实Google大部分的文档都还是够明了的TAT

以下ANDROID代表android代码树的根

ANDROID/frameworks/base/tools/aidl里是aidl工具的C++代码,我没从中获得有用信息。或许很可能是我对C++代码不太有感。。。

一炮打不响就换一个洞,既然Eclipse可以处理aidl文件,我就来上Eclipse的Android插件ADT吧。aidl相关的文件是这个

ANDROID/sdk/eclipse/plugins/com.android.ide.eclipse.adt/src/com/android/ide/eclipse/adt/internal/build/PreCompilerBuilder.java

你问我如何找到的吗?Linux上的经典工具要经常用才能算是经典

$ grep -ri aidl Android/android-source/sdk/eclipse/plugins/com.android.ide.eclipse.adt/

880行有一个private boolean handleAidl函数,看名字就是这地方了。一路摸下去,直到890行

command[index++] = "-p" + Sdk.getCurrent().getTarget(getProject()).getPath( //$NON-NLS-1$
                IAndroidTarget.ANDROID_AIDL);

看到吗,在这儿发现了-p参数,但还不知道它是什么,顺着此条代码追踪,我们找到了这样一个文件

ANDROID/sdk/sdkmanager/libs/sdklib/src/com/android/sdklib/IAndroidTarget.java

现在找到30行,我们终于知道,原来这个-p所指的file created by --preprocess to import就是一个名叫framework.aidl的文件

/** OS Path to the "framework.aidl" file. */
    public final static int ANDROID_AIDL        = 2;

哈哈,让我来验一验你的真身!!!

$ find Android/android-sdk-linux_86/ -name framework.aidl
Android/android-sdk-linux_86/platforms/android-3/framework.aidl
Android/android-sdk-linux_86/platforms/android-4/framework.aidl
Android/android-sdk-linux_86/platforms/android-6/framework.aidl
...

这个framework.aidl文件也很脑残,虽说注释不是很重要,但是这个代码根本就没有半点自我解释能力,也不写个注释说明自己有什么用途--

到此,-p参数确认

革命远未结束,要想解决这个猪血问题,还要知道-I参数的写法。

继续看上面PreCompilerBuilder.java文件的896行

for (IPath p : sourceFolders) {
  IFolder f = wsRoot.getFolder(p);
  command[index++] = "-I" + f.getLocation().toOSString(); //$NON-NLS-1$
}

我想你能看出这是什么意思来了,就是你项目的source目录,一般来说是src那个目录

至此,问题解决,最後总结下使用aidl命令行工具的方法就是

$ aidl -I/path/to/project/src -p/path/to/framework.aidl Interface.aidl

谢谢国家,打完收工

6 COMMENTS >>LEAVE<<

  1. wayne

    有源码看就是好。。。

  2. jojol

    其他参数呢?我看了help后 我不知道 哪些参数必用 哪些参数可选用

  3. ABitNo
    @jojol

    没有必须用的,都是可选的

  4. naturegirl

    我使用你最后总结出来的命令居然还出现了 "aidl: can't open preprocessed file: ~/android-sdk-mac_x86/platforms/android-9/framework.aidl"

    哎 但是你真是说出我的心声了!! 纠结了一整天....继续纠结....

  5. naturegirl

    还是看源代码有用... 补充一句, 路径里面不能使用 "~"
    太感谢作者了!!

  6. yeeach

    具体命令参数可以参考eclipse console输出的编辑信息:
    在eclipse的Preferences > Android > Build 将Build Output设置为Verbose,然后clean一下项目,可以看见eclipse调用aidl命令编译aidl接口文件的具体参数

LEAVE A RESPONSE >>CANCEL<<

loader