HHH-8961 Reduce allocation cost of org.hibernate.cache.spi.CacheKey
instances Remove tenantId field from CacheKey: use a different type when tenants are needed. Also remove the Type as we should be able to rely on the entityOrRoleName String. Conflicts: hibernate-core/src/main/java/org/hibernate/cache/spi/CacheKey.java
This commit is contained in:
parent
0c5c980790
commit
a7910b19a4
54
hibernate-core/src/main/java/org/hibernate/cache/internal/TenantAwareCacheKey.java
vendored
Normal file
54
hibernate-core/src/main/java/org/hibernate/cache/internal/TenantAwareCacheKey.java
vendored
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
|
*
|
||||||
|
* Copyright (c) 2014, Red Hat Inc. or third-party contributors as
|
||||||
|
* indicated by the @author tags or express copyright attribution
|
||||||
|
* statements applied by the authors. All third-party contributions are
|
||||||
|
* distributed under license by Red Hat Inc.
|
||||||
|
*
|
||||||
|
* This copyrighted material is made available to anyone wishing to use, modify,
|
||||||
|
* copy, or redistribute it subject to the terms and conditions of the GNU
|
||||||
|
* Lesser General Public License, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
|
||||||
|
* for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public License
|
||||||
|
* along with this distribution; if not, write to:
|
||||||
|
* Free Software Foundation, Inc.
|
||||||
|
* 51 Franklin Street, Fifth Floor
|
||||||
|
* Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
package org.hibernate.cache.internal;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
import org.hibernate.cache.spi.CacheKey;
|
||||||
|
import org.hibernate.engine.spi.SessionFactoryImplementor;
|
||||||
|
import org.hibernate.type.Type;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Variation of CacheKey to be used when multi-tenancy is applied.
|
||||||
|
* The tenantId field was not simply added to the superclass because of performance considerations:
|
||||||
|
* CacheKey instances are have a very high allocation frequency and this field would enlarge the
|
||||||
|
* size by 50%.
|
||||||
|
*
|
||||||
|
* @author Sanne Grinovero
|
||||||
|
*/
|
||||||
|
public class TenantAwareCacheKey extends CacheKey {
|
||||||
|
|
||||||
|
private final String tenantId;
|
||||||
|
|
||||||
|
public TenantAwareCacheKey(Serializable id, Type type, String entityOrRoleName, SessionFactoryImplementor factory, String tenantId) {
|
||||||
|
super( id, type, entityOrRoleName, factory, tenantId );
|
||||||
|
this.tenantId = tenantId;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getTenantId() {
|
||||||
|
return tenantId;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -33,14 +33,20 @@ import org.hibernate.type.Type;
|
||||||
* Allows multiple entity classes / collection roles to be stored in the same cache region. Also allows for composite
|
* Allows multiple entity classes / collection roles to be stored in the same cache region. Also allows for composite
|
||||||
* keys which do not properly implement equals()/hashCode().
|
* keys which do not properly implement equals()/hashCode().
|
||||||
*
|
*
|
||||||
|
* Note on performance: this class is allocated very heavily!
|
||||||
|
* Make sure the allocation cost stays as low as possible:
|
||||||
|
* if you need to add fields for not-so-common use cases,
|
||||||
|
* it's probably better to create an ad-hoc implementation
|
||||||
|
* extending this one.
|
||||||
|
*
|
||||||
* @author Gavin King
|
* @author Gavin King
|
||||||
* @author Steve Ebersole
|
* @author Steve Ebersole
|
||||||
|
* @author Sanne Grinovero
|
||||||
*/
|
*/
|
||||||
public class CacheKey implements Serializable {
|
public class CacheKey implements Serializable {
|
||||||
|
|
||||||
private final Serializable key;
|
private final Serializable key;
|
||||||
private final Type type;
|
private final Type type;
|
||||||
private final String entityOrRoleName;
|
|
||||||
private final String tenantId;
|
|
||||||
private final int hashCode;
|
private final int hashCode;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -58,16 +64,25 @@ public class CacheKey implements Serializable {
|
||||||
final Serializable id,
|
final Serializable id,
|
||||||
final Type type,
|
final Type type,
|
||||||
final String entityOrRoleName,
|
final String entityOrRoleName,
|
||||||
final String tenantId,
|
|
||||||
final SessionFactoryImplementor factory) {
|
final SessionFactoryImplementor factory) {
|
||||||
this.key = id;
|
this( id, type, entityOrRoleName, factory, null );
|
||||||
this.type = type;
|
|
||||||
this.entityOrRoleName = entityOrRoleName;
|
|
||||||
this.tenantId = tenantId;
|
|
||||||
this.hashCode = calculateHashCode( type, factory );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private int calculateHashCode(Type type, SessionFactoryImplementor factory) {
|
/**
|
||||||
|
* Used by subclasses: need to specify a tenantId
|
||||||
|
*/
|
||||||
|
protected CacheKey(
|
||||||
|
final Serializable id,
|
||||||
|
final Type type,
|
||||||
|
final String entityOrRoleName,
|
||||||
|
final SessionFactoryImplementor factory,
|
||||||
|
final String tenantId) {
|
||||||
|
this.key = id;
|
||||||
|
this.type = type;
|
||||||
|
this.hashCode = calculateHashCode( type, factory, tenantId );
|
||||||
|
}
|
||||||
|
|
||||||
|
private int calculateHashCode(Type type, SessionFactoryImplementor factory, String tenantId) {
|
||||||
int result = type.getHashCode( key, factory );
|
int result = type.getHashCode( key, factory );
|
||||||
result = 31 * result + (tenantId != null ? tenantId.hashCode() : 0);
|
result = 31 * result + (tenantId != null ? tenantId.hashCode() : 0);
|
||||||
return result;
|
return result;
|
||||||
|
@ -77,12 +92,8 @@ public class CacheKey implements Serializable {
|
||||||
return key;
|
return key;
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getEntityOrRoleName() {
|
|
||||||
return entityOrRoleName;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getTenantId() {
|
public String getTenantId() {
|
||||||
return tenantId;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -97,20 +108,14 @@ public class CacheKey implements Serializable {
|
||||||
//hashCode is part of this check since it is pre-calculated and hash must match for equals to be true
|
//hashCode is part of this check since it is pre-calculated and hash must match for equals to be true
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
CacheKey that = (CacheKey) other;
|
//Warning: this equals implementation needs to work correctly also for the subclass
|
||||||
return EqualsHelper.equals( entityOrRoleName, that.entityOrRoleName ) &&
|
final CacheKey that = (CacheKey) other;
|
||||||
type.isEqual( key, that.key ) &&
|
return EqualsHelper.equals( getTenantId(), that.getTenantId() )
|
||||||
EqualsHelper.equals( tenantId, that.tenantId );
|
&& type.isEqual( key, that.key );
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
return hashCode;
|
return hashCode;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
// Mainly for OSCache
|
|
||||||
return entityOrRoleName + '#' + key.toString();//"CacheKey#" + type.toString(key, sf);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -38,6 +38,7 @@ import org.hibernate.ScrollableResults;
|
||||||
import org.hibernate.SessionEventListener;
|
import org.hibernate.SessionEventListener;
|
||||||
import org.hibernate.SessionException;
|
import org.hibernate.SessionException;
|
||||||
import org.hibernate.SharedSessionContract;
|
import org.hibernate.SharedSessionContract;
|
||||||
|
import org.hibernate.cache.internal.TenantAwareCacheKey;
|
||||||
import org.hibernate.cache.spi.CacheKey;
|
import org.hibernate.cache.spi.CacheKey;
|
||||||
import org.hibernate.engine.jdbc.LobCreationContext;
|
import org.hibernate.engine.jdbc.LobCreationContext;
|
||||||
import org.hibernate.engine.jdbc.spi.JdbcConnectionAccess;
|
import org.hibernate.engine.jdbc.spi.JdbcConnectionAccess;
|
||||||
|
@ -250,7 +251,13 @@ public abstract class AbstractSessionImpl implements Serializable, SharedSession
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public CacheKey generateCacheKey(Serializable id, Type type, String entityOrRoleName) {
|
public CacheKey generateCacheKey(Serializable id, Type type, String entityOrRoleName) {
|
||||||
return new CacheKey( id, type, entityOrRoleName, getTenantIdentifier(), getFactory() );
|
final String tenantIdentifier = getTenantIdentifier();
|
||||||
|
if ( tenantIdentifier == null ) {
|
||||||
|
return new CacheKey( id, type, entityOrRoleName, getFactory() );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return new TenantAwareCacheKey( id, type, entityOrRoleName, getFactory(), tenantIdentifier );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private transient JdbcConnectionAccess jdbcConnectionAccess;
|
private transient JdbcConnectionAccess jdbcConnectionAccess;
|
||||||
|
|
|
@ -119,11 +119,11 @@ public class CacheImpl implements CacheImplementor {
|
||||||
}
|
}
|
||||||
|
|
||||||
private CacheKey buildCacheKey(Serializable identifier, EntityPersister p) {
|
private CacheKey buildCacheKey(Serializable identifier, EntityPersister p) {
|
||||||
|
// have to assume non tenancy
|
||||||
return new CacheKey(
|
return new CacheKey(
|
||||||
identifier,
|
identifier,
|
||||||
p.getIdentifierType(),
|
p.getIdentifierType(),
|
||||||
p.getRootEntityName(),
|
p.getRootEntityName(),
|
||||||
null, // have to assume non tenancy
|
|
||||||
sessionFactory
|
sessionFactory
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -197,11 +197,11 @@ public class CacheImpl implements CacheImplementor {
|
||||||
}
|
}
|
||||||
|
|
||||||
private CacheKey buildCacheKey(Serializable ownerIdentifier, CollectionPersister p) {
|
private CacheKey buildCacheKey(Serializable ownerIdentifier, CollectionPersister p) {
|
||||||
|
// have to assume non tenancy
|
||||||
return new CacheKey(
|
return new CacheKey(
|
||||||
ownerIdentifier,
|
ownerIdentifier,
|
||||||
p.getKeyType(),
|
p.getKeyType(),
|
||||||
p.getRole(),
|
p.getRole(),
|
||||||
null, // have to assume non tenancy
|
|
||||||
sessionFactory
|
sessionFactory
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue