All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.bigdata.relation.accesspath.MultiSourceSequentialCloseableIterator Maven / Gradle / Ivy

/**

Copyright (C) SYSTAP, LLC DBA Blazegraph 2006-2016.  All rights reserved.

Contact:
     SYSTAP, LLC DBA Blazegraph
     2501 Calvert ST NW #106
     Washington, DC 20008
     [email protected]

This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; version 2 of the License.

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 General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
*/
/*
 * Created on Oct 19, 2010
 */

package com.bigdata.relation.accesspath;

import java.util.NoSuchElementException;
import java.util.Queue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.locks.ReentrantLock;

import org.apache.log4j.Logger;

import cutthecrap.utils.striterators.ICloseableIterator;

/**
 * Class allows new sources to be attached dynamically. If the existing sources
 * are drained then the iterator will {@link #close()} itself so that new
 * sources can no longer be attached.
 * 
 * @author Bryan Thompson
 * @version $Id$
 */
public class MultiSourceSequentialCloseableIterator implements
        IMultiSourceCloseableIterator {

    private final static Logger log = Logger.getLogger(MultiSourceSequentialCloseableIterator.class);
    
    private final static boolean INFO = log.isInfoEnabled();
    
    private final ReentrantLock lock = new ReentrantLock();

    private final Queue> sources = new LinkedBlockingQueue>();

    /**
     * The current inner iterator. When null the outer iterator has
     * been closed and will not deliver any more results and will not accept any
     * new sources.
     * 

* Note: This can be asynchronously closed if the application invokes * {@link #close()}. Methods which test on this can not assume that it will * be non-null the next time they check unless they are holding * the {@link #lock}. Methods which do not obtain the lock can offer a * weaker atomicity by copying the reference to a local variable and then * testing that variable. */ private volatile ICloseableIterator current; public MultiSourceSequentialCloseableIterator(final ICloseableIterator src) { current = src; } @Override public void close() { lock.lock(); try { /* * Ensure that all sources are eventually closed. */ // close the current source (if any). final ICloseableIterator current = this.current; this.current = null; if (current != null) { if (INFO) log.info("Closing source: " + current); current.close(); } // Close any sources still in the queue. for(ICloseableIterator t : sources) { if (INFO) log.info("Closing source: " + t); t.close(); } // Clear the queue. sources.clear(); } finally { lock.unlock(); } } @Override public boolean add(final ICloseableIterator src) { if (src == null) throw new IllegalArgumentException(); lock.lock(); try { if (current == null) return false; sources.add(src); return true; } finally { lock.unlock(); } } /** * If the current source is not exhausted, then return it immediately. * Otherwise, return the next source which is not exhausted. If no such * sources are available, then {@link #close()} the iterator. The decision * to accept another source or to close the iterator is made atomic by the * use of the {@link #lock} in this method and in {@link #close()}. * * @return The next source -or- null if there are no sources * available. */ private ICloseableIterator nextSource() { final ICloseableIterator tmp = current; if (tmp == null) return null; if (tmp.hasNext()) return current; // Note: MAY be asynchronously cleared! // current is known to be [null]. lock.lock(); try { /** * Close iterator which has been consumed. * * @see MultiSourceSequentialCloseableIterator.nextSource() can * throw NPE */ ICloseableIterator t = this.current; { if (t != null) { if (INFO) log.info("Closing source: " + t); t.close(); } } // remove the head of the queue (non-blocking) while ((t = current = sources.poll()) != null) { if (t.hasNext()) { return t; } else { // Note: should already be closed since exhausted. t.close(); } } // no more sources with data, close while holding lock. close(); return null; } finally { lock.unlock(); } } @Override public boolean hasNext() { while (true) { final ICloseableIterator tmp = nextSource(); if (tmp == null) return false; if (tmp.hasNext()) return true; } } /** * {@inheritDoc} * * @todo Due to the inherent non-atomicity of the while(hasNext()) next() * idiom, it is possible for {@link #hasNext()} to report true and for * {@link #next()} to throw {@link NoSuchElementException} if the * iterator has been concurrently closed. */ @Override public E next() { while (true) { final ICloseableIterator tmp = nextSource(); if (tmp == null) throw new NoSuchElementException(); if (tmp.hasNext()) return tmp.next(); } } @Override public void remove() { throw new UnsupportedOperationException(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy