apache Commons Dbutils泛型应用及类型安全探讨
在使用Apache Commons Dbutils进行数据库操作时,为了提升代码的可复用性和可读性,我们通常会进行封装,并使用泛型。然而,在使用queryOne方法时,泛型容易引发类型安全问题。本文将深入分析其原因,并给出解决方案。
问题根源在于queryOne方法的两种不同泛型实现方式。第一种直接传入class对象作为类型参数:
public <T> T queryOne(Class<T> clazz, String sql, Object... params) throws SQLException { try { return queryRunner.query(connection, sql, new BeanHandler<>(clazz), params); } finally { // 省略代码... } }
第二种方式传入泛型类型的对象实例:
public <T> T queryOne(T t, String sql, Object... params) throws SQLException { try { return queryRunner.query(connection, sql, new BeanHandler<T>((Class<? extends T>) t.getClass()), params); } finally { // 省略代码... } }
第二种方式在new BeanHandler
其根本原因在于Java的类型擦除机制。编译时,泛型信息会被擦除,运行时只保留原始类型。第一种方式中,Class
因此,t.getClass()返回的Class extends Object>无法安全地转换为Class extends T>,需要强制转换。虽然编译可以通过,但运行时可能抛出ClassCastException异常。
使用@SuppressWarnings(“unchecked”)注解可以压制警告,但这并不能解决潜在的类型安全问题。 只有确保传入的t对象的实际类型与期望的T类型一致才能避免运行时异常。 因此,第一种方式,直接传入Class