官方定義:反射主要是指程序可以訪問徒仓、檢測(cè)和修改它本身狀態(tài)或行為的一種能力涤久。在計(jì)算機(jī)科學(xué)領(lǐng)域,反射是一類應(yīng)用癌别,它們能夠自描述和自控制。這類應(yīng)用通過某種機(jī)制來實(shí)現(xiàn)對(duì)自己行為的描述和檢測(cè)蹋笼,并能根據(jù)自身行為的狀態(tài)和結(jié)果展姐,調(diào)整或修改應(yīng)用所描述行為的狀態(tài)和相關(guān)的語義。
功能:
1.在運(yùn)行時(shí)判斷任意一個(gè)對(duì)象所屬的類剖毯。
2.在運(yùn)行時(shí)構(gòu)造任意一個(gè)類的對(duì)象圾笨。
3.在運(yùn)行時(shí)判斷任意一個(gè)類所具有的成員變量和方法。
4.在運(yùn)行時(shí)調(diào)用任意一個(gè)對(duì)象的方法逊谋。
反射的常用類和函數(shù):Class擂达,Object, Constructor,F(xiàn)ield胶滋,Method, Modifier板鬓;
其中Class-類對(duì)象,Constructor-類的構(gòu)造器對(duì)象究恤,F(xiàn)ield-類的屬性對(duì)象俭令,Method-類的方法對(duì)象。
Class獲取類對(duì)象的常用方法
getClass() -- 每一個(gè)Class都有此函數(shù)
Class.getSuperClass() -- 返回子類的類對(duì)象
Class.forName() -- 靜態(tài)方法部宿,通過類的名稱獲取這個(gè)類的對(duì)象
.class -- 通過組件獲取該組件所屬類的對(duì)象
Constructor獲取構(gòu)造器的常用方法
Constructor getConstructor(Class[] params) -- 根據(jù)構(gòu)造函數(shù)的參數(shù)抄腔,返回一個(gè)具體的具有public屬性的構(gòu)造函數(shù)
Constructor[] getConstructors() -- 返回所有具有public屬性的構(gòu)造函數(shù)數(shù)組
Constructor getDeclaredConstructor(Class[] params) -- 根據(jù)構(gòu)造函數(shù)的參數(shù),返回一個(gè)具體的構(gòu)造函數(shù)(不分public和非public屬性)
Constructor[] getDeclaredConstructors() -- 返回該類中所有的構(gòu)造函數(shù)數(shù)組(不分public和非public屬性)
Field獲取字段的常用方法
Field getField(String name) -- 根據(jù)方法名和參數(shù)理张,返回一個(gè)具體的具有public屬性的方法
Field[] getFields() -- 返回所有具有public屬性的方法數(shù)組
Field getDeclaredField(String name) -- 根據(jù)方法名和參數(shù)赫蛇,返回一個(gè)具體的方法(不分public和非public屬性)
Field[] getDeclaredFields() -- 返回該類中的所有的方法數(shù)組(不分public和非public屬性)
Method獲得方法的常用方法
Method getMethod(String name, Class[] params) -- 使用特定的參數(shù)類型,獲得命名的公共方法
Method[] getMethods() -- 用于獲取類的所有的public修飾域的成員方法涯穷,包括從父類繼承的public方法和實(shí)現(xiàn)接口的public方法棍掐;
Method getDeclaredMethod(String name, Class[] params) -- 用于獲取在當(dāng)前類中定義的所有的成員方法和實(shí)現(xiàn)的接口方法,不包括從父類繼承的方法拷况。
Method[] getDeclaredMethods() -- 獲得類聲明的所有方法
類Class作煌、Method、Constructor赚瘦、Field都有一個(gè)public方法int getModifiers()粟誓。該方法返回一個(gè)int類型的數(shù),表示被修飾對(duì)象( Class起意、 Method鹰服、 Constructor、 Field )的修飾類型的組合值。
定義都是抽象的悲酷,相信很多沒有反射概念的人看了定義也是一頭霧水套菜,要徹底搞懂一個(gè)知識(shí)點(diǎn)最好的方式就是結(jié)合實(shí)際的使用場(chǎng)景;
場(chǎng)景一:利用反射關(guān)閉應(yīng)用程序设易,主要代碼:
private ActivityManager manager;
private List<RunningAppProcessInfo> runningProcesses;
private String packName;
private PackageManager pManager;
@Override
public void onReceive(Context context, Intent intent) {
pManager = context.getPackageManager();
manager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
killOthers(context);
}
/**
* 殺死其他正在運(yùn)行的程序
*
* @param context
*/
private void killOthers(Context context) {
runningProcesses = manager.getRunningAppProcesses();
for (RunningAppProcessInfo runningProcess : runningProcesses) {
try {
packName = runningProcess.processName;
ApplicationInfo applicationInfo = pManager.getPackageInfo(packName, 0).applicationInfo;
if (!"com.asms".equals(packName)&&filterApp(applicationInfo)) {
forceStopPackage(packName,context);
System.out.println(packName+"JJJJJJ");
}
}catch(Exception e){
e.printStackTrace();
}
}
}
/**
*強(qiáng)制停止應(yīng)用程序
* @param pkgName
*/
private void forceStopPackage(String pkgName,Context context) throws Exception{
ActivityManager am = (ActivityManager)context.getSystemService(Context.ACTIVITY_SERVICE);
Method method = Class.forName("android.app.ActivityManager").getMethod("forceStopPackage", String.class);
method.invoke(am, pkgName);
}
/**
* 判斷某個(gè)應(yīng)用程序是 不是三方的應(yīng)用程序
* @param info
* @return
*/
public boolean filterApp(ApplicationInfo info) {
if ((info.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0) {
return true;
} else if ((info.flags & ApplicationInfo.FLAG_SYSTEM) == 0) {
return true;
}
return false;
}
說明:如果是在Activity中將onReceiver方法換成onCreate方法即可逗柴。
2.需要在AndroidManifest.xml 把package的userid設(shè)置為system
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.asms"
android:versionCode="1"
android:versionName="1.0" android:sharedUserId="android.uid.system">
場(chǎng)景二:通過反射程序控制Toast信息框
Toast信息框的關(guān)閉是由系統(tǒng)管理的,因?yàn)閔ide方法是隱藏的開發(fā)者沒有辦法直接調(diào)用顿肺,這種情況下可以用發(fā)射機(jī)制獲取這個(gè)方法戏溺,創(chuàng)建一個(gè)顯示和隱藏都由開發(fā)者控制的Toast信息框。
package com.example.reflection;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import android.content.Context;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.TextView;
import android.widget.Toast;
public class MyToast{
Context context= null;
Object obj = null;
public MyToast(Context context,String text){
this.context =context;
Toast toast =Toast.makeText(context, text,1);
try{
Field field = toast.getClass().getDeclaredField("mTN");
field.setAccessible(true);
obj =field.get(toast);
}catch(Exception e) {
// TODO: handle exception
Log.d("AAA","MyToast Exception--->"+e.toString());
}
}
public void show(){
try{
//android4.0以上就要以下處理
// Field mNextViewField=obj.getClass().getDeclaredField("mNextView");
//mNextViewField.setAccessible(true);
//LayoutInflater inflate = (LayoutInflater) context.getSystemService
(Context.LAYOUT_INFLATER_SERVICE);
//View v = inflate.inflate(R.layout.ui_toast, null);
// mNextViewField.set(obj, v);
Method method =obj.getClass().getDeclaredMethod("show",null);
method.invoke(obj,null);
}catch(Exception e) {
// TODO Auto-generated catch block
Log.d("AAA","show Exception--->"+e.toString());
e.printStackTrace();
}
}
public void hide(){
try{
Method method =obj.getClass().getDeclaredMethod("hide",null);
method.invoke(obj,null);
}catch (Exception e) {
// TODO Auto-generated catch block
Log.d("AAA","hide Exception--->"+e.toString());
e.printStackTrace();
}
}
}
顯示toast:
MyToast toast =new MyToast(this,"反射機(jī)制!");
toast.show();
隱藏toast:
toast.hide();
注意在4.0以上的版本中屠尊,還需要對(duì)Toast 中的View進(jìn)行處理旷祸,如代碼中所示;
場(chǎng)景三:ListView組件沒有提供修改快速滑塊圖像的API,因此不能直接修改,但可通過反射實(shí)現(xiàn)
package com.example.reflection;
import java.lang.reflect.Field;
import android.content.Context;
import android.graphics.drawable.Drawable;
import android.util.AttributeSet;
import android.widget.AbsListView;
import android.widget.ListView;
public class MListView extends ListView
{
public MListView(Context context, AttributeSet attrs)
{
super(context, attrs);
// TODO Auto-generated constructor stub
setNewDrawable(context);
}
private void setNewDrawable(Context context)
{
try {
Field field = AbsListView.class.getDeclaredField("mFastScroller");
field.setAccessible(true);
Object obj = field.get(this);
field =field.getType().getDeclaredField("mThumbDrawable");
field.setAccessible(true);
Drawable drawable = (Drawable)field.get(obj);
drawable = context.getResources().getDrawable(R.drawable.ic_launcher);
field.set(obj, drawable);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
反射機(jī)制通過void setAccessible(boolean flag)方法可以得到一個(gè)類的private的方法和屬性讼昆,使用這些private的方法和屬性托享,可以做一些超越限制的事情,所以在使用的時(shí)候控淡,還是要謹(jǐn)慎啊