package ramenj;
import java.math.BigInteger;
import java.util.Optional;
import java.util.OptionalLong;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.stream.LongStream;
/**
* RamenJ (ラーメン素数)制限時間内に算出した最大の素数をラーメンと呼称する (制限時間)ラーメンの計算は3分以内(180秒以内)とし超えてはならない
* (All or Nothingの原則)制限時間内に計算が完了しなかった場合、ラーメンは2とする
* (既知の素数)計算なしに使用してよい既知の素数は2と3とする、それ以外の素数は未知とする
*
* @author noramen
*/
public class Program {
public final static int ramen2 = 2;
public final static int ramen3 = 3;
public final static int noramen = ramen2;
public static void main
(String args
[]) { long start
= System.
currentTimeMillis();
// int ramen = Ramen1(); // NG
// long ramen = Ramen2(); // OK 9223372036854775783
// long ramen = Ramen3(); // OK 9223372036854775783
// long ramen = Ramen3_1(); // OK 9223372036854775783
// long ramen = Ramen3_2(); // OK 9223372036854775783
// BigInteger ramen = Ramen4(); // OK 18446744073709551629
// BigInteger ramen = Ramen5(); // OK 18446744073709551629
// BigInteger ramen = Ramen6(2l, 1l, 1l, 1l); // Ramen5()と同値
// BigInteger ramen = Ramen6(5l, 1l, 2l, 1l); // OK 46116860184273879079
// BigInteger ramen = Ramen6(6l, 1l, 5l, 2l); // OK 55340232221128654843
// BigInteger ramen = Ramen6(13l, 2l, 5l, 2l); // OK 59951918239556042783
// BigInteger ramen = Ramen6(68l, 10l, 5l, 2l); // OK 62718929850612475567
// BigInteger ramen = Ramen7(69l, 10l, 5l, 2l); // OK 63641267054297953129
// BigInteger ramen = Ramen7(9l, 1l, 3l, 1l); // OK 83010348331692982271
// BigInteger ramen = Ramen7(11l, 1l, 3l, 1l); // OK 101457092405402533879
// BigInteger ramen = Ramen7(112l, 10l, 3l, 1l); //OK 103301766812773489067
// BigInteger ramen = Ramen7(114l, 10l, 346l, 100l); // OK 105146441220144444251
// BigInteger ramen = Ramen7(116l, 10l, 346l, 100l); // NG
// BigInteger ramen = Ramen7(118l, 10l, 346l, 100l); // NG
// BigInteger ramen = Ramen7(12l, 1l, 346l, 100l); // NG
// BigInteger ramen = Ramen8(114l, 10l, 346l, 100l); // OK 105146441220144444251
// BigInteger ramen = Ramen8(12l, 1l, 346l, 100l); // OK 110680464442257309779
// BigInteger ramen = Ramen8(13l, 1l, 346l, 100l); // OK 119903836479112085531
// BigInteger ramen = Ramen8(14l, 1l, 346l, 100l); // OK 129127208515966861349
// BigInteger ramen = Ramen8(16l, 1l, 4l, 1l); // OK 147573952589676412931
// BigInteger ramen = Ramen8(18l, 1l, 4l, 1l); // OK 166020696663385964539
//BigInteger ramen = Ramen8(19l, 1l, 4l, 1l); // OK 175244068700240740367
BigInteger ramen
= Ramen8
(20l, 1l, 4l, 1l
); // OK? 184467440737095516163
long end
= System.
currentTimeMillis(); System.
out.
println((end
- start
) / 1000); }
/**
* i,j,kを使った逐次法 (NG)5分経っても終らない
*
* @return noramen
*/
static int Ramen1() {
int ramen = 2;
for (int i
= 3; i
<= Integer.
MAX_VALUE; i
+= 2) { boolean k = true;
for (int j
= 3; j
<= Math.
sqrt(i
); j
+= 2) { if (i % j == 0) {
k = false;
break;
}
}
if (k) {
ramen = i;
}
}
return ramen;
}
/**
* Long.MAX_VALUEから逆順でAll or Nothing (OK)約14秒で出る
*
* @return 9223372036854775783
*/
static long Ramen2() {
for (long i
= Long.
MAX_VALUE; i
>= 3; i
-= 2) { boolean flag = true;
for (long j
= 3; j
<= Math.
sqrt(i
); j
+= 2) { if (i % j == 0) {
flag = false;
break;
}
}
if (flag) {
return i;
}
}
return noramen;
}
/**
* Ramen2()のうちflagをFunctionで書き直したもの (OK)約14秒で出る
*
* @return 9223372036854775783
*/
static long Ramen3() {
Function
<Long, Boolean
> isramen
= t
-> { for (long i
= 3; i
<= Math.
sqrt(t
); i
+= 2) { if (t % i == 0) {
return false;
}
}
return true;
};
for (long i
= Long.
MAX_VALUE; i
>= 3; i
-= 2) { if (isramen.apply(i)) {
return i;
}
}
return noramen;
}
/**
* Ramen3()の素数判定(試し割り法)を+=2ではなく++になるよう書き直したもの (OK)約12秒かかる
*
* @return 9223372036854775783
*/
static long Ramen3_1() {
Function
<Long, Boolean
> isramen
= t
-> { long halfSqrt
= (long) (Math.
sqrt(t
) / 2); for (long i = 1; i <= halfSqrt; i++) {
if (t % (i * 2 + 1) == 0) {
return false;
}
}
return true;
};
for (long i
= Long.
MAX_VALUE; i
>= 3; i
-= 2) { if (isramen.apply(i)) {
return i;
}
}
return noramen;
}
/**
* Ramen3_1()をLongStreamで書き直しparallelしたもの (OK)約7~9秒かかる
*
* @return 9223372036854775783
*/
static long Ramen3_2() {
Function
<Long, Boolean
> isramen
= t
-> { long halfSqrt
= (long) (Math.
sqrt(t
) / 2); OptionalLong findAny = LongStream.rangeClosed(1, halfSqrt).parallel().filter(x -> t % (x * 2 + 1) == 0)
.findAny();
return !findAny.isPresent();
};
for (long i
= Long.
MAX_VALUE; i
>= 3; i
-= 2) { if (isramen.apply(i)) {
return i;
}
}
return noramen;
}
/**
* Ramen3()をBigIntegerに書き直したもの。また (1)sqrtを使わずpowを使用
* (2)ループ処理でをstart3,step2→start5,step6に書き換え
* (3)BigInteger.nextProbablePrimeを使用(OK)約128~140秒かかる
*
* @return 18446744073709551629
*/
while (i.pow(2).compareTo(t) <= 0) {
return false;
}
}
return true;
};
}
while (!isramen.apply(ramen)) {
System.
out.
println(ramen
+ " is not ramen"); ramen = ramen.nextProbablePrime();
}
return ramen;
}
/**
* Ramen4()をStreamで書き直しparallelしたもの ループ処理でをstart3,step2に戻し√x/2に近似で求める
* (OK)約80秒かかる
*
* @return 18446744073709551629
*/
while (xnn.
subtract(xn
).
abs().
compareTo(BigInteger.
ONE) > 0) { xn = xnn;
xnn
= x.
divide(xnn
).
add(xnn
).
divide(BigInteger.
valueOf(2l
)); }
return xnn.
divide(BigInteger.
valueOf(2l
)).
longValueExact(); };
LongStream stream = LongStream.rangeClosed(1, halfsqrt.apply(t));
Optional<BigInteger> findAny = stream.parallel()
.
filter(i
-> t.
remainder(i
) == BigInteger.
ZERO).
findAny(); return !findAny.isPresent();
};
ramen = ramen.nextProbablePrime();
while (!isramen.apply(ramen)) {
System.
out.
println(ramen
+ " is not ramen"); ramen = ramen.nextProbablePrime();
}
return ramen;
}
/**
* Ramen5()にパラメーターを付与し変更しやすいよう変形したもの
*
* @param mul1
* 開始数の乗数
* @param div1
* 開始数の除数
* @param mul2
* ニュートン近似の乗数
* @param div2
* ニュートン近似の除数
* @return Ramen6(13l, 2l, 5l, 2l)=59951918239556042783
*/
static BigInteger Ramen6
(long mul1,
long div1,
long mul2,
long div2
) { while (xnn.
subtract(xn
).
abs().
compareTo(BigInteger.
ONE) > 0) { xn = xnn;
xnn
= x.
divide(xnn
).
add(xnn
).
divide(BigInteger.
valueOf(2l
)); }
return xnn.
divide(BigInteger.
valueOf(2l
)).
longValueExact(); };
LongStream stream = LongStream.rangeClosed(1, halfsqrt.apply(t));
Optional<BigInteger> findAny = stream.parallel()
.
filter(i
-> t.
remainder(i
) == BigInteger.
ZERO).
findAny(); return !findAny.isPresent();
};
ramen = ramen.nextProbablePrime();
while (!isramen.apply(ramen)) {
System.
out.
println(ramen
+ " is not ramen"); ramen = ramen.nextProbablePrime();
}
return ramen;
}
/**
* Ramen6の (1)mapToObj内をBigIntegerからLong (2)findAny.isPresentからnoneMatch
* (3)remainder判定を==からcompareTo==
* (4)isramenをFunction<BigInteger,Boolean>からPredicate<BigInteger> 、で書き直したもの
*
* @param mul1
* 開始数の乗数
* @param div1
* 開始数の除数
* @param mul2
* ニュートン近似の乗数
* @param div2
* ニュートン近似の除数
* @return Ramen7(114l, 10l, 346l, 100l)=105146441220144444251
*/
static BigInteger Ramen7
(long mul1,
long div1,
long mul2,
long div2
) { while (xnn.
subtract(xn
).
abs().
compareTo(BigInteger.
ONE) > 0) { xn = xnn;
xnn
= x.
divide(xnn
).
add(xnn
).
divide(BigInteger.
valueOf(2l
)); }
return xnn.
divide(BigInteger.
valueOf(2l
)).
longValueExact(); };
Predicate<BigInteger> isramen = t -> LongStream.rangeClosed(1, halfsqrt.apply(t)).parallel()
.
noneMatch(i
-> t.
remainder(i
).
compareTo(BigInteger.
ZERO) == 0);
ramen = ramen.nextProbablePrime();
while (!isramen.test(ramen)) {
System.
out.
println(ramen
+ " is not ramen"); ramen = ramen.nextProbablePrime();
}
return ramen;
}
/**
* Ramen7のhalfsqrtからsextantsqrtに書き換えたもの
*
* @param mul1
* 開始数の乗数
* @param div1
* 開始数の除数
* @param mul2
* ニュートン近似の乗数
* @param div2
* ニュートン近似の除数
* @return
*/
static BigInteger Ramen8
(long mul1,
long div1,
long mul2,
long div2
) {
do {
xn = xnn;
xnn
= x.
divide(xnn
).
add(xnn
).
divide(BigInteger.
valueOf(2l
)); } while (xnn.
subtract(xn
).
abs().
compareTo(BigInteger.
ONE) > 0);
return xnn.
max(xn
).
divide(BigInteger.
valueOf(6l
)).
longValueExact(); };
return false;
}
return LongStream.
range(0, sextantsqrt.
apply(t
)).
parallel().
mapToObj(i
-> BigInteger.
valueOf(i
* 6l
+ 5l
)) .
noneMatch(i
-> t.
remainder(i
).
compareTo(BigInteger.
ZERO) == 0 };
ramen = ramen.nextProbablePrime();
while (!isramen.apply(ramen)) {
System.
out.
println(ramen
+ " is not ramen"); ramen = ramen.nextProbablePrime();
}
return ramen;
}
}
package ramenj;

