Android应用开发编译框架流程与IDE及Gradle概要

可以看见,Jack是一个基于Java编译器和ProGuard的工具,但是目前版本还不支持ProGuard的一些高级功能(譬如移除日志代码)。Jill将Java库字节码转化成名为jayce的中间字节码.jack文件,Jack对Java源码和jayce字节码进行编译,生成经过优化的dex字节码。

想尝鲜使用Jack和Jill你需要保证你的Build Tools version是21.1.1版本或者更高。在Gradle中配置如下:

android {

buildToolsRevision ‘21.1.2’

defaultConfig {

// Enable the experimental Jack build tools.

useJack = true

}

}

总归一句话,现在还年轻,有何成就还得观望,看后面的发展趋势吧,反正目前我是没咋使用他,只是试验性的尝鲜了一把而已。

【工匠若水 http://blog.csdn.net/yanbober 未经允许严禁转载,请尊重作者劳动成果。私信联系我

3 Android应用IDE及编译相关特性

=========================

下面介绍一些使用IDE开发过程中高效的代码编写特性,同时包含一些编译相关的注意特性,具体如下会细说。

3-1 Android应用jar包与aar包


我们在Android Studio中对一个自己的库进行编译时会同时生成jar与aar两个包。他们的具体路径如下:

| 包类型 | 在AS中的输出路径 |

| — | — |

| jar | /build/intermediates/bundles/debug[release]/classes.jar |

| aar | /build/outputs/aar/library.aar |

他们两者的区别如下(实质都是压缩包):

| 包类型 | 描述 |

| — | — |

| jar | 只包含class文件,不包含资源文件,用于纯Java编写的库。 |

| aar | 包含所有class及res等全部资源,类似UI库。 |

其实关于他们二者的区别我们通过解压缩即可直观的看出来,这里不再叙述。

3-2 Android Tools Attributes编译说明


关于Android应用编译框架中还需要知道一个有用的工具属性,那就是tools命名空间属性,他的命名空间URI是http://schemas.android.com/tools,可以说这个命名空间是专门为开发者设计的,只在设计阶段有效,运行阶段无效。

3-2-1 绑定tools命名空间的方法

<FrameLayout xmlns:android=“http://schemas.android.com/apk/res/android”

//绑定命名空间

xmlns:tools=“http://schemas.android.com/tools”

android:layout_width=“match_parent”

android:layout_height=“match_parent” >

tools属性可以替换所有Android的属性,只在设计阶段有效,不会被带入最终的apk中,所以运行时无效。整个tools命名空间的属性分两大类,一类是Lint相关的、一类是设计相关的。下面我们列举一下tools相关的一些实用属性。

3-2-2 tools:ignore

告诉Lint忽略xml中某些警告。如下用法:

//忽略Lint对于多语言检测的警告,多个可以用逗号分开

All

3-2-3 tools:targetApi

用来指定API等级,功能和Java文件中的@TargetApi注释类似,值为整数或者含义字符串。如下用法:

3-2-4 tools:locale

默认情况下res/values/strings.xml文件中的字符串会进行拼写检查,如果本地不是英语则会给出警告,我们可以通过这样来指定本地语言然后忽略警告。如下:

3-2-5 tools:context

这个属性其实就是关联Activity属性,在xml中添加该属性后预览该xml文件就能知道采用啥主题来预览,同时关联了Activity文件与xml文件,可以从java文件直接跳转索引。如下:

<android.support.v7.widget.GridLayout xmlns:android=“http://schemas.android.com/apk/res/android” xmlns:tools=“http://schemas.android.com/tools”

tools:context=“.MainActivity” … >

3-2-6 tools:layout

该属性用在xml的fragment节点中,在开发时告诉IDE该fragment在预览模式下显示的样子。如下:

3-2-7 tools:listitem / listheader / listfooter

很明显可以猜到用来给ListView、AdapterView、GridView、ExpandableListView指定预览时的item和头底。如下:

<ListView

android:id=“@android:id/list”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

tools:listitem=“@android:layout/simple_list_item_2” />

3-2-8 tools:showIn

该属性被设置到一给被include的布局的根节点上,预览时可用。如下:

<?xml version="1.0" encoding="utf-8"?>

