(二)利用反射机制实例化类,获取和修改变量,获取和调用方法和构造方法

本文深入探讨Java反射机制的应用,包括实例化对象、修改变量值、调用成员方法及构造方法等。通过代码示例展示了如何操作不同访问级别的变量与方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

上一篇文章初步认识了Java反射机制的基本概念,本文将进一步探究 Java 反射机制的作用。

综述:

获取类之后,可以调用赋值其成员变量,可以调用成员方法,构造方法。对于使用private 修饰的变量、方法和构造方法,需要调用 .setAccessible(true)

直接上代码,一个普通的类,然后待会使用反射机制实例化对象,获取普通常量,普通方法,构造方法,静态方法,静态常量

 

public class Demo4 {
	public Demo4() {
		System.out.println("私有化的无参构造函数被实例化");
	}
	
	private Demo4(String name) {
		System.out.println("public 修饰的有参构造函数被调用:"+name);
	}
	
	private String str1="1";
	
	public String str2="2";
	String str3="3";
	
	
	public void method(String name){
		System.out.println("调用类的方法,输入参数为:"+name);
	}
	
	private void method2(String name){
		System.out.println("调用类的方法,输入参数为:"+name);
	}

	public static String str4="4";
	private static void method3(String name,String msg,int a,Integer b){
		System.out.println("name="+name+";msg="+msg+";a="+a+";b="+b);
	}

}

测试代码

 

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import org.junit.Test;

/**
 * Java 反射机制
 * 实例化类,修改变量值,调用成员方法
 * @author wj
 *
 */
public class Demo4Test {
	
