ez-php / bignum
Arbitrary-precision integer and decimal arithmetic for the ez-php ecosystem — immutable BigInteger and BigDecimal value objects backed by bcmath (gmp optional)
Requires
- php: ^8.5
- ext-gmp: *
Requires (Dev)
- ez-php/docker: ^1.0
- friendsofphp/php-cs-fixer: ^3.94
- phpstan/phpstan: ^2.1
- phpstan/phpstan-deprecation-rules: ^2.0
- phpstan/phpstan-strict-rules: ^2.0
- phpunit/phpunit: ^13.0
Suggests
- ext-bcmath: Alternative integer arithmetic backend (BcMathBackend)
README
Arbitrary-precision integer and decimal arithmetic for PHP 8.5+.
Provides immutable BigInteger and BigDecimal value objects backed by PHP's built-in bcmath extension — no external Composer dependencies, no PECL extensions required. Optionally uses gmp as a faster backend for integer operations.
Installation
composer require ez-php/bignum
Requires ext-bcmath (ships with PHP). Optionally install ext-gmp for faster BigInteger arithmetic.
Quick Start
use EzPhp\BigNum\BigDecimal; use EzPhp\BigNum\BigInteger; use EzPhp\BigNum\RoundingMode; // BigInteger — arbitrary-precision integers $a = BigInteger::of('99999999999999999999999999999999'); $b = BigInteger::of('1'); echo $a->add($b); // 100000000000000000000000000000000 // BigDecimal — exact decimal arithmetic (no floating-point errors) $price = BigDecimal::of('0.1'); $tax = BigDecimal::of('0.2'); echo $price->add($tax); // 0.3 (not 0.30000000000000004) // Division with explicit scale and rounding $result = BigDecimal::of('10')->dividedBy('3', 4, RoundingMode::HALF_UP); echo $result; // 3.3333 // Banker's rounding (HALF_EVEN) echo BigDecimal::of('2.45')->round(1, RoundingMode::HALF_EVEN); // 2.4 echo BigDecimal::of('2.55')->round(1, RoundingMode::HALF_EVEN); // 2.6
API
BigInteger
BigInteger::of(int|string $value): self BigInteger::zero(): self BigInteger::one(): self // Arithmetic (all return new BigInteger) ->add(BigInteger|int|string $other): self ->subtract(BigInteger|int|string $other): self ->multiply(BigInteger|int|string $other): self ->divide(BigInteger|int|string $divisor): self // integer division, truncates towards zero ->mod(BigInteger|int|string $divisor): self ->pow(int $exponent): self // exponent >= 0 ->abs(): self ->negate(): self ->gcd(BigInteger|int|string $other): self ->sqrt(): self // floor(sqrt(n)) // Comparison ->compareTo(BigInteger|int|string $other): int // -1, 0, 1 ->isEqualTo(...) ->isLessThan(...) ->isGreaterThan(...) ->isLessThanOrEqualTo(...) ->isGreaterThanOrEqualTo(...) ->isZero(): bool ->isPositive(): bool ->isNegative(): bool // Conversion ->toInt(): int // throws OverflowException if too large ->toFloat(): float ->toString(): string ->toBigDecimal(): BigDecimal
BigDecimal
BigDecimal::of(int|string|float $value): self BigDecimal::ofUnscaledValue(string $unscaledValue, int $scale): self BigDecimal::zero(): self BigDecimal::one(): self // Accessors ->getScale(): int ->getUnscaledValue(): string // Arithmetic (all return new BigDecimal) ->add(BigDecimal|BigInteger|int|string $other): self ->subtract(BigDecimal|BigInteger|int|string $other): self ->multiply(BigDecimal|BigInteger|int|string $other): self ->divide(BigDecimal|BigInteger|int|string $divisor): self // integer division (scale 0) ->dividedBy($divisor, int $scale, RoundingMode $mode): self // explicit scale + rounding ->mod(BigDecimal|BigInteger|int|string $divisor): self ->pow(int $exponent): self ->abs(): self ->negate(): self // Scale and rounding ->round(int $scale, RoundingMode $mode = HALF_UP): self ->toScale(int $scale, RoundingMode $mode = HALF_UP): self // Comparison (scale-independent) ->compareTo(...) ->isEqualTo(...) ->isLessThan(...) ->isGreaterThan(...) ->isLessThanOrEqualTo(...) ->isGreaterThanOrEqualTo(...) ->isZero(): bool ->isPositive(): bool ->isNegative(): bool // Conversion ->toInt(): int // truncates fractional part; throws OverflowException if too large ->toFloat(): float ->toString(): string ->toBigInteger(): BigInteger ->toScientific(): string // e.g. "1.23456E+2"
RoundingMode
| Mode | Behaviour |
|---|---|
UP |
Round away from zero |
DOWN |
Round towards zero (truncate) |
CEILING |
Round towards positive infinity |
FLOOR |
Round towards negative infinity |
HALF_UP |
Round half away from zero (standard) |
HALF_DOWN |
Round half towards zero |
HALF_EVEN |
Round half to nearest even digit (banker's rounding) |
DivisionByZeroException
Thrown by divide, dividedBy, and mod when the divisor is zero. Extends \ArithmeticError.
Backend Selection
BigInteger auto-selects the best available backend:
- GmpBackend — used when
ext-gmpis loaded (faster for large integers) - BcMathBackend — always available fallback
BigDecimal always uses bcmath directly (GMP does not support decimal arithmetic).
To force a specific backend:
use EzPhp\BigNum\Backend\BcMathBackend; BigInteger::setDefaultBackend(new BcMathBackend());
Design
- Zero framework coupling — usable standalone without bootstrapping the application
- Immutable — all operations return new instances; the original is never modified
- No
mixed— all parameters and return values are strictly typed - No external dependencies — only
ext-bcmath(required) andext-gmp(optional)
License
MIT