java.lang.Object类-研究总结

一、概述:
        Object类是所有Java类的祖先。每个类都使用 Object 作为超类。所有对象(包括数组)都实现这个类的方法。在不明确给出超类的情况下,Java会自动把Object作为要定义类的超类。
      可以使用类型为Object的变量指向任意类型的对象。
      Object类有一个默认构造方法pubilc Object(),在构造子类实例时,都会先调用这个默认构造方法。
      Object类的变量只能用作各种值的通用持有者。要对他们进行任何专门的操作,都需要知道它们的原始类型并进行类型转换。       例如:
      Object obj = new MyObject();

      MyObject x = (MyObject)obj

      Object类是所有Java类的超类。Class类也是Object类的子类。

   Class类的官方定义:public final class Class<T> extends Object  implements Serializable, GenericDeclaration, Type, AnnotatedElement

      Class 类的实例表示正在运行的 Java 应用程序中的类和接口。枚举是一种类,注释是一种接口。每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共享该 Class 对象。基本的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象。

  Class 没有公共构造方法。Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的。

    Class类的定义详见:http://www.javaweb.cc/JavaAPI1.6/java/lang/Class.html

二 方法摘要

/*
 *类 Object 是类层次结构的根类。每个类都使用 Object 作为超类。所有对象(包括数组)都实现这个类的方法。 
*/
public class Object{
	public Object()//构造方法
	protected Object clone()throws CloneNotSupportedException//创建并返回创建并返回此对象的一个副本
	public boolean equals(Object obj)//比较当前对象与obj是否相等
	protected void finalize()throws Throwable//析构方法:当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。
	public final Class<?> getClass()//返回该对象的运行时类,即返回当前对象所在的类
	public int hashCode()//返回该对象的哈希码值
	public String toString()//返回当前对象的描述字符串
	public final void notify()//唤醒在此对象监视器上等待的单个线程
	public final void notifyAll()//唤醒在此对象监视器上等待的所有线程。
	//在其他线程调用此对象的 notify() 方法或 notifyAll() 方法,或者超过指定的时间量前,导致当前线程等待。
	public final void wait(long timeout)throws InterruptedException
	/*
	 * 在其他线程调用此对象的 notify() 方法或 notifyAll() 方法,
	 * 或者其他某个线程中断当前线程,或者已超过某个实际时间量前,导致当前线程等待。
	*/
	public final void wait(long timeout,int nanos)throws InterruptedException
	//在其他线程调用此对象的 notify() 方法或 notifyAll() 方法前,导致当前线程等待。
	public final void wait()throws InterruptedException
}
三 重点解读

1.hashCode()

   public int hashCode()
返回该对象的哈希码值。支持此方法是为了提高哈希表(例如 java.util.Hashtable 提供的哈希表)的性能。

hashCode 的常规协定是:

  • 在 Java 应用程序执行期间,在对同一对象多次调用 hashCode 方法时,必须一致地返回相同的整数,前提是将对象进行 equals 比较时所用的信息没有被修改。从某一应用程序的一次执行到同一应用程序的另一次执行,该整数无需保持一致。
  • 如果根据 equals(Object) 方法,两个对象是相等的,那么对这两个对象中的每个对象调用 hashCode 方法都必须生成相同的整数结果。
  • 如果根据 equals(java.lang.Object) 方法,两个对象不相等,那么对这两个对象中的任一对象上调用 hashCode 方法不 要求一定生成不同的整数结果。但是,程序员应该意识到,为不相等的对象生成不同整数结果可以提高哈希表的性能。

实际上,由 Object 类定义的 hashCode 方法确实会针对不同的对象返回不同的整数。(这一般是通过将该对象的内部地址转换成一个整数来实现的,但是 JavaTM 编程语言不需要这种实现技巧。)

返回:

此对象的一个哈希码值。
另请参见:
equals(java.lang.Object)Hashtable
2.  equals

     public boolean equals(Object obj)

指示其他某个对象是否与此对象“相等”。

equals 方法在非空对象引用上实现相等关系:

  • 自反性:对于任何非空引用值 x,x.equals(x) 都应返回 true。
  • 对称性:对于任何非空引用值 x 和 y,当且仅当 y.equals(x) 返回 true 时,x.equals(y) 才应返回 true。
  • 传递性:对于任何非空引用值 x、y 和 z,如果 x.equals(y) 返回 true,并且 y.equals(z) 返回 true,那么 x.equals(z) 应返回 true。
  • 一致性:对于任何非空引用值 x 和 y,多次调用 x.equals(y) 始终返回 true 或始终返回 false,前提是对象上 equals 比较中所用的信息没有被修改。
  • 对于任何非空引用值 x,x.equals(null) 都应返回 false。

Object 类的 equals 方法实现对象上差别可能性最大的相等关系;即,对于任何非空引用值 x 和 y,当且仅当 x 和 y 引用同一个对象时,此方法才返回 true(x == y 具有值 true)。

注意:当此方法被重写时,通常有必要重写 hashCode 方法,以维护 hashCode 方法的常规协定,该协定声明相等对象必须具有相等的哈希码。

参数:
obj - 要与之比较的引用对象。
返回:
如果此对象与 obj 参数相同,则返回 true;否则返回 false。
另请参见:
hashCode()Hashtable
3.toString

     public String toString()

