Problem
The blocksWorld is a well-known problem in AI searching area. A description can be found at https://en.wikipedia.org/wiki/Blocks_world. We have several stacks and blocks. Each blocks is assigned a unique letter. The goal state this program wants to achieve is that all blocks are in the first stack and in alphabetical order.
Program
Code
The code can be found at https://github.com/melonskin/blocksWorldAstarSearch.
Compile
Since I am using maven, the easiest way to compile and run it is using IDE (I use IDEA). However, we can always compile and run it in command line. The steps are as follows.
cd
to the project foldercd src
javac com/tangshijin/blocksWorld/*.java
Run
To run the program, execute following command with the arguments of your own choice.
java -cp . com.tangshijin.blocksWorld.Main no_blocks no_stacks no_iter
The user-defined arguments are
no_blocks
: number of blocks in this caseno_stacks
: number of stacksno_iter
: maximum number of iterations allowed
Output examples
An output example is shown as below.
$ java -cp . com.tangshijin.blocksWorld.Main 6 4 100
initial state:
1 | A
2 | E
3 | F B D
4 | C
iter = 0, queueSize = 0, f = -1, depth = 0
iter = 1, queueSize = 11, f = 8, depth = 1
iter = 2, queueSize = 19, f = 7, depth = 2
iter = 3, queueSize = 27, f = 6, depth = 3
iter = 4, queueSize = 32, f = 5, depth = 4
iter = 5, queueSize = 37, f = 6, depth = 4
iter = 6, queueSize = 45, f = 6, depth = 5
iter = 7, queueSize = 50, f = 6, depth = 5
iter = 8, queueSize = 54, f = 6, depth = 5
iter = 9, queueSize = 55, f = 6, depth = 5
iter = 10, queueSize = 59, f = 6, depth = 5
iter = 11, queueSize = 64, f = 6, depth = 3
iter = 12, queueSize = 68, f = 6, depth = 6
success! depth = 6, total_goal_tests = 12, max_queue_size = 69
solution path:
1 | A
2 | E
3 | F B D
4 | C
1 | A
2 | E D
3 | F B
4 | C
1 | A B
2 | E D
3 | F
4 | C
1 | A B C
2 | E D
3 | F
4 |
1 | A B C D
2 | E
3 | F
4 |
1 | A B C D E
2 |
3 | F
4 |
1 | A B C D E F
2 |
3 |
4 |
Heuristic function
My heuristic function is basically the summation of the number of moves to move single block to its target position for all blocks.
At the target state, all blocks will be in order in the first stack. At other states, for the first stack, there are actually two case. One is the current block is above the target position. The measurement is number of moves that we takes all blocks above the target position(inclusive) away and put the current blocks. The other is current block is under the target position. The heuristic function valus is number of moves that taking all blocks above the current block (inclusive) away, and filling blocks back until reaching target position, then putting current block back.
For other stacks, the distance is actually the summation of the number of blocks above the target position in the first stack and the number of blocks above the current position in the current stack, plus one. That is to say, number of moves if we move all blocks above current position (exclusive) and target position (inclusive) to other position, and then move current block to the target position.
Furthermore, to improve the heuristic function, we may consider adding weight parameter to each block. Obviously, getting “A” in place first is better than getting “B” in place first then dealing with “A”. Because we don’t need to touch “A” any longer. Therefore, we can improve our heuristic function by add addtional term to the number of moves of every blocks to make sure “A” is placed in the right position first and so on.
If admissible?
This heuristic function is not admissible. Because during experiment, decreasing f(n)
values are observed sometimes. However, for admissible heuristic functions, A* explore nodes in the order of increasing f(n)
values. This function will overestimate true distance to goal. Taking fewer steps other than what has been described is totally possible.
Experiments
Performance metrics for the first heuristic
15 cases are chosen to run. Every case is repeated for 20 times. Max number of iteration is set to be 20000. The table below shows the mean performance.
No. blocks | No. stacks | Mean depth | Inital moves | No. successes | Mean No. iterations | Mean max frontier size |
---|---|---|---|---|---|---|
5 | 3 | 5.45 | 15 | 20/20 | 8.60 | 18.75 |
6 | 3 | 6.40 | 18 | 20/20 | 11.10 | 27.85 |
7 | 3 | 8.75 | 21 | 20/20 | 43.45 | 86.05 |
8 | 3 | 10.00 | 24 | 20/20 | 57.65 | 140.45 |
9 | 3 | 10.40 | 27 | 20/20 | 364.45 | 910.15 |
10 | 3 | 8.80 | 30 | 20/20 | 34.40 | 89.35 |
10 | 4 | 13.45 | 40 | 20/20 | 752.90 | 3544.25 |
10 | 5 | 16.25 | 50 | 20/20 | 2104.50 | 14121.90 |
10 | 6 | 17.26 | 60 | 19/20 | 1660.74 | 12332.37 |
10 | 7 | 18.81 | 70 | 16/20 | 5657.13 | 52990.88 |
10 | 8 | 18.80 | 80 | 10/20 | 6306.90 | 70438.80 |
15 | 7 | 23.40 | 105 | 5/20 | 5196.80 | 52627.40 |
20 | 7 | 22.50 | 140 | 4/20 | 7635.00 | 70571.50 |
25 | 7 | NaN | 175 | 0/20 | NaN | NaN |
30 | 7 | 19.00 | 210 | 1/20 | 1189.00 | 11662.00 |
Performance metrics for the second heuristic
For second heuristic function, larger cases are chosen to run.
No. blocks | No. stacks | Mean depth | Inital moves | No. successes | Mean No. iterations | Mean max frontier size |
---|---|---|---|---|---|---|
5 | 3 | 6.30 | 15 | 20/20 | 9.95 | 21.05 |
6 | 3 | 6.35 | 18 | 20/20 | 10.10 | 24.80 |
7 | 3 | 8.50 | 21 | 20/20 | 15.80 | 38.50 |
8 | 3 | 8.20 | 24 | 20/20 | 15.85 | 40.30 |
9 | 3 | 7.70 | 27 | 20/20 | 14.95 | 38.05 |
10 | 3 | 10.40 | 30 | 20/20 | 17.85 | 47.45 |
10 | 4 | 13.70 | 40 | 20/20 | 178.85 | 885.65 |
10 | 5 | 13.40 | 50 | 20/20 | 300.35 | 1999.80 |
10 | 6 | 17.89 | 60 | 18/20 | 1640.50 | 11468.22 |
10 | 7 | 17.88 | 70 | 16/20 | 2950.56 | 22574.31 |
11 | 7 | 20.56 | 77 | 18/20 | 3351.78 | 27974.33 |
15 | 7 | 26.50 | 105 | 10/20 | 7602.10 | 70549.70 |
20 | 7 | 28.71 | 140 | 7/20 | 7256.29 | 67380.14 |
25 | 7 | 32.00 | 175 | 4/20 | 7061.25 | 65670.75 |
30 | 7 | 31.50 | 210 | 2/20 | 6359.00 | 57032.50 |
35 | 7 | 34.00 | 245 | 3/20 | 12145.00 | 96536.67 |
15 | 8 | 18.33 | 120 | 3/20 | 5763.67 | 64798.67 |
15 | 9 | 14.00 | 135 | 1/20 | 2366.00 | 33107.00 |
15 | 10 | 24.00 | 150 | 1/20 | 2233.00 | 31626.00 |
15 | 11 | 35.00 | 165 | 1/20 | 12504.00 | 197140.00 |
Discussion
From the table of performance metrics, we can observe that this heuristic function works well. The mean path depths are far more less than corresponding numbers of initial moves we made.
For every case with 3 stacks, all 20 repeated tests succeeded with a mean depth less than or almost 10 and relatively small number of iterations and frontier size. Even for the difficult case with 3 stacks and 10 blocks, this program performs well.
For the cases with 10 blocks and more than 3 stacks, this program has about 90% chance to succeed. The mean depth is still reasonable, far less than the number of initial moves. Compared with the benchmark depth 18 of the case with 5 stacks and 10 blocks, our mean depth is around 16. However, the number of iterations and frontier size will increasing dramatically as the number of stacks rises.
As the number of stacks goes up, the branching parameter will increasing exponentially. We also have more initial random moves so that the depth of potential goal is very likely to be large. Therefore, the problem will be harder to solve. As we can observe in the table, for the initial heuristic function, case with 20 blocks and 7 stacks has the largest mean number of iterations and frontier size. It has 20% chance to find the goal, less than other cases. Meanwhile, since the cases with more blocks and stacks will have more initial moves, it can also make the problem harder.
The largest problem is considered th be the most computationally difficult. Therefore, it should be the case with large number of stacks, which means large branching parameter and more initial random moves. The largest problem we could solve is the case with 10 blocks and 8 stacks for initial heuristic functions. It will be slow when we have 9 stacks. For more difficult problem, the program has the risk running out of memory.
The second one performs better on large problems since the corresponding number of iterations and maximum frontier size are less than the cases with the initial heuristics. As we can see, the goals it can reach are far deeper than those with initial heuristic function. It can deal with the cases with more blocks (35) and more stacks (11) with less iterations and frontier size.
As mentioned above, our mean path depth for the case with 10 blocks and 5 stacks is pretty similar to the benchmark optimal depth provided. The depth is also far more smaller than the number of initial random moves.
To improve the heuristic function, we may consider modify the weight for every letter more precisely based on performance.