-
ElementType.TYPE_PARAMETER JDK1.8新加的,类型参数声明
-
ElementType.TYPE_USE JDK1.8新加的,类型使用声明
@Retention
表示注解的生命周期
-
RetentionPolicy.SOURCE 源码阶段,在编译时会去除
-
RetentionPolicy.CLASS 注解会保存在class文件中,运行时会去除
-
RetentionPolicy.RUNTIME 注解会一直存在,在运行时可以利用反射去获取注解上面对应的值
@Inherited
表示允许子类继承父类的注解类型
@Documented
表示文档注解
今天我们就是利用运行时注解的特性,来做一些骚操作。
自定义注解
=====
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface AnnotationTest {
int value();
}
上面的代码自定义了一个AnnotationTest注解,@Target是ElementType.TYPE,表示是作用在类、接口或者枚举上面的。@Retention是RetentionPolicy.RUNTIME,表示在运行时期。内部定义了一个返回值为int的方法,表示使用注解时要传递一个int型的参数。那下面我们建个类来看看如何使用。
@AnnotationTest(1)
public class Test {
}
Test test = new Test();
//获取Test的Class对象
Class<? extends Test> clazz = test.getClass();
//使用Class对象获取对应的注解
AnnotationTest annotation = clazz.getAnnotation(AnnotationTest.class);
if (annotation != null) {
//获取注解传递的值
int value = annotation.value();
Log.d(TAG, “initView: >>>>>>>>>>>” + value);
}
上面代码首先定义了一个Test类,然后使用@AnnotationTest注解,传递数值1.然后使用反射获取该类的注解,接着获取注解传递的值,如上图打印出来的是1。
那下面开始进入我们今天的主题,使用注解去设置Activity的页面布局,不在用setContentView了。
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface ContentView {
int value();
}
定义了一个ContentView注解,返回值是int型的方法。
public class BaseActivity extends AppCompatActivity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
InjectUtils.injectLayout(this);
initView();
}
protected void initView() {
}
}
非常简单的一个BaseActivity
@ContentView(R.layout.activity_main)
public class MainActivity extends BaseActivity {
}
MainActivity,使用ContentView注解,然后注解传递布局
<?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:layout_width=“match_parent”
android:layout_height=“match_parent”
android:orientation=“vertical”
tools:context=“.MainActivity”>
<Button
android:id=“@+id/btn”
android:layout_width=“match_parent”
android:layout_height=“wrap_content” />
<Button
android:id=“@+id/btn2”
android:layout_width=“match_parent”
android:layout_height=“wrap_content” />
MainActivity的布局
还有一个InjectUtils类,获取MainActivity的注解和布局,然后在设置给MainActivity,主要的操作就是放在InjectUtils类中。
public class InjectUtils {
public static void injectLayout(Object context) {
//获取Class对象,这里context就是传递过来的MainActivity
Class<?> clazz = context.getClass();
//获取@ContentView注解
ContentView contentView = clazz.getAnnotation(ContentView.class);
if (contentView != null) {
//获取注解中的布局文件
int layoutId = contentView.value();
try {
//利用反射调用Activity中的setContentView方法,将布局设置给MainActivity
Method setContentViewMethod = clazz.getMethod(“setContentView”, int.class);
setContentViewMethod.invoke(context, layoutId);
} catch (NoSuchMethodException | IllegalAccessException | InvocationTargetException e) {
e.printStackTrace();
【附】相关架构及资料
往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,群内还有技术大牛一起讨论交流解决问题。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
2687)]
[外链图片转存中…(img-mZ2MaZgK-1715330932688)]
往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,群内还有技术大牛一起讨论交流解决问题。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!