abstractdestinationcache.java

来自「OpenJMS是一个开源的Java Message Service API 1.」· Java 代码 · 共 400 行

JAVA
400
字号
/** * Redistribution and use of this software and associated documentation * ("Software"), with or without modification, are permitted provided * that the following conditions are met: * * 1. Redistributions of source code must retain copyright *    statements and notices.  Redistributions must also contain a *    copy of this document. * * 2. Redistributions in binary form must reproduce the *    above copyright notice, this list of conditions and the *    following disclaimer in the documentation and/or other *    materials provided with the distribution. * * 3. The name "Exolab" must not be used to endorse or promote *    products derived from this Software without prior written *    permission of Exoffice Technologies.  For written permission, *    please contact info@exolab.org. * * 4. Products derived from this Software may not be called "Exolab" *    nor may "Exolab" appear in their names without prior written *    permission of Exoffice Technologies. Exolab is a registered *    trademark of Exoffice Technologies. * * 5. Due credit should be given to the Exolab Project *    (http://www.exolab.org/). * * THIS SOFTWARE IS PROVIDED BY EXOFFICE TECHNOLOGIES AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT * NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL * EXOFFICE TECHNOLOGIES OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED * OF THE POSSIBILITY OF SUCH DAMAGE. * * Copyright 2001-2005 (C) Exoffice Technologies Inc. All Rights Reserved. * * $Id: AbstractDestinationCache.java,v 1.4 2007/01/24 12:00:28 tanderson Exp $ */package org.exolab.jms.messagemgr;import java.util.Collections;import java.util.HashMap;import java.util.Map;import javax.jms.JMSException;import org.apache.commons.logging.Log;import org.apache.commons.logging.LogFactory;import org.exolab.jms.client.JmsDestination;import org.exolab.jms.lease.LeaseEventListenerIfc;import org.exolab.jms.lease.LeaseManager;import org.exolab.jms.message.MessageImpl;import org.exolab.jms.persistence.PersistenceException;import org.exolab.jms.persistence.DatabaseService;/** * Abstract implementation of the {@link DestinationCache} interface. * * @author <a href="mailto:tma@netspace.net.au">Tim Anderson</a> * @version $Revision: 1.4 $ $Date: 2007/01/24 12:00:28 $ */public abstract class AbstractDestinationCache implements DestinationCache,        LeaseEventListenerIfc {    /**     * The destination to cache messages for.     */    private final JmsDestination _destination;    /**     * The message cache for this destination.     */    private DefaultMessageCache _cache = new DefaultMessageCache();    /**     * The set of consumers that have subscribed to this cache, keyed on id.     */    private Map _consumers = Collections.synchronizedMap(new HashMap());    /**     * A map of String -> MessageLease objects, representing the active leases     * keyed on JMSMessageID.     */    private final HashMap _leases = new HashMap();    /**     * The database service.     */    private final DatabaseService _database;    /**     * The lease manager.     */    private final LeaseManager _leaseMgr;    /**     * The logger.     */    private static final Log _log = LogFactory.getLog(            AbstractDestinationCache.class);    /**     * Construct a new <code>AbstractDestinationCache</code>.     *     * @param destination the destination to cache messages for     * @param database    the database service     * @param leases      the lease manager     */    public AbstractDestinationCache(JmsDestination destination,                                    DatabaseService database,                                    LeaseManager leases) {        if (destination == null) {            throw new IllegalArgumentException(                    "Argument 'destination' is null");        }        if (database == null) {            throw new IllegalArgumentException("Argument 'database' is null");        }        if (leases == null) {            throw new IllegalArgumentException("Argument 'leases' is null");        }        _destination = destination;        _database = database;        _leaseMgr = leases;    }    /**     * Returns the destination that messages are being cached for.     *     * @return the destination that messages are being cached for     */    public JmsDestination getDestination() {        return _destination;    }    /**     * Register a consumer with this cache.     *     * @param consumer the message consumer for this destination     * @return <code>true</code> if registered; otherwise <code>false</code>     */    public boolean addConsumer(ConsumerEndpoint consumer) {        boolean result = false;        // check to see that the consumer is actually one for this        // destination        if (consumer.getDestination().equals(getDestination())) {            Long key = new Long(consumer.getId());            if (!_consumers.containsKey(key)) {                _consumers.put(key, consumer);                result = true;            }        }        return result;    }    /**     * Remove the consumer for the list of registered consumers.     *     * @param consumer the consumer to remove     */    public void removeConsumer(ConsumerEndpoint consumer) {        Long key = new Long(consumer.getId());        _consumers.remove(key);    }    /**     * Determines if the cache has any consumers.     *     * @return <code>true</code> if the cache has consumers; otherwise     *         <code>false</code>     */    public boolean hasConsumers() {        return !_consumers.isEmpty();    }    /**     * Returns the number of messages in the cache.     *     * @return the number of messages in the cache     */    public int getMessageCount() {        return _cache.getMessageCount();    }    /**     * Determines if this cache can be destroyed. This implementation returns     * <code>true</code> if there are no active consumers.     *     * @return <code>true</code> if the cache can be destroyed, otherwise     *         <code>false</code>     */    public boolean canDestroy() {        return !hasConsumers();    }    /**     * Destroy this cache.     */    public synchronized void destroy() {        // clear the cache        _cache.clear();        // remove the consumers        _consumers.clear();        // remove the leases        MessageLease[] leases;        synchronized (_leases) {            leases = (MessageLease[]) _leases.values().toArray(                    new MessageLease[0]);            _leases.clear();        }        for (int i = 0; i < leases.length; ++i) {            MessageLease lease = leases[i];            _leaseMgr.removeLease(lease);        }    }    /**     * Invoked when a message lease has expired.     *     * @param object an instance of {@link MessageRef}     */    public void onLeaseExpired(Object object) {        MessageRef reference = (MessageRef) object;        String messageId = reference.getMessageId();        synchronized (_leases) {            _leases.remove(messageId);        }        // determine whether the message is persistent or not and take        // the corresponding action        try {            _database.begin();            if (reference.isPersistent()) {                persistentMessageExpired(reference);            } else {                messageExpired(reference);            }            reference.destroy();            _database.commit();        } catch (Exception exception) {            _log.error("Failed to expire message", exception);            try {                _database.rollback();            } catch (PersistenceException error) {                _log.warn("Failed to rollback", error);            }        }    }    public void collectGarbage(boolean aggressive) {        if (aggressive) {            // clear all persistent messages in the cache            _cache.clearPersistentMessages();            if (_log.isDebugEnabled()) {                _log.debug("Evicted all persistent messages from cache "                           + getDestination().getName());            }        }        if (_log.isDebugEnabled()) {            _log.debug("DESTCACHE -" + getDestination().getName()                       + " Messages: P[" + _cache.getPersistentCount()                       + "] T[" + _cache.getTransientCount() + "] Total: ["                       + _cache.getMessageCount() + "]");        }    }    /**     * Add a message reference and its corresponding message to the cache     *     * @param reference the reference to the message     * @param message   the message     */    protected void addMessage(MessageRef reference, MessageImpl message) {        if (_log.isDebugEnabled()) {            _log.debug("addMessage(reference=[JMSMessageID="                    + reference.getMessageId() + "])");        }        _cache.addMessage(reference, message);    }    /**     * Returns the message cache.     *     * @return the message cache     */    protected DefaultMessageCache getMessageCache() {        return _cache;    }    /**     * Returns a consumer endpoint, given its id.     *     * @param consumerId the consumer identity     * @return the consumer corresponding to <code>id</code>, or     *         <code>null</code> if none is registered     */    protected ConsumerEndpoint getConsumerEndpoint(long consumerId) {        return (ConsumerEndpoint) _consumers.get(new Long(consumerId));    }    /**     * Helper to return the consumers as an array.     *     * @return the consumers of this cache     */    protected ConsumerEndpoint[] getConsumerArray() {        return (ConsumerEndpoint[]) _consumers.values().toArray(                new ConsumerEndpoint[0]);    }    /**     * Remove an expired non-peristent message, and notify any listeners.     *     * @param reference the reference to the expired message     * @throws JMSException for any error     */    protected void messageExpired(MessageRef reference)            throws JMSException {        // notify consumers        String messageId = reference.getMessageId();        ConsumerEndpoint[] consumers = getConsumerArray();        for (int i = 0; i < consumers.length; ++i) {            consumers[i].messageRemoved(messageId);        }    }    /**     * Remove an expired persistent message, and notify any listeners.     *     * @param reference  the reference to the expired message     * @throws JMSException         if a listener fails to handle the     *                              expiration     * @throws PersistenceException if there is a persistence related problem     */    protected void persistentMessageExpired(MessageRef reference)            throws JMSException, PersistenceException {        // notify consumers        String messageId = reference.getMessageId();        ConsumerEndpoint[] consumers = getConsumerArray();        for (int i = 0; i < consumers.length; ++i) {            consumers[i].persistentMessageRemoved(messageId);        }    }    /**     * Check to see if the message has a TTL. If so then set up a lease for it.     * An expiry time of 0 means that the message never expires     *     * @param reference a reference to the message     * @param message   the message     * @throws JMSException if the JMSExpiration property can't be accessed     */    protected void checkMessageExpiry(MessageRef reference,                                      MessageImpl message) throws JMSException {        checkMessageExpiry(reference, message.getJMSExpiration());    }    /**     * Check to see if the message has a TTL. If so then set up a lease for it.     * An expiry time of 0 means that the message never expires     *     * @param reference  a reference to the message     * @param expiryTime the time when the message expires     */    protected void checkMessageExpiry(MessageRef reference,                                      long expiryTime) {        if (expiryTime != 0) {            synchronized (_leases) {                // ensure that a lease for this message does not already exist.                if (!_leases.containsKey(reference.getMessageId())) {                    long duration = expiryTime - System.currentTimeMillis();                    if (duration <= 0) {                        duration = 1;                    }                    MessageLease lease = new MessageLease(reference, duration,                                                          this);                    _leaseMgr.addLease(lease);                    _leases.put(reference.getMessageId(), lease);                }            }        }    }}

⌨️ 快捷键说明

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