Android 注解的一些应用以及原理
在这边文章之前你首先需要对java 的注解部分有一个基本的了解(不需要太过的深入)。
简单来说,注解这个东西就是用于辅助我们开发java代码的,注解本身无法干扰java源代码的执行。
在android 里面 注解主要用来干这么几件事:
1.和编译器一起给你一些提示警告信息。
2.配合一些ide 可以更加方便快捷 安全有效的编写java代码。谷歌出的support-annotations这个库 就是主要干这个的。
3.和反射一起 提供一些类似于spring 可配置的功能,方便简洁(这部分有过j2ee 开发经验的可以直接跳过了)。
首先来看一下官方提供的库吧。首先在build文件里 增加一句话
compile 'com.android.support:support-annotations:22.2.0'
这个地方要注意,我现在是取得version22.2的版本,你们引用的时候如果有更新,android studio可能会提示你sync失败,这个时候你要自己去谷歌官网查询一下最新的版本号自己替换。
然后我们就可以看看这个谷歌的注解库 为我们提供了哪些功能。
http://tools.android.com/tech-docs/support-annotations 这个是官方的文档 我也是主要根据这个来讲解这个库的使用。
首先来看 Nullness 注解 这个主要是有两种@Nullable 和 @NonNull。
前者表示可以为空 后者表示不能为空。
我只演示后者(前者可以自己测试)定义一个函数

然后我们去调用他 看看。
String a=null;
sayHello(a);
这个时候IDE就自动会给我们警告信息了(其实ide 的警告信息多数也是从编译器而来)
Argument 'a' might be null less... (Ctrl+F1)
This inspection analyzes method control and data flow to report possible conditions that are always true or false, expressions whose value is statically proven to be constant, and situations that can lead to nullability contract violations.
Variables, method parameters and return values marked as @Nullable or @NotNull are treated as nullable (or not-null, respectively) and used during the analysis to check nullability contracts, e.g. report possible NullPointerException errors.
More complex contracts can be defined using @Contract annotation, for example:
@Contract("_, null -> null") — method returns null if its second argument is null @Contract("_, null -> null; _, !null -> !null") — method returns null if its second argument is null and not-null otherwise @Contract("true -> fail") — a typical assertFalse method which throws an exception if true is passed to it
The inspection can be configured to use custom @Nullable
@NotNull annotations (by default the ones from annotations.jar will be used)
再介绍一下
Resource Type Annotations
定义一个方法
private String sayHello(@StringRes int resId) {
return "";
}
然后调用
sayHello(R.layout.activity_main);
这个地方这个注解的意思是你只能传string 类型的id进去 而我们传了layout类型的,如果你不加那个注解的话 ide和编译器是没有反应的,到运行期才会有错误。
但是你加了以后就会发现

IDE也是直接报错的!
再比如这样一段代码
class TestAnoataion {
public void testAnoation() {
Log.v("main", "test anoation");
}
}
class TestAnoataion2 extends TestAnoataion
{
public void testAnoation()
{
}
}
假使我们父类的设计的时候本意是 如果你要重写testAnoation这个方法一定得调用父类的testAnation这个方法。就跟我们activity的oncreate方法一样。
哪如果子类没有写super.testAnoation的话 你根本就不知道 就很容易报错。但是如果你加了注解。那么ide就会很明确的提示你错误

是不是很强大?当你调用了super语句以后 一切就和谐了!