	@Test
	public void test7(){
		System.out.println("综述:获取类之后,可以调用赋值其成员变量,可以调用成员方法,构造方法。对于使用private 修饰的变量、方法和构造方法,需要调用 .setAccessible(true)");
	}
	
	
	/**
	 * 私有化的构造函数被实例化
	 * public 修饰的无参构造函数 方法有三种 http://blog.csdn.net/bestcxx/article/details/79134315
	 * Class.forName("demo3.Demo3Class1")
	 * Demo3Class1.class
	 * demo31.getClass()
	 */
	@Test
	public void test6(){
		Class<?> obj=Demo4.class;
		Constructor<?> c;//无参构造函数
		Constructor<?> c2;//有参构造函数
		try {
			//获取无参的构造函数
			c = obj.getDeclaredConstructor();
			
			//调用构造函数
			c.newInstance();
			
			//---------------------------------------------
			
			//获取有参构造函数
			c2=obj.getDeclaredConstructor(String.class);
			
			//允许反射调用 private 的构造函数
			c2.setAccessible(true);
			
			//调用构造函数
			c2.newInstance("123");
			
		} catch (NoSuchMethodException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (SecurityException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InstantiationException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalArgumentException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InvocationTargetException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		
		
	}
	
	/**
	 * Java 反射机制 调用 静态方法
	 * 静态方法本身即代表已被实例化的对象
	 * method.invoke(method, 传入参数)
	 */
	@Test
	public void test5(){
		Class<?> obj=Demo4.class;
		
		try {
			Method method=obj.getDeclaredMethod("method3", String.class,String.class,int.class,Integer.class);
			method.setAccessible(true);
			System.out.println("method.invoke(method, \"1\",\"2\",3,4)=调用静态方法");
			method.invoke(method, "1","2",3,4);
		} catch (NoSuchMethodException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (SecurityException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalArgumentException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InvocationTargetException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
	
	/**
	 *反射机制 获取 静态变量 
	 *直接以自己为对象进行取值和赋值
	 *field.get(field)
	 *field.set(field, "123")
	 */
	@Test
	public void test4(){
		Class<?> obj=Demo4.class;
		
		try {
			Field field=obj.getDeclaredField("str4");
			field.setAccessible(true);	
			System.out.println("field.get(field)=获取静态变量,field 即表示自身和已被实例化的类"+field.get(field));
			field.set(field, "123");
			System.out.println("field.set(field, \"123\")=给静态变量赋值,field 即表示自身和已被实例化的类"+field.get(field));
		} catch (NoSuchFieldException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (SecurityException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		
	}
	
	/**
	 * 运用反射机制调用方法
	 */
	@Test 
	public void test3(){
		Class<?> obj=Demo4.class;
		
		try {
			//获得方法
			Method method=obj.getDeclaredMethod("method2", String.class);
			
			//获取一个对象
			Object demo4= obj.newInstance();
			
			//允许反射方法调用 private 修饰的方法
			method.setAccessible(true);
			
			//反射 调用 方法,并传入参数
			method.invoke(demo4, "name2");
			
		} catch (NoSuchMethodException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (SecurityException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InstantiationException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalArgumentException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InvocationTargetException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		
	}
	
	/**
	 * 反射机制 修改 非 public 修饰的变量,比如private、默认、protected
	 * Field field=obj.getDeclaredField("str1");//如果获取的是 非public 修饰的变量,需要使用 DeclaredField
	 * field.setAccessible(true);//允许通过反射机制获取和修改 private 修饰的变量
	 * 
	 */
	@Test
	public void test2(){
		Class<?> obj=Demo4.class;
		try {
			//获得变量,如果获取的是 非public 修饰的变量,需要使用 DeclaredField
			Field field=obj.getDeclaredField("str1");
			
			//用反射机制获取一个对象
			Object demo4=obj.newInstance();//这里并没有转化为具体的类的对象
		
			//使用反射机制将 获取的成员变量 赋给 demo4 并且赋新值
			field.setAccessible(true);//允许通过反射机制获取和修改 private 修饰的变量
			
			//输出 demo4 中 变量 field 的变量值
			System.out.println(field.get(demo4));
			
			//将 demo4 中 变量 field 的值修改为 "aaa"
			field.set(demo4, "aaa");
			
			//验证值被修改
			System.out.println(field.get(demo4));
			
		} catch (NoSuchFieldException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (SecurityException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InstantiationException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
	
	/**
	 * 反射机制 修改 public 修饰的变量,将结果赋给指定对象
	 */
	@Test
	public void test1(){
		Class<?> obj=Demo4.class;
		try {
			Field field=obj.getField("str2");
			
			//用反射机制获取一个对象
			Demo4 demo4=(Demo4) obj.newInstance();
			System.out.println(demo4.str2);
			
			//使用反射机制将 获取的成员变量 赋给 demo4 并且赋新值
			field.set(demo4, "aaa");
			
			//验证值被修改
			System.out.println(demo4.str2);
			
		} catch (NoSuchFieldException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (SecurityException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InstantiationException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
	
	
	
	/**
	 * 反射机制 实例化对象
	 * 输出成员常量,调用成员方法
	 */
	@Test
	public void test(){
		System.out.println("obj.newInstance()=实例化一个 obj类型的对象,可以输出成员常量,调用成员方法");
		Class<?> obj=Demo4.class;
		try {
			Demo4 demo4=(Demo4) obj.newInstance();
			System.out.println(demo4.str2);
			
			demo4.method("Jecket");
		} catch (InstantiationException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		System.out.println();
	}
}

运行结果

 

obj.newInstance()=实例化一个 obj类型的对象,可以输出成员常量,调用成员方法
私有化的无参构造函数被实例化
2
调用类的方法,输入参数为:Jecket

私有化的无参构造函数被实例化
2
aaa
私有化的无参构造函数被实例化
1
aaa
私有化的无参构造函数被实例化
调用类的方法,输入参数为:name2
field.get(field)=获取静态变量,field 即表示自身和已被实例化的类4
field.set(field, "123")=给静态变量赋值,field 即表示自身和已被实例化的类123
method.invoke(method, "1","2",3,4)=调用静态方法
name=1;msg=2;a=3;b=4
私有化的无参构造函数被实例化
public 修饰的有参构造函数被调用:123
综述:获取类之后,可以调用赋值其成员变量,可以调用成员方法,构造方法。对于使用private 修饰的变量、方法和构造方法,需要调用 .setAccessible(true)

 

 

 

 

 

 

 

 

<think>嗯,用户让我详细介绍Redis的详细使用说明。首先,我需要确定用户的需求是什么。可能他们是一个开发者,刚开始接触Redis,或者想深入了解它的高级功能。接下来,我应该从基础开始,逐步深入,覆盖主要功能和常见使用场景。 首先,Redis是什么?它是一个内存数据结构存储,支持多种数据结构,比如字符串、哈希、列表等等。需要解释它的主要特点,比如高性能、持久化、支持事务和发布订阅等。然后,可能需要分几个部分来介绍:安装配置、数据结构、持久化、集群、应用场景等等。 安装部分,可能需要给出不同操作系统的安装步骤,比如Ubuntu用apt-get,CentOS用yum,macOS用Homebrew。然后基本命令,比如启动服务端和客户端,还有常用的配置项,比如端口、持久化设置。 数据结构是Redis的核心,每个结构都需要详细说明,比如字符串的常用命令set、get,哈希的hset、hget,列表的lpush、rpop,集合的交并差操作,有序集合的排序,还有位图和地理空间这些高级结构。需要给出每个数据结构的典型使用场景,比如缓存、计数器、消息队列等。 持久化部分,用户可能会关心RDB和AOF的区别,以及如何配置。需要解释两者的优缺点,比如RDB适合备份,AOF更安全但可能更慢。事务和Lua脚本也是重要部分,说明事务的执行方式和注意事项,比如没有回滚,以及Lua脚本的原子性。 集群和高可用方面,主从复制、哨兵、Cluster模式需要介绍,特别是Cluster的自动分片和故障转移。性能优化部分,内存管理和批量操作是关键,比如使用Pipeline减少网络延迟,避免大键等。 应用场景要具体,比如缓存、会话存储、排行榜、消息队列,这些例子帮助用户理解如何在实际项目中使用Redis。安全性方面,密码认证和网络隔离也是必须提到的。 最后,注意事项部分,内存不足的问题,持久化数据安全,以及集群配置的复杂性,都是用户可能遇到的挑战。需要提醒他们根据需求选择合适的策略。 可能用户没有明说的需求是,他们可能在设计系统时需要选择合适的技术栈,或者遇到了性能问题想优化。所以除了基本介绍,还需要强调最佳实践和常见陷阱,比如避免大键、合理选择数据结构。另外,安全性容易被忽视,特别是生产环中设置密码的重要性。 总之,结构需要清晰,由浅入深,覆盖基础到高级内容,同时结合实际例子,帮助用户更好地理解和应用Redis。可能还需要检查是否有最新的Redis版本更新内容,确保信息准确。比如最新版本是否引入了新数据结构或命令,但目前考虑用户可能使用的是较常见版本,所以先覆盖主流功能。以下</think>是对 **Redis** 的详细介绍和使用说明,内容涵盖核心概念、使用场景、数据结构及操作指南等: --- ### **1. Redis 简介** Redis(**Re**mote **Di**ctionary **S**ervice)是一个开源的 **内存数据结构存储系统**,支持多种数据结构(如字符串、哈希、列表等),常用于缓存、消息队列、实时数据处理等场景。其特点包括: - **高性能**:数据存储在内存中,读写速度极快(微秒级响应)。 - **持久化**:支持 RDB(快照)和 AOF(追加日志)两种持久化方式。 - **高可用**:通过哨兵(Sentinel)和集群(Cluster)模式实现故障转移和扩展。 - **多数据结构**:支持字符串、列表、哈希、集合、有序集合等。 --- ### **2. 安装启动** #### **安装 Redis** - **Linux (Ubuntu/Debian)**: ```bash sudo apt-get install redis-server ``` - **macOS**: ```bash brew install redis ``` - **Windows**:官方不支持原生版本,可通过 WSL 或 Docker 运行。 #### **启动 Redis** - **启动服务端**: ```bash redis-server ``` - **启动客户端**: ```bash redis-cli ``` #### **基本配置** 配置文件通常位于 `/etc/redis/redis.conf`,常用配置项: - `port 6379`:默认端口号。 - `bind 127.0.0.1`:绑定 IP。 - `save 60 1000`:60 秒内有 1000 次修改则触发 RDB 持久化。 - `appendonly yes`:启用 AOF 持久化。 --- ### **3. 核心数据结构操作** #### **(1) 字符串(String)** - **用途**:缓存、计数器、分布式锁。 - **常用命令**: ```bash SET key value # 设置值 GET key # 获取值 INCR key # 自增计数器 EXPIRE key 60 # 设置过期时间(秒) ``` #### **(2) 哈希(Hash)** - **用途**:存储对象(如用户信息)。 - **常用命令**: ```bash HSET user:1 name "Alice" age 30 # 设置字段 HGET user:1 name # 获取字段值 HGETALL user:1 # 获取所有字段 ``` #### **(3) 列表(List)** - **用途**:消息队列、最新动态。 - **常用命令**: ```bash LPUSH queue task1 # 左侧插入元素 RPOP queue # 右侧弹出元素 LRANGE queue 0 -1 # 获取全部元素 ``` #### **(4) 集合(Set)** - **用途**:标签、好友关系、去重。 - **常用命令**: ```bash SADD tags "redis" "db" # 添加元素 SMEMBERS tags # 获取所有元素 SINTER set1 set2 # 求交集 ``` #### **(5) 有序集合(Sorted Set)** - **用途**:排行榜、优先级队列。 - **常用命令**: ```bash ZADD rank 100 "user1" # 添加带分数的元素 ZRANGE rank 0 -1 WITHSCORES # 按分数升序获取 ZREVRANGE rank 0 10 # 按分数降序获取前10 ``` #### **(6) 其他数据结构** - **位图(Bitmap)**:用于二值统计(如每日签到)。 - **HyperLogLog**:基数统计(如 UV 去重)。 - **地理空间(GEO)**:存储经纬度,支持距离计算。 --- ### **4. 持久化高可用** #### **(1) 持久化方式** - **RDB**:定期生成数据快照,恢复速度快,但可能丢失部分数据。 - **AOF**:记录所有写操作命令,数据更安全,但文件较大。 #### **(2) 主从复制** 配置从节点同步主节点数据: ```bash # 在从节点配置文件中添加: replicaof <master-ip> <master-port> ``` #### **(3) 哨兵模式(Sentinel)** 实现自动故障转移: ```bash # 哨兵配置文件 sentinel.conf sentinel monitor mymaster 127.0.0.1 6379 2 sentinel down-after-milliseconds mymaster 5000 ``` #### **(4) 集群模式(Cluster)** 分片存储数据,支持横向扩展: ```bash # 创建集群(至少3主3从) redis-cli --cluster create 127.0.0.1:7000 127.0.0.1:7001 ... ``` --- ### **5. 典型应用场景** - **缓存加速**:缓存数据库查询结果,降低后端负载。 - **会话存储(Session Store)**:集中管理用户会话。 - **排行榜**:使用有序集合实时更新排名。 - **消息队列**:通过列表或 Stream 数据结构实现。 - **分布式锁**:通过 `SET key value NX EX` 实现互斥锁。 --- ### **6. 性能优化注意事项** - **内存管理**:监控内存使用,避免 OOM(`maxmemory` 配置)。 - **Pipeline 批量操作**:减少网络往返次数。 - **避免大 Key**:单个 Key 的 Value 不宜过大(如超过 10KB)。 - **合理选择数据结构**:例如用 Hash 代替多个 String 存储对象。 --- ### **7. 安全性** - **密码认证**:在配置文件中设置 `requirepass yourpassword`。 - **禁用危险命令**:禁止使用 `FLUSHALL`、`CONFIG` 等。 - **网络隔离**:通过防火墙限制访问 IP。 --- ### **8. 常见问题** - **缓存穿透**:使用布隆过滤器或缓存空值。 - **缓存雪崩**:设置随机过期时间,保证高可用架构。 - **数据一致性**:结合数据库的 Binlog 或延迟双删策略。 --- 通过以上内容,可以全面掌握 Redis 的核心功能和使用技巧。实际应用中需根据场景选择合适的数据结构和架构模式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值