Skip to content

Instantly share code, notes, and snippets.

@vivekascoder
Created May 23, 2024 18:48
Show Gist options
  • Save vivekascoder/15fa210e4685a1f32bd97a105a9a380f to your computer and use it in GitHub Desktop.
Save vivekascoder/15fa210e4685a1f32bd97a105a9a380f to your computer and use it in GitHub Desktop.
contract Root {
// calculates a^(1/n) to dp decimal places
// maxIts bounds the number of iterations performed
function nthRoot(uint _a, uint _n, uint _dp, uint _maxIts) pure public returns(uint) {
assert (_n > 1);
// The scale factor is a crude way to turn everything into integer calcs.
// Actually do (a * (10 ^ ((dp + 1) * n))) ^ (1/n)
// We calculate to one extra dp and round at the end
uint one = 10 ** (1 + _dp);
uint a0 = one ** _n * _a;
// Initial guess: 1.0
uint xNew = one;
uint iter = 0;
while (xNew != x && iter < _maxIts) {
uint x = xNew;
uint t0 = x ** (_n - 1);
if (x * t0 > a0) {
xNew = x - (x - a0 / t0) / _n;
} else {
xNew = x + (a0 / t0 - x) / _n;
}
++iter;
}
// Round to nearest in the last dp.
return (xNew + 5) / 10;
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment