/src/main/java/org/apache/commons/io/LineIterator.java

https://github.com/apache/commons-io · Java · 187 lines · 76 code · 13 blank · 98 comment · 12 complexity · 5735c19a4bcc856e759d5aedf2a7153e MD5 · raw file

  1. /*
  2. * Licensed to the Apache Software Foundation (ASF) under one or more
  3. * contributor license agreements. See the NOTICE file distributed with
  4. * this work for additional information regarding copyright ownership.
  5. * The ASF licenses this file to You under the Apache License, Version 2.0
  6. * (the "License"); you may not use this file except in compliance with
  7. * the License. You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. package org.apache.commons.io;
  18. import java.io.BufferedReader;
  19. import java.io.Closeable;
  20. import java.io.IOException;
  21. import java.io.Reader;
  22. import java.util.Iterator;
  23. import java.util.NoSuchElementException;
  24. /**
  25. * An Iterator over the lines in a {@code Reader}.
  26. * <p>
  27. * {@code LineIterator} holds a reference to an open {@code Reader}.
  28. * When you have finished with the iterator you should close the reader
  29. * to free internal resources. This can be done by closing the reader directly,
  30. * or by calling the {@link #close()} or {@link #closeQuietly(LineIterator)}
  31. * method on the iterator.
  32. * <p>
  33. * The recommended usage pattern is:
  34. * <pre>
  35. * LineIterator it = FileUtils.lineIterator(file, "UTF-8");
  36. * try {
  37. * while (it.hasNext()) {
  38. * String line = it.nextLine();
  39. * // do something with line
  40. * }
  41. * } finally {
  42. * it.close();
  43. * }
  44. * </pre>
  45. *
  46. * @since 1.2
  47. */
  48. public class LineIterator implements Iterator<String>, Closeable {
  49. // N.B. This class deliberately does not implement Iterable, see https://issues.apache.org/jira/browse/IO-181
  50. /**
  51. * Closes a {@code LineIterator} quietly.
  52. *
  53. * @param iterator The iterator to close, or {@code null}.
  54. * @deprecated As of 2.6 deprecated without replacement. Please use the try-with-resources statement or handle
  55. * suppressed exceptions manually.
  56. * @see Throwable#addSuppressed(java.lang.Throwable)
  57. */
  58. @Deprecated
  59. public static void closeQuietly(final LineIterator iterator) {
  60. IOUtils.closeQuietly(iterator);
  61. }
  62. /** The reader that is being read. */
  63. private final BufferedReader bufferedReader;
  64. /** The current line. */
  65. private String cachedLine;
  66. /** A flag indicating if the iterator has been fully read. */
  67. private boolean finished;
  68. /**
  69. * Constructs an iterator of the lines for a {@code Reader}.
  70. *
  71. * @param reader the {@code Reader} to read from, not null
  72. * @throws IllegalArgumentException if the reader is null
  73. */
  74. public LineIterator(final Reader reader) throws IllegalArgumentException {
  75. if (reader == null) {
  76. throw new IllegalArgumentException("Reader must not be null");
  77. }
  78. if (reader instanceof BufferedReader) {
  79. bufferedReader = (BufferedReader) reader;
  80. } else {
  81. bufferedReader = new BufferedReader(reader);
  82. }
  83. }
  84. /**
  85. * Closes the underlying {@code Reader}.
  86. * This method is useful if you only want to process the first few
  87. * lines of a larger file. If you do not close the iterator
  88. * then the {@code Reader} remains open.
  89. * This method can safely be called multiple times.
  90. *
  91. * @throws IOException if closing the underlying {@code Reader} fails.
  92. */
  93. @Override
  94. public void close() throws IOException {
  95. finished = true;
  96. cachedLine = null;
  97. IOUtils.close(bufferedReader);
  98. }
  99. /**
  100. * Indicates whether the {@code Reader} has more lines.
  101. * If there is an {@code IOException} then {@link #close()} will
  102. * be called on this instance.
  103. *
  104. * @return {@code true} if the Reader has more lines
  105. * @throws IllegalStateException if an IO exception occurs
  106. */
  107. @Override
  108. public boolean hasNext() {
  109. if (cachedLine != null) {
  110. return true;
  111. }
  112. if (finished) {
  113. return false;
  114. }
  115. try {
  116. while (true) {
  117. final String line = bufferedReader.readLine();
  118. if (line == null) {
  119. finished = true;
  120. return false;
  121. }
  122. if (isValidLine(line)) {
  123. cachedLine = line;
  124. return true;
  125. }
  126. }
  127. } catch(final IOException ioe) {
  128. IOUtils.closeQuietly(this, ioe::addSuppressed);
  129. throw new IllegalStateException(ioe);
  130. }
  131. }
  132. /**
  133. * Overridable method to validate each line that is returned.
  134. * This implementation always returns true.
  135. * @param line the line that is to be validated
  136. * @return true if valid, false to remove from the iterator
  137. */
  138. protected boolean isValidLine(final String line) {
  139. return true;
  140. }
  141. /**
  142. * Returns the next line in the wrapped {@code Reader}.
  143. *
  144. * @return the next line from the input
  145. * @throws NoSuchElementException if there is no line to return
  146. */
  147. @Override
  148. public String next() {
  149. return nextLine();
  150. }
  151. /**
  152. * Returns the next line in the wrapped {@code Reader}.
  153. *
  154. * @return the next line from the input
  155. * @throws NoSuchElementException if there is no line to return
  156. */
  157. public String nextLine() {
  158. if (!hasNext()) {
  159. throw new NoSuchElementException("No more lines");
  160. }
  161. final String currentLine = cachedLine;
  162. cachedLine = null;
  163. return currentLine;
  164. }
  165. /**
  166. * Unsupported.
  167. *
  168. * @throws UnsupportedOperationException always
  169. */
  170. @Override
  171. public void remove() {
  172. throw new UnsupportedOperationException("remove not supported");
  173. }
  174. }