黑马程序员——反射

------- android培训java培训iOS培训.Net培训期待与您交流! ----------


类的加载

当程序要使用某个类时如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类进行初始化。

类初始化时机

创建类的实例

访问类的静态变量,或者为静态变量赋值

调用类的静态方法

使用反射方式来强制创建某个类或接口对应的java.lang.Class对象

初始化某个类的子类

直接使用java.exe命令来运行某个主类

类加载器

负责将.class文件加载到内存中,并为之生成对应的Class对象

虽然我们不需要关心类加载机制,但是了解这个机制我们就能更好的理解程序的运行

类加载器的组成

Bootstrap ClassLoader根类加载器、Extension ClassLoader扩展类加载器、System ClassLoader系统类加载器

反射

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对应任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调节对象的方法的功能称为java语言的反射机制。

要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法,所以先要获取到每一个字节码文件对应的Class类型的对象

就是通过class文件对象,去使用该文件中的成员变量、构造方法、成员方法

要想这样使用,首先必须得到class文件对象,其实也就是得到Class类的对象

Class类:

成员变量 Field

构造方法 Constructor

成员方法 Method

获取class文件对象的方式:

一、Object类的getClass()方法

Person p1=new Person();

Class c1=p1.getClass();

二、数据类型的静态属性class

Class c2=Person.class;

三、Class类中的静态方法 :public static Class forName(String className)

Class c3=Class.forName("Person");

注:要写类名的全路径(带包名的)

开发时用第三种,因为第三种是一个字符串,而不是一个具体的类名,这样我们就可以把这样的字符串配置到配置文件中

以下都以Person类为例:

class Person
{
	private String name;
	int age;
	public String address; 
	
	public Person()
	{
	
	}
	Person(String name,int age)
	{
		this.name=name;
		this.age=age;
	}
	public Person(String name,int age,String address)
	{
		this.name=name;
		this.age=age;
		this.address=address;
	}

	public void show()
	{
		System.out.println("show");
	}
	public void method(String s)
	{
		System.out.println("method"+s);
	}
	public String getString(String s,int i)
	{
		return s+"---"+i;
	}
	private void function()
	{
		System.out.println("function");
	}	
	public String toString()
	{
		return "Person [name="+name+",age="+age+",address="+address+"]";
	}
	
}


通过反射获取构造方法并使用

public Constructor[] getConstructors():获取所有公共构造方法

public Constructor[] getDeclaredConstructors():所有构造方法

public Constructor getConstructor(Class... parameterTypes):获取单个构造方法

...表示可传参数也可不传

参数表示的是:你要获取的构造方法的构造参数个数及数据类型的class字节码文件对象

public T newInstance(Object... initargs)

使用此Constructor对象表示的构造方法来创建该构造方法的声明类的新实例,并用指定的初始化参数初始化该实例

public Constructor getConstructor(Class... parameterTypes):获取带参构造方法对象

暴力访问

con.setAccessible(true);//值为true则指示反射的对象在使用时应该取消JAVA语言访问检查

 

import java.lang.reflect.Constructor;
class ReflectDemo 
{
	public static void main(String[] args) throws Exception
	{	
		//获取字节码文件对象
		Class c=Class.forName("Person");

		//Constructor[] cons=c.getConstructors();
		//Constructor[] cons=c.getDeclaredConstructors();
		//for(Constructor con : cons)
		//{
		//	System.out.println(con);
		//}

		Constructor con=c.getConstructor();//返回的是构造方法对象

		Object obj=con.newInstance();
		System.out.println(obj);//底层调用toString方法
	}
}

/*
需求:<strong>通过反射去获取该构造方法并使用</strong>:public Person(String name,int age,String address)
*/
import java.lang.reflect.Constructor;
class  ReflectDemo
{
	public static void main(String[] args) throws Exception
	{
		//获取字节码文件对象
		Class c=Class.forName("Person");

		Constructor con=c.getConstructor(String.class,int.class,String.class);

		Object obj=con.newInstance("张三",29,"北京");
		System.out.println(obj);
	}
}