返回该对象的字符串表示。通常,toString 方法会返回一个“以文本方式表示”此对象的字符串。结果应是一个简明但易于读懂的信息表达式。 建议所有子类都重写此方法。

Object 类的 toString 方法返回一个字符串,该字符串由类名(对象是该类的一个实例)、at 标记符“@”和此对象哈希码的无符号十六进制表示组成。换句话说,该方法返回一个字符串,它的值等于:

getClass().getName() + '@' + Integer.toHexString(hashCode())
返回:
该对象的字符串表示形式。比如:Test@de6ced
4.getClass
   public final Class<?> getClass()
返回此 Object 的运行时类。返回的 Class 对象是由所表示类的 static synchronized 方法锁定的对象。

实际结果类型是 Class<? extends |X|>,其中 |X| 表示清除表达式中的静态类型,该表达式调用 getClass。 例如,以下代码片段中不需要强制转换:

Number n = 0; 
Class<? extends Number> c = n.getClass();

返回:
表示此对象运行时类的 Class 对象。
另请参见:
The Java Language Specification, Third Edition (15.8.2 Class Literals)
5. 更多请参考:http://www.javaweb.cc/JavaAPI1.6/java/lang/Object.html

<think>嗯,我现在遇到了一个问题,项目在从Java 8升级到Java 21之后,使用Spring框架的时候,出现了CGLIB相关的错误,具体是CodeGenerationException,原因是InaccessibleObjectException,提示说java.base模块没有开放java.lang包给未命名模块。这让我有点困惑,得仔细分析一下。 首先,用户提到的错误信息显示CGLIB尝试访问ClassLoader的defineClass方法时被阻止了。根据引用[2],Java 9引入了模块系统,限制了内部API的访问。特别是java.base模块没有开放java.lang包,而CGLIB需要访问这个包里的方法生成代理,所以导致了问题。这说明模块系统的限制是主要原因。 然后,引用[4]提到代码中调用了BeanMap.create(Object),这会生成动态代理,可能涉及到缓存的问题。但用户的问题可能更多是模块权限的问题,而不是缓存失效。但可能需要检查是否有重复生成代理的情况,不过根据错误信息,更可能还是模块权限的问题。 解决方案方面,引用[2]和引用[3]都提到了需要开放模块。可能的解决方法包括使用JVM参数--add-opens来开放java.base/java.lang给未命名模块。此外,引用[5]提到Spring可能已经更新了CGLIB的版本,可能需要升级Spring或CGLIB的版本来兼容Java 21的模块系统。 另外,可能需要检查是否在使用Spring的内置CGLIB(如spring-core中的repackaged版本),如果是的话,可能需要升级Spring版本,因为新版本可能已经解决了这个问题。或者,如果使用的是旧版Spring,可能需要手动添加JVM参数。 还有,用户可能需要检查他们的代码是否直接或间接依赖了旧版的CGLIB,导致与新Java版本不兼容。升级到Spring的最新版本可能包含修复后的CGLIB,从而避免这个问题。 总结一下,解决方案可能包括:添加JVM参数开放模块,升级Spring或CGLIB版本,检查依赖关系确保使用兼容的库,或者调整代码避免使用需要反射访问内部API的部分。</think>### 问题分析 在Java 8升级到Java 21后,使用Spring框架的CGLIB生成代理时,会因模块系统(Jigsaw)的限制触发`InaccessibleObjectException`,具体表现为: ``` Caused by: net.sf.cglib.core.CodeGenerationException: java.lang.reflect.InaccessibleObjectException --> Unable to make protected final java.lang.Class java.lang.ClassLoader.defineClass(...) accessible: module java.base does not "opens java.lang" to unnamed module @5a10411 ``` **核心原因**: 1. **模块化限制**:Java 9+ 的模块系统默认禁止未命名模块访问`java.base`模块的`java.lang`包[^2][^3]。 2. **CGLIB依赖反射**:CGLIB需通过反射调用`ClassLoader.defineClass`生成动态代理,但Java 21强化了对反射访问内部API的限制[^4]。 3. **库兼容性**:旧版CGLIB或Spring未适配Java模块化系统[^5]。 --- ### 解决方案 #### 方法1:通过JVM参数开放模块权限 在启动命令中添加以下参数,允许未命名模块访问`java.base/java.lang`: ```bash --add-opens java.base/java.lang=ALL-UNNAMED ``` **适用场景**:临时调试或快速修复,需确保安全性。 #### 方法2:升级Spring/CGLIB版本 - **Spring 5.3.10+**:已内置修复模块化问题的CGLIB版本。 - **CGLIB 3.3.0+**:支持Java模块化系统。 ```xml <!-- Maven依赖示例 --> <dependency> <groupId>cglib</groupId> <artifactId>cglib</artifactId> <version>3.3.0</version> </dependency> ``` #### 方法3:改用JDK动态代理 若业务允许,可优先使用JDK动态代理(需接口): ```java Proxy.newProxyInstance( target.getClass().getClassLoader(), target.getClass().getInterfaces(), new MyInvocationHandler() ); ``` #### 方法4:检查代理生成逻辑 确认代码中是否重复生成代理(如`BeanMap.create`调用),优化缓存机制以避免冗余操作[^4]。 --- ### 验证步骤 1. **添加JVM参数**:验证是否消除异常。 2. **升级依赖**:检查Spring和CGLIB版本是否兼容Java 21。 3. **日志分析**:查看代理生成日志,确认是否涉及非法反射调用。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值