From c8f1682e076db24a2c431d1a8438b000a45184b2 Mon Sep 17 00:00:00 2001 From: Zach Reizner Date: Mon, 18 May 2015 16:17:19 -0700 Subject: [PATCH] Use std::bsearch in LookupByKey for binary search --- include/flatbuffers/flatbuffers.h | 42 +++++++++++++++---------------- 1 file changed, 21 insertions(+), 21 deletions(-) diff --git a/include/flatbuffers/flatbuffers.h b/include/flatbuffers/flatbuffers.h index f77b13eed..8f0671bc7 100644 --- a/include/flatbuffers/flatbuffers.h +++ b/include/flatbuffers/flatbuffers.h @@ -305,28 +305,17 @@ public: } template return_type LookupByKey(K key) const { - auto span = size(); - uoffset_t start = 0; - // Perform binary search for key. - while (span) { - // Compare against middle element of current span. - auto middle = span / 2; - auto table = Get(start + middle); - auto comp = table->KeyCompareWithValue(key); - if (comp > 0) { - // Greater than. Adjust span and try again. - span = middle; - } else if (comp < 0) { - // Less than. Adjust span and try again. - middle++; - start += middle; - span -= middle; - } else { - // Found element. - return table; - } + std::size_t count = size(); + void *search_result = std::bsearch(&key, Data(), count, + IndirectHelper::element_stride, KeyCompare); + + if (!search_result) { + return nullptr; // Key not found. } - return nullptr; // Key not found. + + const uint8_t *data = reinterpret_cast(search_result); + + return IndirectHelper::Read(data, 0); } protected: @@ -335,6 +324,17 @@ protected: Vector(); uoffset_t length_; + +private: + template static int KeyCompare(const void *ap, const void *bp) { + const K *key = reinterpret_cast(ap); + const uint8_t *data = reinterpret_cast(bp); + auto table = IndirectHelper::Read(data, 0); + + // std::bsearch compares with the operands transposed, so we negate the + // result here. + return -table->KeyCompareWithValue(*key); + } }; // Convenient helper function to get the length of any vector, regardless