Skip to content

Commit 49ddfac

Browse files
mp911deschauder
authored andcommitted
Performance improvements.
Introduce caching for configured RowMapper/ResultSetExtractor. We now create RowMapper/ResultSetExtractor instances only once if they are considered static configuration. A configured RowMapper ref/class is always static. A configured ResultSetExtractor ref/class is static when the extractor does not accept a RowMapper or if the RowMapper is configured. Default mappers or projection-specific ones require ResultSetExtractor recreation of the ResultSetExtractor is configured as Class. Reuse TypeInformation as much as possible to avoid Class -> TypeInformation conversion. Introduce LRU cache for DefaultAggregatePath to avoid PersistentPropertyPath lookups. Introduce best-effort quoted cache for SqlIdentifier to avoid excessive string object creation. Closes #1721 Original pull request #1722
1 parent 8a94345 commit 49ddfac

File tree

16 files changed

+588
-151
lines changed

16 files changed

+588
-151
lines changed

spring-data-jdbc/src/main/java/org/springframework/data/jdbc/core/convert/EntityRowMapper.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -79,8 +79,8 @@ public T mapRow(ResultSet resultSet, int rowNumber) throws SQLException {
7979
RowDocument document = RowDocumentResultSetExtractor.toRowDocument(resultSet);
8080

8181
return identifier == null //
82-
? converter.readAndResolve(entity.getType(), document) //
83-
: converter.readAndResolve(entity.getType(), document, identifier);
82+
? converter.readAndResolve(entity.getTypeInformation(), document, Identifier.empty()) //
83+
: converter.readAndResolve(entity.getTypeInformation(), document, identifier);
8484
}
8585

8686
}

spring-data-jdbc/src/main/java/org/springframework/data/jdbc/core/convert/JdbcConverter.java

+35-4
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@
2626
import org.springframework.data.relational.core.mapping.RelationalPersistentEntity;
2727
import org.springframework.data.relational.core.mapping.RelationalPersistentProperty;
2828
import org.springframework.data.relational.domain.RowDocument;
29+
import org.springframework.data.util.TypeInformation;
2930
import org.springframework.lang.Nullable;
3031

3132
/**
@@ -47,7 +48,21 @@ public interface JdbcConverter extends RelationalConverter {
4748
* @return The converted value wrapped in a {@link JdbcValue}. Guaranteed to be not {@literal null}.
4849
* @since 2.4
4950
*/
50-
JdbcValue writeJdbcValue(@Nullable Object value, Class<?> type, SQLType sqlType);
51+
default JdbcValue writeJdbcValue(@Nullable Object value, Class<?> type, SQLType sqlType) {
52+
return writeJdbcValue(value, TypeInformation.of(type), sqlType);
53+
}
54+
55+
/**
56+
* Convert a property value into a {@link JdbcValue} that contains the converted value and information how to bind it
57+
* to JDBC parameters.
58+
*
59+
* @param value a value as it is used in the object model. May be {@code null}.
60+
* @param type {@link TypeInformation} into which the value is to be converted. Must not be {@code null}.
61+
* @param sqlType the {@link SQLType} to be used if non is specified by a converter.
62+
* @return The converted value wrapped in a {@link JdbcValue}. Guaranteed to be not {@literal null}.
63+
* @since 3.2.6
64+
*/
65+
JdbcValue writeJdbcValue(@Nullable Object value, TypeInformation<?> type, SQLType sqlType);
5166

5267
/**
5368
* Read the current row from {@link ResultSet} to an {@link RelationalPersistentEntity#getType() entity}.
@@ -84,7 +99,7 @@ default <T> T mapRow(RelationalPersistentEntity<T> entity, ResultSet resultSet,
8499
default <T> T mapRow(PersistentPropertyPathExtension path, ResultSet resultSet, Identifier identifier, Object key) {
85100

86101
try {
87-
return (T) readAndResolve(path.getRequiredLeafEntity().getType(),
102+
return (T) readAndResolve(path.getRequiredLeafEntity().getTypeInformation(),
88103
RowDocumentResultSetExtractor.toRowDocument(resultSet), identifier);
89104
} catch (SQLException e) {
90105
throw new RuntimeException(e);
@@ -118,15 +133,31 @@ default <R> R readAndResolve(Class<R> type, RowDocument source) {
118133
* @since 3.2
119134
* @see #read(Class, RowDocument)
120135
*/
121-
<R> R readAndResolve(Class<R> type, RowDocument source, Identifier identifier);
136+
default <R> R readAndResolve(Class<R> type, RowDocument source, Identifier identifier) {
137+
return readAndResolve(TypeInformation.of(type), source, identifier);
138+
}
139+
140+
/**
141+
* Read a {@link RowDocument} into the requested {@link TypeInformation aggregate type} and resolve references by
142+
* looking these up from {@link RelationResolver}.
143+
*
144+
* @param type target aggregate type.
145+
* @param source source {@link RowDocument}.
146+
* @param identifier identifier chain.
147+
* @return the converted object.
148+
* @param <R> aggregate type.
149+
* @since 3.2.6
150+
* @see #read(Class, RowDocument)
151+
*/
152+
<R> R readAndResolve(TypeInformation<R> type, RowDocument source, Identifier identifier);
122153

123154
/**
124155
* The type to be used to store this property in the database. Multidimensional arrays are unwrapped to reflect a
125156
* top-level array type (e.g. {@code String[][]} returns {@code String[]}).
126157
*
127158
* @return a {@link Class} that is suitable for usage with JDBC drivers.
128159
* @see org.springframework.data.jdbc.support.JdbcUtil#targetSqlTypeFor(Class)
129-
* @since 2.0
160+
* @since 2.0 TODO: Introduce variant returning TypeInformation.
130161
*/
131162
Class<?> getColumnType(RelationalPersistentProperty property);
132163

spring-data-jdbc/src/main/java/org/springframework/data/jdbc/core/convert/MapEntityRowMapper.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,7 @@ public Map.Entry<Object, T> mapRow(ResultSet rs, int rowNum) throws SQLException
6464
@SuppressWarnings("unchecked")
6565
private T mapEntity(RowDocument document, Object key) {
6666

67-
return (T) converter.readAndResolve(path.getRequiredLeafEntity().getType(), document,
67+
return (T) converter.readAndResolve(path.getRequiredLeafEntity().getTypeInformation(), document,
6868
identifier.withPart(keyColumn, key, Object.class));
6969
}
7070
}

spring-data-jdbc/src/main/java/org/springframework/data/jdbc/core/convert/MappingJdbcConverter.java

+3-3
Original file line numberDiff line numberDiff line change
@@ -231,14 +231,14 @@ private boolean canWriteAsJdbcValue(@Nullable Object value) {
231231
}
232232

