Previous Contents Next

Finding Least Cost Paths

Many applications need to find least cost paths through weighted directed graphs. The problem is to find a path through a graph in which non-negative weights are associated with the arcs. We will use Dijkstra's algorithm to determine the path.

This will be an opportunity to use several previously introduced libraries. In the order of appearance, the following modules will be used: Genlex and Printf for input and output, the module Weak to implement a cache, the module Sys to track the time saved by a cache, and the Awi library to construct a graphical user interface. The module Sys is also used to construct a standalone application that takes the name of a file describing the graph as a command line argument.

Graph Representions

A weighted directed graph is defined by a set of nodes, a set of edges, and a mapping from the set of edges to a set of values. There are numerous implementations of the data type weighted directed graph. The behavior of the representations depends on the size and the number of edges in the graph. Since one goal of this application is to show how to cache certain previously executed computations without using all of memory, an adjacency matrix is used to represent weighted directed graphs. In this way, memory usage will not be increased by list manipulations.


# type cost = Nan | Cost of float;;
# type adj_mat = cost array array;;
# type 'a graph = { mutable ind : int;
size : int;
nodes : 'a array;
m : adj_mat};;
The field size indicates the maximal number of nodes, the field ind the actual number of nodes.

We define functions to let us create graphs edge by edge.

The function to create a graph takes as arguments a node and the maximal number of nodes.

# let create_graph n s =
{ ind = 0; size = s; nodes = Array.create s n;
m = Array.create_matrix s s Nan } ;;
val create_graph : 'a -> int -> 'a graph = <fun>


The function belongs_to checks if the node n is contained in the graph g.

# let belongs_to n g =
let rec aux i =
(i < g.size) & ((g.nodes.(i) = n) or (aux (i+1)))
in aux 0;;
val belongs_to : 'a -> 'a graph -> bool = <fun>


The function index returns the index of the node n in the graph g. If the node does not exist, a Not_found exception is thrown.

# let index n g =
let rec aux i =
if i >= g.size then raise Not_found
else if g.nodes.(i) = n then i
else aux (i+1)
in aux 0 ;;
val index : 'a -> 'a graph -> int = <fun>


The next two functions are for adding nodes and edges of cost c to graphs.

# let add_node n g =
if g.ind = g.size then failwith "the graph is full"
else if belongs_to n g then failwith "the node already exists"
else (g.nodes.(g.ind) <- n; g.ind <- g.ind + 1) ;;
val add_node : 'a -> 'a graph -> unit = <fun>
# let add_edge e1 e2 c g =
try
let x = index e1 g and y = index e2 g in
g.m.(x).(y) <- Cost c
with Not_found -> failwith "node does not exist" ;;
val add_edge : 'a -> 'a -> float -> 'a graph -> unit = <fun>


Now it is easy to create a complete weighted directed graph starting with a list of nodes and edges. The function test_aho constructs the graph of figure 13.8:

# let test_aho () =
let g = create_graph "nothing" 5 in
List.iter (fun x -> add_node x g) ["A"; "B"; "C"; "D"; "E"];
List.iter (fun (a,b,c) -> add_edge a b c g)
["A","B",10.;
"A","D",30.;
"A","E",100.0;
"B","C",50.;
"C","E",10.;
"D","C",20.;
"D","E",60.];
for i=0 to g.ind -1 do g.m.(i).(i) <- Cost 0.0 done;
g;;
val test_aho : unit -> string graph = <fun>
# let a = test_aho();;
val a : string graph =
{ind=5; size=5; nodes=[|"A"; "B"; "C"; "D"; "E"|];
m=[|[|Cost 0; Cost 10; Nan; Cost 30; Cost ...|]; ...|]}




Figure 13.8: The test graph


Constructing Graphs

It is tedious to directly construct graphs in a program. To avoid this, we define a concise textual representation for graphs. We can define the graphs in text files and construct them in applications by reading the text files.

