oracle 取昨日24点,24点运算 - 一贱书生的个人空间 - OSCHINA - 中文开源技术交流社区...

import java.util.Scanner;

/**

* Declaration: All Rights Reserved !!!

*/

public class Main {

public static void main(String[] args) {

// Scanner scanner = new Scanner(System.in);

Scanner scanner = new Scanner(Main.class.getClassLoader().getResourceAsStream("data2.txt"));

while (scanner.hasNextLine()) {

String input = scanner.nextLine();

System.out.println(solve(input));

}

scanner.close();

}

/**

* 进行24点运算,假设输入的牌都是合法的

*

* @param input 输入的牌

* @return 结果

*/

private static String solve(String input) {

// 有王

if (input.contains("joker") || input.contains("JOKER")) {

return "ERROR";

}

String[] parts = input.split("(\\s)+");

// 不是四个牌

if (parts.length != 4) {

return "ERROR";

}

// 将牌转换成数字

int[] ints = new int[parts.length];

for (int i = 0; i < ints.length; i++) {

ints[i] = convert(parts[i]);

}

String[] r = {""};

boolean result = point24(ints, r);

// 有结果

if (result) {

return r[0];

}

return "ERROR";

}

/**

* 4个[1, 13]的能否通过加减乘除,得到数字为24

*

* @param opd 四个元素的数组

* @param r 用于保存结果

* @return true,可以组成24,false不可以组成24

*/

private static boolean point24(int[] opd, String[] r) {

char[] opt = new char[opd.length - 1];

char[] all = {'+', '-', '*', '/'};

return point24(opd, opt, all, 0, r);

}

/**

* @param opd 操作数

* @param opt 用于计算的操作符

* @param all 可以使用的全部操作符

* @param n 当前处理的数

* @param r 用于保存表达式结果

* @return true,计算的值为24,false,计算的值不是24

*/

private static boolean point24(int[] opd, char[] opt, char[] all, int n, String[] r) {

// 处理最后一个数字

if (n == opd.length - 1) {

if (calculate(opd, opt, r)) {

return true;

}

} else {

for (int i = n; i < opd.length; i++) {

int temp = opd[n];

opd[n] = opd[i];

opd[i] = temp;

for (char a : all) {

opt[n] = a;

boolean find = point24(opd, opt, all, n + 1, r);

if (find) {

return true;

}

}

// 现场还原

temp = opd[n];

opd[n] = opd[i];

opd[i] = temp;

}

}

return false;

}

/**

* 计算值

*

* @param opd 操作数

* @param opt 操作符

* @param r 保存结果表达式

* @return true,结果为24,false结果不是24

*/

private static boolean calculate(int[] opd, char[] opt, String[] r) {

double v = opd[0];

for (int i = 0; i < opt.length; i++) {

v = calculate(v, opd[i + 1], opt[i]);

}

boolean eq = Math.abs(v - 24) < 0.0000001;

if (eq) {

r[0] = convert(opd[0]);

for (int i = 0; i < opt.length; i++) {

r[0] = r[0] + opt[i] + convert(opd[i + 1]);

}

}

return eq;

}

/**

* 将牌转换成对应的数值

*

* @param s 牌

* @return 数值

*/

private static int convert(String s) {

if (s.length() == 1) {

char c = s.charAt(0);

if (c >= '2' && c <= '9') {

return c - '0';

}

switch (c) {

case 'J':

return 11;

case 'Q':

return 12;

case 'K':

return 13;

case 'A':

return 1;

default:

// do nothing

}

} else {

switch (s) {

case "10":

return 10;

default:

// do nothing

}

}

throw new RuntimeException("参数错误:" + s + ",只能输入[2, 10]、J、Q、K、A");

}

/**

* 将数值转换成牌

*

* @param i 数值

* @return 牌

*/

private static String convert(int i) {

if (i >= 2 && i <= 10) {

return "" + i;

}

switch (i) {

case 1:

return "A";

case 11:

return "J";

case 12:

return "Q";

case 13:

return "K";

default:

// do nothing

}

throw new RuntimeException("参数错误:" + i + ",牌值[1, 13]的整数");

}

/**

* 表达式求值

*

* @param a 操作数一

* @param b 操作数二

* @param opt 操作符

* @return 运算结果

*/

private static double calculate(double a, double b, int opt) {

switch (opt) {

case '+':

return a + b;

case '-':

return a - b;

case '*':

return a * b;

case '/':

return a / b;

default:

// do nothing

}

throw new RuntimeException("参数错误:" + ((char) opt) + "计算操作只支持加(+)、减(-)、*(乘)、除(、)");

}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值