233233
@Override
234-
public JdbcValue writeJdbcValue(@Nullable Object value, Class<?> columnType, SQLType sqlType) {
234+
public JdbcValue writeJdbcValue(@Nullable Object value, TypeInformation<?> columnType, SQLType sqlType) {
235235

236236
JdbcValue jdbcValue = tryToConvertToJdbcValue(value);
237237
if (jdbcValue != null) {
238238
return jdbcValue;
239239
}
240240

241-
Object convertedValue = writeValue(value, TypeInformation.of(columnType));
241+
Object convertedValue = writeValue(value, columnType);
242242

243243
if (convertedValue == null || !convertedValue.getClass().isArray()) {
244244

@@ -275,7 +275,7 @@ private JdbcValue tryToConvertToJdbcValue(@Nullable Object value) {
275275

276276
@SuppressWarnings("unchecked")
277277
@Override
278-
public <R> R readAndResolve(Class<R> type, RowDocument source, Identifier identifier) {
278+
public <R> R readAndResolve(TypeInformation<R> type, RowDocument source, Identifier identifier) {
279279

280280
RelationalPersistentEntity<R> entity = (RelationalPersistentEntity<R>) getMappingContext()
281281
.getRequiredPersistentEntity(type);

spring-data-jdbc/src/main/java/org/springframework/data/jdbc/repository/query/AbstractJdbcQuery.java

+9-8
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818
import java.sql.ResultSet;
1919
import java.sql.SQLException;
2020
import java.util.List;
21+
import java.util.function.Supplier;
2122
import java.util.stream.Stream;
2223

2324
import org.springframework.core.convert.converter.Converter;
@@ -83,9 +84,9 @@ public JdbcQueryMethod getQueryMethod() {
8384
*/
8485
@Deprecated(since = "3.1", forRemoval = true)
8586
// a better name would be createReadingQueryExecution
86-
protected JdbcQueryExecution<?> getQueryExecution(JdbcQueryMethod queryMethod,
87+
JdbcQueryExecution<?> getQueryExecution(JdbcQueryMethod queryMethod,
8788
@Nullable ResultSetExtractor<?> extractor, RowMapper<?> rowMapper) {
88-
return createReadingQueryExecution(extractor, rowMapper);
89+
return createReadingQueryExecution(extractor, () -> rowMapper);
8990
}
9091

9192
/**
@@ -96,21 +97,21 @@ protected JdbcQueryExecution<?> getQueryExecution(JdbcQueryMethod queryMethod,
9697
* @param rowMapper must not be {@literal null}.
9798
* @return a JdbcQueryExecution appropriate for {@literal queryMethod}. Guaranteed to be not {@literal null}.
9899
*/
99-
protected JdbcQueryExecution<?> createReadingQueryExecution(@Nullable ResultSetExtractor<?> extractor,
100-
RowMapper<?> rowMapper) {
100+
JdbcQueryExecution<?> createReadingQueryExecution(@Nullable ResultSetExtractor<?> extractor,
101+
Supplier<RowMapper<?>> rowMapper) {
101102

102103
if (getQueryMethod().isCollectionQuery()) {
103-
return extractor != null ? createSingleReadingQueryExecution(extractor) : collectionQuery(rowMapper);
104+
return extractor != null ? createSingleReadingQueryExecution(extractor) : collectionQuery(rowMapper.get());
104105
}
105106

106107
if (getQueryMethod().isStreamQuery()) {
107-
return extractor != null ? createSingleReadingQueryExecution(extractor) : streamQuery(rowMapper);
108+
return extractor != null ? createSingleReadingQueryExecution(extractor) : streamQuery(rowMapper.get());
108109
}
109110

110-
return extractor != null ? createSingleReadingQueryExecution(extractor) : singleObjectQuery(rowMapper);
111+
return extractor != null ? createSingleReadingQueryExecution(extractor) : singleObjectQuery(rowMapper.get());
111112
}
112113

113-
protected JdbcQueryExecution<Object> createModifyingQueryExecutor() {
114+
JdbcQueryExecution<Object> createModifyingQueryExecutor() {
114115

115116
return (query, parameters) -> {
116117

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,99 @@
1+
/*
2+
* Copyright 2018-2024 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.jdbc.repository.query;
17+
18+
import java.sql.SQLType;
19+
import java.util.List;
20+
21+
import org.springframework.core.MethodParameter;
22+
import org.springframework.data.jdbc.core.convert.JdbcColumnTypes;
23+
import org.springframework.data.jdbc.support.JdbcUtil;
24+
import org.springframework.data.relational.repository.query.RelationalParameters;
25+
import org.springframework.data.repository.query.Parameter;
26+
import org.springframework.data.repository.query.ParametersSource;
27+
import org.springframework.data.util.Lazy;
28+
import org.springframework.data.util.TypeInformation;
29+
30+
/**
31+
* Custom extension of {@link RelationalParameters}.
32+
*
33+
* @author Mark Paluch
34+
* @since 3.2.6
35+
*/
36+
public class JdbcParameters extends RelationalParameters {
37+
38+
/**
39+
* Creates a new {@link JdbcParameters} instance from the given {@link ParametersSource}.
40+
*
41+
* @param parametersSource must not be {@literal null}.
42+
*/
43+
public JdbcParameters(ParametersSource parametersSource) {
44+
super(parametersSource,
45+
methodParameter -> new JdbcParameter(methodParameter, parametersSource.getDomainTypeInformation()));
46+
}
47+
48+
@SuppressWarnings({ "rawtypes", "unchecked" })
49+
private JdbcParameters(List<JdbcParameter> parameters) {
50+
super((List) parameters);
51+
}
52+
53+
@Override
54+
public JdbcParameter getParameter(int index) {
55+
return (JdbcParameter) super.getParameter(index);
56+
}
57+
58+
@Override
59+
@SuppressWarnings({ "rawtypes", "unchecked" })
60+
protected JdbcParameters createFrom(List<RelationalParameter> parameters) {
61+
return new JdbcParameters((List) parameters);
62+
}
63+
64+
/**
65+
* Custom {@link Parameter} implementation.
66+
*
67+
* @author Mark Paluch
68+
* @author Chirag Tailor
69+
*/
70+
public static class JdbcParameter extends RelationalParameter {
71+
72+
private final SQLType sqlType;
73+
private final Lazy<SQLType> actualSqlType;
74+
75+
/**
76+
* Creates a new {@link RelationalParameter}.
77+
*
78+
* @param parameter must not be {@literal null}.
79+
*/
80+
JdbcParameter(MethodParameter parameter, TypeInformation<?> domainType) {
81+
super(parameter, domainType);
82+
83+
TypeInformation<?> typeInformation = getTypeInformation();
84+
85+
sqlType = JdbcUtil.targetSqlTypeFor(JdbcColumnTypes.INSTANCE.resolvePrimitiveType(typeInformation.getType()));
86+
87+
actualSqlType = Lazy.of(() -> JdbcUtil
88+
.targetSqlTypeFor(JdbcColumnTypes.INSTANCE.resolvePrimitiveType(typeInformation.getActualType().getType())));
89+
}
90+
91+
public SQLType getSqlType() {
92+
return sqlType;
93+
}
94+
95+
public SQLType getActualSqlType() {
96+
return actualSqlType.get();
97+
}
98+
}
99+
}

spring-data-jdbc/src/main/java/org/springframework/data/jdbc/repository/query/JdbcQueryMethod.java

+18-4
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@
4040
import org.springframework.lang.Nullable;
4141
import org.springframework.util.ClassUtils;
4242
import org.springframework.util.ConcurrentReferenceHashMap;
43+
import org.springframework.util.ObjectUtils;
4344
import org.springframework.util.StringUtils;
4445

4546
/**
@@ -59,6 +60,7 @@ public class JdbcQueryMethod extends QueryMethod {
5960
private final Map<Class<? extends Annotation>, Optional<Annotation>> annotationCache;
6061
private final NamedQueries namedQueries;
6162
private @Nullable RelationalEntityMetadata<?> metadata;
63+
private final boolean modifyingQuery;
6264

6365
// TODO: Remove NamedQueries and put it into JdbcQueryLookupStrategy
6466
public JdbcQueryMethod(Method method, RepositoryMetadata metadata, ProjectionFactory factory,
@@ -70,11 +72,12 @@ public JdbcQueryMethod(Method method, RepositoryMetadata metadata, ProjectionFac
7072
this.method = method;
7173
this.mappingContext = mappingContext;
7274
this.annotationCache = new ConcurrentReferenceHashMap<>();
75+
this.modifyingQuery = AnnotationUtils.findAnnotation(method, Modifying.class) != null;
7376
}
7477

7578
@Override
7679
protected Parameters<?, ?> createParameters(ParametersSource parametersSource) {
77-
return new RelationalParameters(parametersSource);
80+
return new JdbcParameters(parametersSource);
7881
}
7982

8083
@Override
@@ -108,8 +111,8 @@ public RelationalEntityMetadata<?> getEntityInformation() {
108111
}
109112

110113
@Override
111-
public RelationalParameters getParameters() {
112-
return (RelationalParameters) super.getParameters();
114+
public JdbcParameters getParameters() {
115+
return (JdbcParameters) super.getParameters();
113116
}
114117

115118
/**
@@ -124,6 +127,17 @@ String getDeclaredQuery() {
124127
return StringUtils.hasText(annotatedValue) ? annotatedValue : getNamedQuery();
125128
}
126129

130+
String getRequiredQuery() {
131+
132+
String query = getDeclaredQuery();
133+
134+
if (ObjectUtils.isEmpty(query)) {
135+
throw new IllegalStateException(String.format("No query specified on %s", getName()));
136+
}
137+
138+
return query;
139+
}
140+
127141
/**
128142
* Returns the annotated query if it exists.
129143
*
@@ -210,7 +224,7 @@ String getResultSetExtractorRef() {
210224
*/
211225
@Override
212226
public boolean isModifyingQuery() {
213-
return AnnotationUtils.findAnnotation(method, Modifying.class) != null;
227+
return modifyingQuery;
214228
}
215229

216230
@SuppressWarnings("unchecked")

0 commit comments

Comments
 (0)