The textual representation for a graph consists of lines of the following forms: For example, the following file, aho.dat, describes the graph of figure 13.8 :
SIZE 5
NODE A
NODE B
NODE C
NODE D
NODE E
EDGE A B 10.0
EDGE A D 30.0
EDGE A E 100.0
EDGE B C 50.
EDGE C E 10.
EDGE D C 20.
EDGE D E 60.
To read graph files, we use the lexical analysis module Genlex. The lexical analyser is constructed from a list of keywords keywords.

The function parse_line executes the actions associated to the key words by modifying the reference to a graph.

# let keywords = [ "SIZE"; "NODE"; "EDGE"; "#"];;
val keywords : string list = ["SIZE"; "NODE"; "EDGE"; "#"]
# let lex_line l = Genlex.make_lexer keywords (Stream.of_string l);;
val lex_line : string -> Genlex.token Stream.t = <fun>
# let parse_line g s = match s with parser
[< '(Genlex.Kwd "SIZE"); '(Genlex.Int n) >] ->
g := create_graph "" n
| [< '(Genlex.Kwd "NODE"); '(Genlex.Ident name) >] ->
add_node name !g
| [< '(Genlex.Kwd "EDGE"); '(Genlex.Ident e1);
'(Genlex.Ident e2); '(Genlex.Float c) >] ->
add_edge e1 e2 c !g
| [< '(Genlex.Kwd "#") >] -> ()
| [<>] -> () ;;
val parse_line : string graph ref -> Genlex.token Stream.t -> unit = <fun>


The analyzer is used to define the function creating a graph from the description in the file:

# let create_graph name =
let g = ref {ind=0; size=0; nodes =[||]; m = [||]} in
let ic = open_in name in
try
print_string ("Loading "^name^": ");
while true do
print_string ".";
let l = input_line ic in parse_line g (lex_line l)
done;
!g
with End_of_file -> print_newline(); close_in ic; !g ;;
val create_graph : string -> string graph = <fun>
The following command constructs a graph from the file aho.dat.

# let b = create_graph "PROGRAMMES/aho.dat" ;;
Loading PROGRAMMES/aho.dat: ..............
val b : string graph =
{ind=5; size=5; nodes=[|"A"; "B"; "C"; "D"; "E"|];
m=[|[|Nan; Cost 10; Nan; Cost 30; Cost 100|]; ...|]}


Dijkstra's Algorithm

Dijkstra's algorithm finds a least cost path between two nodes. The cost of a path between node n1 and node n2 is the sum of the costs of the edges on that path. The algorithm requires that costs always be positive, so there is no benefit in passing through a node more than once.

Dijkstra's algorithm effectively computes the minimal cost paths of all nodes of the graph which can be reached from a source node n1. The idea is to consider a set containing only nodes of which the least cost path to n1 is already known. This set is enlarged successively, considering nodes which can be accessed directly by an edge from one of the nodes already contained in the set. From these candidates, the one with the best cost path to the source node is added to the set.

To keep track of the state of the computation, the type comp_state is defined, as well as a function for creating an initial state:

# type comp_state = { paths : int array;
already_treated : bool array;
distances : cost array;
source : int;
nn : int};;
# let create_state () = { paths = [||]; already_treated = [||]; distances = [||];
nn = 0; source = 0};;
The field source contains the start node. The field already_treated indicates the nodes whose optimal path from the source is already known. The field nn indicates the total number of the graph's nodes. The vector distances holds the minimal distances between the source and the other nodes. For each node, the vector path contains the preceding node on the least cost path. The path to the source can be reconstructed from each node by using path.

Cost Functions

Four functions on costs are defined: a_cost to test for the existence of an edge, float_of_cost to return the floating point value, add_cost to add two costs and less_cost to check if one cost is smaller than another.


# let a_cost c = match c with Nan -> false | _-> true;;
val a_cost : cost -> bool = <fun>
# let float_of_cost c = match c with
Nan -> failwith "float_of_cost"
| Cost x -> x;;
val float_of_cost : cost -> float = <fun>
# let add_cost c1 c2 = match (c1,c2) with
Cost x, Cost y -> Cost (x+.y)
| Nan, Cost y -> c2
| Cost x, Nan -> c1
| Nan, Nan -> c1;;
val add_cost : cost -> cost -> cost = <fun>
# let less_cost c1 c2 = match (c1,c2) with
Cost x, Cost y -> x < y
| Cost x, Nan -> true
| _, _ -> false;;
val less_cost : cost -> cost -> bool = <fun>
The value Nan plays a special role in the computations and in the comparison. We will come back to this when we have presented the main function (page ??).

Implementing the Algorithm

The search for the next node with known least cost path is divided into two functions. The first, first_not_treated, selects the first node not already contained in the set of nodes with known least cost paths. This node serves as the initial value for the second function, least_not_treated, which returns a node not already in the set with a best cost path to the source. This path will be added to the set.

# exception Found of int;;
exception Found of int
# let first_not_treated cs =
try
for i=0 to cs.nn-1 do
if not cs.already_treated.(i) then raise (Found i)
done;
raise Not_found;
0
with Found i -> i ;;
val first_not_treated : comp_state -> int = <fun>
# let least_not_treated p cs =
let ni = ref p
and nd = ref cs.distances.(p) in
for i=p+1 to cs.nn-1 do
if not cs.already_treated.(i) then
if less_cost cs.distances.(i) !nd then
( nd := cs.distances.(i);
ni := i )
done;
!ni,!nd;;
val least_not_treated : int -> comp_state -> int * cost = <fun>


The function one_round selects a new node, adds it to the set of treated nodes and computes the distances for any next candidates.

# exception No_way;;
exception No_way
# let one_round cs g =
let p = first_not_treated cs in
let np,nc = least_not_treated p cs in
if not(a_cost nc ) then raise No_way
else
begin
cs.already_treated.(np) <- true;
for i = 0 to cs.nn -1 do
if not cs.already_treated.(i) then
if a_cost g.m.(np).(i) then
let ic = add_cost cs.distances.(np) g.m.(np).(i) in
if less_cost ic cs.distances.(i) then (
cs.paths.(i) <- np;
cs.distances.(i) <- ic
)
done;
cs
end;;
val one_round : comp_state -> 'a graph -> comp_state = <fun>


The only thing left in the implementation of Dijkstra's algorithm is to iterate the preceding function. The function dij takes a node and a graph as arguments and returns a value of type comp_state, with the information from which the least cost paths from the source to all the reachable nodes of the graph can be deduced.

# let dij s g =
if belongs_to s g then
begin
let i = index s g in
let cs = { paths = Array.create g.ind (-1) ;
already_treated = Array.create g.ind false;
distances = Array.create g.ind Nan;
nn = g.ind;
source = i} in
cs.already_treated.(i) <- true;
for j=0 to g.ind-1 do
let c = g.m.(i).(j) in
cs.distances.(j) <- c;
if a_cost c then cs.paths.(j) <- i
done;
try
for k = 0 to cs.nn-2 do
ignore(one_round cs g)
done;
cs
with No_way -> cs
end
else failwith "dij: node unknown";;
val dij : 'a -> 'a graph -> comp_state = <fun>


Nan is the initial value of the distances. It represents an infinite distance, which conforms to the comparison function less_cost. In contrast, for the addition of costs (function add_cost), this value is treated as a zero. This allows a simple implementation of the table of distances.

Now the search with Dijkstra's algorithm can be tested.

# let g = test_aho ();;
# let r = dij "A" g;;


The return values are:

# r.paths;;
- : int array = [|0; 0; 3; 0; 2|]
# r.distances;;
- : cost array = [|Cost 0; Cost 10; Cost 50; Cost 30; Cost 60|]


