Current issue

Vol.26 No.4

Vol.26 No.4

Volumes

© 1984-2024
British APL Association
All rights reserved.

Archive articles posted online on request: ask the archivist.

archive/21/4

Volume 21, No.4

At Play with J: Metlov’s Triumph

by Eugene McDonnell

A puzzle was recently announced by Frank Buss on the Internet that led to some interesting discoveries. The puzzle is to be found by Googling “Frank Buss Triangle Problem” and then clicking on “Triangles Challenge” or browsing directly to http://www.frank-buss.de/challenge/ if you prefer. It says:

The challenge is to write a program, which counts all triangles with area >0 in this figure:

diagram

But count only the triangles, which are bounded by lines, like (P0, P7, P8), not all possible connections between the points, like (P7, P8, P9). If anything is unclear, the solution is 27 and looks like this: (see below).

Graphic output is not needed, but you can do what you want. If a GUI or something else is included, it would be nice to write: how long you needed for the pure algorithm and for the rest.

This is not a quantitative, but more a qualitative challenge. Neither the number of lines nor the time (which I can’t verify anyway) is important, but I’m interested in good solutions, which show the advantages of the chosen language.

Every program should be documented enough to understand how it works and it should not simply print 27, but somewhere it should read from a file or integrate the points and geometry, so that it is easy to change it for similar problems, for example if another line is added, but it need not to be so general as to count the number of squares.

diagrams

There were 31 entries: The languages they used, the number of entries in that language, and the average number of lines in the programs are tabulated below:

Language
Number of entries Average number of lines

C++

3

115

Java

4

105

Python

1

94

Haskell

1

93

Ruby

1

75

Scheme

1

66

Awk

1

59

Lisp

17

56

Kogut

1

29

J

1

1


Most of these had a generous amount of documentation along with the actual program. I don’t know most of the languages used, but I could come to some conclusions about them. It seems to me that most of the authors were more programmers than mathematicians. Almost all of them tackled the problem as one of establishing the proper way to represent the points, lines, and intersections in the triangle. Most of them gave solutions which were wired in, and their programs could not easily be extended to variations of the problem.

Since Haskell is supposed to be a functional programming language, I thought it might give an interesting and useful result, but I was disappointed. It hard-coded the geometry of the problem, so that it, like many others, couldn’t be extended.

The J solution was submitted by Dr. K. L. Metlov. Here it is:

* -: @ * +

Metlov is a physicist, with many publications in his field, and he obviously studied the triangle puzzle as a mathematical one. In his notes, one sees that he experiments with variations of the problem, and in a relatively short time had concluded that a simple expression could be formed that would apply to a triangle with any number of lines.

This is a fork, and a dyad, and it is better understood by emphasizing its forkiness.

     *   +
      \ / 
       |
       *
       |
       -:

The arguments are multiplied and added, this product and sum are multiplied, and the product is halved.

I give Metlov’s Documentation below:

“When both sides of the triangle are divided into an equal number of steps (let’s call this number – n), the number of triangles is n^3 (n to the third power). For the example Frank Buss gives n=3 and the answer is
3^3 = 27.

When sides are subdivided into a different number of subdivisions, say, n and m, the number of triangles is equal to

½m × n(m + n)

which is integer for any integer m and n.

In J language (see http://www.jsoftware.com/ for description and download) the first formula is coded and invoked as

    nt =: ^&3
    nt 3
27

The second formula is coded and invoked as

    nt =: * -: @ * +
    3 nt 3
27
    2 nt 5
35

The first variant of the program is three characters, the second is 6 characters.

It took me 15-20 minutes of drawing rectangles to derive the formula. J is an array-oriented language, descendent of APL. Therefore, the above programs (without change) can indeed be used to process millions of rectangles very fast. In order to achieve this the arguments must be arrays (of equal length in the second case). For example:

    (3 2) nt (3 5)
27 35 

How the formula was developed:

Here is the link to the page of notes I made when thinking about the problem. http://www.livejournal.com/users/dr_klm/51584.html?thread=435072#t435072 and [overleaf] is a copy of the page.

The direct link is here: http://galaxy.fzu.cz/~metlov/Triangles_Deriv.gif

page from notebook

I do not know if that will be enough to communicate the basic idea used for deriving the formula. On the other hand I do not have time to explain it in full detail.

The interesting part occupies the lower left quarter of the page. Triangles are counted separately for two lower corners of the big triangle (left and right) and then the result is multiplied by 2 (if n=m), or added up with exchanging n<->m (if n!=m). To count triangles for one corner I sum up the triangles, occupying all single sub-sectors, the triangles, occupying all pairs of two consecutive sub-sectors, ... three sub-sectors... etc... In this sum, the triangles, which include both left and right corners of the big triangle are counted with weight 1/2 (to note that they will be counted again, in the sum for the other corner).

I ran this procedure for n=3, m=3 approximately in the middle of the page. Then, by induction, wrote a general formula with the sum. The sum is nothing else but an arithmetic progression, which is immediately summed up. Then, with very basic algebra, the final formulas are obtained.”

Comment from Frank Buss: This is a nice solution and the language looks interesting. It is the same concept as the Scheme solution, which uses a formula instead of counting the triangles, but this formula is much easier than the one used in the Scheme solution.


script began 7:07:26
caching off
debug mode off
cache time 3600 sec
indmtime not found in cache
cached index is fresh
recompiling index.xml
index compiled in 0.2435 secs
read index
read issues/index.xml
identified 26 volumes, 101 issues
array (
  'id' => '10005770',
)
regenerated static HTML
article source is 'HTML'
source file encoding is 'ASCII'
read as 'Windows-1252'
URL: http://www.frank-buss.de/challenge/ => http://www.frank-buss.de/challenge/
URL: metlov1.gif => trad/v214/metlov1.gif
URL: metlov2.gif => trad/v214/metlov2.gif
URL: http://www.jsoftware.com/ => http://www.jsoftware.com/
URL: http://www.livejournal.com/users/dr_klm/51584.html?thread=435072#t435072 => http://www.livejournal.com/users/dr_klm/51584.html?thread=435072#t435072
URL: http://galaxy.fzu.cz/%7emetlov/triangles_deriv.gif%20 => http://galaxy.fzu.cz/%7Emetlov/Triangles_Deriv.gif%20
URL: metlov3.gif => trad/v214/metlov3.gif
completed in 0.2678 secs