java中反射的使用
结合demo理解反射:
import java.lang.reflect.*;
/**
* 反射使用
**/
public class ReflectDemo{
public static void main(String[] args){
ReflectDemo demo = new ReflectDemo();
if(args.length == 0){
demo.showClassInfo(null);
}else{
demo.showClassInfo(args[0]);
}
demo.getAndSetPrivateFieldValue();
demo.callPrivateMethod();
} private void showClassInfo(String className){
new showReflectionInfo().showClassInfo(className != null && className.length() > 0 ? className : "Test");
} /**
* 利用反射获取并修改私有域的值
**/
private void getAndSetPrivateFieldValue(){
try{
// 利用反射创建实例
Class _testClass = Class.forName("Test");// Class.forName()方法获取Class对象
Object _t = _testClass.newInstance();// 使用Class对象.newInstance()创建对象实例 Field _f = Test.class.getDeclaredField("name");
_f.setAccessible(true);// 设置私有的域可被访问
Object _value = _f.get(_t); // 使用get(Object obj)方法获取私有域的值
System.out.println(_value);// 输出 zhangsan _f.set(_t, "wangwu");// 使用set(Object obj, Object newValue)方法修改私有域的值
_value = _f.get(_t);
System.out.println(_value);// 输出 wangwu
System.out.println();
}catch(Exception e){
e.printStackTrace();
}
} /**
* 利用反射调用其他类的私有方法
**/
private void callPrivateMethod(){
Test _t= new Test(); System.out.println(_t.getName());
//_t.setName("lisi");// setName为Test类的私有方法,外部不可以直接这样调用// 使用反射机制调用Test类的私有方法setName()
Method _m = null;
try{
//
_m = Test.class.getDeclaredMethod("setName", String.class);
/*
方法二:使用反射得到setName()方法
Class _cl = _t.getClass();
Method[] _ms = _cl.getDeclaredMethods();
for(int i=0;i<_ms.length;i++){
if(_ms[i].getName().equals("setName")){
_m = _ms[i];
break;
}
}
*/ _m.setAccessible(true);// 不设置可访问权限,会报错java.lang.IllegalAccessException: Class ReflectDemo can not access a member of class Test with modifiers "private"
_m.invoke(_t, "lisi");// invoke表示调用setName()方法,这里m就是setName()方法
}catch(Exception e){
e.printStackTrace();
}
System.out.println(_t.getName());
}
} class Test{
private String name = "zhangsan";
public String getName(){
return name;
} private void setName(String paramName){
name = paramName;
}
} class showReflectionInfo{ /**
* 利用反射分析类的能力
**/
public void showClassInfo(String paramClassName){
if(paramClassName != null && paramClassName.length() > 0){
try{
StringBuilder _sb = new StringBuilder("");
Class _class = Class.forName(paramClassName);
Class _superClass = _class.getSuperclass(); String _modifier = Modifier.toString(_class.getModifiers());
if(_modifier != null && _modifier.length() > 0){
_sb.append(_modifier.concat(" "));
}
_sb.append("class ".concat(_class.getName()));
if(_superClass != null && _superClass != Object.class){
_sb.append(" extends ".concat(_superClass.getName()));
}
_sb.append(" {\n"); printFields(_sb, _class);
_sb.append("\n");
printConstructors(_sb, _class);
_sb.append("\n");
printMethods(_sb, _class); _sb.append("}\n"); System.out.println(_sb.toString());
}catch(Exception e){
e.printStackTrace();
} }
} /**
* 打印这个类的所有域
**/
private void printFields(StringBuilder paramStringBuilder, Class paramClass){
if(paramStringBuilder == null || paramClass == null){
return;
}
// getDeclaredFields()返回这个类或接口的全部域(含私有域);getFields()返回本类/接口及其所有父类/接口的所有公有域
Field[] _fields = paramClass.getDeclaredFields();
String _modifier = "";
for(Field _f:_fields){
paramStringBuilder.append(" ");
_modifier = Modifier.toString(_f.getModifiers());
if(_modifier != null && _modifier.length() > 0){
paramStringBuilder.append(_modifier.concat(" "));
}
Class _type = _f.getType();
paramStringBuilder.append(_type.getName().concat(" "));
paramStringBuilder.append(_f.getName().concat(";\n"));
}
} /**
* 打印这个类的所有构造方法
**/
private void printConstructors(StringBuilder paramStringBuilder, Class paramClass){
if(paramStringBuilder == null || paramClass == null){
return;
}
// getDeclaredConstructors()返回这个类的全部构造方法(含私有的);getConstructors()返回这个类所有公有构造方法
Constructor[] _constructors = paramClass.getDeclaredConstructors();
String _modifier = "";
for(Constructor _c:_constructors){
paramStringBuilder.append(" ");
_modifier = Modifier.toString(_c.getModifiers());
if(_modifier != null && _modifier.length() > 0){
paramStringBuilder.append(_modifier.concat(" "));
} paramStringBuilder.append(_c.getName().concat("("));
Class[] _paramTypes = _c.getParameterTypes();
int _len = _paramTypes.length;
for(int i = 0;i< _len;i++){
paramStringBuilder.append(_paramTypes[i].getName());
if(i<_len - 1){
paramStringBuilder.append(",");
}
}
paramStringBuilder.append(");\n");
}
} /**
* 打印这个类的所有非构造方法
**/
private void printMethods(StringBuilder paramStringBuilder, Class paramClass){
if(paramStringBuilder == null || paramClass == null){
return;
} // getDeclaredMethods()返回这个类或接口的全部方法(含私有方法);getMethods()返回本类/接口及其所有父类/接口的所有公有方法
Method[] _methods = paramClass.getDeclaredMethods();
String _modifier = "";
for(Method _m:_methods){
paramStringBuilder.append(" ");
// Modifier中静态方法:Modifier.toString()、Modifier.isAbstract()、Modifier.isPublic()、Modifier.isPrivate()...
_modifier = Modifier.toString(_m.getModifiers());
if(_modifier != null && _modifier.length() > 0){
paramStringBuilder.append(_modifier.concat(" "));
}
paramStringBuilder.append(_m.getReturnType().getName().concat(" ").concat(_m.getName()).concat("(")); Class[] _paramTypes = _m.getParameterTypes();
int _len = _paramTypes.length;
for(int i = 0;i< _len;i++){
paramStringBuilder.append(_paramTypes[i].getName());
if(i<_len - 1){
paramStringBuilder.append(",");
}
}
paramStringBuilder.append(");\n");
}
}
}
输出结果如下:
第2个例子:
import java.util.*;
import java.nio.file.*;
import java.nio.charset.*;
import java.io.*;
import java.lang.reflect.*;
/**
* 利用反射机制使用Scanner类中私有构造方法private Scanner(Path source, Charset charset)读取打印文件内容
**/
public class ReflectDemo1{ public static void main(String[] args){
try{ // ###### 获取构造函数Constructor对象 方法一:######
// 注意:使用getDeclaredConstructor()方法获取构造函数时声明处 Class<?> cl 切记一定要加<?>
Class<?> cl = Class.forName("java.util.Scanner"); // 也可以Class<?> cl = Scanner.class;
Constructor cons = cl.getDeclaredConstructor(Path.class,Charset.class); /*
// ###### 获取构造函数Constructor对象 方法二:######
Class cl = Class.forName("java.util.Scanner");
Constructor cons = null; Constructor[] cts = cl.getDeclaredConstructors();
Class[] cls;
for(Constructor ct:cts){
cls = ct.getParameterTypes();
List<String> ll = new ArrayList<String>(cls.length);
for(Class c:cls){
ll.add(c.getName());
} if(ll.contains(Path.class.getName()) && ll.contains(Charset.class.getName()) ){
cons = ct;
break;
}
}
*/ if(cons !=null){
System.out.println("find it");
cons.setAccessible(true);// 因为这个构造方法是私有的,需要设置权限
Scanner scanner = (Scanner) cons.newInstance(Paths.get("ReflectDemo1.java"),Charset.forName("utf-8"));
String str = "";
while((str = scanner.nextLine())!=null){
System.out.println(str);
}
}
}catch(Exception e){
}
}
}
java中反射的使用的更多相关文章
- Java中反射的三种常用方式
Java中反射的三种常用方式 package com.xiaohao.test; public class Test{ public static void main(String[] args) t ...
- java中反射学习整理
转载请注明:http://blog.csdn.net/j903829182/article/details/38405735 反射主要是指程序能够訪问.检測和改动它本身的状态或行为的一种能力. jav ...
- Java中反射的实现方式
所谓反射,是指在运行时状态中,获取类中的属性和方法,以及调用其中的方法的一种机制.这种机制的作用在于获取运行时才知道的类(Class)及其中的属性(Field).方法(Method)以及调用其中的方法 ...
- 简单模拟Java中反射的应用场景
有人说Java是一门静态语言.那么何为静态语言,动态语言又是什么? 1.动态语言 是一类在运行时可以改变其结构的语言:例如新的函数.对象.甚至代码可以 被引进,已有的函数可以被删除或是其他结构上的变化 ...
- Java中反射机制和Class.forName、实例对象.class(属性)、实例对象getClass()的区别
一.Java的反射机制 每个Java程序执行前都必须经过编译.加载.连接.和初始化这几个阶段,后三个阶段如下图: 其中
- Java中反射与常用方法
java通常是先有类再有对象,有对象我就可以调用方法或者属性. 反射其实是通过Class对象来调用类里面的方法.通过反射可以调用私有方法和私有属性.大部分框架都是运用反射原理. 如何获得Class ...
- Java中反射机制详解
序言 在学习java基础时,由于学的不扎实,讲的实用性不强,就觉得没用,很多重要的知识就那样一笔带过了,像这个马上要讲的反射机制一样,当时学的时候就忽略了,到后来学习的知识中,很多东西动不动就用反射, ...
- java中反射讲解及实例
Java反射机制详解 java 反射 定义 功能 示例 概要: Java反射机制详解 | |目录 1反射机制是什么 2反射机制能做什么 3反射机制的相关API ·通过一个对象获得完整的包名和类名 ·实 ...
- Java中反射的理解
反射 一.什么是反射 Java 反射是Java语言的一个很重要的特征,它使得Java具体了"动态性". 反射用在 Java 身上指的是我们可以于运行时加载.探知.使用编译期间完全未 ...
- java中反射
Person.java===>>person.class ==>>jvm中的类加载器===>>class对象:代表内存中Person.class ==>> ...
随机推荐
- GoCN每日新闻(2019-10-11)
GoCN每日新闻(2019-10-11) GoCN每日新闻(2019-10-11) 1. golang 将数据库转换为gorm结构 https://studygolang.com/articles/2 ...
- Tkinter 之使用PAGE工具开发GUI界面
一.安装 1.官网下载 PAGE http://page.sourceforge.net/ Tcl(8.6+) https://www.activestate.com/activetcl/downlo ...
- es6学习2:变量的解构赋值
一:数组的解构赋值 ES6 允许按照一定模式,从数组和对象中提取值,对变量进行赋值,这被称为解构 let [foo, [[bar], baz]] = [1, [[2], 3]]; foo bar ba ...
- Linux 磁盘的分区
如果我们想在系统中新增一块硬盘,需要做什么呢? 1. 对磁盘进行分区,新建可用分区 2. 对该分区进行格式化,以创建系统可用的文件系统 3. 若想要仔细一点,可以对刚才新建好的文件系统进行检验 4. ...
- md5加密后不能解密
MD5加密原理是散列算法,散列算法也称哈希算法.计算机专业学的数据结构就有哈希表这一知识点.比如10除以3余数为一,4除以3余数也为一,但余数为一的就不知道这个数是哪个了.所以md5不能解密.就算是设 ...
- 【转】Fiddler抓包和修改WebSocket数据,支持wss
记录一下用Fiddler对WebSocket收发的数据进行抓包分析和篡改数据,只找到这么一个方法,能用就行吧.时间:2019-3-29环境: win7 + Fiddler 5.0 Fiddler抓取W ...
- jmeter测试 flask 接口请求
jmeter测试 flask 接口请求 flask的代码如下: #!/usr/bin/env python # -*- coding: utf-8 -*- from flask import Flas ...
- openstack错误问题定位及调试
- Xamarin图表开发基础教程(11)OxyPlot框架支持的图表类型
Xamarin图表开发基础教程(11)OxyPlot框架支持的图表类型 OxyPlot组件中支持7种类型的条型图表,分别为普通条形图.线型条形图.矩形条形图.差值图.龙卷风图.普通柱形图和柱形误差图, ...
- activiti 自定义用户
https://blog.csdn.net/meng564764406/article/details/53789958 此文目的: 对网络上的关于对activiti 使用做一个总结,因为很难找到一个 ...