6

也许你们中的大多数人都知道发送 + 更多 = 金钱。好吧,我目前正在学习 Java,其中一个练习是我必须解决 HES + THE = BEST。

现在,到目前为止,我可以/应该使用 if-for-while-do 循环,仅此而已。虽然我确信有不同的方法可以解决它,但这不是我要进行的练习的重点。我必须能够以最有效的方式使用 if-for-while-do 循环。

我的问题?我似乎想不出一种有效的方法来解决它!我想出了这个,它解决了这个难题,但也许是最糟糕的有效方法:

public class Verbalarithmetics {

    public static void main (String args[]) {
        // Countint Variables
        int index_h = 0;
        int index_e = 0;
        int index_s = 0;
        int index_t = 0;
        int index_b = 0;

        // Start with h = 1 and increase until the else-if statement is true
        for(int h = 1; h <= 9; h++) { // h = 1, because first Symbol can't be zero
            index_h++;
                // Increase e so long until e equals h
                for(int e = 0; e <= 9; e++) {
                    index_e++;
                 if (e == h) {
                    continue;
                 }

                 // Increase s so long until s equals h or e
                 for(int s = 0; s <= 9; s++) {
                     index_s++;
                    if (s == h || s == e) {
                       continue;
                    }//end if

                    // Increase t so long until t equals h or e or s.
                    for(int t = 1; t <= 9; t++) { // t = 1, because 1st Symbol cant be zero
                        index_t++;
                      if(t == h || t == e || t == s) {
                         continue;
                      }// end if

                      // Increase b so long until b equals h, e, s or t.
                      for(int b = 1; b <= 9; b++) { // b = 1, weil das 1. Symbol nicht für eine 0 stehen darf
                          index_b++;
                          if (b == h || b == e || b == s || b == t) {
                              continue;
                          }// end if

                          // x = 100*h + 10*e + s
                          // y = 100*t + 10*h + e
                          // z = 1000*b + 100*e + 10*s + t
                          // Check if x+y=z, if true -> Print out Solution, else continue with the upper most loop
                          else 
                              if (100*h + 10*e + s + 100*t + 10*h + e == 1000*b + 100*e +10*s + t) {
                                  System.out.println("HES + THE = BEST => " + h + e + s + " + " + t + h + e + " = " + b + e + s + t);
                                  System.out.println("With H=" + h + ", E=" + e + ", S=" + s + ", T=" + t + ", und B=" + b + ".");
                                  System.out.println("It took " + index_h + 
                                          " Loop-Cycles to find 'h' !");
                                  System.out.println("It took " + index_e + 
                                          " Loop-Cycles to find 'e' !");
                                  System.out.println("It took " + index_s + 
                                          " Loop-Cycles to find 's' !");
                                  System.out.println("It took " + index_t + 
                                          " Loop-Cycles to find 't' !");
                                  System.out.println("It took " + index_b + 
                                          " Loop-Cycles to find 'b' !");
                                  System.out.println("This is a total of " + (index_h + index_e + index_s + index_t + index_b) + 
                                          " Loop-Cycles");
                          }// end else if
                      }//end for
                    }//end for
                 }//end for
              }//end for
        }//end for
    }   
}

解决这个难题总共需要大约 15000 个奇数循环周期。这在我看来很多。请问有什么指点吗?

4

7 回答 7

4

这里最大的问题是:你能(你想)从逻辑上推导出某些约束并将它们应用到你的算法中,还是你想暴力破解它?假设是前者,其中一些非常明显:

  • B = 1
  • T 不能为 0(因为它在 THE 中是第一个),因此 S 和 E 都不能为 0。
  • T = E + S % 10

因此,您有 S、E、H 循环,最多为您提供 9 * 8 * 8 个组合,即 576。再加上 H + T 必须大于或等于 9 的事实,您将进一步减少这一点。

更新这是一个快速而丑陋的解决方案。它仅基于上面列出的 3 个约束。

public class Puzzle {
  public static void main(String[] args) {
    for (int S = 1; S<10; S++) {
      for (int E = 1; E<10; E++) {
        if (S==E) continue; // all letters stand for different digits
        for (int H = 1; H<10; H++) {
          if (H==E || H==S) continue; // all letters stand for different digits
          checkAndPrint(S, E, H);
        }
      } // for
    } // for
  } // main

  private static boolean checkAndPrint(int S, int E, int H) {
    int T = (S + E) % 10;
    int S1 = (E + H) + (S + E) / 10; // calculates value for 'S' in 'BEST' (possibly + 10)
    if (S1 % 10 != S) return false;
    int E1 = H + T + S1 / 10; // calculates value for 'E' in 'BEST' (possibly + 10)
    if (E1 % 10 != E) return false;
    System.out.println(H + "" + E + "" + S + " + " + T + "" + H + "" + E + " = 1" + E + "" + S + "" + T);
    return true;
  }
}
于 2009-11-19T20:50:34.243 回答
2

也许您可能想查看这个存储库:它是使用 JavaFX 解决语言算术问题的解决方案。用于语言算术问题的萤火虫算法 [GitHub]

于 2016-06-11T20:40:17.410 回答
1

不是循环遍历字母的所有值,而是遍历 S、E 和 T 的可能值。S + E % 10 应该是 T。一旦你有一组潜在的 S、E、T 解决方案,找到循环可能的 E+H+(0 或 1,取决于 S+E 是否大于 9)=S 解...等等。

于 2009-11-19T20:35:52.443 回答
1

我不是专家,但可能值得研究管理约束的语言,例如 Prolog。这里有一个非常相似的问题:

计算不同奇数的列表(如果存在),使得它们的总和等于给定数

Prolog 是一种不同类型的语言,但如果您这样做是为了自己的教育,那么它肯定会锻炼您的大脑:-)

将有可能编写字母表的通用方法 - 而不仅仅是这里相当简单的方法。

另一种方法(不能保证给出结果)是使用优化技术,例如遗传算法。猜测一些解,并计算它们与正确解的接近程度,然后调整它们。您可以通过这种方法获得部分解决方案。

于 2009-11-29T17:05:54.290 回答
0

嗯,你可以在你的方法中以优化的形式做很多事情。

首先,获得“BEST”的最大值。假设“HES”具有最高可能值 987,那么“THE”将是 X98,因此“THE”的最高值为 698,即 987+698=1685。

如果“THE”具有最高值,THE 将是 987,HES 将是 876 -> 876+987=1863,高于 1685,因此 1863 是“最佳”的上限。因此,您可以让您的程序将“B”的上限调整为 1(在这种情况下,这已经为您提供了第一个数字..)。BEST 的下限很简单,因为它是 1023。

然后你做这样的事情:

for(i=102;i<=987;i++)
{
    for(j=1023-i;j<=(1863-i < 987 ? 1863-i:987);j++)
    {
        //check if the solution matches and doesn't have duplicate digits
    }
}

这样,您可以通过内部 for 循环中的值立即丢弃许多不可能的组合。我敢打赌,有类似的方法可以更多地限制可能解决方案的空间。

这样程序就不那么复杂了。

于 2009-11-19T20:55:43.233 回答
0

如果标准方法是暴力破解,效率就会消失,正如这里所建议的那样。仅使用循环的最有效方法可能涉及计算详尽的可能性集,存储它们,然后遍历每个可能性以查看它是否有效。

于 2009-11-19T20:31:26.620 回答
0

这类问题是查询优化的典型代表。Java 不是为了那个。

如果您的状态少于数百亿,请使用蛮力。与创建优化查询引擎相比,运行蛮力搜索所需的时间要少得多

于 2009-11-19T21:22:59.607 回答