@@ -110,7 +110,7 @@ public abstract class RepositoryFactorySupport
110
110
CONVERSION_SERVICE .removeConvertible (Object .class , Object .class );
111
111
}
112
112
113
- private final Map <RepositoryInformationCacheKey , RepositoryInformation > repositoryInformationCache ;
113
+ private final Map <RepositoryInformationCacheKey , RepositorySub > repositoryInformationCache ;
114
114
private final List <RepositoryProxyPostProcessor > postProcessors ;
115
115
116
116
private @ Nullable Class <?> repositoryBaseClass ;
@@ -130,7 +130,7 @@ public abstract class RepositoryFactorySupport
130
130
@ SuppressWarnings ("null" )
131
131
public RepositoryFactorySupport () {
132
132
133
- this .repositoryInformationCache = new HashMap <>(16 );
133
+ this .repositoryInformationCache = new HashMap <>(8 );
134
134
this .postProcessors = new ArrayList <>();
135
135
136
136
this .namedQueries = PropertiesBasedNamedQueries .EMPTY ;
@@ -294,16 +294,6 @@ protected RepositoryFragments getRepositoryFragments(RepositoryMetadata metadata
294
294
return RepositoryFragments .empty ();
295
295
}
296
296
297
- /**
298
- * Creates {@link RepositoryComposition} based on {@link RepositoryMetadata} for repository-specific method handling.
299
- *
300
- * @param metadata the repository metadata to use.
301
- * @return repository composition.
302
- */
303
- private RepositoryComposition getRepositoryComposition (RepositoryMetadata metadata ) {
304
- return RepositoryComposition .fromMetadata (metadata );
305
- }
306
-
307
297
/**
308
298
* Returns a repository instance for the given interface.
309
299
*
@@ -362,8 +352,9 @@ public <T> T getRepository(Class<T> repositoryInterface, RepositoryFragments fra
362
352
repositoryInterface );
363
353
repositoryCompositionStep .tag ("fragment.count" , String .valueOf (fragments .size ()));
364
354
365
- RepositoryComposition composition = getRepositoryComposition (metadata , fragments );
366
- RepositoryInformation information = getRepositoryInformation (metadata , composition );
355
+ RepositorySub stub = getRepositoryStub (metadata , fragments );
356
+ RepositoryComposition composition = stub .composition ();
357
+ RepositoryInformation information = stub .information ();
367
358
368
359
repositoryCompositionStep .tag ("fragments" , () -> {
369
360
@@ -493,47 +484,35 @@ protected RepositoryMetadata getRepositoryMetadata(Class<?> repositoryInterface)
493
484
* @return will never be {@literal null}.
494
485
*/
495
486
protected RepositoryInformation getRepositoryInformation (RepositoryMetadata metadata , RepositoryFragments fragments ) {
496
- return getRepositoryInformation (metadata , getRepositoryComposition ( metadata , fragments ));
487
+ return getRepositoryStub (metadata , fragments ). information ( );
497
488
}
498
489
499
490
/**
500
- * Returns the {@link RepositoryComposition} for the given {@link RepositoryMetadata} and extra
501
- * {@link RepositoryFragments}.
502
- *
503
- * @param metadata must not be {@literal null}.
504
- * @param fragments must not be {@literal null}.
505
- * @return will never be {@literal null}.
506
- */
507
- private RepositoryComposition getRepositoryComposition (RepositoryMetadata metadata , RepositoryFragments fragments ) {
508
-
509
- Assert .notNull (metadata , "RepositoryMetadata must not be null" );
510
- Assert .notNull (fragments , "RepositoryFragments must not be null" );
511
-
512
- RepositoryComposition composition = getRepositoryComposition (metadata );
513
- RepositoryFragments repositoryAspects = getRepositoryFragments (metadata );
514
-
515
- return composition .append (fragments ).append (repositoryAspects );
516
- }
517
-
518
- /**
519
- * Returns the {@link RepositoryInformation} for the given repository interface.
491
+ * Returns the cached {@link RepositorySub} for the given repository and composition. {@link RepositoryMetadata} is a
492
+ * strong cache key while {@link RepositoryFragments} contributes a light-weight caching component by using only the
493
+ * fragments hash code. In a typical Spring scenario, that shouldn't impose issues as one repository factory produces
494
+ * only a single repository instance for one repository interface. Things might be different when using various
495
+ * fragments for the same repository interface.
520
496
*
521
497
* @param metadata
522
- * @param composition
498
+ * @param fragments
523
499
* @return
524
500
*/
525
- private RepositoryInformation getRepositoryInformation (RepositoryMetadata metadata ,
526
- RepositoryComposition composition ) {
501
+ private RepositorySub getRepositoryStub (RepositoryMetadata metadata , RepositoryFragments fragments ) {
527
502
528
- RepositoryInformationCacheKey cacheKey = new RepositoryInformationCacheKey (metadata , composition );
503
+ RepositoryInformationCacheKey cacheKey = new RepositoryInformationCacheKey (metadata , fragments );
529
504
530
505
synchronized (repositoryInformationCache ) {
531
506
532
507
return repositoryInformationCache .computeIfAbsent (cacheKey , key -> {
533
508
509
+ RepositoryComposition composition = RepositoryComposition .fromMetadata (metadata );
510
+ RepositoryFragments repositoryAspects = getRepositoryFragments (metadata );
511
+ composition = composition .append (fragments ).append (repositoryAspects );
512
+
534
513
Class <?> baseClass = repositoryBaseClass != null ? repositoryBaseClass : getRepositoryBaseClass (metadata );
535
514
536
- return new DefaultRepositoryInformation (metadata , baseClass , composition );
515
+ return new RepositorySub ( new DefaultRepositoryInformation (metadata , baseClass , composition ) , composition );
537
516
});
538
517
}
539
518
}
@@ -816,6 +795,18 @@ public List<QueryMethod> getQueryMethods() {
816
795
}
817
796
}
818
797
798
+ /**
799
+ * Repository stub holding {@link RepositoryInformation} and {@link RepositoryComposition}.
800
+ *
801
+ * @param information
802
+ * @param composition
803
+ * @author Mark Paluch
804
+ * @since 3.4.4
805
+ */
806
+ record RepositorySub (RepositoryInformation information , RepositoryComposition composition ) {
807
+
808
+ }
809
+
819
810
/**
820
811
* Simple value object to build up keys to cache {@link RepositoryInformation} instances.
821
812
*
@@ -825,31 +816,26 @@ public List<QueryMethod> getQueryMethods() {
825
816
private static final class RepositoryInformationCacheKey {
826
817
827
818
private final String repositoryInterfaceName ;
828
- private final long compositionHash ;
819
+ private final long fragmentsHash ;
829
820
830
821
/**
831
- * Creates a new {@link RepositoryInformationCacheKey} for the given {@link RepositoryMetadata} and composition .
822
+ * Creates a new {@link RepositoryInformationCacheKey} for the given {@link RepositoryMetadata} and fragments .
832
823
*
833
824
* @param metadata must not be {@literal null}.
834
- * @param composition must not be {@literal null}.
825
+ * @param fragments must not be {@literal null}.
835
826
*/
836
- public RepositoryInformationCacheKey (RepositoryMetadata metadata , RepositoryComposition composition ) {
827
+ public RepositoryInformationCacheKey (RepositoryMetadata metadata , RepositoryFragments fragments ) {
837
828
838
829
this .repositoryInterfaceName = metadata .getRepositoryInterface ().getName ();
839
- this .compositionHash = composition .hashCode ();
840
- }
841
-
842
- public RepositoryInformationCacheKey (String repositoryInterfaceName , long compositionHash ) {
843
- this .repositoryInterfaceName = repositoryInterfaceName ;
844
- this .compositionHash = compositionHash ;
830
+ this .fragmentsHash = fragments .getFragments ().hashCode ();
845
831
}
846
832
847
833
public String getRepositoryInterfaceName () {
848
834
return this .repositoryInterfaceName ;
849
835
}
850
836
851
- public long getCompositionHash () {
852
- return this .compositionHash ;
837
+ public long getFragmentsHash () {
838
+ return this .fragmentsHash ;
853
839
}
854
840
855
841
@ Override
@@ -860,7 +846,7 @@ public boolean equals(Object o) {
860
846
if (!(o instanceof RepositoryInformationCacheKey that )) {
861
847
return false ;
862
848
}
863
- if (compositionHash != that .compositionHash ) {
849
+ if (fragmentsHash != that .fragmentsHash ) {
864
850
return false ;
865
851
}
866
852
return ObjectUtils .nullSafeEquals (repositoryInterfaceName , that .repositoryInterfaceName );
@@ -869,14 +855,14 @@ public boolean equals(Object o) {
869
855
@ Override
870
856
public int hashCode () {
871
857
int result = ObjectUtils .nullSafeHashCode (repositoryInterfaceName );
872
- result = 31 * result + ( int ) ( compositionHash ^ ( compositionHash >>> 32 ) );
858
+ result = 31 * result + Long . hashCode ( fragmentsHash );
873
859
return result ;
874
860
}
875
861
876
862
@ Override
877
863
public String toString () {
878
864
return "RepositoryFactorySupport.RepositoryInformationCacheKey(repositoryInterfaceName="
879
- + this .getRepositoryInterfaceName () + ", compositionHash =" + this .getCompositionHash () + ")" ;
865
+ + this .getRepositoryInterfaceName () + ", fragmentsHash =" + this .getFragmentsHash () + ")" ;
880
866
}
881
867
}
882
868
0 commit comments