/* Copyright (c) 2013 James Ahlborn Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package com.healthmarketscience.jackcess; import java.io.IOException; import java.util.Collection; import java.util.Iterator; import java.util.Map; import java.util.stream.Stream; import java.util.stream.StreamSupport; import com.healthmarketscience.jackcess.util.ColumnMatcher; import com.healthmarketscience.jackcess.util.ErrorHandler; import com.healthmarketscience.jackcess.util.IterableBuilder; /** * Manages iteration for a {@link Table}. Different cursors provide different * methods of traversing a table. Cursors should be fairly robust in the face * of table modification during traversal (although depending on how the table * is traversed, row updates may or may not be seen). Multiple cursors may * traverse the same table simultaneously. *
* Basic cursors will generally iterate table data in the order it appears in * the database and searches will require scanning the entire table. * Additional features are available when utilizing an {@link Index} backed * {@link IndexCursor}. *
* The {@link CursorBuilder} provides a variety of static utility methods to * construct cursors with given characteristics or easily search for specific * values as well as friendly and flexible construction options. *
* A Cursor instance is not thread-safe (see {@link Database} for more
* thread-safety details).
*
* @author James Ahlborn
* @usage _general_class_
*/
public interface Cursor extends Iterable
* Savepoints may be used across different cursor instances for the same
* table, but they must have the same {@link Id}.
*/
public Savepoint getSavepoint();
/**
* Moves the cursor to a savepoint previously returned from
* {@link #getSavepoint}.
* @throws IllegalArgumentException if the given savepoint does not have a
* cursorId equal to this cursor's id
*/
public void restoreSavepoint(Savepoint savepoint)
throws IOException;
/**
* Resets this cursor for forward traversal. Calls {@link #beforeFirst}.
*/
public void reset();
/**
* Resets this cursor for forward traversal (sets cursor to before the first
* row).
*/
public void beforeFirst();
/**
* Resets this cursor for reverse traversal (sets cursor to after the last
* row).
*/
public void afterLast();
/**
* Returns {@code true} if the cursor is currently positioned before the
* first row, {@code false} otherwise.
*/
public boolean isBeforeFirst() throws IOException;
/**
* Returns {@code true} if the cursor is currently positioned after the
* last row, {@code false} otherwise.
*/
public boolean isAfterLast() throws IOException;
/**
* Returns {@code true} if the row at which the cursor is currently
* positioned is deleted, {@code false} otherwise (including invalid rows).
*/
public boolean isCurrentRowDeleted() throws IOException;
/**
* Calls {@link #beforeFirst} on this cursor and returns a modifiable
* Iterator which will iterate through all the rows of this table. Use of
* the Iterator follows the same restrictions as a call to
* {@link #getNextRow}.
*
* For more flexible iteration see {@link #newIterable}.
* @throws RuntimeIOException if an IOException is thrown by one of the
* operations, the actual exception will be contained within
*/
@Override
public Iterator
* Note, re-deleting an already deleted row is allowed (it does nothing).
* @throws IllegalStateException if the current row is not valid (at
* beginning or end of table)
*/
public void deleteCurrentRow() throws IOException;
/**
* Update the current row.
* @return the given row values if long enough, otherwise a new array,
* updated with the current row values
* @throws IllegalStateException if the current row is not valid (at
* beginning or end of table), or deleted.
*/
public Object[] updateCurrentRow(Object... row) throws IOException;
/**
* Update the current row.
* @return the given row, updated with the current row values
* @throws IllegalStateException if the current row is not valid (at
* beginning or end of table), or deleted.
*/
public
* Warning, this method always starts searching from the beginning of
* the Table (you cannot use it to find successive matches).
*
* @param columnPattern column from the table for this cursor which is being
* matched by the valuePattern
* @param valuePattern value which is equal to the corresponding value in
* the matched row. If this object is an instance of
* {@link java.util.function.Predicate}, it will be
* applied to the potential row value instead
* (overriding any configured ColumnMatcher)
* @return {@code true} if a valid row was found with the given value,
* {@code false} if no row was found
*/
public boolean findFirstRow(Column columnPattern, Object valuePattern)
throws IOException;
/**
* Moves to the next row (as defined by the cursor) where the given column
* has the given value. This may be more efficient on some cursors than
* others. If a match is not found (or an exception is thrown), the cursor
* is restored to its previous state.
*
* @param columnPattern column from the table for this cursor which is being
* matched by the valuePattern
* @param valuePattern value which is equal to the corresponding value in
* the matched row. If this object is an instance of
* {@link java.util.function.Predicate}, it will be
* applied to the potential row value instead
* (overriding any configured ColumnMatcher)
* @return {@code true} if a valid row was found with the given value,
* {@code false} if no row was found
*/
public boolean findNextRow(Column columnPattern, Object valuePattern)
throws IOException;
/**
* Moves to the first row (as defined by the cursor) where the given columns
* have the given values. This may be more efficient on some cursors than
* others. If a match is not found (or an exception is thrown), the cursor
* is restored to its previous state.
*
* Warning, this method always starts searching from the beginning of
* the Table (you cannot use it to find successive matches).
*
* @param rowPattern column names and values which must be equal to the
* corresponding values in the matched row. If a value is
* an instance of {@link java.util.function.Predicate}, it
* will be applied to the potential row value instead
* (overriding any configured ColumnMatcher)
* @return {@code true} if a valid row was found with the given values,
* {@code false} if no row was found
*/
public boolean findFirstRow(Map