diff options
author | Luboš Luňák <l.lunak@collabora.com> | 2021-11-25 00:07:03 +0100 |
---|---|---|
committer | Luboš Luňák <l.lunak@collabora.com> | 2021-11-25 14:27:36 +0100 |
commit | 7b0aabe71d2455f6f643553a07f1056935cf190f (patch) | |
tree | 75c44a062d93e32086db23fd8cc6b779bd319a9d /sc/source | |
parent | c41b0bf4c32e1934021d2d607f3f2fe7bc755cc8 (diff) |
sort, cache and binary search query items if they're many (tdf#136838)
This makes autofilter even with tdf#136838 almost instanteous.
Change-Id: I94b4b6d6ab6f8e73312d88c8b88c0f393707f117
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/125795
Tested-by: Jenkins
Reviewed-by: Luboš Luňák <l.lunak@collabora.com>
Diffstat (limited to 'sc/source')
-rw-r--r-- | sc/source/core/data/table3.cxx | 68 |
1 files changed, 54 insertions, 14 deletions
diff --git a/sc/source/core/data/table3.cxx b/sc/source/core/data/table3.cxx index 270c80b9ed73..4da4f23c59b4 100644 --- a/sc/source/core/data/table3.cxx +++ b/sc/source/core/data/table3.cxx @@ -2971,7 +2971,7 @@ public: std::pair<bool,bool> validQueryProcessEntry(SCROW nRow, SCCOL nCol, SCTAB nTab, const ScQueryParam& rParam, ScRefCellValue& aCell, bool* pbTestEqualCondition, const ScInterpreterContext* pContext, QueryEvaluator& aEval, - const ScQueryEntry& rEntry ) + ScTable::ValidQueryCache* pValidQueryCache, const ScQueryEntry& rEntry ) { std::pair<bool,bool> aRes(false, false); const ScQueryEntry::QueryItemsType& rItems = rEntry.GetQueryItems(); @@ -3007,14 +3007,35 @@ std::pair<bool,bool> validQueryProcessEntry(SCROW nRow, SCCOL nCol, SCTAB nTab, valid = false; if(valid) { - for (const auto& rItem : rItems) + if(rItems.size() >= 100 && pValidQueryCache) { - // For speed don't bother comparing approximately here, usually there either - // will be an exact match or it wouldn't match anyway. - if (rItem.meType == ScQueryEntry::ByValue - && value == rItem.mfVal) + // Sort, cache and binary search for the value in items. + // Don't bother comparing approximately. + auto& values = pValidQueryCache->mCachedSortedItemValues; + if(!pValidQueryCache->mCachedSortedItemValuesReady) { + values.reserve(rItems.size()); + for (const auto& rItem : rItems) + if (rItem.meType == ScQueryEntry::ByValue) + values.push_back(rItem.mfVal); + std::sort(values.begin(), values.end()); + pValidQueryCache->mCachedSortedItemValuesReady = true; + } + auto it = std::lower_bound(values.begin(), values.end(), value); + if( it != values.end() && *it == value ) return std::make_pair(true, true); + } + else + { + for (const auto& rItem : rItems) + { + // For speed don't bother comparing approximately here, usually there either + // will be an exact match or it wouldn't match anyway. + if (rItem.meType == ScQueryEntry::ByValue + && value == rItem.mfVal) + { + return std::make_pair(true, true); + } } } } @@ -3042,17 +3063,34 @@ std::pair<bool,bool> validQueryProcessEntry(SCROW nRow, SCCOL nCol, SCTAB nTab, // generous as isQueryByString() but it should be enough and better be safe. if(cellSharedString != nullptr) { - if (rParam.bCaseSens) + if(rItems.size() >= 100 && pValidQueryCache) { - for (const auto& rItem : rItems) + // Sort, cache and binary search for the string in items. + // Since each SharedString is identified by pointer value, + // sorting by pointer value is enough. + auto& values = pValidQueryCache->mCachedSortedItemStrings; + if(!pValidQueryCache->mCachedSortedItemStringsReady) { - if ((rItem.meType == ScQueryEntry::ByString - || (compareByValue && rItem.meType == ScQueryEntry::ByValue)) - && cellSharedString->getData() == rItem.maString.getData()) + values.reserve(rItems.size()); + for (const auto& rItem : rItems) { - return std::make_pair(true, true); + if (rItem.meType == ScQueryEntry::ByString + || (compareByValue && rItem.meType == ScQueryEntry::ByValue)) + { + values.push_back(rParam.bCaseSens + ? rItem.maString.getData() + : rItem.maString.getDataIgnoreCase()); + } } + std::sort(values.begin(), values.end()); + pValidQueryCache->mCachedSortedItemStringsReady = true; } + const rtl_uString* string = rParam.bCaseSens + ? cellSharedString->getData() + : cellSharedString->getDataIgnoreCase(); + auto it = std::lower_bound(values.begin(), values.end(), string); + if( it != values.end() && *it == string ) + return std::make_pair(true, true); } else { @@ -3060,7 +3098,9 @@ std::pair<bool,bool> validQueryProcessEntry(SCROW nRow, SCCOL nCol, SCTAB nTab, { if ((rItem.meType == ScQueryEntry::ByString || (compareByValue && rItem.meType == ScQueryEntry::ByValue)) - && cellSharedString->getDataIgnoreCase() == rItem.maString.getDataIgnoreCase()) + && ( rParam.bCaseSens + ? cellSharedString->getData() == rItem.maString.getData() + : cellSharedString->getDataIgnoreCase() == rItem.maString.getDataIgnoreCase())) { return std::make_pair(true, true); } @@ -3176,7 +3216,7 @@ bool ScTable::ValidQuery( aCell = GetCellValue(nCol, nRow); std::pair<bool,bool> aRes = validQueryProcessEntry(nRow, nCol, nTab, rParam, aCell, - pbTestEqualCondition, pContext, aEval, rEntry); + pbTestEqualCondition, pContext, aEval, pValidQueryCache, rEntry); if (nPos == -1) { |