⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 batchfetchqueue.java

📁 hibernate-3.1.3-all-src.zip 面向对象的访问数据库工具
💻 JAVA
字号:
//$Id: BatchFetchQueue.java 7205 2005-06-19 03:48:49Z oneovthafew $
package org.hibernate.engine;

import java.io.Serializable;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

import org.apache.commons.collections.SequencedHashMap;
import org.hibernate.EntityMode;
import org.hibernate.collection.PersistentCollection;
import org.hibernate.persister.collection.CollectionPersister;
import org.hibernate.persister.entity.EntityPersister;
import org.hibernate.util.MarkerObject;

/**
 * Tracks entity and collection keys that are available for batch
 * fetching, and the queries which were used to load entities, which
 * can be re-used as a subquery for loading owned collections.
 * 
 * @author Gavin King
 */
public class BatchFetchQueue {

	public static final Object MARKER = new MarkerObject("MARKER");
	
	// A set of entity keys that we predict we might need to load soon
	// TODO: this would be better as a SequencedReferenceSet, but no such beast exists!
	private final Map batchLoadableEntityKeys = new SequencedHashMap(8); //actually, a Set
	
	// The subqueries that were used to load the entity with the given key
	private final Map subselectsByEntityKey = new HashMap(8); //new ReferenceMap(ReferenceMap.HARD, ReferenceMap.SOFT);
	
	// The owning persistence context
	private final PersistenceContext context;
	
	public BatchFetchQueue(PersistenceContext context) {
		this.context = context;
	}
	
	public void clear() {
		batchLoadableEntityKeys.clear();
		subselectsByEntityKey.clear();
	}

	public SubselectFetch getSubselect(EntityKey key) {
		return (SubselectFetch) subselectsByEntityKey.get(key);
	}

	public void addSubselect(EntityKey key, SubselectFetch subquery) {
		subselectsByEntityKey.put(key, subquery);
	}

	public void clearSubselects() {
		subselectsByEntityKey.clear();
	}
	
	/**
	 * After evicting or deleting or loading an entity, we don't 
	 * need to batch fetch it anymore, remove it from the queue
	 * if necessary
	 */
	public void removeBatchLoadableEntityKey(EntityKey key) {
		if ( key.isBatchLoadable() ) batchLoadableEntityKeys.remove(key);
	}
	
	/**
	 * After evicting or deleting an entity, we don't need to 
	 * know the query that was used to load it anymore (don't 
	 * call this after loading the entity, since we might still
	 * need to load its collections)
	 */
	public void removeSubselect(EntityKey key) {
		subselectsByEntityKey.remove(key);
	}
	
	/**
	 * If an EntityKey represents a batch loadable entity, add
	 * it to the queue.
	 */
	public void addBatchLoadableEntityKey(EntityKey key) {
		if ( key.isBatchLoadable() ) batchLoadableEntityKeys.put(key, MARKER);
	}

	/**
	 * Get a batch of uninitialized collection keys for this role
	 * @param collectionPersister the collection role
	 * @param id a key that must be included
	 * @param batchSize the maximum number of keys to return
	 * @return an array of collection keys, of length batchSize (padded with nulls)
	 */
	public Serializable[] getCollectionBatch(
			final CollectionPersister collectionPersister, 
			final Serializable id, 
			final int batchSize,
			final EntityMode entityMode
	) {
		Serializable[] keys = new Serializable[batchSize];
		keys[0] = id;
		int i = 1;
		//int count = 0;
		int end = -1;
		boolean checkForEnd = false;
		// this only works because collection entries are kept in a sequenced
		// map by persistence context (maybe we should do like entities and
		// keep a separate sequences set...)
		Iterator iter = context.getCollectionEntries().entrySet().iterator(); //TODO: calling entrySet on an IdentityMap is SLOW!!
		while ( iter.hasNext() ) {
			Map.Entry me = (Map.Entry) iter.next();
			
			CollectionEntry ce = (CollectionEntry) me.getValue();
			PersistentCollection collection = (PersistentCollection) me.getKey();
			if ( !collection.wasInitialized() && ce.getLoadedPersister() == collectionPersister ) {
				
				if ( checkForEnd && i == end ) return keys; //the first key found after the given key
				
				//if ( end == -1 && count > batchSize*10 ) return keys; //try out ten batches, max
				
				final boolean isEqual = collectionPersister.getKeyType().isEqual( 
						id, 
						ce.getLoadedKey(), 
						entityMode, 
						collectionPersister.getFactory() 
					);
				
				if ( isEqual ) {
					end = i;
					//checkForEnd = false;
				}
				else {
					keys[i++] = ce.getLoadedKey();
					//count++;
				}
				
				if ( i == batchSize ) {
					i = 1; //end of array, start filling again from start
					if (end!=-1) checkForEnd = true;
				}
			}
			
		}
		return keys; //we ran out of keys to try
	}

	/**
	 * Get a batch of unloaded identifiers for this class, using a slightly
	 * complex algorithm that tries to grab keys registered immediately after
	 * the given key.
	 * 
	 * @param entityName The name of the persistent class
	 * @param id an identifier that must be included
	 * @param batchSize the maximum number of keys to return
	 * @return an array of identifiers, of length batchSize (padded with nulls)
	 */
	public Serializable[] getEntityBatch(
			final EntityPersister persister, 
			final Serializable id, 
			final int batchSize, 
			final EntityMode entityMode
	) {
		Serializable[] ids = new Serializable[batchSize];
		ids[0] = id; //first element of array is reserved for the actual instance we are loading!
		int i = 1;
		int end = -1;
		boolean checkForEnd = false;
		//int count = 0;
		Iterator iter = batchLoadableEntityKeys.keySet().iterator();
		while ( iter.hasNext() ) {
			
			EntityKey key = (EntityKey) iter.next();
			if ( key.getEntityName().equals( persister.getEntityName() ) ) { //TODO: this needn't exclude subclasses...
				
				if ( checkForEnd && i == end ) return ids; //the first id found after the given id

				//if ( end == -1 && count > batchSize*10 ) return ids; //try out ten batches, max

				if ( persister.getIdentifierType().isEqual( id, key.getIdentifier(), entityMode ) ) {
					end = i;
					//checkForEnd = false;
				}
				else {
					ids[i++] = key.getIdentifier();
					//count++;
				}
				
				if ( i == batchSize ) {
					i = 1; //end of array, start filling again from start
					if (end!=-1) checkForEnd = true;
				}
				
			}
			
		}
		return ids; //we ran out of ids to try
	}

}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -