mirror of https://github.com/apache/poi.git
Apply patch from bug #51460 (with some related generics tweaks) - Improve HSSF performance when loading very long rows, by switching the CellValue array to an iterator
git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@1141970 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
663068fb50
commit
5d2797bff2
|
@ -34,7 +34,8 @@
|
||||||
|
|
||||||
<changes>
|
<changes>
|
||||||
<release version="3.8-beta4" date="2011-??-??">
|
<release version="3.8-beta4" date="2011-??-??">
|
||||||
<action dev="poi-developers" type="add">51444 - Prevent corrupted output when saving files created by LibreOffice 3.3 </action>
|
<action dev="poi-developers" type="add">51460 - Improve HSSF performance when loading very long rows, by switching the CellValue array to an iterator</action>
|
||||||
|
<action dev="poi-developers" type="fix">51444 - Prevent corrupted output when saving files created by LibreOffice 3.3 </action>
|
||||||
<action dev="poi-developers" type="add">51422 - Support using RecalcIdRecord to trigger a full formula recalculation on load </action>
|
<action dev="poi-developers" type="add">51422 - Support using RecalcIdRecord to trigger a full formula recalculation on load </action>
|
||||||
<action dev="poi-developers" type="add">50474 - Example demonstrating how to update Excel workbook embedded in a WordprocessingML document </action>
|
<action dev="poi-developers" type="add">50474 - Example demonstrating how to update Excel workbook embedded in a WordprocessingML document </action>
|
||||||
<action dev="poi-developers" type="fix">51431 - Avoid IndexOutOfBoundException when removing freeze panes in XSSF </action>
|
<action dev="poi-developers" type="fix">51431 - Avoid IndexOutOfBoundException when removing freeze panes in XSSF </action>
|
||||||
|
|
|
@ -360,13 +360,13 @@ public final class InternalSheet {
|
||||||
|
|
||||||
private static final class RecordCloner implements RecordVisitor {
|
private static final class RecordCloner implements RecordVisitor {
|
||||||
|
|
||||||
private final List<RecordBase> _destList;
|
private final List<Record> _destList;
|
||||||
|
|
||||||
public RecordCloner(List<RecordBase> destList) {
|
public RecordCloner(List<Record> destList) {
|
||||||
_destList = destList;
|
_destList = destList;
|
||||||
}
|
}
|
||||||
public void visitRecord(Record r) {
|
public void visitRecord(Record r) {
|
||||||
_destList.add((RecordBase)r.clone());
|
_destList.add((Record)r.clone());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -378,7 +378,7 @@ public final class InternalSheet {
|
||||||
* belongs to a sheet.
|
* belongs to a sheet.
|
||||||
*/
|
*/
|
||||||
public InternalSheet cloneSheet() {
|
public InternalSheet cloneSheet() {
|
||||||
List<RecordBase> clonedRecords = new ArrayList<RecordBase>(_records.size());
|
List<Record> clonedRecords = new ArrayList<Record>(_records.size());
|
||||||
for (int i = 0; i < _records.size(); i++) {
|
for (int i = 0; i < _records.size(); i++) {
|
||||||
RecordBase rb = _records.get(i);
|
RecordBase rb = _records.get(i);
|
||||||
if (rb instanceof RecordAggregate) {
|
if (rb instanceof RecordAggregate) {
|
||||||
|
@ -725,8 +725,8 @@ public final class InternalSheet {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* get the NEXT value record (from LOC). The first record that is a value record
|
* Get all the value records (from LOC). Records will be returned from the first
|
||||||
* (starting at LOC) will be returned.
|
* record (starting at LOC) which is a value record.
|
||||||
*
|
*
|
||||||
* <P>
|
* <P>
|
||||||
* This method is "loc" sensitive. Meaning you need to set LOC to where you
|
* This method is "loc" sensitive. Meaning you need to set LOC to where you
|
||||||
|
@ -735,8 +735,27 @@ public final class InternalSheet {
|
||||||
* at what this sets it to. For this method, set loc to dimsloc to start with,
|
* at what this sets it to. For this method, set loc to dimsloc to start with,
|
||||||
* subsequent calls will return values in (physical) sequence or NULL when you get to the end.
|
* subsequent calls will return values in (physical) sequence or NULL when you get to the end.
|
||||||
*
|
*
|
||||||
* @return CellValueRecordInterface representing the next value record or NULL if there are no more
|
* @return Iterator of CellValueRecordInterface representing the value records
|
||||||
*/
|
*/
|
||||||
|
public Iterator<CellValueRecordInterface> getCellValueIterator(){
|
||||||
|
return _rowsAggregate.getCellValueIterator();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all the value records (from LOC). Records will be returned from the first
|
||||||
|
* record (starting at LOC) which is a value record.
|
||||||
|
*
|
||||||
|
* <P>
|
||||||
|
* This method is "loc" sensitive. Meaning you need to set LOC to where you
|
||||||
|
* want it to start searching. If you don't know do this: setLoc(getDimsLoc).
|
||||||
|
* When adding several rows you can just start at the last one by leaving loc
|
||||||
|
* at what this sets it to. For this method, set loc to dimsloc to start with,
|
||||||
|
* subsequent calls will return values in (physical) sequence or NULL when you get to the end.
|
||||||
|
*
|
||||||
|
* @return Array of CellValueRecordInterface representing the remaining value records
|
||||||
|
* @deprecated use {@link #getValueIterator()} instead
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
public CellValueRecordInterface[] getValueRecords() {
|
public CellValueRecordInterface[] getValueRecords() {
|
||||||
return _rowsAggregate.getValueRecords();
|
return _rowsAggregate.getValueRecords();
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,7 +27,7 @@ import org.apache.poi.hssf.record.Record;
|
||||||
*/
|
*/
|
||||||
public final class RecordStream {
|
public final class RecordStream {
|
||||||
|
|
||||||
private final List _list;
|
private final List<Record> _list;
|
||||||
private int _nextIndex;
|
private int _nextIndex;
|
||||||
private int _countRead;
|
private int _countRead;
|
||||||
private final int _endIx;
|
private final int _endIx;
|
||||||
|
@ -35,14 +35,14 @@ public final class RecordStream {
|
||||||
/**
|
/**
|
||||||
* Creates a RecordStream bounded by startIndex and endIndex
|
* Creates a RecordStream bounded by startIndex and endIndex
|
||||||
*/
|
*/
|
||||||
public RecordStream(List inputList, int startIndex, int endIx) {
|
public RecordStream(List<Record> inputList, int startIndex, int endIx) {
|
||||||
_list = inputList;
|
_list = inputList;
|
||||||
_nextIndex = startIndex;
|
_nextIndex = startIndex;
|
||||||
_endIx = endIx;
|
_endIx = endIx;
|
||||||
_countRead = 0;
|
_countRead = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public RecordStream(List records, int startIx) {
|
public RecordStream(List<Record> records, int startIx) {
|
||||||
this(records, startIx, records.size());
|
this(records, startIx, records.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -61,7 +61,7 @@ public final class RecordStream {
|
||||||
/**
|
/**
|
||||||
* @return the {@link Class} of the next Record. <code>null</code> if this stream is exhausted.
|
* @return the {@link Class} of the next Record. <code>null</code> if this stream is exhausted.
|
||||||
*/
|
*/
|
||||||
public Class peekNextClass() {
|
public Class<? extends Record> peekNextClass() {
|
||||||
if(!hasNext()) {
|
if(!hasNext()) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
|
@ -448,6 +448,16 @@ public final class RowRecordsAggregate extends RecordAggregate {
|
||||||
return startHidden;
|
return startHidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an iterator for the cell values
|
||||||
|
*/
|
||||||
|
public Iterator<CellValueRecordInterface> getCellValueIterator() {
|
||||||
|
return _valuesAgg.iterator();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated use {@link #getCellValueIterator()} instead
|
||||||
|
*/
|
||||||
public CellValueRecordInterface[] getValueRecords() {
|
public CellValueRecordInterface[] getValueRecords() {
|
||||||
return _valuesAgg.getValueRecords();
|
return _valuesAgg.getValueRecords();
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
package org.apache.poi.hssf.record.aggregates;
|
package org.apache.poi.hssf.record.aggregates;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.apache.poi.hssf.model.RecordStream;
|
import org.apache.poi.hssf.model.RecordStream;
|
||||||
|
@ -40,7 +41,7 @@ import org.apache.poi.ss.formula.ptg.Ptg;
|
||||||
* @author Glen Stampoultzis (glens at apache.org)
|
* @author Glen Stampoultzis (glens at apache.org)
|
||||||
* @author Jason Height (jheight at chariot dot net dot au)
|
* @author Jason Height (jheight at chariot dot net dot au)
|
||||||
*/
|
*/
|
||||||
public final class ValueRecordsAggregate {
|
public final class ValueRecordsAggregate implements Iterable<CellValueRecordInterface> {
|
||||||
private static final int MAX_ROW_INDEX = 0XFFFF;
|
private static final int MAX_ROW_INDEX = 0XFFFF;
|
||||||
private static final int INDEX_NOT_SET = -1;
|
private static final int INDEX_NOT_SET = -1;
|
||||||
private int firstcell = INDEX_NOT_SET;
|
private int firstcell = INDEX_NOT_SET;
|
||||||
|
@ -301,10 +302,67 @@ public final class ValueRecordsAggregate {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* iterator for CellValueRecordInterface
|
||||||
|
*/
|
||||||
|
class ValueIterator implements Iterator<CellValueRecordInterface> {
|
||||||
|
|
||||||
|
int curRowIndex = 0, curColIndex = -1;
|
||||||
|
int nextRowIndex = 0, nextColIndex = -1;
|
||||||
|
|
||||||
|
public ValueIterator() {
|
||||||
|
getNextPos();
|
||||||
|
}
|
||||||
|
|
||||||
|
void getNextPos() {
|
||||||
|
if (nextRowIndex >= records.length)
|
||||||
|
return; // no next already
|
||||||
|
|
||||||
|
while (nextRowIndex < records.length) {
|
||||||
|
++nextColIndex;
|
||||||
|
if (records[nextRowIndex] == null || nextColIndex >= records[nextRowIndex].length) {
|
||||||
|
++nextRowIndex;
|
||||||
|
nextColIndex = -1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (records[nextRowIndex][nextColIndex] != null)
|
||||||
|
return; // next cell found
|
||||||
|
}
|
||||||
|
// no next found
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean hasNext() {
|
||||||
|
return nextRowIndex < records.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
public CellValueRecordInterface next() {
|
||||||
|
if (!hasNext())
|
||||||
|
throw new IndexOutOfBoundsException("iterator has no next");
|
||||||
|
|
||||||
|
curRowIndex = nextRowIndex;
|
||||||
|
curColIndex = nextColIndex;
|
||||||
|
final CellValueRecordInterface ret = records[curRowIndex][curColIndex];
|
||||||
|
getNextPos();
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void remove() {
|
||||||
|
records[curRowIndex][curColIndex] = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** value iterator */
|
||||||
|
public Iterator<CellValueRecordInterface> iterator() {
|
||||||
|
return new ValueIterator();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets all the cell records contained in this aggregate.
|
* Gets all the cell records contained in this aggregate.
|
||||||
* Note {@link BlankRecord}s appear separate (not in {@link MulBlankRecord}s).
|
* Note {@link BlankRecord}s appear separate (not in {@link MulBlankRecord}s).
|
||||||
|
* @deprecated use {@link #iterator()} instead
|
||||||
*/
|
*/
|
||||||
|
@Deprecated
|
||||||
public CellValueRecordInterface[] getValueRecords() {
|
public CellValueRecordInterface[] getValueRecords() {
|
||||||
List<CellValueRecordInterface> temp = new ArrayList<CellValueRecordInterface>();
|
List<CellValueRecordInterface> temp = new ArrayList<CellValueRecordInterface>();
|
||||||
|
|
||||||
|
|
|
@ -142,7 +142,7 @@ public final class HSSFSheet implements org.apache.poi.ss.usermodel.Sheet {
|
||||||
row = sheet.getNextRow();
|
row = sheet.getNextRow();
|
||||||
}
|
}
|
||||||
|
|
||||||
CellValueRecordInterface[] cvals = sheet.getValueRecords();
|
Iterator<CellValueRecordInterface> iter = sheet.getCellValueIterator();
|
||||||
long timestart = System.currentTimeMillis();
|
long timestart = System.currentTimeMillis();
|
||||||
|
|
||||||
if (log.check( POILogger.DEBUG ))
|
if (log.check( POILogger.DEBUG ))
|
||||||
|
@ -151,8 +151,8 @@ public final class HSSFSheet implements org.apache.poi.ss.usermodel.Sheet {
|
||||||
HSSFRow lastrow = null;
|
HSSFRow lastrow = null;
|
||||||
|
|
||||||
// Add every cell to its row
|
// Add every cell to its row
|
||||||
for (int i = 0; i < cvals.length; i++) {
|
while (iter.hasNext()) {
|
||||||
CellValueRecordInterface cval = cvals[i];
|
CellValueRecordInterface cval = iter.next();
|
||||||
|
|
||||||
long cellstart = System.currentTimeMillis();
|
long cellstart = System.currentTimeMillis();
|
||||||
HSSFRow hrow = lastrow;
|
HSSFRow hrow = lastrow;
|
||||||
|
|
Loading…
Reference in New Issue