2024-10-03 16:32:34 +08:00
|
|
|
|
/*
|
2025-06-02 00:18:29 +08:00
|
|
|
|
* Copyright 2022-2025 the original author or authors.
|
2024-10-03 16:32:34 +08:00
|
|
|
|
*
|
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
|
*
|
|
|
|
|
|
* https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
*
|
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
2024-10-03 16:27:19 +08:00
|
|
|
|
package xyz.zhouxy.jdbc;
|
|
|
|
|
|
|
|
|
|
|
|
import java.beans.BeanInfo;
|
|
|
|
|
|
import java.beans.IntrospectionException;
|
|
|
|
|
|
import java.beans.Introspector;
|
|
|
|
|
|
import java.beans.PropertyDescriptor;
|
|
|
|
|
|
import java.lang.reflect.Constructor;
|
|
|
|
|
|
import java.lang.reflect.InvocationTargetException;
|
|
|
|
|
|
import java.lang.reflect.Method;
|
|
|
|
|
|
import java.sql.ResultSet;
|
|
|
|
|
|
import java.sql.ResultSetMetaData;
|
|
|
|
|
|
import java.sql.SQLException;
|
|
|
|
|
|
import java.util.Arrays;
|
|
|
|
|
|
import java.util.Map;
|
|
|
|
|
|
import java.util.function.Function;
|
|
|
|
|
|
import java.util.stream.Collectors;
|
|
|
|
|
|
|
|
|
|
|
|
import javax.annotation.Nullable;
|
|
|
|
|
|
|
2024-10-03 17:12:32 +08:00
|
|
|
|
import com.google.common.base.CaseFormat;
|
|
|
|
|
|
|
2024-11-02 10:55:23 +08:00
|
|
|
|
import xyz.zhouxy.plusone.commons.annotation.StaticFactoryMethod;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* DefaultBeanRowMapper
|
|
|
|
|
|
*
|
|
|
|
|
|
* <p>
|
|
|
|
|
|
* 默认实现的将 {@link ResultSet} 转换为 Java Bean 的 {@link RowMapper}。
|
|
|
|
|
|
* </p>
|
|
|
|
|
|
*
|
|
|
|
|
|
* <p>
|
|
|
|
|
|
* <b>NOTE: 使用反射获取类型信息,也是使用反射调用无参构造器和 {@code setter} 方法。
|
|
|
|
|
|
* 实际使用中还是建议针对目标类型自定义 {@link RowMapper}。</b>
|
|
|
|
|
|
* </p>
|
|
|
|
|
|
* @author <a href="http://zhouxy.xyz:3000/ZhouXY108">ZhouXY</a>
|
|
|
|
|
|
* @since 1.0.0
|
|
|
|
|
|
*/
|
2024-11-01 16:46:17 +08:00
|
|
|
|
public class DefaultBeanRowMapper<T> implements RowMapper<T> {
|
2024-10-03 16:27:19 +08:00
|
|
|
|
|
2024-11-02 10:55:23 +08:00
|
|
|
|
/** Bean 的无参构造器 */
|
2024-10-03 16:27:19 +08:00
|
|
|
|
private final Constructor<T> constructor;
|
2024-11-02 10:55:23 +08:00
|
|
|
|
|
|
|
|
|
|
/** 列名到属性的映射 */
|
2024-10-03 16:27:19 +08:00
|
|
|
|
private final Map<String, PropertyDescriptor> colPropertyMap;
|
|
|
|
|
|
|
2024-11-01 16:46:17 +08:00
|
|
|
|
private DefaultBeanRowMapper(Constructor<T> constructor, Map<String, PropertyDescriptor> colPropertyMap) {
|
2024-10-03 16:27:19 +08:00
|
|
|
|
this.constructor = constructor;
|
|
|
|
|
|
this.colPropertyMap = colPropertyMap;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-02 10:55:23 +08:00
|
|
|
|
/**
|
2025-06-02 00:18:29 +08:00
|
|
|
|
* 创建一个 {@code DefaultBeanRowMapper}
|
2024-11-02 10:55:23 +08:00
|
|
|
|
*
|
|
|
|
|
|
* @param <T> Bean 类型
|
|
|
|
|
|
* @param beanType Bean 类型
|
|
|
|
|
|
* @return DefaultBeanRowMapper 对象
|
2025-06-02 00:18:29 +08:00
|
|
|
|
* @throws SQLException 创建 {@code DefaultBeanRowMapper} 出现错误的异常时抛出
|
2024-11-02 10:55:23 +08:00
|
|
|
|
*/
|
|
|
|
|
|
@StaticFactoryMethod(DefaultBeanRowMapper.class)
|
2024-11-01 16:46:17 +08:00
|
|
|
|
public static <T> DefaultBeanRowMapper<T> of(Class<T> beanType) throws SQLException {
|
2024-10-03 16:27:19 +08:00
|
|
|
|
return of(beanType, null);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-02 10:55:23 +08:00
|
|
|
|
/**
|
2025-06-02 00:18:29 +08:00
|
|
|
|
* 创建一个 {@code DefaultBeanRowMapper}
|
2024-11-02 10:55:23 +08:00
|
|
|
|
*
|
|
|
|
|
|
* @param <T> Bean 类型
|
|
|
|
|
|
* @param beanType Bean 类型
|
|
|
|
|
|
* @param propertyColMap Bean 字段与列名的映射关系。key 是字段,value 是列名。
|
2025-06-02 00:18:29 +08:00
|
|
|
|
* @return {@code DefaultBeanRowMapper} 对象
|
|
|
|
|
|
* @throws SQLException 创建 {@code DefaultBeanRowMapper} 出现错误的异常时抛出
|
2024-11-02 10:55:23 +08:00
|
|
|
|
*/
|
|
|
|
|
|
@StaticFactoryMethod(DefaultBeanRowMapper.class)
|
2024-11-01 16:46:17 +08:00
|
|
|
|
public static <T> DefaultBeanRowMapper<T> of(Class<T> beanType, @Nullable Map<String, String> propertyColMap)
|
2024-10-03 16:27:19 +08:00
|
|
|
|
throws SQLException {
|
|
|
|
|
|
try {
|
2024-10-03 17:12:32 +08:00
|
|
|
|
// 获取无参构造器
|
|
|
|
|
|
Constructor<T> constructor = beanType.getDeclaredConstructor();
|
|
|
|
|
|
constructor.setAccessible(true); // NOSONAR
|
|
|
|
|
|
|
2025-05-28 17:30:28 +08:00
|
|
|
|
final Map<String, PropertyDescriptor> colPropertyMap = buildColPropertyMap(beanType, propertyColMap);
|
2024-11-01 16:46:17 +08:00
|
|
|
|
return new DefaultBeanRowMapper<>(constructor, colPropertyMap);
|
2024-10-03 16:27:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
catch (IntrospectionException e) {
|
|
|
|
|
|
throw new SQLException("There is an exception occurs during introspection.", e);
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (NoSuchMethodException e) {
|
|
|
|
|
|
throw new SQLException("Could not find a no-args constructor in " + beanType.getName(), e);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-02 10:55:23 +08:00
|
|
|
|
/** {@inheritDoc} */
|
2024-10-03 16:27:19 +08:00
|
|
|
|
@Override
|
2024-11-01 16:46:17 +08:00
|
|
|
|
public T mapRow(ResultSet rs, int rowNumber) throws SQLException {
|
2024-10-03 16:27:19 +08:00
|
|
|
|
try {
|
2024-11-02 10:55:23 +08:00
|
|
|
|
// 调用无参构造器创建实例
|
2024-10-03 16:27:19 +08:00
|
|
|
|
T newInstance = this.constructor.newInstance();
|
|
|
|
|
|
ResultSetMetaData metaData = rs.getMetaData();
|
2024-11-02 10:55:23 +08:00
|
|
|
|
// 遍历结果的每一列
|
2024-10-03 16:27:19 +08:00
|
|
|
|
for (int i = 1; i <= metaData.getColumnCount(); i++) {
|
|
|
|
|
|
String colName = metaData.getColumnName(i);
|
2024-11-02 10:55:23 +08:00
|
|
|
|
// 获取查询结果列名对应的属性,调用 setter
|
2024-10-03 16:27:19 +08:00
|
|
|
|
PropertyDescriptor propertyDescriptor = this.colPropertyMap.get(colName);
|
|
|
|
|
|
if (propertyDescriptor != null) {
|
|
|
|
|
|
Method setter = propertyDescriptor.getWriteMethod();
|
|
|
|
|
|
if (setter != null) {
|
|
|
|
|
|
Class<?> propertyType = propertyDescriptor.getPropertyType();
|
|
|
|
|
|
setter.setAccessible(true); // NOSONAR
|
|
|
|
|
|
setter.invoke(newInstance, rs.getObject(colName, propertyType));
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return newInstance;
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (IllegalAccessException | InstantiationException | InvocationTargetException e) {
|
|
|
|
|
|
throw new SQLException(e);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-05-28 17:30:28 +08:00
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 构建 column name 和 PropertyDescriptor 的 映射
|
|
|
|
|
|
*
|
|
|
|
|
|
* @param <T> Java bean 类型
|
|
|
|
|
|
* @param beanType Java bean 类型
|
|
|
|
|
|
* @param propertyColMap 属性与列名的映射
|
|
|
|
|
|
* @return column name 和 PropertyDescriptor 的映射
|
|
|
|
|
|
* @throws IntrospectionException if an exception occurs during introspection.
|
|
|
|
|
|
*/
|
|
|
|
|
|
private static <T> Map<String, PropertyDescriptor> buildColPropertyMap(
|
|
|
|
|
|
Class<T> beanType, Map<String, String> propertyColMap) throws IntrospectionException {
|
|
|
|
|
|
|
|
|
|
|
|
BeanInfo beanInfo = Introspector.getBeanInfo(beanType);
|
|
|
|
|
|
PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();
|
|
|
|
|
|
|
|
|
|
|
|
// Bean 的属性名为小驼峰,对应的列名为下划线
|
|
|
|
|
|
Function<? super PropertyDescriptor, String> keyMapper;
|
|
|
|
|
|
if (propertyColMap == null || propertyColMap.isEmpty()) {
|
|
|
|
|
|
keyMapper = p -> CaseFormat.LOWER_CAMEL.to(CaseFormat.LOWER_UNDERSCORE, p.getName());
|
|
|
|
|
|
}
|
|
|
|
|
|
else {
|
|
|
|
|
|
keyMapper = p -> {
|
|
|
|
|
|
String propertyName = p.getName();
|
|
|
|
|
|
String colName = propertyColMap.get(propertyName);
|
|
|
|
|
|
return colName != null ? colName
|
|
|
|
|
|
: CaseFormat.LOWER_CAMEL.to(CaseFormat.LOWER_UNDERSCORE, propertyName);
|
|
|
|
|
|
};
|
|
|
|
|
|
}
|
|
|
|
|
|
return Arrays.stream(propertyDescriptors)
|
|
|
|
|
|
.collect(Collectors.toMap(keyMapper, Function.identity(), (a, b) -> b));
|
|
|
|
|
|
}
|
2024-10-03 16:27:19 +08:00
|
|
|
|
}
|