13
13
*/
14
14
package de .tilman_neumann .jml .powers ;
15
15
16
- import java .io .BufferedReader ;
17
- import java .io .InputStreamReader ;
18
16
import java .math .BigInteger ;
19
- import java .security .SecureRandom ;
20
- import java .util .ArrayList ;
21
17
22
18
import org .apache .logging .log4j .Logger ;
23
19
import org .apache .logging .log4j .LogManager ;
27
23
import de .tilman_neumann .jml .primes .exact .AutoExpandingPrimesArray ;
28
24
import de .tilman_neumann .jml .roots .Roots ;
29
25
import de .tilman_neumann .jml .roots .SqrtExact ;
30
- import de .tilman_neumann .util .Ensure ;
31
- import de .tilman_neumann .util .ConfigUtil ;
32
26
33
27
import static de .tilman_neumann .jml .base .BigIntConstants .*;
34
28
43
37
public class PurePowerTest {
44
38
private static final Logger LOG = LogManager .getLogger (PurePowerTest .class );
45
39
40
+ private static final boolean DEBUG = false ;
41
+
46
42
private static final double LN_2 = Math .log (2 );
47
43
private static final double LN_3 = Math .log (3 );
48
44
@@ -97,7 +93,7 @@ public Result test_v01(BigInteger N) {
97
93
// N is even -> we can test bit patterns before the full i.th root
98
94
BigInteger N_reduced = N .shiftRight (lsb );
99
95
for (int b = 3 ; b <log3N ; b =primesArray .getPrime (++bIndex )) {
100
- // LOG.debug("test b = " + b);
96
+ if ( DEBUG ) LOG .debug ("test b = " + b );
101
97
// if the number of trailing zeros in N is not equal to 0 (mod b), then N is not a b.th power
102
98
if (lsb % b != 0 ) continue ;
103
99
// full b.th root required
@@ -110,7 +106,7 @@ public Result test_v01(BigInteger N) {
110
106
} else {
111
107
// N is odd
112
108
for (int b = 3 ; b <log3N ; b =primesArray .getPrime (++bIndex )) {
113
- // LOG.debug("test b = " + b);
109
+ if ( DEBUG ) LOG .debug ("test b = " + b );
114
110
BigInteger floor_bthRoot = Roots .ithRoot (N , b )[0 ];
115
111
if (floor_bthRoot .pow (b ).equals (N )) {
116
112
// found exact power!
@@ -191,11 +187,11 @@ public Result test_v01(BigInteger N) {
191
187
// Sieve out some powers: If (2b+1) is prime, then N = x^b is only possible if N^2 == 1 (mod (2b+1)) or if (2b+1) | N
192
188
final int b2p = (b <<1 ) + 1 ;
193
189
while (b2 < b2p ) b2 = primesArray .getPrime (b2Index ++);
194
- // LOG.debug("test b = " + b + ", b2 = " + b2);
190
+ if ( DEBUG ) LOG .debug ("test b = " + b + ", b2 = " + b2 );
195
191
if (b2 == b2p ) {
196
192
// 2*b+1 is prime
197
193
final int mod = N_square .mod (b2p );
198
- // LOG.debug("N = " + N + ", N^2 % b2 = " + mod);
194
+ if ( DEBUG ) LOG .debug ("N = " + N + ", N^2 % b2 = " + mod );
199
195
if (mod > 1 ) continue ;
200
196
}
201
197
// Full root required.
@@ -220,17 +216,17 @@ public Result test_v01(BigInteger N) {
220
216
int b2 = 3 , b2Index = 1 ; // skip 2
221
217
int bIndex = 1 ; // skip 2
222
218
for (int b = 3 ; b <=lsb ; b =primesArray .getPrime (++bIndex )) {
223
- // LOG.debug("test b = " + b);
219
+ if ( DEBUG ) LOG .debug ("test b = " + b );
224
220
// N can only be a b.th power if b | lsb
225
221
if (lsb % b != 0 ) continue ;
226
222
// Sieve out some powers: If (2b+1) is prime, then N = x^b is only possible if N^2 == 1 (mod (2b+1)) or if (2b+1) | N
227
223
final int b2p = (b <<1 ) + 1 ;
228
224
while (b2 < b2p ) b2 = primesArray .getPrime (b2Index ++);
229
- // LOG.debug("test b = " + b + ", b2 = " + b2);
225
+ if ( DEBUG ) LOG .debug ("test b = " + b + ", b2 = " + b2 );
230
226
if (b2 == b2p ) {
231
227
// 2*b+1 is prime
232
228
final int mod = N_reduced_square .mod (b2p );
233
- // LOG.debug("N = " + N + ", N^2 % b2 = " + mod);
229
+ if ( DEBUG ) LOG .debug ("N = " + N + ", N^2 % b2 = " + mod );
234
230
if (mod > 1 ) continue ;
235
231
}
236
232
// full b.th root required
@@ -242,93 +238,4 @@ public Result test_v01(BigInteger N) {
242
238
}
243
239
return null ; // no pure power
244
240
}
245
-
246
- private static void testCorrectness (int nCount ) {
247
- PurePowerTest powTest = new PurePowerTest ();
248
-
249
- // create test set for performance test
250
- SecureRandom rng = new SecureRandom ();
251
- for (int bits =10 ; bits <=50 ; bits +=5 ) {
252
- LOG .info ("Test correctness with " + nCount + " " + bits + "-bit numbers" );
253
- ArrayList <BigInteger > testSet = new ArrayList <BigInteger >();
254
- for (int i =0 ; i <nCount ; i ++) {
255
- testSet .add (new BigInteger (bits , rng ));
256
- }
257
-
258
- // test correctness:
259
- // pure powers are not unique, e.g. 3^9 == 27^3, thus we can only check if the final result is correct
260
- for (BigInteger testNum : testSet ) {
261
- Result r1 = powTest .test_v01 (testNum );
262
- if (r1 !=null ) Ensure .ensureEquals (testNum , r1 .base .pow (r1 .exponent ));
263
-
264
- Result r2 = powTest .test /*_v02*/ (testNum );
265
- Ensure .ensureEquals (r1 ==null , r2 ==null );
266
- if (r2 !=null ) Ensure .ensureEquals (testNum , r2 .base .pow (r2 .exponent ));
267
- }
268
- }
269
- LOG .info ("" );
270
- }
271
-
272
- private static void testPerformance (int nCount ) {
273
- PurePowerTest powTest = new PurePowerTest ();
274
-
275
- // create test set for performance test
276
- SecureRandom rng = new SecureRandom ();
277
- for (int bits =50 ; ; bits +=50 ) {
278
- ArrayList <BigInteger > testSet = new ArrayList <BigInteger >();
279
- for (int i =0 ; i <nCount ; i ++) {
280
- testSet .add (new BigInteger (bits , rng ));
281
- }
282
-
283
- // test performance
284
- long t0 , t1 ;
285
- t0 = System .currentTimeMillis ();
286
- for (BigInteger testNum : testSet ) {
287
- powTest .test_v01 (testNum );
288
- }
289
- t1 = System .currentTimeMillis ();
290
- LOG .info ("v01: Testing " + nCount + " " + bits + "-bit numbers took " + (t1 -t0 ) + " ms" );
291
-
292
- t0 = System .currentTimeMillis ();
293
- for (BigInteger testNum : testSet ) {
294
- powTest .test /*_v02*/ (testNum );
295
- }
296
- t1 = System .currentTimeMillis ();
297
- LOG .info ("v02: Testing " + nCount + " " + bits + "-bit numbers took " + (t1 -t0 ) + " ms" );
298
- LOG .info ("" );
299
- }
300
- }
301
-
302
- private static void testInputs () {
303
- PurePowerTest powTest = new PurePowerTest ();
304
- while (true ) {
305
- try {
306
- LOG .info ("Insert test argument N:" );
307
- BufferedReader in = new BufferedReader (new InputStreamReader (System .in ));
308
- String line = in .readLine ();
309
- String input = line !=null ? line .trim () : "" ;
310
- //LOG.debug("input = >" + input + "<");
311
- BigInteger N = new BigInteger (input );
312
- Result purePower = powTest .test (N );
313
- if (purePower == null ) {
314
- LOG .info ("N = " + N + " is not a pure power." );
315
- } else {
316
- LOG .info ("N = " + N + " = " + purePower .base + "^" + purePower .exponent + " is a pure power!" );
317
- }
318
- } catch (Exception ex ) {
319
- LOG .error ("Error " + ex , ex );
320
- }
321
- }
322
- }
323
-
324
- /**
325
- * Test.
326
- * @param args ignored
327
- */
328
- public static void main (String [] args ) {
329
- ConfigUtil .initProject ();
330
- testCorrectness (100000 );
331
- testPerformance (1000000 );
332
- testInputs ();
333
- }
334
241
}
0 commit comments