<TextView xmlns:android=“http://schemas.android.com/apk/res/android”

xmlns:tools=“http://schemas.android.com/tools”

android:text=“@string/hello_world”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

tools:showIn=“@layout/activity_main” />

3-2-9 tools:menu

用来告诉IDE在预览时当前xml布局显示指定的menu样式。其实如果我们指定了tools:context属性,IDE会很智能的在我们指定Activity文件中的onCreateOptionsMenu方法中寻找menu样式预览。当然了,我们可以通过该属性覆盖Activity中的menu预览。如下:

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

xmlns:tools=“http://schemas.android.com/tools”

android:orientation=“vertical”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

tools:menu=“menu1,menu2” />

3-2-10 tools:actionBarNavMode

指定actionbar的预览模式。可选值为”standard”、”list”、”tabs”。如下:

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

xmlns:tools=“http://schemas.android.com/tools”

android:orientation=“vertical”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

tools:actionBarNavMode=“tabs” />

3-2-11 Designtime设计时属性

tools属性可以替换所有Android的属性,只在设计阶段有效,不会被带入最终的apk中,所以运行时无效。譬如我们最常见的在写一个TextView时不想在xml中给他初始文本内容,又想预览,这时候就可以用tools属性替代。如下:

<TextView

tools:text=“Name:”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

tools:visibility=“invisible” />

可以看见,tools命名空间属性可以加速我们开发的灵活度和可预测度,同时在编译框架中又会忽略这些属性,不把他们带入最终产物apk中,我们可以在开发中酌情选择是否使用这些属性。

3-3 Android Annotations注解说明


在Android Support Library的19.1版本中加入了一个新的注解包,这个包用来注解代码,方便捕获程序中存在的问题和bug,这个包内部的代码就已经使用了该注解,在22.2版本中又增加了13个新的注解,这些注解可以方便我们代码开发与调运规范和bug检查,具体用法下面会一一讲解。

3-3-1 注解Library依赖引用

annotations注解包默认不会自动被包含,不过如果使用appcompat包则会自动包含,因为appcompat包里使用了注解。

在Android工程的Gradle文件中引入注解包如下:

dependencies {

compile ‘com.android.support:support-annotations:22.2.0’

}

如果不是Android工程中想使用该注解包则可以如下写法(url为本地android sdk的注解包路径):

repositories {

jcenter()

maven { url ‘/extras/android/m2repository’ }

}

3-3-2 执行注解

当我们在用Android Studio和IntelliJ时如果给使用了注解的方法传递错误类型参数,则IDE会实时标记提醒错误。如果使用的是Gradle 1.3.0版本以上且安装了Android M Preview Tools以上工具则可以通过命令行调用gradle的lint任务进行检查(nullness注解会被忽略检查)。

3-3-3 Null空类型判断注解

@Nullable 注解用来标注给定的参数或返回值可以为null。

@NonNull 注解用来标注给定的参数或返回值不能为null。

假设一个本地变量值为null,且把它作为参数传递给一个方法,且该方法的参数被@NonNull标注,则AS会提醒存在一个潜在的崩溃。如下:

import android.support.annotation.NonNull;

import android.support.annotation.Nullable;

/**

  • Add support for inflating the tag.

*/

@Nullable

@Override