/*
需求:<strong>通过反射获取私有构造方法并使用</strong>:private Person(String name){}
*/
import java.lang.reflect.Constructor;
class  ReflectDemo
{
	public static void main(String[] args) throws Exception
	{
		//获取字节码文件对象
		Class c=Class.forName("Person");

		//获取私有构造方法对象
		//会报异常:NoSuchMethodException  没有这个方法异常
		//原因是一开始我们使用的方法只能获取公共的
		//Constructor con=c.getConstructor(String.class);

		//用该私有构造方法创建对象
		//也会报异常:IllegalAccessException 非法的访问异常
		Constructor con=c.getDeclaredConstructor(String.class);

		//暴力访问
		con.setAccessible(true);		

		//通过私有构造方法对象创建对象	
		Object obj=con.newInstance("张三");
		System.out.println(obj);
	}
}


通过反射获取成员变量并使用

获取所有成员:getFields getDeclaredFields

获取单个成员:getField getDeclaredField

修改成员变量的值:set(Object obj,Object value)

将指定对象变量上此Field对象表示的字段设置为指定的新值

 

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
class ReflectDemo 
{
	public static void main(String[] args) throws Exception
	{
		//获取字节码文件对象
		Class c=Class.forName("Person");

		//获取公共成员变量
		//Field[] fields=c.getFields();

		//获取所有成员变量
		//Field[] fields=c.getDeclaredFields();
		//for(Field field : fields)
		//{
	//		System.out.println(field);
	//	}

		//通过无参构造方法创建对象
		Constructor con=c.getConstructor();
		Object obj=con.newInstance();
		System.out.println(obj);

		//获取单个的成员变量:获取address并对其赋值
		Field addressField=c.getField("address");

		//public void set(Object obj,Object value);
		//将指定对象变量上此Field对象表示的字段设置为指定的新值
		addressField.set(obj,"北京");//给obj对象的addressField字段设置值为“北京”
		System.out.println(obj);


		//获取name(私有)并对其赋值
		Field nameField=c.getDeclaredField("name");
		nameField.setAccessible(true);//暴力访问
		nameField.set(obj,"张三");
		System.out.println(obj);

		//获取age并对其赋值(不管私不私有,这样写都能访问)
		Field ageField=c.getDeclaredField("age");
		ageField.setAccessible(true);//暴力访问
		ageField.set(obj,27);
		System.out.println(obj);
		
	}
}


通过反射获取成员方法并使用

获取所有方法:getMethods getDeclaredMethods

获取单个方法:getMethod getDeclaredMethod

暴力访问:method.setAccessible(true);

 

import java.lang.reflect.Method;
import java.lang.reflect.Constructor;
class  ReflectDemo
{
	public static void main(String[] args) throws Exception
	{
		//获取字节码文件对象
		Class c=Class.forName("Person");

		/*
		//获取公共成员方法  <strong>特殊之处</strong>:获取自己的包括父亲的公共方法
		//Method[] methods=c.getMethods();

		//获取所有成员方法 <strong> 特殊之处</strong>:这里只获取自己的所有方法,不包括父亲的
		Method[] methods=c.getDeclaredMethods();
		for(Method method :methods)
		{
			System.out.println(method);
		}
		*/

		//通过无参构造方法创建对象
		Constructor con=c.getConstructor();
		Object obj=con.newInstance();		


		//获取单个方法并使用:public void show()
		//public Method getMethod(String name,class... parameterTypes)
		//第一个参数表示的是方法名,第二个参数表示的是方法的参数的class类型
		Method m1=c.getMethod("show");

		//public Object invoke(Object obj,Object...args)
		//返回值是Object接收,第一个参数表示对象是谁,第二个参数表示调用该方法的实际参数
		m1.invoke(obj);//调用obj对象的m1方法(Person对象的show方法)
		
	}
}

 

