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

src.org.python.modules.PyTeeIterator Maven / Gradle / Ivy

There is a newer version: 2.7.1.1
Show newest version
package org.python.modules;

import java.util.Map;

import org.python.core.Py;
import org.python.core.PyException;
import org.python.core.PyIterator;
import org.python.core.PyNewWrapper;
import org.python.core.PyObject;
import org.python.core.PyType;
import org.python.expose.ExposedMethod;
import org.python.expose.ExposedNew;
import org.python.expose.ExposedType;
import org.python.util.Generic;

@ExposedType(name = "itertools.tee", base = PyObject.class, isBaseType = false)
public class PyTeeIterator extends PyIterator {

    private final int position;
    private int count = 0;
    private final PyObject iterator;
    private final Map buffer;
    private final int[] offsets;

    PyTeeIterator(PyObject iterator, Map buffer, int[] offsets, int position) {
        this.iterator = iterator;
        this.buffer = buffer;
        this.offsets = offsets;
        this.position = position;
    }
    
    @ExposedNew
    final static PyObject tee___new__ (PyNewWrapper new_, boolean init,
            PyType subtype, PyObject[] args, String[] keywords) {
        final int nargs = args.length;
        // CPython tee ignores keywords, so we do too!
        if (nargs < 1 || nargs > 1) {
            throw Py.TypeError("tee expected 1 arguments, got " + nargs);
        }
        return makeTees(args[0], 1)[0];
    }

    public static PyTeeIterator[] makeTees(PyObject iterable, int n) {
        if (n < 0) {
            throw Py.ValueError("n must be >= 0");
        }
        PyObject iterator = iterable.__iter__();
        Map buffer = Generic.concurrentMap();
        int[] offsets = new int[n];
        PyTeeIterator[] tees = new PyTeeIterator[n];
        for (int i = 0; i < n; i++) {
            offsets[i] = -1;
            tees[i] = new PyTeeIterator(iterator, buffer, offsets, i);
        }
        return tees;
    }

    protected PyObject nextElement(PyObject pyIter) {
        PyObject element = null;
        try {
            element = pyIter.__iternext__();//next();
        } catch (PyException pyEx) {
            if (pyEx.match(Py.StopIteration)) {
                // store exception - will be used by PyIterator.next()
                stopException = pyEx;
            } else {
                throw pyEx;
            }
        }
        return element;
    }

    @ExposedMethod
    public final PyObject tee_next() {
        return next();
    }
    
    public PyObject __iternext__() {
        final PyObject item;
        int max = Integer.MIN_VALUE;
        int min = Integer.MAX_VALUE;
        for (int j = 0; j < offsets.length; j++) {
            if (max < offsets[j]) {
                max = offsets[j];
            }
            if (min > offsets[j]) {
                min = offsets[j];
            }
        }
        if (count > max) {
            item = nextElement(iterator);
            if (item != null) {
                buffer.put(count, item);
            }
        } else if (count < min) {
            item = buffer.remove(count);
        } else {
            item = buffer.get(count);
        }
        offsets[position] = count;
        count++;
        return item;
    }
}






© 2015 - 2025 Weber Informatics LLC | Privacy Policy