import java.math.BigInteger;
import java.util.Optional;
import java.util.OptionalLong;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.stream.LongStream;

/**
 * RamenJ (ラーメン素数)制限時間内に算出した最大の素数をラーメンと呼称する (制限時間)ラーメンの計算は3分以内(180秒以内)とし超えてはならない
 * (All or Nothingの原則)制限時間内に計算が完了しなかった場合、ラーメンは2とする
 * (既知の素数)計算なしに使用してよい既知の素数は2と3とする、それ以外の素数は未知とする
 * 
 * @author noramen
 */
public class Program {
	public final static int ramen2 = 2;
	public final static int ramen3 = 3;
	public final static int noramen = ramen2;

	public static void main(String args[]) {
		long start = System.currentTimeMillis();

		// int ramen = Ramen1(); // NG
		// long ramen = Ramen2(); // OK 9223372036854775783
		// long ramen = Ramen3(); // OK 9223372036854775783
		// long ramen = Ramen3_1(); // OK 9223372036854775783
		// long ramen = Ramen3_2(); // OK 9223372036854775783

		// BigInteger ramen = Ramen4(); // OK 18446744073709551629
		// BigInteger ramen = Ramen5(); // OK 18446744073709551629

		// BigInteger ramen = Ramen6(2l, 1l, 1l, 1l); // Ramen5()と同値
		// BigInteger ramen = Ramen6(5l, 1l, 2l, 1l); // OK 46116860184273879079
		// BigInteger ramen = Ramen6(6l, 1l, 5l, 2l); // OK 55340232221128654843
		// BigInteger ramen = Ramen6(13l, 2l, 5l, 2l); // OK 59951918239556042783
		// BigInteger ramen = Ramen6(68l, 10l, 5l, 2l); // OK 62718929850612475567

		// BigInteger ramen = Ramen7(69l, 10l, 5l, 2l); // OK 63641267054297953129
		// BigInteger ramen = Ramen7(9l, 1l, 3l, 1l); // OK 83010348331692982271
		// BigInteger ramen = Ramen7(11l, 1l, 3l, 1l); // OK 101457092405402533879
		// BigInteger ramen = Ramen7(112l, 10l, 3l, 1l); //OK 103301766812773489067
		// BigInteger ramen = Ramen7(114l, 10l, 346l, 100l); // OK 105146441220144444251
		// BigInteger ramen = Ramen7(116l, 10l, 346l, 100l); // NG
		// BigInteger ramen = Ramen7(118l, 10l, 346l, 100l); // NG
		// BigInteger ramen = Ramen7(12l, 1l, 346l, 100l); // NG

		// BigInteger ramen = Ramen8(114l, 10l, 346l, 100l); // OK 105146441220144444251
		// BigInteger ramen = Ramen8(12l, 1l, 346l, 100l); // OK 110680464442257309779
		// BigInteger ramen = Ramen8(13l, 1l, 346l, 100l); // OK 119903836479112085531
		// BigInteger ramen = Ramen8(14l, 1l, 346l, 100l); // OK 129127208515966861349
		// BigInteger ramen = Ramen8(16l, 1l, 4l, 1l); // OK 147573952589676412931
		// BigInteger ramen = Ramen8(18l, 1l, 4l, 1l); // OK 166020696663385964539
		//BigInteger ramen = Ramen8(19l, 1l, 4l, 1l); // OK 175244068700240740367
		BigInteger ramen = Ramen8(20l, 1l, 4l, 1l); // OK? 184467440737095516163

		System.out.println();
		System.out.println(ramen);

		long end = System.currentTimeMillis();
		System.out.println((end - start) / 1000);
	}