import java.lang.reflect.Method;
import java.lang.reflect.Constructor;
class ReflectDemo 
{
	public static void main(String[] args) throws Exception
	{
		

		//获取字节码文件对象
		Class c=Class.forName("Person");

		//通过无参构造方法创建对象
		Constructor con=c.getConstructor();
		Object obj=con.newInstance();

//需求:获取方法:public void method(String s)

		//public Method getMethod(String name,class... parameterTypes)
		//第一个参数表示的是方法名,第二个参数表示的是方法的参数的class类型
		Method m1=c.getMethod("method",String.class);

		//public Object invoke(Object obj,Object...args)
		//返回值是Object接收,第一个参数表示对象是谁,第二个参数表示调用该方法的实际参数
		m1.invoke(obj,"hello");

//需求:获取方法:public String getString(String s,int i)
		Method m2=c.getMethod("getString",String.class,int.class);
		Object objString=m2.invoke(obj,"hello",100);
		System.out.println(objString);
	}
}


动态代理

代理:本来应该自己做的事情,却请了别人来做,被请的人就是代理对象

动态代理:在程序运行过程中产生的这个对象,动态代理其实就是通过反射来生成一个代理

JAVAjava.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口,通过使用这个类和接口就可以生成动态代理对象。JDK提供的代理只能针对接口做代理,更强大的代理cglib

Proxy类中的方法创建动态代理类对象

public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)

最终会调用InvocationHandler的方法

InocationHandler

Object invoke(Object proxy,Method method,Object[] args)

 

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

class MyInvocationHandler implements InvocationHandler
{
	private Object target;//目标对象
	 MyInvocationHandler(Object target)
	{
		this.target=target;
	}
	public Object invoke(Object proxy,Method method,Object[] args) throws Throwable
	{
		System.out.println("权限校验");
		Object result=method.invoke(target,args);
		System.out.println("日志记录");
		return result;//返回的是代理对象
	}
}

import java.lang.reflect.Proxy;
class Test 
{
	public static void main(String[] args) 
	{

		//我们要创建一个动态代理对象
		//Proxy类中有一个方法可以创建动态代理对象
		//public static Object newProxyInstance(ClassLoader loader,Class[] interfaces,
		//InvocationHandler h)
		//返回值Object返回的其实就是动态代理对象

		//我准备对ud对象做一个代理对象
		MyInvocationHandler handler=new MyInvocationHandler(ud);//想让谁当代理对象就传谁
		UserDao proxy=(UserDao)Proxy.newProxyInstance(ud.getClass().getClassLoader(),
			ud.getClass().getInterfaces(),handler);
		proxy.add();
		proxy.delete();
		proxy.update();
		proxy.find();

		System.out.println("---------");
		StudentDao sd=new StudentDaoImpl();
		MyInvocationHandler handler2=new MyInvocationHandler(sd);//想让谁当代理对象就传谁
		StudentDao proxy2=(StudentDao)Proxy.newProxyInstance(sd.getClass().getClassLoader(),
			sd.getClass().getInterfaces(),handler2);
		proxy2.login();
		proxy2.regist();		
	}
}


 


 


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
黑马程序员的tb_brand是指在JavaWeb基础教程中创建的一个表。这个表是用来存储品牌信息的,具体的表结构和数据类型需要和JavaBean类中的成员变量保持一致。\[1\]在这个教程中,使用了Maven来构建项目,并且使用了MyBatis作为持久层框架,通过配置pom.xml文件来引入相关依赖。\[2\] Maven是一个用于管理和构建Java项目的工具,它提供了一套标准化的项目结构、构建流程和依赖管理机制。\[3\] #### 引用[.reference_title] - *1* [【JAVAWEB开发】黑马程序员java web案例资料(含Element的删除与修改)](https://blog.csdn.net/aasd23/article/details/126940147)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [黑马程序员-MyBatis 框架-最全入门笔记、阿伟看了都得说真大、真细、真全!!!](https://blog.csdn.net/qq_57383364/article/details/128103058)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [JavaWeb——黑马程序员课程笔记](https://blog.csdn.net/King_ZACC/article/details/128573804)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值