- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章JAVA编程不能不知道的反射用法总结由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
首先定义一个MyBase类,其中有私有字段,也有公有字段。同时也有公有方法和私有方法。MyBase类示例如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
package
com.hwdev.demo;
/**
* 基类示例
* @author wangming
*/
public
class
MyBase {
//公有字段
public
int
version =
1
;
//私有字段
private
String date =
"2021-05-18"
;
//公有方法
public
void
say2(String msg){
System.out.println(
"Hello "
+ msg);
}
//私有方法
private
String getData(){
return
this
.date;
}
}
//加入Java开发交流君样:756584822一起吹水聊天
|
这里再定义一个Hello类,它继承自MyBase类,通过继承主要用于验证一下反射对于父类、子类的反射用法.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
package
com.hwdev.demo;
/**
*
* @author wangming
*/
public
class
Hello
extends
MyBase {
public
String author =
"JackWang"
;
public
int
version =
1
;
private
String company =
"kzcloud"
;
public
void
say(String msg){
System.out.println(
"Hello "
+ msg);
}
public
void
setAuthor(String author){
this
.author = author;
}
public
String getAuthor(){
return
this
.author;
}
private
int
getVersion(){
return
this
.version;
}
}
|
关于Java反射,功能强大的就是可以通过字符串配置来动态从系统中调用方法或者修改其中某个对象的字段值,而Class.forName方法即可以通过传入类全路径字符串名称来获取对应的Class对象,非常的方便。另外通过getField方法和GetMethod方法可以获取指定字段和方法,并动态调用.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
|
package
com.hwdev.demo;
import
java.lang.reflect.*;
import
java.util.Arrays;
/**
* 反射第一种用法 Class.forName
* @author wangming
*/
public
class
ReflectDemo01 {
public
static
void
Test() {
try
{
//通过字符串全路径类名查找Class
Class helloC = Class.forName(
"com.hwdev.demo.Hello"
);
//获取所有公有的字段数组,私有的无法获取
Field [] fields = helloC.getFields();
//打印字段数组内容//加入Java开发交流君样:756584822一起吹水聊天
System.out.println(Arrays.toString(fields));
//[public java.lang.String com.hwdev.demo.Hello.author, public int com.hwdev.demo.Hello.version]
//实例化
Object obj = helloC.newInstance();
//获取特定字段,比遍历Field[]效率更高
Field f = helloC.getField(
"author"
);
if
(f !=
null
){
//关闭安全检查,提高效率
f.setAccessible(
true
);
//获取字段author内容
String author = (String)f.get(obj);
System.out.println(
"author="
+ author);
//author=JackWang
}
//获取所有公有的方法数组,私有的无法获取
Method [] methods = helloC.getMethods();
//打印方法数组内容,子类等方法也可以获取到
System.out.println(Arrays.toString(methods));
//本类所有方法
Method [] methods2 = helloC.getDeclaredMethods();
//打印方法数组内容
System.out.println(Arrays.toString(methods2));
//获取特定方法,第二个参数String.class为say方法的参数类型
//say(java.lang.String)
Method m = helloC.getDeclaredMethod(
"say"
,String.
class
);
if
(m !=
null
){
//关闭安全检查,提高效率
m.setAccessible(
true
);
//获取字段author内容
Object returnValue = m.invoke(obj,
new
Object[]{
"Java"
});
//Hello Java
if
(returnValue!=
null
){
System.out.println(
"returnValue ="
+ returnValue);
}
}
//加入Java开发交流君样:756584822一起吹水聊天
}
catch
(ClassNotFoundException | SecurityException ex){
ex.printStackTrace();
}
catch
(Exception ex){
ex.printStackTrace();
}
}
}
|
这里需要注意:xxx.getMethods()方法默认情况下,会返回本类、父类、父接口的公有方法,而xxx.getDeclaredMethods()返回本类的 所有方法,包括私有的方法。同理,反射API中其他getXXX和getDeclaredXXX的用法类似.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
package
com.hwdev;
import
com.hwdev.demo.ReflectDemo01;
/**
*
* @author wangming
*/
public
class
Main {
/**
* @param args the command line arguments
*/
public
static
void
main(String[] args) {
//反射第一种用法 Class.forName
ReflectDemo01.Test();
}
}
|
执行程序,输出的结果如下:
[public java.lang.String com.hwdev.demo.Hello.author, public int com.hwdev.demo.Hello.version, public int com.hwdev.demo.MyBase.version] author=JackWang [public void com.hwdev.demo.Hello.say(java.lang.String), public void com.hwdev.demo.Hello.setAuthor(java.lang.String), public java.lang.String com.hwdev.demo.Hello.getAuthor(), public void com.hwdev.demo.MyBase.say2(java.lang.String), public final void java.lang.Object.wait() throws java.lang.InterruptedException, public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException, public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException, public boolean java.lang.Object.equals(java.lang.Object), public java.lang.String java.lang.Object.toString(), public native int java.lang.Object.hashCode(), public final native java.lang.Class java.lang.Object.getClass(), public final native void java.lang.Object.notify(), public final native void java.lang.Object.notifyAll()] [public void com.hwdev.demo.Hello.say(java.lang.String), public void com.hwdev.demo.Hello.setAuthor(java.lang.String), public java.lang.String com.hwdev.demo.Hello.getAuthor(), private int com.hwdev.demo.Hello.getVersion()] Hello Java 。
从输出结果上来看,Field [] fields = helloC.getFields();不但可以获取Hello类的公有字段,还可以获取到父类MyBase的公有字段:com.hwdev.demo.MyBase.version 。
而Method [] methods2 = helloC.getDeclaredMethods();则可以获取本类,即Hello类所有的方法,包括公有的方法和私有的方法。因此,Java反射可以访问类的私有字段和方法,从而暴露内部的信息,这也是Java反射有安全问题的原因.
由于Java方法支持重载,因此同名的方法可以存在多个,即参数不同,因此在用反射调用方法时,需要指定方法的参数类型,这样就可以明确调到的具体是哪个方法签名,如`Method m = helloC.getDeclaredMethod("say",String.class); `调用的是`public void com.hwdev.demo.Hello.say(java.lang.String) `.
除了可以用Class.forName来进行反射外,还可以通过如下方式来获取反射对象:
1
2
3
4
5
6
|
Hello hello =
new
Hello();
Class helloC = hello.getClass();
Field [] fields = helloC.getFields();
//
Class helloC = Hello.
class
;
Field [] fields = helloC.getFields();
|
下面介绍一下如何用Java反射修改私有字段和调用私有方法的示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
|
package
com.hwdev.demo;
import
java.lang.reflect.*;
/**
* 反射访问私有字段和方法
* @author wangming
*/
public
class
ReflectDemo02 {
//加入Java开发交流君样:756584822一起吹水聊天
public
static
void
Test() {
try
{
//通过已有类查找Class
Class helloC = Hello.
class
;
//实例化
Object obj = helloC.newInstance();
//获取特定私有字段
Field f = helloC.getDeclaredField(
"company"
);
if
(f !=
null
){
//私有必须开启
f.setAccessible(
true
);
//设置私有字段值
f.set(obj,
"newKZ"
);
//获取字段author内容
String fv = (String)f.get(obj);
System.out.println(
"company="
+ fv);
//company=newKZ
}
//获取私有方法
Method m = helloC.getDeclaredMethod(
"getVersion"
,
null
);
if
(m !=
null
){
//私有必须开启
m.setAccessible(
true
);
Object returnValue = m.invoke(obj,
null
);
if
(returnValue!=
null
){
//returnValue =1
System.out.println(
"returnValue ="
+ returnValue);
}
}
}
catch
(SecurityException ex){
ex.printStackTrace();
}
catch
(Exception ex){
ex.printStackTrace();
}
}
}
|
另外,Java反射可以获取注解信息,这个对于ORM框架来讲,用的非常多.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
package
com.hwdev.demo;
import
java.lang.annotation.ElementType;
import
java.lang.annotation.Retention;
import
java.lang.annotation.RetentionPolicy;
import
java.lang.annotation.Target;
/**
* 注解示例
* @author wangming
*/
@Retention
(RetentionPolicy.RUNTIME)
@Target
(ElementType.FIELD)
public
@interface
ORMAnnotation {
public
String FieldName();
public
String FieldType();
}
|
其中,@Retention(RetentionPolicy.RUNTIME)表示注解可以在运行时通过反射访问。@Target(ElementType.FIELD) 表示这个注解只能用在字段上面。同理,可以把FIELD改为Type或者Method等.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
package
com.hwdev.demo;
import
java.lang.annotation.Annotation;
import
java.lang.reflect.*;
/**
* 反射或者字段注解
* @author wangming
*/
//加入Java开发交流君样:
756584822
一起吹水聊天
public
class
ReflectDemo03 {
public
static
void
Test() {
try
{
Class helloC = Class.forName(
"com.hwdev.demo.HelloAnnotation"
);
Field[] fields = helloC.getDeclaredFields();
for
(Field f : fields){
//关闭安全检查,提高效率
f.setAccessible(
true
);
Annotation ann = f.getAnnotation(ORMAnnotation.
class
);
if
(ann
instanceof
ORMAnnotation){
ORMAnnotation ormAnn = (ORMAnnotation) ann;
System.out.println(
"FieldName="
+ ormAnn.FieldName());
System.out.println(
"FieldType="
+ ormAnn.FieldType());
}
}
}
catch
(ClassNotFoundException | SecurityException ex){
ex.printStackTrace();
}
catch
(Exception ex){
ex.printStackTrace();
}
}
}
|
执行此示例,则输出如下:
FieldName=f_author FieldType=varchar(50) FieldName=f_ver FieldType=int 。
再次,介绍一下如何用反射获取方法参数个数和类型,其中包含泛型的信息获取:
。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
|
package
com.hwdev.demo;
import
java.util.ArrayList;
import
java.util.List;
/**
* 泛型示例
* @author wangming
*/
public
class
GenericCls {
protected
List<String> myList =
new
ArrayList();
public
GenericCls(
int
size){
for
(
int
i =
0
;i<size;i++){
myList.add(
"item"
+i);
}
}
public
List<String> getList(){
return
this
.myList;
}
public
String getList(
int
idx){
return
this
.myList.get(idx);
}
}
package
com.hwdev.demo;
import
java.lang.reflect.*;
/**
* 反射获取方法参数
* @author wangming
*/
public
class
ReflectDemo05 {
public
static
void
Test() {
try
{
//加入Java开发交流君样:756584822一起吹水聊天
Class helloC = Class.forName(
"com.hwdev.demo.GenericCls"
);
//构造函数调用
Object obj = helloC.getConstructor(
int
.
class
).newInstance(
3
);
Method method = helloC.getMethod(
"getList"
,
int
.
class
);
Class<?> returnType = method.getReturnType();
System.out.println(
"ReturnType = "
+ returnType.getName());
Parameter[] params = method.getParameters();
for
(Parameter p : params){
System.out.println(
"ParameterizedType = "
+ p.getParameterizedType());
System.out.println(
"getModifiers = "
+ p.getModifiers());
System.out.println(
"getName = "
+ p.getName());
System.out.println(
"getType = "
+ p.getType());
}
//调用方法
Object ret = method.invoke(obj,
new
Object[]{
2
});
System.out.println(
"ret = "
+ ret.toString());
Method method2 = helloC.getMethod(
"getList"
,
null
);
Type greturnType = method2.getGenericReturnType();
System.out.println(
"getGenericReturnType = "
+ returnType.getName());
if
(greturnType
instanceof
ParameterizedType){
ParameterizedType type = (ParameterizedType) greturnType;
System.out.println(
"type = "
+ type.getTypeName());
Type[] typeArguments = type.getActualTypeArguments();
for
(Type typeArgument : typeArguments){
Class typeArgClass = (Class) typeArgument;
System.out.println(
"typeArgClass = "
+ typeArgClass);
}
}
}
catch
(ClassNotFoundException | SecurityException ex){
ex.printStackTrace();
}
catch
(Exception ex){
ex.printStackTrace();
}
}
}
|
。
执行此示例,则输出如下:
R`eturnType = java.lang.String ParameterizedType = int getModifiers = 0 getName = arg0 getType = int ret = item2 getGenericReturnType = java.lang.String type = java.util.List<java.lang.String> typeArgClass = class java.lang.String 。
本篇文章就到这里了,希望能给你带来帮助,也希望您能够多多关注我的更多内容! 。
原文链接:https://blog.csdn.net/wj1314250/article/details/117135271 。
最后此篇关于JAVA编程不能不知道的反射用法总结的文章就讲到这里了,如果你想了解更多关于JAVA编程不能不知道的反射用法总结的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我正在尝试打印 timeval 类型的值。实际上我可以打印它,但我收到以下警告: 该行有多个标记 格式“%ld”需要“long int”类型,但参数 2 的类型为“struct timeval” 程序
我正在编写自己的 unix 终端,但在执行命令时遇到问题: 首先,我获取用户输入并将其存储到缓冲区中,然后我将单词分开并将它们存储到我的 argv[] 数组中。IE命令是“firefox”以启动存储在
我是 CUDA 的新手。我有一个关于一个简单程序的问题,希望有人能注意到我的错误。 __global__ void ADD(float* A, float* B, float* C) { con
我有一个关于 C 语言 CGI 编程的一般性问题。 我使用嵌入式 Web 服务器来处理 Web 界面。为此,我在服务器中存储了一个 HTML 文件。在此 HTML 文件中包含 JavaScript 和
**摘要:**在代码的世界中,是存在很多艺术般的写法,这可能也是部分程序员追求编程这项事业的内在动力。 本文分享自华为云社区《【云驻共创】用4种代码中的艺术试图唤回你对编程的兴趣》,作者: break
我有一个函数,它的任务是在父对象中创建一个变量。我想要的是让函数在调用它的级别创建变量。 createVariable testFunc() [1] "test" > testFunc2() [1]
以下代码用于将多个连续的空格替换为1个空格。虽然我设法做到了,但我对花括号的使用感到困惑。 这个实际上运行良好: #include #include int main() { int ch, la
我正在尝试将文件写入磁盘,然后自动重新编译。不幸的是,某事似乎不起作用,我收到一条我还不明白的错误消息(我是 C 初学者 :-)。如果我手动编译生成的 hello.c,一切正常吗?! #include
如何将指针值传递给结构数组; 例如,在 txt 上我有这个: John Doe;xxxx@hotmail.com;214425532; 我的代码: typedef struct Person{
我尝试编写一些代码来检索 objectID,结果是 2B-06-01-04-01-82-31-01-03-01-01 . 这个值不正确吗? // Send a SysObjectId SNMP req
您好,提前感谢您的帮助, (请注意评论部分以获得更多见解:即,以下示例中的成本列已添加到此问题中;西蒙提供了一个很好的答案,但成本列本身并未出现在他的数据响应中,尽管他提供的功能与成本列一起使用) 我
我想知道是否有人能够提出一些解决非线性优化问题的软件包的方法,而非线性优化问题可以为优化解决方案提供整数变量?问题是使具有相等约束的函数最小化,该函数受某些上下边界约束的约束。 我已经在R中使用了'n
我是 R 编程的初学者,正在尝试向具有 50 列的矩阵添加一个额外的列。这个新列将是该行中前 10 个值的平均值。 randomMatrix <- generateMatrix(1,5000,100,
我在《K&R II C 编程 ANSI C》一书中读到,“>>”和“0; nwords--) sum += *buf++; sum = (sum >>
当下拉列表的选择发生变化时,我想: 1) 通过 div 在整个网站上显示一些 GUI 阻止覆盖 2)然后处理一些代码 3) 然后隐藏叠加层。 问题是,当我在事件监听器函数中编写此逻辑时,将执行 onC
我正在使用 Clojure 和 RESTEasy 设计 JAX-RS REST 服务器. 据我了解,用 Lisp 系列语言编写的应用程序比用“传统”命令式语言编写的应用程序更多地构建为“特定于领域的语
我目前正在研究一种替代出勤监控系统作为一项举措。目前,我设计的用户表单如下所示: Time Stamp Userform 它的工作原理如下: 员工将选择他/她将使用的时间戳类型:开始时间、超时、第一次
我是一名学生,试图自学编程,从在线资源和像您这样的人那里获得帮助。我在网上找到了一个练习来创建一个小程序来执行此操作: 编写一个程序,读取数字 a 和 b(长整型)并列出 a 和 b 之间有多少个数字
我正在尝试编写一个 shell 程序,给定一个参数,打印程序的名称和参数中的每个奇数词(即,不是偶数词)。但是,我没有得到预期的结果。在跟踪我的程序时,我注意到,尽管奇数词(例如,第 5 个词,5 %
只是想知道是否有任何 Java API 可以让您控制台式机/笔记本电脑外壳上的 LED? 或者,如果不可能,是否有可能? 最佳答案 如果你说的是前面的 LED 指示电源状态和 HDD 繁忙状态,恐怕没
我是一名优秀的程序员,十分优秀!