	/**
	 * i,j,kを使った逐次法 (NG)5分経っても終らない
	 * 
	 * @return noramen
	 */
	static int Ramen1() {
		int ramen = 2;

		for (int i = 3; i <= Integer.MAX_VALUE; i += 2) {
			boolean k = true;
			for (int j = 3; j <= Math.sqrt(i); j += 2) {
				if (i % j == 0) {
					k = false;
					break;
				}
			}

			if (k) {
				ramen = i;
				System.out.println(i);
			}
		}

		return ramen;
	}

	/**
	 * Long.MAX_VALUEから逆順でAll or Nothing (OK)約14秒で出る
	 * 
	 * @return 9223372036854775783
	 */
	static long Ramen2() {
		for (long i = Long.MAX_VALUE; i >= 3; i -= 2) {
			boolean flag = true;
			for (long j = 3; j <= Math.sqrt(i); j += 2) {
				if (i % j == 0) {
					flag = false;
					break;
				}
			}

			if (flag) {
				return i;
			}
		}
		return noramen;
	}

	/**
	 * Ramen2()のうちflagをFunctionで書き直したもの (OK)約14秒で出る
	 * 
	 * @return 9223372036854775783
	 */
	static long Ramen3() {
		Function<Long, Boolean> isramen = t -> {
			for (long i = 3; i <= Math.sqrt(t); i += 2) {
				if (t % i == 0) {
					return false;
				}
			}
			return true;
		};

		for (long i = Long.MAX_VALUE; i >= 3; i -= 2) {
			if (isramen.apply(i)) {
				return i;
			}
		}

		return noramen;
	}