Displaying the Results

To make the results more readable, we now define a display function.

The table paths of the state returned by dij only contains the last edges of the computed paths. In order to get the entire paths, it is necessary to recursively go back to the source.

# let display_state f (g,st) dest =
if belongs_to dest g then
let d = index dest g in
let rec aux is =
if is = st.source then Printf.printf "%a" f g.nodes.(is)
else (
let old = st.paths.(is) in
aux old;
Printf.printf " -> (%4.1f) %a" (float_of_cost g.m.(old).(is))
f g.nodes.(is)
)
in
if not(a_cost st.distances.(d)) then Printf.printf "no way\n"
else (
aux d;
Printf.printf " = %4.1f\n" (float_of_cost st.distances.(d)));;
val display_state :
(out_channel -> 'a -> unit) -> 'a graph * comp_state -> 'a -> unit = <fun>
This recursive function uses the command stack to display the nodes in the right order. Note that the use of the format "a" requires the function parameter f to preserve the polymorphism of the graphs for the display.

The optimal path between the nodes "A" (index 0) and "E" (index 4) is displayed in the following way:

# display_state (fun x y -> Printf.printf "%s!" y) (a,r) "E";;
A! -> (30.0) D! -> (20.0) C! -> (10.0) E! = 60.0
- : unit = ()


The different nodes of the path and the costs of each route are shown.

Introducing a Cache

Dijkstra's algorithm computes all least cost paths starting from a source. The idea of preserving these least cost paths for the next inquiry with the same source suggests itself. However, this storage could occupy a considerable amount of memory. This suggests the use of ``weak pointers.'' If the results of a computation starting from a source are stored in a table of weak pointers, it will be possible for the next computation to check if the computation has already been done. Because the pointers are weak, the memory occupied by the states can be freed by the garbage collector if needed. This avoids interrupting the rest of the program through the allocation of too much memory. In the worst case, the computation has to be repeated for a future inquiry.

Implementing a Cache

A new type 'a comp_graph is defined:

# type 'a comp_graph =
{ g : 'a graph; w : comp_state Weak.t } ;;
The fields g and w correspond to the graph and to the table of weak pointers, pointing to the computation states for each possible source.

Such values are constructed by the function create_comp_graph.

# let create_comp_graph g =
{ g = g;
w = Weak.create g.ind } ;;
val create_comp_graph : 'a graph -> 'a comp_graph = <fun>


The function dij_quick checks to see if the computation has already been done. If it has, the stored result is returned. Otherwise, the computation is executed and the result is registered in the table of weak pointers.

# let dij_quick s cg =
let i = index s cg.g in
match Weak.get cg.w i with
None -> let cs = dij s cg.g in
Weak.set cg.w i (Some cs);
cs
| Some cs -> cs;;
val dij_quick : 'a -> 'a comp_graph -> comp_state = <fun>


The display function still can be used:

# let cg_a = create_comp_graph a in
let r = dij_quick "A" cg_a in
display_state (fun x y -> Printf.printf "%s!" y) (a,r) "E" ;;
A! -> (30.0) D! -> (20.0) C! -> (10.0) E! = 60.0
- : unit = ()


Performance Evaluation

We will test the performance of the functions dij and dij_quick by iterating each one on a random list of sources. In this way an application which frequently computes least cost paths is simulated (for example a railway route planning system).

We define the following function to time the calculations:

# let exe_time f g ss =
let t0 = Sys.time() in
Printf.printf "Start (%5.2f)\n" t0;
List.iter (fun s -> ignore(f s g)) ss;
let t1 = Sys.time() in
Printf.printf "End (%5.2f)\n" t1;
Printf.printf "Duration = (%5.2f)\n" (t1 -. t0) ;;
val exe_time : ('a -> 'b -> 'c) -> 'b -> 'a list -> unit = <fun>


We create a random list of 20000 nodes and measure the performance on the graph a:

# let ss =
let ss0 = ref [] in
let i0 = int_of_char 'A' in
let new_s i = Char.escaped (char_of_int (i0+i)) in
for i=0 to 20000 do ss0 := (new_s (Random.int a.size))::!ss0 done;
!ss0 ;;
val ss : string list =
["A"; "B"; "D"; "A"; "E"; "C"; "B"; "B"; "D"; "E"; "B"; "E"; "C"; "E"; "E";
"D"; "D"; "A"; "E"; ...]
# Printf.printf"Function dij :\n";
exe_time dij a ss ;;
Function dij :
Start ( 1.14)
End ( 1.46)
Duration = ( 0.32)
- : unit = ()
# Printf.printf"Function dij_quick :\n";
exe_time dij_quick (create_comp_graph a) ss ;;
Function dij_quick :
Start ( 1.46)
End ( 1.50)
Duration = ( 0.04)
- : unit = ()


The results confirm our assumption. The direct access to a result held in the cache is considerably faster than a second computation of the result.

A Graphical Interface

We use the Awi library to construct a graphical interface to display graphs. The interface allows selection of the source and destination nodes of the path. When the path is found, it is displayed graphically. We define the type 'a gg, containing fields describing the graph and the computation, as well as fields of the graphical interface.


# #load "PROGRAMMES/awi.cmo";;

# type 'a gg = { mutable src : 'a * Awi.component;
mutable dest : 'a * Awi.component;
pos : (int * int) array;
cg : 'a comp_graph;
mutable state : comp_state;
mutable main : Awi.component;
to_string : 'a -> string;
from_string : string -> 'a } ;;


The fields src and dest are tuples (node, component), associating a node and a component. The field pos contains the position of each component. The field main is the main container of the set of components. The two functions to_string and from_string are conversion functions between type 'a and strings. The elements necessary to construct these values are the graph information, the position table and the conversion functions.


# let create_gg cg vpos ts fs =
{src = cg.g.nodes.(0),Awi.empty_component;
dest = cg.g.nodes.(0),Awi.empty_component;
pos = vpos;
cg = cg;
state = create_state () ;
main = Awi.empty_component;
to_string = ts;
from_string = fs};;
val create_gg :
'a comp_graph ->
(int * int) array -> ('a -> string) -> (string -> 'a) -> 'a gg = <fun>


Visualisation

In order to display the graph, the nodes have to be drawn, and the edges have to be traced. The nodes are represented by button components of the Awi library. The edges are traced directly in the main window. The function display_edge displays the edges. The function display_shortest_path displays the found path in a different color.

Drawing Edges
An edge connects two nodes and has an associated weight. The connection between two nodes can be represented by a line. The main difficulty is indicating the orientation of the line. We choose to represent it by an arrow. The arrow is rotated by the angle the line has with the abscissa (the x-axis) to give it the proper orientation. Finally, the costs are displayed beside the edge.

To draw the arrow of an edge we define the functions rotate and translate which care respectively for rotation and shifting. The function display_arrow draws the arrow.

# let rotate l a =
let ca = cos a and sa = sin a in
List.map (function (x,y) -> ( x*.ca +. -.y*.sa, x*.sa +. y*.ca)) l;;
val rotate : (float * float) list -> float -> (float * float) list = <fun>
# let translate l (tx,ty) =
List.map (function (x,y) -> (x +. tx, y +. ty)) l;;
val translate :
(float * float) list -> float * float -> (float * float) list = <fun>
# let display_arrow (mx,my) a =
let triangle = [(5.,0.); (-3.,3.); (1.,0.); (-3.,-3.); (5.,0.)] in
let tr = rotate triangle a in
let ttr = translate tr (mx,my) in
let tt = List.map (function (x,y) -> (int_of_float x, int_of_float y)) ttr
in
Graphics.fill_poly (Array.of_list tt);;
val display_arrow : float * float -> float -> unit = <fun>


The position of the text indicating the weight of an edge depends on the angle of the edge.

# let display_label (mx,my) a lab =
let (sx,sy) = Graphics.text_size lab in
let pos = [ float(-sx/2),float(-sy) ] in
let pr = rotate pos a in
let pt = translate pr (mx,my) in
let px,py = List.hd pt in
let ox,oy = Graphics.current_point () in
Graphics.moveto ((int_of_float mx)-sx-6)
((int_of_float my) );
Graphics.draw_string lab;
Graphics.moveto ox oy;;
val display_label : float * float -> float -> string -> unit = <fun>


The preceding functions are now used by the function display_edge. Parameters are the graphical interface gg, the nodes i and j, and the color (col) to use.

# let display_edge gg col i j =
let g = gg.cg.g in
let x,y = gg.main.Awi.x,gg.main.Awi.y in
if a_cost g.m.(i).(j) then (
let (a1,b1) = gg.pos.(i)
and (a2,b2) = gg.pos.(j) in
let x0,y0 = x+a1,y+b1 and x1,y1 = x+a2,y+b2 in
let rxm = (float(x1-x0)) /. 2. and rym = (float(y1-y0)) /. 2. in
let xm = (float x0) +. rxm and ym = (float y0) +. rym in
Graphics.set_color col;
Graphics.moveto x0 y0;
Graphics.lineto x1 y1;
let a = atan2 rym rxm in
display_arrow (xm,ym) a;
display_label (xm,ym) a
(string_of_float(float_of_cost g.m.(i).(j))));;
val display_edge : 'a gg -> Graphics.color -> int -> int -> unit = <fun>


Displaying a Path
To display a path, all edges along the path are displayed. The graphical display of a path towards a destination uses the same technique as the textual display.

# let rec display_shortest_path gg col dest =
let g = gg.cg.g in
if belongs_to dest g then
let d = index dest g in
let rec aux is =
if is = gg.state.source then ()
else (
let old = gg.state.paths.(is) in
display_edge gg col old is;
aux old )
in
if not(a_cost gg.state.distances.(d)) then Printf.printf "no way\n"
else aux d;;
val display_shortest_path : 'a gg -> Graphics.color -> 'a -> unit = <fun>


Displaying a Graph
The function display_gg displays a complete graph. If the destination node is not empty, the path between the source and the destination is traced.

# let display_gg gg () =
Awi.display_rect gg.main ();
for i=0 to gg.cg.g.ind -1 do
for j=0 to gg.cg.g.ind -1 do
if i<> j then display_edge gg (Graphics.black) i j
done
done;
if snd gg.dest != Awi.empty_component then
display_shortest_path gg Graphics.red (fst gg.dest);;
val display_gg : 'a gg -> unit -> unit = <fun>


The Node Component

The nodes still need to be drawn. Since the user is allowed to choose the source and destination nodes, we define a component for nodes.

The user's main action is choosing the end nodes of the path to be found. Thus a node must be a component that reacts to mouse clicks, using its state to indicate if it has been chosen as a source or destination. We choose the button component, which reacts to mouse clicks.

Node Actions
It is necessary to indicate node selection. To show this, the background color of a node is changed by the function inverse.

# let inverse b =
let gc = Awi.get_gc b in
let fcol = Awi.get_gc_fcol gc
and bcol = Awi.get_gc_bcol gc in
Awi.set_gc_bcol gc fcol;
Awi.set_gc_fcol gc bcol;;
val inverse : Awi.component -> unit = <fun>


The function action_click effects this selection. It is called when a node is clicked on by the mouse. As parameters it takes the node associated with the button and the graph to modify the source or the destination of the search. When both nodes are selected, the function dij_quick finds a least cost path.
            
# let action_click node gg b bs =
let (s1,s) = gg.src
and (s2,d) = gg.dest in
if s == Awi.empty_component then (
gg.src <- (node,b); inverse b )
else
if d == Awi.empty_component then (
inverse b;
gg.dest <- (node,b);
gg.state <- dij_quick s1 gg.cg;
display_shortest_path gg (Graphics.red) node
)
else (inverse s; inverse d;
gg.dest <- (s2,Awi.empty_component);
gg.src <- node,b; inverse b);;
val action_click : 'a -> 'a gg -> Awi.component -> 'b -> unit = <fun>


Creating an Interface
The main function to create an interface takes an interface graph and a list of options, creates the different components and associates them with the graph. The parameters are the graph (gg), its dimensions (gw and gh), a list of graph and node options (lopt) and a list of node border options (lopt2).
  
# let main_gg gg gw gh lopt lopt2 =
let gc = Awi.make_default_context () in
Awi.set_gc gc lopt;
(* compute the maximal button size *)
let vs = Array.map gg.to_string gg.cg.g.nodes in
let vsize = Array.map Graphics.text_size vs in
let w = Array.fold_right (fun (x,y) -> max x) vsize 0
and h = Array.fold_right (fun (x,y) -> max y) vsize 0 in
(* create the main panel *)
gg.main <- Awi.create_panel true gw gh lopt;
gg.main.Awi.display <- display_gg gg;
(* create the buttons *)
let vb_bs =
Array.map (fun x -> x,Awi.create_button (" "^(gg.to_string x)^" ")
lopt)
gg.cg.g.nodes in
let f_act_b = Array.map (fun (x,(b,bs)) ->
let ac = action_click x gg b
in Awi.set_bs_action bs ac) vb_bs in
let bb =
Array.map (function (_,(b,_)) -> Awi.create_border b lopt2) vb_bs
in
Array.iteri
(fun i (b) -> let x,y = gg.pos.(i) in
Awi.add_component gg.main b
["PosX",Awi.Iopt (x-w/2);
"PosY", Awi.Iopt (y-h/2)]) bb;
();;
val main_gg :
'a gg ->
int ->
int -> (string * Awi.opt_val) list -> (string * Awi.opt_val) list -> unit =
<fun>
The buttons are created automatically. They are positioned on the main window.

Testing the Interface
Everything is ready to create an interface now. We use a graph whose nodes are character strings to simplify the conversion functions. We construct the graph gg as follows:

# let id x = x;;
# let pos = [| 200, 300; 80, 200 ; 100, 100; 200, 100; 260, 200 |];;
# let gg = create_gg (create_comp_graph (test_aho())) pos id id;;
# main_gg gg 400 400 ["Background", Awi.Copt (Graphics.rgb 130 130 130);
"Foreground",Awi.Copt Graphics.green]
[ "Relief", Awi.Sopt "Top";"Border_size", Awi.Iopt 2];;


Calling Awi.loop true false gg.main;; starts the interaction loop of the Awi library.



Figure 13.9: Selecting the nodes for a search


Figure 13.9 shows the computed path between the nodes "A" and "E". The edges on the path have changed their color.

Creating a Standalone Application

We will now show the steps needed to construct a standalone application. The application takes the name of a file describing the graph as an argument. For standalone applications, it is not necesary to have an Objective CAML distribution on the execution machine.

A Graph Description File

The file containes information about the graph as well as information used for the graphical interface. For the latter information, we define a second format. From this graphical description, we construct a value of the type g_info.

# type g_info = {npos : (int * int) array;
mutable opt : Awi.lopt;
mutable g_w : int;
mutable g_h : int};;


The format for the graphical information is described by the four key words of list key2.

# let key2 = ["HEIGHT"; "LENGTH"; "POSITION"; "COLOR"];;
val key2 : string list = ["HEIGHT"; "LENGTH"; "POSITION"; "COLOR"]
# let lex2 l = Genlex.make_lexer key2 (Stream.of_string l);;
val lex2 : string -> Genlex.token Stream.t = <fun>

# let pars2 g gi s = match s with parser
[< '(Genlex.Kwd "HEIGHT"); '(Genlex.Int i) >] -> gi.g_h <- i
| [< '(Genlex.Kwd "LENGTH"); '(Genlex.Int i) >] -> gi.g_w <- i
| [< '(Genlex.Kwd "POSITION"); '(Genlex.Ident s);
'(Genlex.Int i); '(Genlex.Int j) >] -> gi.npos.(index s g) <- (i,j)
| [< '(Genlex.Kwd "COLOR"); '(Genlex.Ident s);
'(Genlex.Int r); '(Genlex.Int g); '(Genlex.Int b) >] ->
gi.opt <- (s, Awi.Copt (Graphics.rgb r g b))::gi.opt
| [<>] -> ();;
val pars2 : string graph -> g_info -> Genlex.token Stream.t -> unit = <fun>


Creating the Application

The function create_graph takes the name of a file as input and returns a couple composed of a graph and associated graphical information.

# let create_gg_graph name =
let g = create_graph name in
let gi = {npos = Array.create g.size (0,0); opt=[]; g_w =0; g_h = 0;} in
let ic = open_in name in
try
print_string ("Loading (pass 2) " ^name ^" : ");
while true do
print_string ".";
let l = input_line ic in pars2 g gi (lex2 l)
done ;
g,gi
with End_of_file -> print_newline(); close_in ic; g,gi;;
val create_gg_graph : string -> string graph * g_info = <fun>


The function create_app constructs the interface of a graph.

# let create_app name =
let g,gi = create_gg_graph name in
let size = (string_of_int gi.g_w) ^ "x" ^ (string_of_int gi.g_h) in
Graphics.open_graph (" "^size);
let gg = create_gg (create_comp_graph g) gi.npos id id in
main_gg gg gi.g_w gi.g_h
[ "Background", Awi.Copt (Graphics.rgb 130 130 130) ;
"Foreground", Awi.Copt Graphics.green ]
[ "Relief", Awi.Sopt "Top" ; "Border_size", Awi.Iopt 2 ] ;
gg;;
val create_app : string -> string gg = <fun>


Finally, the function main takes the name of the file from the command line, constructs a graph with an interface and starts the interaction loop on the main component of the graph interface.

# let main () =
if (Array.length Sys.argv ) <> 2
then Printf.printf "Usage: dij.exe filename\n"
else
let gg = create_app Sys.argv.(1) in
Awi.loop true false gg.main;;
val main : unit -> unit = <fun>


The last expression of that program starts the function main.

The Executable

The motivation for making a standalone application is to support its distribution. We collect the types and functions described in this section in the file dij.ml. Then we compile the file, adding the different libraries which are used. Here is the command to compile it under Linux.
ocamlc -custom -o dij.exe graphics.cma awi.cmo graphs.ml \
    -cclib -lgraphics -cclib -L/usr/X11/lib -cclib -lX11
Compiling standalone applications using the Graphics library is described in chapters 5 and 7.

Final Notes

The skeleton of this application is sufficiently general to be used in contexts other than the search for traveling paths. Different types of problems can be represented by a weighted graph. For example the search for a path in a labyrinth can be coded in a graph where each intersection is a node. Finding a solution corresponds to computing the shortest path between the start and the goal.

To compare the performance betwen C and Objective CAML, we wrote Dijkstra's algorithm in C. The C program uses the Objective CAML data structures to perform the calculations.

To improve the graphical interface, we add a textfield for the name of the file and two buttons to load and to store a graph. The user may then modify the positions of the nodes by mouse to improve the appearance.

A second improvement of the graphical interface is the ability to choose the form of the nodes. To display a button, a function tracing a rectangle is called. The display functions can be specialized to use polygons for nodes.






Previous Contents Next