那这个support库 就暂时介绍到这里,个人建议大家有时间一定要自己过一遍一个官方文档。
这里定义的注解 对提高大家的代码质量 会非常非常有帮助。一定要掌握学会。
最后再上一个小例子吧,很多android 开发者都喜欢依赖注入的一些开源库,来绑定你的控件id。
这样写法不但优雅而且好用。
Butterknife
这个开源库大家一定很多人都用过,我就写一个小demo 来演示一下这些库的原理,其实还蛮简单的。
需要注意的是 要理解这个小demo 需要你除了知道注解的基础知识以外还需要有反射的基础知识。
关于反射我很久之前也写过一个教程
http://www.cnblogs.com/punkisnotdead/p/3384464.html
不会的可以赶紧补一下。
首先定义一个InjectView注解
package com.example.administrator.testfab; import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; /**
* Created by Administrator on 2015/8/5.
*/ @Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface InjectView {
//id就表示哪些控件,-1就表示取不到时候的默认值
int id() default -1;
}
然后定义一个解释器
package com.example.administrator.testfab; import android.app.Activity;
import android.view.View; import java.lang.reflect.Field; /**
* Created by Administrator on 2015/8/5.
*/
public class InjectViewParser { public static void inject(Object object) { try {
parse(object);
} catch (Exception e) {
e.printStackTrace();
}
} public static void parse(Object object) throws Exception {
final Class<?> clazz = object.getClass();
View view = null;
Field[] fields = clazz.getDeclaredFields();
for (Field field : fields) {
if (field.isAnnotationPresent(InjectView.class)) {
InjectView injectView = field.getAnnotation(InjectView.class);
int id = injectView.id();
if (id < 0) {
throw new Exception("id must not be null");
} else {
field.setAccessible(true);
if (object instanceof View) {
view = ((View) object).findViewById(id);
} else if (object instanceof Activity) {
view = ((Activity) object).findViewById(id);
}
field.set(object, view);
} } } }
}
最后在actity里使用即可
package com.example.administrator.testfab; import android.os.AsyncTask;
import android.os.Bundle;
import android.support.annotation.CallSuper;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.annotation.StringRes;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.Button; public class MainActivity extends AppCompatActivity { @InjectView(id = R.id.bt)
private Button bt; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//开始注入
InjectViewParser.inject(this);
//这个主要是测试注入id 成功没有 成功了就不会报错~
bt.setText("inject done"); } @Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_main, menu);
return true;
} @Override
public boolean onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int id = item.getItemId(); //noinspection SimplifiableIfStatement
if (id == R.id.action_settings) {
return true;
} return super.onOptionsItemSelected(item);
} }
Android 注解的一些应用以及原理的更多相关文章
- Java Android 注解(Annotation) 及几个常用开源项目注解原理简析
不少开源库(ButterKnife.Retrofit.ActiveAndroid等等)都用到了注解的方式来简化代码提高开发效率. 本文简单介绍下 Annotation 示例.概念及作用.分类.自定义. ...
- Android注解使用之通过annotationProcessor注解生成代码实现自己的ButterKnife框架
前言: Annotation注解在Android的开发中的使用越来越普遍,例如EventBus.ButterKnife.Dagger2等,之前使用注解的时候需要利用反射机制势必影响到运行效率及性能,直 ...
- Android注解使用之使用Support Annotations注解优化代码
前言: 前面学习总结了Java注解的使用,博客地址详见Java学习之注解Annotation实现原理,从本质上了解到什么注解,以及注解怎么使用?不要看见使用注解就想到反射会影响性能之类,今天我们就来学 ...
- android注解使用详解(图文)
在使用Java的SSH框架的时候,一直在感叹注解真是方便啊,关于注解的原理,大家可以参考我的另一片文章Java注解详解.最近有时间研究了android注解的使用,今天与大家分享一下. android中 ...
- Android注解框架实战-ButterKnife
文章大纲 Android注解框架介绍 ButterKnife实战 项目源码下载 一.框架介绍 为什么要用注解框架? 在Android开发过程中,我们经常性地需要操作组件,操作方法有findVie ...
- android注解使用具体解释(图文)
在使用Java的SSH框架的时候,一直在感叹注解真是方便啊,关于注解的原理,大家能够參考我的还有一片文章Java注解具体解释. 近期有时间研究了android注解的使用,今天与大家分享一下. andr ...
- Android注解支持(Support Annotations)
注解支持(Support Annotations) Android support library从19.1版本开始引入了一个新的注解库,它包含很多有用的元注解,你能用它们修饰你的代码,帮助你发现bu ...
- 【转】android camera(二):摄像头工作原理、s5PV310 摄像头接口(CAMIF)
关键词:android camera CMM 模组 camera参数 CAMIF平台信息:内核:linux系统:android 平台:S5PV310(samsung exynos 4210) 作者 ...
- Java注解的基本概念和原理及其简单实用
一.注解的基本概念和原理及其简单实用 注解(Annotation)提供了一种安全的类似注释的机制,为我们在代码中添加信息提供了一种形式化得方法,使我们可以在稍后某个时刻方便的使用这些数据(通过解析 ...
随机推荐
- 李洪强iOS开发之OC[008] -创建一个对象并访问实例变量
// // main.m // 07 - 创建一个对象并且访问实例变量 // // Created by vic fan on 16/7/3. // Copyright © 2016年 李洪强 ...
- lintcode:排颜色 II
排颜色 II 给定一个有n个对象(包括k种不同的颜色,并按照1到k进行编号)的数组,将对象进行分类使相同颜色的对象相邻,并按照1,2,...k的顺序进行排序. 样例 给出colors=[3, 2, 2 ...
- PHP组合模式、策略模式
一.问题 模拟不同课程有不同的收费方式,并且能灵活改变(新增或删减),如讲座可以固定收费也可改为按时收费,研讨会也是. 二.模式简介及关键点 1.在父类代码中使用条件语句是一种退倒,可以用多态来代替条 ...
- 物联网操作系统Hello China移植mile stone之一:移植基础版本V1.76发布
Hello China V1.76版发布,这是向ARM系列CPU移植的基础版本.相对V1.75版,该版本主要做了如下的一些调整: 1. 通过宏定义的方式对内核实现了模块化,开发者可以通过开启或关闭预 ...
- Qt中的多线程技术(列表总结比较,多线程创建和销毁其实是有开销的,只是增加了用户体验而已)
http://blog.csdn.net/u011012932/article/details/52943811
- DP 子序列问题
函数lower_bound()在first和last中的前闭后开区间进行二分查找,返回大于或等于val的第一个元素位置.如果所有元素都小于val,则返回last的位置举例如下:一个数组number序列 ...
- C++:构造函数默认的参数声明
C++函数的默认参数指的是在函数声明或者定义时给形式参数指定默认值,从而在调用参数时可以少写参数,少掉的参数用默认值代替.LZ的Display()函数的代码看起来似乎是可以有s2和s3两个默认参数,那 ...
- Android 实现全屏、无标题栏
实现全屏无标题栏: 1.在xml文件中进行配置 AndroidManifest.xml中,找到需要全屏或设置成无标题栏的Activity,在该Activity进行如下配置即可. 实现全屏效果: and ...
- 通用sqlserver分页存储过程
来自:http://www.cnblogs.com/vagerent/archive/2007/10/17/927825.html 单主键: CREATE PROC P_viewPage /** ...
- 如何使用通用Mapper
集成方法请看上面的文档,集成后,可以继续阅读本页文档. 1. 继承通用的Mapper<T>,必须指定泛型<T> 例如下面的例子: public interface UserIn ...