/* * PROJECT: NyARToolkit(Extension) * -------------------------------------------------------------------------------- * The NyARToolkit is Java edition ARToolKit class library. * Copyright (C)2008-2009 Ryo Iizuka * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. * * For further information please contact. * http://nyatla.jp/nyatoolkit/ * <airmail(at)ebony.plala.or.jp> or <nyatla(at)nyatla.jp> * */ package jp.nyatla.nyartoolkit.core.utils; import jp.nyatla.nyartoolkit.*; /** * このクラスには、方程式を解く関数を定義します。 */ public class NyAREquationSolver { /** * この関数は、 二次方程式(ax^2+bx+c=0)を解いて、実根のみを返します。 * @param i_a * 係数aの値 * @param i_b * 係数bの値 * @param i_c * 係数cの値 * @param o_result * 解を受け取る配列。2要素以上である事。 * @return * o_resultに格納した実根の数。 */ public static int solve2Equation(double i_a, double i_b, double i_c,double[] o_result) { assert i_a!=0; return solve2Equation(i_b/i_a,i_c/i_a,o_result,0); } /** * この関数は、 二次方程式(x^2+bx+c=0)を解いて、実根のみを返します。 * @param i_b * 係数bの値 * @param i_c * 係数cの値 * @param o_result * 解を受け取る配列。2要素以上である事。 * @return * o_resultに格納した実根の数。 */ public static int solve2Equation(double i_b, double i_c,double[] o_result) { return solve2Equation(i_b,i_c,o_result,0); } /** * この関数は、 二次方程式(x^2+bx+c=0)を解いて、実根のみを返します。 * @param i_b * 係数bの値 * @param i_c * 係数cの値 * @param o_result * 解を受け取る配列。 * @param i_result_st * o_resultの格納を開始するインデクス番号 * @return * o_resultに格納した実根の数。 */ public static int solve2Equation(double i_b, double i_c,double[] o_result,int i_result_st) { double t=i_b*i_b-4*i_c; if(t<0){ //虚数根 return 0; } if(t==0){ //重根 o_result[i_result_st+0]=-i_b/(2); return 1; } //実根2個 t=Math.sqrt(t); o_result[i_result_st+0]=(-i_b+t)/(2); o_result[i_result_st+1]=(-i_b-t)/(2); return 2; } /** * この関数は、3次方程式(a*x^3+b*x^2+c*x+d=0)の実根を計算します。 * このコードは、http://aoki2.si.gunma-u.ac.jp/JavaScript/src/3jisiki.htmlを元にしています。 * @param i_a * X^3の係数 * @param i_b * X^2の係数 * @param i_c * X^1の係数 * @param i_d * X^0の係数 * @param o_result * 実根。double[3]を指定すること。 * @return * 返却した実根の数。 */ public static int solve3Equation(double i_a, double i_b, double i_c, double i_d,double[] o_result) { assert (i_a != 0); return solve3Equation(i_b/i_a,i_c/i_a,i_d/i_a,o_result); } /** * この関数は、3次方程式(x^3+b*x^2+c*x+d=0)の実根を計算します。 * このコードは、http://aoki2.si.gunma-u.ac.jp/JavaScript/src/3jisiki.htmlを元にしています。 * @param i_b * X^2の係数 * @param i_c * X^1の係数 * @param i_d * X^0の係数 * @param o_result * 実根。double[3]を指定すること。 * @return * 返却した実根の数。 */ public static int solve3Equation(double i_b, double i_c, double i_d,double[] o_result) { double tmp,b, p, q; b = i_b/(3); p = b * b - i_c / 3; q = (b * (i_c - 2 * b * b) - i_d) / 2; if ((tmp = q * q - p * p * p) == 0) { // 重根 q = Math.cbrt(q); o_result[0] = 2 * q - b; o_result[1] = -q - b; return 2; } else if (tmp > 0) { // 実根1,虚根2 double a3 = Math.cbrt(q + ((q > 0) ? 1 : -1) * Math.sqrt(tmp)); double b3 = p / a3; o_result[0] = a3 + b3 - b; // 虚根:-0.5*(a3+b3)-b,Math.abs(a3-b3)*Math.sqrt(3.0)/2 return 1; } else { // 実根3 tmp = 2 * Math.sqrt(p); double t = Math.acos(q / (p * tmp / 2)); o_result[0] = tmp * Math.cos(t / 3) - b; o_result[1] = tmp * Math.cos((t + 2 * Math.PI) / 3) - b; o_result[2] = tmp * Math.cos((t + 4 * Math.PI) / 3) - b; return 3; } } /** * この関数は、4次方程式(ax^4+bx^3+b*cx^2+d*x+e=0)の実根を計算します。 * @param i_a * X^4の係数 * @param i_b * X^3の係数 * @param i_c * X^2の係数 * @param i_d * X^1の係数 * @param i_e * X^0の係数 * @param o_result * 実根。double[4]を指定すること。 * @return * 返却した実根の数。 */ public static int solve4Equation(double i_a, double i_b, double i_c, double i_d,double i_e,double[] o_result) throws NyARException { assert (i_a != 0); double A3,A2,A1,A0,B3; A3=i_b/i_a; A2=i_c/i_a; A1=i_d/i_a; A0=i_e/i_a; B3=A3/4; double p,q,r; double B3_2=B3*B3; p=A2-6*B3_2;//A2-6*B3*B3; q=A1+B3*(-2*A2+8*B3_2);//A1-2*A2*B3+8*B3*B3*B3; r=A0+B3*(-A1+A2*B3)-3*B3_2*B3_2;//A0-A1*B3+A2*B3*B3-3*B3*B3*B3*B3; if(q==0){ double result_0,result_1; //複二次式 int res=solve2Equation(p,r,o_result,0); switch(res){ case 0: //全て虚数解 return 0; case 1: //重根 //解は0,1,2の何れか。 result_0=o_result[0]; if(result_0<0){ //全て虚数解 return 0; } //実根1個 if(result_0==0){ //NC o_result[0]=0-B3; return 1; } //実根2個 result_0=Math.sqrt(result_0); o_result[0]=result_0-B3; o_result[1]=-result_0-B3; return 2; case 2: //実根2個だからt==t2==0はありえない。(case1) //解は、0,2,4の何れか。 result_0=o_result[0]; result_1=o_result[1]; int number_of_result=0; if(result_0>0){ //NC result_0=Math.sqrt(result_0); o_result[0]= result_0-B3; o_result[1]=-result_0-B3; number_of_result+=2; } if(result_1>0) { //NC result_1=Math.sqrt(result_1); o_result[number_of_result+0]= result_1-B3; o_result[number_of_result+1]=-result_1-B3; number_of_result+=2; } return number_of_result; default: throw new NyARException(); } }else{ //それ以外 //最適化ポイント: //u^3 + (2*p)*u^2 +((- 4*r)+(p^2))*u -q^2= 0 double u=solve3Equation_1((2*p),(- 4*r)+(p*p),-q*q); if(u<0){ //全て虚数解 return 0; } double ru=Math.sqrt(u); //2次方程式を解いてyを計算(最適化ポイント) int result_1st,result_2nd; result_1st=solve2Equation(-ru,(p+u)/2+ru*q/(2*u),o_result,0); //配列使い回しのために、変数に退避 switch(result_1st){ case 0: break; case 1: o_result[0]=o_result[0]-B3; break; case 2: o_result[0]=o_result[0]-B3; o_result[1]=o_result[1]-B3; break; default: throw new NyARException(); } result_2nd=solve2Equation(ru,(p+u)/2-ru*q/(2*u),o_result,result_1st); //0,1番目に格納 switch(result_2nd){ case 0: break; case 1: o_result[result_1st+0]=o_result[result_1st+0]-B3; break; case 2: o_result[result_1st+0]=o_result[result_1st+0]-B3; o_result[result_1st+1]=o_result[result_1st+1]-B3; break; default: throw new NyARException(); } return result_1st+result_2nd; } } /** * 3次方程式の実根を1個だけ求める。 * 4字方程式で使う。 * @param i_b * @param i_c * @param i_d * @param o_result * @return */ private static double solve3Equation_1(double i_b, double i_c, double i_d) { double tmp,b, p, q; b = i_b/(3); p = b * b - i_c / 3; q = (b * (i_c - 2 * b * b) - i_d) / 2; if ((tmp = q * q - p * p * p) == 0) { // 重根 q = Math.cbrt(q); return 2 * q - b; } else if (tmp > 0) { // 実根1,虚根2 double a3 = Math.cbrt(q + ((q > 0) ? 1 : -1) * Math.sqrt(tmp)); double b3 = p / a3; return a3 + b3 - b; } else { // 実根3 tmp = 2 * Math.sqrt(p); double t = Math.acos(q / (p * tmp / 2)); return tmp * Math.cos(t / 3) - b; } } /* public static void main(String[] args) { NyAREquationSolver n = new NyAREquationSolver(); int l=0; double[] r = new double[10]; try{ l=n.solve4Equation(1, 9, -18, -68, 120, r); }catch(Exception e){ e.printStackTrace(); } System.out.println(l); }*/ }