	/**
	 * Ramen3()の素数判定(試し割り法)を+=2ではなく++になるよう書き直したもの (OK)約12秒かかる
	 * 
	 * @return 9223372036854775783
	 */
	static long Ramen3_1() {
		Function<Long, Boolean> isramen = t -> {
			long halfSqrt = (long) (Math.sqrt(t) / 2);
			for (long i = 1; i <= halfSqrt; i++) {
				if (t % (i * 2 + 1) == 0) {
					return false;
				}
			}
			return true;
		};

		for (long i = Long.MAX_VALUE; i >= 3; i -= 2) {
			if (isramen.apply(i)) {
				return i;
			}
		}

		return noramen;
	}

	/**
	 * Ramen3_1()をLongStreamで書き直しparallelしたもの (OK)約7～9秒かかる
	 * 
	 * @return 9223372036854775783
	 */
	static long Ramen3_2() {
		Function<Long, Boolean> isramen = t -> {
			long halfSqrt = (long) (Math.sqrt(t) / 2);
			OptionalLong findAny = LongStream.rangeClosed(1, halfSqrt).parallel().filter(x -> t % (x * 2 + 1) == 0)
					.findAny();
			return !findAny.isPresent();
		};

		for (long i = Long.MAX_VALUE; i >= 3; i -= 2) {
			if (isramen.apply(i)) {
				return i;
			}
		}

		return noramen;
	}

	/**
	 * Ramen3()をBigIntegerに書き直したもの。また (1)sqrtを使わずpowを使用
	 * (2)ループ処理でをstart3,step2→start5,step6に書き換え
	 * (3)BigInteger.nextProbablePrimeを使用(OK)約128～140秒かかる
	 * 
	 * @return 18446744073709551629
	 */
	static BigInteger Ramen4() {
		Function<BigInteger, Boolean> isramen = t -> {
			BigInteger i = BigInteger.valueOf(5l);
			while (i.pow(2).compareTo(t) <= 0) {
				if (t.remainder(i) == BigInteger.ZERO
						|| t.remainder(i.add(BigInteger.valueOf(2l))) == BigInteger.ZERO) {
					return false;
				}
				i = i.add(BigInteger.valueOf(6l));
			}
			return true;
		};

		BigInteger ramen = BigInteger.valueOf(Long.MAX_VALUE).multiply(BigInteger.valueOf(2l));
		if (ramen.remainder(BigInteger.valueOf(2l)) == BigInteger.ZERO) {
			ramen = ramen.add(BigInteger.ONE);
		}
		while (!isramen.apply(ramen)) {
			System.out.println(ramen + " is not ramen");
			ramen = ramen.nextProbablePrime();
		}
		return ramen;
	}

	/**
	 * Ramen4()をStreamで書き直しparallelしたもの ループ処理でをstart3,step2に戻し√x/2に近似で求める
	 * (OK)約80秒かかる
	 * 
	 * @return 18446744073709551629
	 */
	static BigInteger Ramen5() {
		Function<BigInteger, Long> halfsqrt = x -> {
			BigInteger xnn = BigInteger.valueOf((long) Math.sqrt(Long.MAX_VALUE));
			BigInteger xn = BigInteger.ZERO;
			while (xnn.subtract(xn).abs().compareTo(BigInteger.ONE) > 0) {
				xn = xnn;
				xnn = x.divide(xnn).add(xnn).divide(BigInteger.valueOf(2l));
			}
			return xnn.divide(BigInteger.valueOf(2l)).longValueExact();
		};

		Function<BigInteger, Boolean> isramen = t -> {
			LongStream stream = LongStream.rangeClosed(1, halfsqrt.apply(t));
			Optional<BigInteger> findAny = stream.parallel()
					.mapToObj(i -> BigInteger.valueOf(i).multiply(BigInteger.valueOf(2l).add(BigInteger.ONE)))
					.filter(i -> t.remainder(i) == BigInteger.ZERO).findAny();
			return !findAny.isPresent();
		};

		BigInteger ramen = BigInteger.valueOf(Long.MAX_VALUE).multiply(BigInteger.valueOf(2l));
		ramen = ramen.nextProbablePrime();
		while (!isramen.apply(ramen)) {
			System.out.println(ramen + " is not ramen");
			ramen = ramen.nextProbablePrime();
		}
		return ramen;
	}

	/**
	 * Ramen5()にパラメーターを付与し変更しやすいよう変形したもの
	 * 
	 * @param mul1
	 *            開始数の乗数
	 * @param div1
	 *            開始数の除数
	 * @param mul2
	 *            ニュートン近似の乗数
	 * @param div2
	 *            ニュートン近似の除数
	 * @return Ramen6(13l, 2l, 5l, 2l)=59951918239556042783
	 */
	static BigInteger Ramen6(long mul1, long div1, long mul2, long div2) {
		Function<BigInteger, Long> halfsqrt = x -> {
			BigInteger xnn = BigInteger.valueOf((long) Math.sqrt(Long.MAX_VALUE)).multiply(BigInteger.valueOf(mul2))
					.divide(BigInteger.valueOf(div2));
			BigInteger xn = BigInteger.ZERO;
			while (xnn.subtract(xn).abs().compareTo(BigInteger.ONE) > 0) {
				xn = xnn;
				xnn = x.divide(xnn).add(xnn).divide(BigInteger.valueOf(2l));
			}
			return xnn.divide(BigInteger.valueOf(2l)).longValueExact();
		};

		Function<BigInteger, Boolean> isramen = t -> {
			LongStream stream = LongStream.rangeClosed(1, halfsqrt.apply(t));
			Optional<BigInteger> findAny = stream.parallel()
					.mapToObj(i -> BigInteger.valueOf(i).multiply(BigInteger.valueOf(2l).add(BigInteger.ONE)))
					.filter(i -> t.remainder(i) == BigInteger.ZERO).findAny();
			return !findAny.isPresent();
		};

		BigInteger ramen = BigInteger.valueOf(Long.MAX_VALUE).multiply(BigInteger.valueOf(mul1))
				.divide(BigInteger.valueOf(div1));
		ramen = ramen.nextProbablePrime();
		while (!isramen.apply(ramen)) {
			System.out.println(ramen + " is not ramen");
			ramen = ramen.nextProbablePrime();
		}
		return ramen;
	}

	/**
	 * Ramen6の (1)mapToObj内をBigIntegerからLong (2)findAny.isPresentからnoneMatch
	 * (3)remainder判定を==からcompareTo==
	 * (4)isramenをFunction<BigInteger,Boolean>からPredicate<BigInteger> 、で書き直したもの
	 * 
	 * @param mul1
	 *            開始数の乗数
	 * @param div1
	 *            開始数の除数
	 * @param mul2
	 *            ニュートン近似の乗数
	 * @param div2
	 *            ニュートン近似の除数
	 * @return Ramen7(114l, 10l, 346l, 100l)=105146441220144444251
	 */
	static BigInteger Ramen7(long mul1, long div1, long mul2, long div2) {
		Function<BigInteger, Long> halfsqrt = x -> {
			BigInteger xnn = BigInteger.valueOf((long) Math.sqrt(Long.MAX_VALUE)).multiply(BigInteger.valueOf(mul2))
					.divide(BigInteger.valueOf(div2));
			BigInteger xn = BigInteger.ZERO;
			while (xnn.subtract(xn).abs().compareTo(BigInteger.ONE) > 0) {
				xn = xnn;
				xnn = x.divide(xnn).add(xnn).divide(BigInteger.valueOf(2l));
			}
			return xnn.divide(BigInteger.valueOf(2l)).longValueExact();
		};

		Predicate<BigInteger> isramen = t -> LongStream.rangeClosed(1, halfsqrt.apply(t)).parallel()
				.mapToObj(i -> BigInteger.valueOf(i * 2l + 1l))
				.noneMatch(i -> t.remainder(i).compareTo(BigInteger.ZERO) == 0);

		BigInteger ramen = BigInteger.valueOf(Long.MAX_VALUE).multiply(BigInteger.valueOf(mul1))
				.divide(BigInteger.valueOf(div1));
		ramen = ramen.nextProbablePrime();
		while (!isramen.test(ramen)) {
			System.out.println(ramen + " is not ramen");
			ramen = ramen.nextProbablePrime();
		}
		return ramen;
	}

	/**
	 * Ramen7のhalfsqrtからsextantsqrtに書き換えたもの
	 * 
	 * @param mul1
	 *            開始数の乗数
	 * @param div1
	 *            開始数の除数
	 * @param mul2
	 *            ニュートン近似の乗数
	 * @param div2
	 *            ニュートン近似の除数
	 * @return
	 */
	static BigInteger Ramen8(long mul1, long div1, long mul2, long div2) {
		Function<BigInteger, Long> sextantsqrt = x -> {
			BigInteger xnn = BigInteger.valueOf((long) Math.sqrt(Long.MAX_VALUE)).multiply(BigInteger.valueOf(mul2))
					.divide(BigInteger.valueOf(div2));
			BigInteger xn = BigInteger.ZERO;

			do {
				xn = xnn;
				xnn = x.divide(xnn).add(xnn).divide(BigInteger.valueOf(2l));
			} while (xnn.subtract(xn).abs().compareTo(BigInteger.ONE) > 0);

			return xnn.max(xn).divide(BigInteger.valueOf(6l)).longValueExact();
		};

		Function<BigInteger, Boolean> isramen = t -> {
			if (t.remainder(BigInteger.valueOf(2l)).compareTo(BigInteger.ZERO) == 0
					|| t.remainder(BigInteger.valueOf(3l)).compareTo(BigInteger.ZERO) == 0) {
				return false;
			}

			return LongStream.range(0, sextantsqrt.apply(t)).parallel().mapToObj(i -> BigInteger.valueOf(i * 6l + 5l))
					.noneMatch(i -> t.remainder(i).compareTo(BigInteger.ZERO) == 0
							|| t.remainder(i.add(BigInteger.valueOf(2l))).compareTo(BigInteger.ZERO) == 0);
		};

		BigInteger ramen = BigInteger.valueOf(Long.MAX_VALUE).multiply(BigInteger.valueOf(mul1))
				.divide(BigInteger.valueOf(div1));
		ramen = ramen.nextProbablePrime();
		while (!isramen.apply(ramen)) {
			System.out.println(ramen + " is not ramen");
			ramen = ramen.nextProbablePrime();
		}
		return ramen;
	}
}
