前段时间由于项目需要实现了展开通知栏的功能,所谓展开通知栏,就是通过expand  statusbar的方式将所有通知显示出来,也就是手势由顶部下滑展开的界面,statusbar属于全局且系统级别的界面,API中并没有开放独立的接口供我们调用以完成通知栏的展开,所以这里需要用到反射,通过反射调用方法,使用的方法是invoke(),这个方法是Method的方法,可能大家不太理解,我对这部分也知之甚少,用我自己最粗浅的理解就是,通过方法名创建Method对象,通过该对象的invoke()方法反射出方法(这一步其实就是方法的执行),供我们调用,这样说起来很苍白,先上一段代码解释下吧。


public class Test { 

    public static void main(String args[]){ 
        A a = new A; 
        Class class = A.getClass(); 
        Method m1 = class.getDeclaredMethod("outInfo"); 
        Method m2 = clazz.getDeclaredMethod("setMsg", String.class); 
        Method m3 = clazz.getDeclaredMethod("getMsg"); 
        m1.invoke(a); 
        m2.invoke(a, "重新设置msg信息!"); 
        String msg = (String) m3.invoke(a); 
        System.out.println(msg); 
    } 
} 

class A { 
    private String msg; 

    public A(String msg) { 
        this.msg = msg; 
    } 

    public void setMsg(String msg) { 
        this.msg = msg; 
    } 

    public String getMsg() { 
        return msg; 
    } 

    public void outInfo() { 
        System.out.println("这是测试Java反射的测试类"); 
    } 
}


这里用到的是最简单的反射调用方法的实现,更深入的反射的使用还需要我们继续深入的学习,下面就是解决 android 展开通知栏的问题,先上代码再粗浅的解释下。


<span style="font-family: Arial; background-color: rgb(255, 255, 255);">首先我们需要获得系统服务,前面说过,</span><span style="font-family: 'Times New Roman'; background-color: rgb(255, 255, 255);">statusbar</span><span style="font-family: 宋体; background-color: rgb(255, 255, 255);">是全局且系统级别的布局,然而对其实现展开还需要增加一项权限,这个不能忽略</span>

<uses-permission android:name="android.permission.EXPAND_STATUS_BAR" />

StatusBarManager的包名反射得到statusbarManager对象,再通过expand方法名得到Method这个类的对象expand,下面执行该方法,expand.invoke().这样便能实现展开通知栏,这个方法在android4.2以下版本中测试良好,可是近来发现4.2及以上版本中这个方法失效了,4.2中对通知栏做了改变,由两部分组成,一个是我们通常意义上的通知栏,如下图

android Activity暴露风险_android

仔细看右上角有个头像按钮,点击这个,通知栏便会翻转至另外一个界面,同样是通过通知栏的形式展示的,如下图


android Activity暴露风险_android_02

API的更新,于是我上了神网stackoverflow,反编译了别的APK,最后终于被我试出来了,现在把代码给大家贴一下。


public void OpenNotify() {
		// TODO Auto-generated method stub
		try {
			Object service = getSystemService("statusbar");
			Class<?> statusbarManager = Class
					.forName("android.app.StatusBarManager");
			Method expand = null;
			if (service != null) {
				expand = statusbarManager.getMethod("expand");
				expand.setAccessible(true);
				expand.invoke(service);
			}

		} catch (Exception e) {
		}

	}
public void OpenNotify() {
		// TODO Auto-generated method stub
		int currentApiVersion = android.os.Build.VERSION.SDK_INT;
		try {
			Object service = getSystemService("statusbar");
			Class<?> statusbarManager = Class
					.forName("android.app.StatusBarManager");
			Method expand = null;
			if (service != null) {
				if (currentApiVersion <= 16) {
					expand = statusbarManager.getMethod("expand");
				} else {
					expand = statusbarManager
							.getMethod("expandNotificationsPanel");
				}
				expand.setAccessible(true);
				expand.invoke(service);
			}

		} catch (Exception e) {
		}

	}




就是在上面代码的基础上做了点改进,很一目了然,不做多的解释了。要说声抱歉,第一次发的文章格式很有问题,现在已经改正了。

demo下载地址。