public View onCreateView(String name, @NonNull Context context, @NonNull AttributeSet attrs) {

3-3-4 资源类型注解

Android的资源值通常都是通过R文件映射的整型id来关联的,也就是说获取一个layout类型的资源参数很容易被误传递一个其他类型的资源参数,因为他们都是整型的资源id,编译器很难区分。为了解决这种问题可以使用资源类型注解,因为注解提供类型检查。譬如下面是一个被@LayoutRes注解的整型参数却传递了一个string类型的资源参数,此时IDE会给出错误提示:

调运setContentView方法时传递错误参数:

这里写图片描述

setContentView的资源注解实现方法:

这里写图片描述

实际上有很多不同的资源类型注解,譬如@AnimatorRes、@AnimRes、@AnyRes、@ArrayRes、@AttrRes、@BoolRes、@ColorRes、@DimenRes、@DrawableRes、@FractionRes、@IdRes、@IntegerRes、@InterpolatorRes、@LayoutRes、@MenuRes、@PluralsRes、@RawRes、@StringRes、@StyleableRes、@StyleRes、@XmlRes等,一般一个foo类型资源的相应资源类型注解就是@FooRes。除此之外,还有一个名为@AnyRes的特殊资源类型注解,它被用来标注一个未知特殊类型的资源,且必须是一个资源类型。譬如在Resources.getResourceName(@AnyRes int resId)上使用的时候,我们可以通过getResources().getResourceName(R.drawable.icon)和getResources().getResourceName(R.string.app_name)等方式来使用,但却不能通过getResources().getResourceName(42)来使用。

3-3-5 IntDef/StringDef类型注解

这种类型的注解是基于Intellij的魔数检查机制的,因为Android开发中很多时候出于性能考虑,我们会使用整型常量代替枚举类型。譬如AppCompat库里的一个例子:

import android.support.annotation.IntDef;

public abstract class ActionBar {

@IntDef({NAVIGATION_MODE_STANDARD, NAVIGATION_MODE_LIST, NAVIGATION_MODE_TABS})

@Retention(RetentionPolicy.SOURCE)

public @interface NavigationMode {}

public static final int NAVIGATION_MODE_STANDARD = 0;

public static final int NAVIGATION_MODE_LIST = 1;

public static final int NAVIGATION_MODE_TABS = 2;

@NavigationMode

public abstract int getNavigationMode();

public abstract void setNavigationMode(@NavigationMode int mode);

上面非注解的部分是原本的代码和API,我们通过@interface创建了一个新注解NavigationMode,并且用@IntDef标注它可能的取值,还添加了@Retention(RetentionPolicy.SOURCE)告诉编译器这个新定义的注解不需要被记录在生成的.class文件中;使用这个注解后,如果我们返回或传递的参数不在指定的常量值中则IDE会给出明显的错误提示。

我们也可以指定整型常量为一个标记性质的类型,因为这样就可以通过|、&等操作符同时传递多个整型常量。如下例子:

@IntDef(flag=true, value={

DISPLAY_USE_LOGO,

DISPLAY_SHOW_HOME,

DISPLAY_HOME_AS_UP,

DISPLAY_SHOW_TITLE,

DISPLAY_SHOW_CUSTOM

})

@Retention(RetentionPolicy.SOURCE)

public @interface DisplayOptions {}

@StringDef和@IntDef的作用基本上一样,不同的是它针对的是字符串。关于类型注解更多信息点我获取

3-3-6 线程注解@UiThread、@WorkerThread等

线程注解是在Support Library 22.2及更高版本中才被支持的,如果我们的方法只能在指定的线程类型中被调用,则就可以使用以下4个注解来标注它们:

  • @UiThread

  • @MainThread

  • @WorkerThread

  • @BinderThread

如果一个类中的所有方法都有相同的线程需求则可以注解类本身。

关于线程注解使用的一个例子是AsyncTask,如下:

@WorkerThread

protected abstract Result doInBackground(Params… params);

@MainThread

protected void onProgressUpdate(Progress… values) {

}

如果在使用该方法时没有按照注解线程执行则会报错,如下:

这里写图片描述

3-3-7 约束值注解@Size、@IntRange、@FloatRange

如果我们的参数是float或double类型且限制在一个范围内,则可以使用@FloatRange注解,如下:

public void setAlpha(@FloatRange(from=0.0, to=1.0) float alpha) {

如果我们的参数是int或long类型则可以使用@IntRange注解来约束值的范围,如下:

public void setAlpha(@IntRange(from=0,to=255) int alpha) { … }

对于数据集合或字符串,我们可以用@Size注解来限定集合大小或字符串长度。譬如:

//集合不能为空

@Size(min=1)

//字符串最多只能有23个字符

@Size(max=23)

//数组只能有2个元素

@Size(2)

//数组大小必须是2的倍数

@Size(multiple=2)

3-3-8 权限注解@RequiresPermission

如果我们的方法调用时需要特定权限,则可以用@RequiresPermission进行注解。如下:

@RequiresPermission(Manifest.permission.SET_WALLPAPER)

public abstract void setWallpaper(Bitmap bitmap) throws IOException;

如果方法至少需要权限集合中的一个则可以使用anyOf属性。如下:

@RequiresPermission(anyOf = {

Manifest.permission.ACCESS_COARSE_LOCATION,

Manifest.permission.ACCESS_FINE_LOCATION})

public abstract Location getLastKnownLocation(String provider);

如果方法同时需要多个权限则可以使用allOf属性。如下:

@RequiresPermission(allOf = {

Manifest.permission.READ_HISTORY_BOOKMARKS,

Manifest.permission.WRITE_HISTORY_BOOKMARKS})

public static final void updateVisitedHistory(ContentResolver cr, String url, boolean real) {

对于Intent的权限可以直接在定义的Intent常量字符串字段上标注权限需求(通常都已经被@SdkConstant注解标注过了)。如下:

@RequiresPermission(android.Manifest.permission.BLUETOOTH)

public static final String ACTION_REQUEST_DISCOVERABLE =

“android.bluetooth.adapter.action.REQUEST_DISCOVERABLE”;

有时对于ContentProvider的权限可能需要单独的标注读和写权限,所以可以用@Read或者@Write注解。如下:

@RequiresPermission.Read(@RequiresPermission(READ_HISTORY_BOOKMARKS))

@RequiresPermission.Write(@RequiresPermission(WRITE_HISTORY_BOOKMARKS))

public static final Uri BOOKMARKS_URI = Uri.parse(“content://browser/bookmarks”);

3-3-9 重写方法注解@CallSuper

如果想在重写方法中要求必须调运父类的super方法则可以用@CallSuper标注。如下:

@CallSuper

protected void onCreate(@Nullable Bundle savedInstanceState) {

3-3-10 @CheckResult、@VisibleForTesting、@Keep

如果我们的方法返回一个值且期望调用者用这个值做些事情则可以使用@CheckResult注解标注这个方法。

如果我们想在测试时使用一个类、方法或者字段则可以用@VisibleForTesting注解进行标明。

特别要说的就是我们还在注解库里添加了@Keep注解,但是Gradle插件目前版本支持的还不是很好,这个注解标注的东西不会被混淆。

3-3-11 其他注解相关

如果我们在自己的lib库中使用了上面这些注解,且通过Gradle构建生成了aar包,则在构建时Android Gradle插件会提取注解信息放在aar文件中供使用库的地方使用。在aar文件中你可以看到一个名为annotations.zip的文件,这个文件记录的就是使用IntelliJ扩展注解XML格式的注解信息。

PS一句,IntelliJ其实还有他自己的注解,这里不再详细说明。

可以看见,有了这种注解我们的IDE看起来就更加的智能便捷强大了,同时也帮助我们的编译框架及早的发现问题,给出警告或者错误提示。

【工匠若水 http://blog.csdn.net/yanbober 未经允许严禁转载,请尊重作者劳动成果。私信联系我

4 Gradle Android构建

======================

其实Android的Gradle构建没啥太多说的,掌握Groovy语法和Gradle框架后看看Gradle Android Plugs相关DSL特性就行了,用的时候去查API和Android DSL规则即可,重点明白结构和常用的DSL规则即可,这里给出一个常用模板,补充常用实战基础示例参见《Android Studio入门到精通》一文的第7小节。

4-1 Gradle Android编译构建配置基础


Android Studio的project在project树根和每个module根都包含一个叫build.gradle的文件,我们所谓的编译构建配置其实就是通过Gradle的Android插件进行编写脚本,大多数情况下我们一般只用修改相关module下的build.gradle文件就能满足需求。如下一个例子:

//添加Gradle的Android构建插件,默认包含一些task和android {…}等特定元素

apply plugin: ‘com.android.application’

//android元素用来配置Android编译特有选项

学习福利

【Android 详细知识点思维脑图(技能树)】

其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。

虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。

这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。

由于篇幅有限,这里以图片的形式给大家展示一小部分。

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

/android元素用来配置Android编译特有选项

学习福利

【Android 详细知识点思维脑图(技能树)】

[外链图片转存中…(img-MoBED7Za-1714407961635)]

其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。

虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。

这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。

由于篇幅有限,这里以图片的形式给大家展示一小部分。

[外链图片转存中…(img-BLbUjFku-1714407961636)]

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 17
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值