首先建立一个布局文件
?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.example.user.myapplication.MainActivity"> <TextView android:id="@+id/tv" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/app_name"/> <Button android:onClick="click" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Hello World!" /> </RelativeLayout>大体思路,一个textview,一个button,botton采用原始的方法,点击button然后获取到我们用注解定义的textview,然后打出该textview展示的值!
接着我们编写一个自定义注解,没基础的去看一下注解
@Target(ElementType.FIELD) @Retention(RetentionPolicy.RUNTIME) public @interface InjectView { int value() default -1; }这是一个运行时注解,只作用于属性
接着我们在MainActivity里面完成相关的逻辑实现,大体思路,通过反射获取到当前Activity的所有属性,然后根据属性来判断是否有自己所有定义的注解,如果有即获取,并读取注解里的值,然后将值赋给属性,即可代替findviewById方法!
//为了使代码结构更加清晰我多套了一层方法方便阅读 private void superInitView(Activity activity){ try { injectView(activity); } catch (Exception e) { e.printStackTrace(); } } private void injectView(Activity activity) throws Exception{ Class<Activity> clazz = (Class<Activity>) activity.getClass();//获取该Activity的字节码 Field[] fields = clazz.getDeclaredFields();//通过反射获取该类的所有属性,即当前Activity的所有属性 for (Field field:fields) {//遍历所有属性 if (field.isAnnotationPresent(InjectView.class)){//判断该属性是否自己定义的InjectView使用注解 InjectView inject=field.getAnnotation(InjectView.class);//如果是的话就获得该类上注解 int id=inject.value();//读取该ID if(id>0) { field.setAccessible(true); field.set(activity, activity.findViewById(id));//给view赋值,相当于 view = findviewById(R.id.tv); } } }
调用:
@InjectView(R.id.tv) private TextView tv;//需要调用的注解属性 @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); superInitView(this); } public void click(View v){ Toast.makeText(this,tv.getText().toString(),Toast.LENGTH_LONG).show(); }