Big integers in C#
Asked Answered
D

13

63

Currently I am borrowing java.math.BigInteger from the J# libraries as described here. Having never used a library for working with large integers before, this seems slow, on the order of 10 times slower, even for ulong length numbers. Does anyone have any better (preferably free) libraries, or is this level of performance normal?

Dynamo answered 7/10, 2008 at 0:16 Comment(5)
nice link. A couple of gems there.Rahn
Only problem is that they require the J# redistributables to be installed. The fact that J# is all but dead (it wasn't in VS 2008 atleast) probably doesn't help with promoting that.Dynamo
J# is just to facilitate the migration of existing Java projects to .NET. I definitely wouldn't incorporate any of its libraries into a new project.Miley
possible duplicate of [How can I represent a very large integer in .NET? ](#25875)Toms
@Roger: I'd rather close the other, as this one has more up to information attached to it.Dynamo
C
66

As of .NET 4.0 you can use the System.Numerics.BigInteger class. See documentation here: http://msdn.microsoft.com/en-us/library/system.numerics.biginteger(v=vs.110).aspx

Another alternative is the IntX class.

IntX is an arbitrary precision integers library written in pure C# 2.0 with fast - O(N * log N) - multiplication/division algorithms implementation. It provides all the basic operations on integers like addition, multiplication, comparing, bitwise shifting etc.

Chuipek answered 27/4, 2009 at 19:6 Comment(2)
For the record, The System.Numerics assembly is not referenced by default in new projects, so this needs to be added before BigInteger can be used.Hixon
In Solution Explorer do right click on References > Add Reference > Search for numerics and check the found reference. Now you can add "using System.Numerics;" at the top of your .cs fileGlazer
H
9

F# also ships with one. You can get it at Microsoft.FSharp.Math.

Hemihedral answered 31/1, 2009 at 13:22 Comment(0)
L
8

The System.Numerics.BigInteger class in .NET 4.0 is based on Microsoft.SolverFoundation.Common.BigInteger from Microsoft Research.

The Solver Foundation's BigInteger class looks very performant. I am not sure about which license it is released under, but you can get it here (download and install Solver Foundation and find the Microsoft.Solver.Foundation.dll).

Lyell answered 19/6, 2009 at 17:48 Comment(0)
B
4

I reckon you could optimize the implementation if you perform all the operations on BigInts that are going to return results smaller than a native type (Eg. int64) on the native types and only deal with the big array if you are going to overflow.

edit This implementation on codeproject, seems only 7 times slower ... But with the above optimization you could get it to perform almost identically to native types for small numbers.

Bernt answered 7/10, 2008 at 0:33 Comment(5)
I believe the J# library uses Byte's internally, it has a ToByteArray() function atleast, and no other ToArray() function. This might be an idea if I wanted to roll my own, I'm not too thrilled with that idea either though.Dynamo
What size data are you working with, in general? Will stuff overflow the Int64 size on a regular basis, or is it an exception?Bernt
I'm doing it mostly on a case by case basis, setting it to BigInt when I run into an overflow with a checked{} block. If it does that once, then there's a pretty good chance it'll do it repeatedly and often.Dynamo
monoxide, I don't think it will be easy to get this to perform much faster than the implementation on codeproject for large numbers.Bernt
Like I said in my original post, I've never used a BigInteger implementation before, so I wouldn't know what sort of performance they get. That was more the question I was asking. Damn, I still want to roll my own as an intellectual excercise now though :(Dynamo
B
4

Here are several implementations of BigInteger in C#. I've used Mono's BigInteger implementation, works pretty fast (I've used it in CompactFramework)

Bouncy Castle

Mono

Bathypelagic answered 23/6, 2009 at 9:48 Comment(0)
P
3

I'm not sure about the performance, but IronPython also has a BigInteger class. It is in the Microsoft.Scripting.Math namespace.

Pressurize answered 7/10, 2008 at 0:20 Comment(0)
E
2

Yes, it will be slow, and 10x difference is about what I'd expect. BigInt uses an array to represent an arbitrary length, and all the operations have to be done manually (as opposed to most math which can be done directly with the CPU)

I don't even know if hand-coding it in assembly will give you much of a performance gain over 10x, that's pretty damn close. I'd look for other ways to optimize it--sometimes depending on your math problem there are little tricks you can do to make it quicker.

Escolar answered 7/10, 2008 at 0:23 Comment(0)
R
2

I used Biginteger at a previous job. I don't know what kind of performance needs you have. I did not use it in a performance-intensive situation, but never had any problems with it.

Redwood answered 7/10, 2008 at 0:29 Comment(0)
S
2

This may sound like a strange suggestion, but have you tested the decimal type to see how fast it works?

The decimal range is ±1.0 × 10^−28 to ±7.9 × 10^28, so it may still not be large enough, but it is larger than a ulong.

There was supposed to be a BigInteger class in .NET 3.5, but it got cut.

Solfatara answered 7/10, 2008 at 0:31 Comment(6)
Yes, I saw that anouncement when originally looking for a BigInt library. Makes me sad. Oh well.Dynamo
You would have to be careful - using a decimal might cause rounding issues.Christy
Decimal doesn't cause rounding errors. It's not floating point.Hanley
Decimal isn't float. It's a precision type as far as I'm aware.Dynamo
Decimal only tries to minimize errors due to rounding. It is not immune to rounding. msdn.microsoft.com/en-us/library/system.decimal.aspxFluoridate
Decimal is a floating point type. It's just that it's a floating decimal point instead of a floating binary point. See pobox.com/~skeet/csharp/decimal.html I wouldn't expect rounding issues if all values are integers within the appropriate range though.Wastage
J
1

This won't help you, but there was supposed to be a BigInteger class in .Net 3.5; it got cut, but from statements made at PDC, it will be in .Net 4.0. They apparently have spent a lot of time optimizing it, so the performance should be much better than what you're getting now.

Further, this question is essentially a duplicate of How can I represent a very large integer in .NET?

Jacksonjacksonville answered 8/11, 2008 at 20:41 Comment(1)
BigInteger is in 3.5 but its an internal class. ;) Its there just not ready for prime time.Rectocele
D
1

See the answers in this thread. You will need to use one of the third-party big integer libraries/classes available or wait for C# 4.0 which will include a native BigInteger datatype.

Dairymaid answered 10/11, 2008 at 20:28 Comment(0)
D
1

This Looks very promising. It is a C# Wrapper over GMP.

http://web.rememberingemil.org/Projects/GnuMpDotNet/GnuMpDotNet.html

There are also other BigInteger options for .Net here in particular, Mpir.Net

Diegodiehard answered 25/8, 2015 at 1:39 Comment(1)
Mpir.NET is partly based on Emil's GMP wrapper. The other part is X-MPIR, and fused together you get the convenience of Emil's wrapper and the performance of X-MPIR.Croupier
D
1

You can also use the Math.Gmp.Native Nuget package that I wrote. Its source code is available on GitHub, and documentation is available here. It exposes to .NET all of the functionality of the GMP library which is known as a highly-optimized arbitrary-precision arithmetic library.

Arbitrary-precision integer are represented by the mpz_t type. Operations on these integers all begin with the mpz_ prefix. For examples, mpz_add or mpz_cmp. Source code examples are given for each operation.

Destrier answered 18/5, 2019 at 9:10 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.