A pesquisa binária clássica procura por um elemento em uma lista que "case" com um dado item, aproveitando-se do fato de que a lista está ordenada. O "casamento" pode ser dado por um teste de igualdade, ou por uma comparação por um campo do elemento, ou outras possibilidades. O ponto importante é que o elemento desejado "compara igual" com o item, todos os elementos anteriores "comparam menor" com o item e todos os elementos posteriores "comparam maior" com o item.
Mas esse não é o único tipo de pesquisa binária possível. Ao invés de casarmos os elementos da lista com um item, podemos guiar a pesquisa por um predicado que é verdadeiro para todos os elementos a partir de um dado ponto na lista, e falso para todos os elementos anteriores. Não há sequer o requerimento de ordem na lista; apenas que o predicado particione a lista em dois segmentos contíguos. A pesquisa então retorna o índice do primeiro elemento para o qual o predicado é verdadeiro:
function malandrous_binary_search(cmp, item, list) { // Retorna o índice do primeiro elemento da lista para o qual // cmp(elem, item) é verdadeiro. [Uma definição alternativa seria // passar um predicado de um argumento só e não passar 'item'.] var min=0, max=list.length, mid; while (min != max) { mid = Math.floor((min+max)/2); if (cmp(list[mid], item)) max = mid; // Se list[mid] satisfaz o teste, então a posição // que queremos é mid ou está antes de mid. else min = mid + 1; // Se list[mid] não satisfaz o teste, então o que // queremos só pode estar depois de mid. } return min; } // Exemplo. fives_start = malandrous_binary_search(function(x,y) x>=y, 5, [1,1,2,3,5,5,5,8,13,13,21]); fives_end = malandrous_binary_search(function(x,y) x>y, 5, [1,1,2,3,5,5,5,8,13,13,21]);
Eu denomino este algoritmo pesquisa binária malandra.
A primeira (e única) vez em que me deparei com esta maravilha foi no código do mwetoolkit, no qual eu estava trabalhando ano passado como bolsista. Lá, a pesquisa binária malandra é usada para encontrar um range de frases ordenadas que começam com um mesmo prefixo. Para isso, procura-se o primeiro elemento da lista que seja maior ou igual ao prefixo (onde um elemento é "igual" ao prefixo se inicia com ele), e o primeiro elemento que seja maior (i.e., o primeiro elemento fora do range). [Na verdade, a pesquisa é feita sobre todos os sufixos de um texto, usando uma array de sufixos, outra invenção especial de primeira que descobri lá. Até que eu aprendi com essa bolsa, vou lhes contar.]
Copyright © 2010-2024 Vítor De Araújo
O conteúdo deste blog, a menos que de outra forma especificado, pode ser utilizado segundo os termos da licença Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International.
Powered by Blognir.