bvector: much faster unranking
from O(t*n) to O(t*log(n)) for given McE params.
This commit is contained in:
parent
805df12176
commit
889970349e
|
@ -202,9 +202,10 @@ void bvector::colex_unrank (bvector&res, uint n, uint k) const
|
||||||
res.clear();
|
res.clear();
|
||||||
res.resize (n, 0);
|
res.resize (n, 0);
|
||||||
|
|
||||||
uint p;
|
|
||||||
for (uint i = k; i > 0; --i) {
|
for (uint i = k; i > 0; --i) {
|
||||||
p = i;
|
/* Original code:
|
||||||
|
|
||||||
|
uint p = i;
|
||||||
for (;;) {
|
for (;;) {
|
||||||
combination_number (p, i, t);
|
combination_number (p, i, t);
|
||||||
|
|
||||||
|
@ -212,10 +213,34 @@ void bvector::colex_unrank (bvector&res, uint n, uint k) const
|
||||||
++p;
|
++p;
|
||||||
}
|
}
|
||||||
|
|
||||||
combination_number (p - 1, i, t);
|
* ...that kindof lacks speed. We're actually trying to find
|
||||||
|
* the smallest value of p for which comb(p,i)>r.
|
||||||
|
*
|
||||||
|
* Computing all combination numbers is KIND OF slow and cache
|
||||||
|
* (as suggested by Barreto and others) doesn't really make big
|
||||||
|
* difference here (we're usually doing only one or two runs of
|
||||||
|
* this stuff in one program run). Storing about 50megs of
|
||||||
|
* precalculated combination numbers is weird as well.
|
||||||
|
*
|
||||||
|
* Therefore, with the knowledge that i <= p <= n, we're
|
||||||
|
* halving the search interval as usual.
|
||||||
|
*/
|
||||||
|
|
||||||
|
uint p, a = i, b = n;
|
||||||
|
while (a < b) {
|
||||||
|
p = (a + b) / 2;
|
||||||
|
|
||||||
|
combination_number (p, i, t);
|
||||||
|
if (mpz_cmp (t, r) > 0) b = p;
|
||||||
|
else a = p + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
combination_number (b - 1, i, t);
|
||||||
mpz_swap (t2, r);
|
mpz_swap (t2, r);
|
||||||
mpz_sub (r, t2, t);
|
mpz_sub (r, t2, t);
|
||||||
if (p > n) continue; //overflow protection
|
|
||||||
|
//overflow protection (result's wrong anyway now)
|
||||||
|
if (p > n) continue;
|
||||||
res[p - 1] = 1;
|
res[p - 1] = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue