147 lines
4.8 KiB
Java
147 lines
4.8 KiB
Java
import java.io.BufferedReader;
|
|
import java.io.File;
|
|
import java.io.FileReader;
|
|
import java.io.IOException;
|
|
import java.math.BigInteger;
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.regex.Matcher;
|
|
import java.util.regex.Pattern;
|
|
|
|
public class CheckMachinFormula {
|
|
|
|
private static String FILE_NAME = "MachinFormula.txt";
|
|
|
|
public static void main(String[] args) {
|
|
try {
|
|
runPrivate();
|
|
} catch (Exception e) {
|
|
e.printStackTrace();
|
|
}
|
|
}
|
|
|
|
private static void runPrivate() throws IOException {
|
|
try (BufferedReader reader = new BufferedReader(new FileReader(new File(FILE_NAME)));) {
|
|
String inLine = null;
|
|
while ( (inLine = reader.readLine()) != null ) {
|
|
String[] split = inLine.split("=");
|
|
System.out.println(tanLeft(split[0].trim()) + " = " + split[1].trim().replaceAll("\\s+", " ") + " = " + tanRight(split[1].trim()));
|
|
}
|
|
}
|
|
}
|
|
|
|
private static String tanLeft(String formula) {
|
|
if ( formula.compareTo("pi/4") == 0 ) {
|
|
return "1";
|
|
}
|
|
throw new RuntimeException("ERROR 104: Unknown left side: " + formula);
|
|
}
|
|
|
|
private static final Pattern ARCTAN_PATTERN = Pattern.compile("(-{0,1}\\d+)\\*arctan\\((\\d+)/(\\d+)\\)");
|
|
|
|
private static Fraction tanRight(String formula) {
|
|
Matcher matcher = ARCTAN_PATTERN.matcher(formula);
|
|
List<Term> terms = new ArrayList<>();
|
|
while ( matcher.find() ) {
|
|
terms.add(new Term(Integer.parseInt(matcher.group(1)), new Fraction(matcher.group(2), matcher.group(3))));
|
|
}
|
|
return evaluateArctan(terms);
|
|
}
|
|
|
|
private static Fraction evaluateArctan(List<Term> terms) {
|
|
if ( terms.size() == 1 ) {
|
|
Term term = terms.get(0);
|
|
return evaluateArctan(term.coefficient, term.fraction);
|
|
}
|
|
int size = terms.size();
|
|
List<Term> left = terms.subList(0, (size+1) / 2);
|
|
List<Term> right = terms.subList((size+1) / 2, size);
|
|
return arctanFormula(evaluateArctan(left), evaluateArctan(right));
|
|
}
|
|
|
|
private static Fraction evaluateArctan(int coefficient, Fraction fraction) {
|
|
//System.out.println("C = " + coefficient + ", F = " + fraction);
|
|
if ( coefficient == 1 ) {
|
|
return fraction;
|
|
}
|
|
else if ( coefficient < 0 ) {
|
|
return evaluateArctan(-coefficient, fraction).negate();
|
|
}
|
|
if ( coefficient % 2 == 0 ) {
|
|
Fraction f = evaluateArctan(coefficient/2, fraction);
|
|
return arctanFormula(f, f);
|
|
}
|
|
Fraction a = evaluateArctan(coefficient/2, fraction);
|
|
Fraction b = evaluateArctan(coefficient - (coefficient/2), fraction);
|
|
return arctanFormula(a, b);
|
|
}
|
|
|
|
private static Fraction arctanFormula(Fraction f1, Fraction f2) {
|
|
return f1.add(f2).divide(Fraction.ONE.subtract(f1.multiply(f2)));
|
|
}
|
|
|
|
private static class Fraction {
|
|
|
|
public static final Fraction ONE = new Fraction("1", "1");
|
|
|
|
private BigInteger numerator;
|
|
private BigInteger denominator;
|
|
|
|
public Fraction(String num, String den) {
|
|
numerator = new BigInteger(num);
|
|
denominator = new BigInteger(den);
|
|
}
|
|
|
|
public Fraction(BigInteger num, BigInteger den) {
|
|
numerator = num;
|
|
denominator = den;
|
|
}
|
|
|
|
public Fraction negate() {
|
|
return new Fraction(numerator.negate(), denominator);
|
|
}
|
|
|
|
public Fraction add(Fraction f) {
|
|
BigInteger gcd = denominator.gcd(f.denominator);
|
|
BigInteger first = numerator.multiply(f.denominator.divide(gcd));
|
|
BigInteger second = f.numerator.multiply(denominator.divide(gcd));
|
|
return new Fraction(first.add(second), denominator.multiply(f.denominator).divide(gcd));
|
|
}
|
|
|
|
public Fraction subtract(Fraction f) {
|
|
return add(f.negate());
|
|
}
|
|
|
|
public Fraction multiply(Fraction f) {
|
|
BigInteger num = numerator.multiply(f.numerator);
|
|
BigInteger den = denominator.multiply(f.denominator);
|
|
BigInteger gcd = num.gcd(den);
|
|
return new Fraction(num.divide(gcd), den.divide(gcd));
|
|
}
|
|
|
|
public Fraction divide(Fraction f) {
|
|
return multiply(new Fraction(f.denominator, f.numerator));
|
|
}
|
|
|
|
@Override
|
|
public String toString() {
|
|
if ( denominator.compareTo(BigInteger.ONE) == 0 ) {
|
|
return numerator.toString();
|
|
}
|
|
return numerator + " / " + denominator;
|
|
}
|
|
}
|
|
|
|
private static class Term {
|
|
|
|
private int coefficient;
|
|
private Fraction fraction;
|
|
|
|
public Term(int c, Fraction f) {
|
|
coefficient = c;
|
|
fraction = f;
|
|
}
|
|
}
|
|
|
|
}
|