Cooperative Learning in Neural Networks Using Particle Swarm Optimizers

Download as pdf or txt
Download as pdf or txt
You are on page 1of 8

Cooperative Learning in Neural Networks using Particle Swarm Optimizers

a F.
a Department

van den Bergh, a A.P. Engelbrecht

of Computer Science, University of Pretoria, South Africa, [email protected], [email protected]

Abstract
This paper presents a method to employ particle swarms optimizers in a cooperative conguration. This is achieved by splitting the input vector into several sub-vectors, each which is optimized cooperatively in its own swarm. The application of this technique to neural network training is investigated, with promising results. Keywords: Particle swarms, cooperative learning, optimization Computing Review Categories: G.1.6, I.2.6

1 Introduction
Particle Swarm Optimizers (PSOs) have previously been used to train neural networks[6, 10] and generally met with success. The advantage of the PSO over many of the other optimization algorithms is its relative simplicity. This paper aims to improve the performance of the basic PSO by partitioning the input vector into several subvectors. Each sub-vector is then allocated its own swarm. In Section 2, a brief overview of PSOs is presented, followed by a discussion of how cooperative behavior can be implemented through a splitting technique in Section 3. Implementations, applied to both neural network training and function minimization, are described in Section 4, followed by results obtained from various case studies in Section 5. This paper is concluded by some ideas that can serve as future research directions.

worse performance on others. Kennedy [7] used cluster analysis to modify the update equations so that particles attempt to conform to the center of their clusters rather than attempting to conform to a global best. This approach resulted in improved performance on some problems, but slightly worse performance on others.

2.1 Inside the Particle Swarm


A swarm consists of several particles, where each particle keeps track of its own attributes. The most important attribute of each particle is its current position, as given by an n-dimensional vector, corresponding to a potential solution of the function to be minimized. Along with the position the particle has a current velocity, keeping track of the speed and direction in which the particle is currently traveling. Each particle also has a current tness value (analogous to population members in a GA), which is obtained by evaluating the error function at the particles current position. For a neural network implementation, this value corresponds to a forward propagation through the network, assuming the position vector corresponds to the weight vector of the network. Additionally, each particle also remembers its own personal best position (yielding the lowest error) so that potentially good solutions can be used to guide the construction of new solutions. At swarm level, the best overall position amongst all particles is also recorded; upon termination of the algorithm this will serve as the answer. During each epoch every particle is accelerated towards its own personal best as well as in the direction of the global best position. This is achieved by calculating a new velocity term for each particle based on the distance from its personal best, as well as its distance from the global best position. These two components (personal and global velocities) are then randomly weighted to produce the new velocity value for this particle, which will in turn affect the next position of the particle during the next epoch.

2 Particle Swarms
Particle swarms, as rst described by Eberhart and Kennedy [3, 5], are best classied under the heading of evolutionary computing. The basic operational principle of the particle swarm is reminiscent of the behaviour of a ock of birds or the sociological behaviour of a group of people. Various attempts have been made to improve the performance of the baseline Particle Swarm Optimizer. Work done by Eberhart and Shi [9, 4] focus on optimizing the update equations for the particles. Their improvements generally leads to better performance on a very large class of problems. These improvements can be (and have been) applied to the technique presented here. Angeline [1] used a selection mechanism (as used in genetic algorithms) in an attempt to improve the general quality of the particles in a swarm. This approach lead to improved performance on some problems, but somewhat

2.2 Particle Swarm Parameters


A number of factors will affect the performance of the PSO. Firstly, the number of particles in the swarm affects the run-time signicantly, thus a balance between variety (more particles) and speed (fewer particles) must be sought. Another important factor in the convergence speed of the algorithm is the maximum velocity parameter. This parameter limits the maximum jump that a particle can make in one step, thus a too large value for this parameter will result in oscillation. A small value, on the other hand, can cause the particle to become trapped in local minima. Current research indicates that using a constriction coefcient leads to improved performance [4]. For the implementation used in this paper, the maximum velocity parameter is set to the default value of 2 for neural network training sessions (the value 2 was suggested in the earlier works in PSOs [5]), but increased appropriately for the function minimization tests (typically half the range of the function [4]). To aid the convergence of the algorithm, an inertia term (introduced in [9]) is used to attenuate the magnitude of the velocity updates over time. The attenuation factor is a linear function of the current epoch number.

In a cooperative learning framework, several populations will be considered simultaneously [8]. The solution vector is then split amongst the different populations according to some rule; the simplest of schemes does not allow any overlap between the space covered by different populations. Thus, to nd a solution to the original problem, representatives from all the populations are combined to form the potential solution vector, which is passed on to the error function. A new dimension is now added to the survival game: cooperation between different populations. The same concept can easily be applied to PSOs. Theres already some form of cooperation between the elements in the swarm: they all share the global best position. By using multiple swarms and splitting the vector across these swarms inter-swarm cooperative behavior emerges. Two new problems are introduced with this approach, however: Selection: The solution vector is split into N parts, each part being optimized by a swarm with M particles, leaving N M possible vectors to choose from. The simplest approach is to select the best particle from each swarm (how to calculate which particle is best will be discussed later). Note that this might not be the optimal choice, it is analogous to pairing highly t members of a GA population only with other highly t members, which is known to have a detrimental effect on training performance. Credit assignment: This is analogous to the real-world problem of handing out group projects to students: Who gets the credit for the nal product? In terms of swarms, how much credit should each swarm be awarded when the combined vector (built from all the swarms) results in a better solution? A simple solution is to give all swarms an equal amount of credit. The next section will present an argument to help explain why splitting the vector results in improved performance.

2.3 Swarm Behavior


Each particle will y in the direction of a better solution, weighted by some random factor, possibly causing it to overshoot its target, possibly nding a new personal or even global minimum. The interaction of the particles in the swarm creates a very good balance between straying off-course and staying close to the optimal solution. This type of behavior seems to be ideal when exploring large error surfaces, especially with a relatively large maximum velocity parameter. Some particles can explore far beyond the current minimum, while the population still remembers the global best solution. This solves one of the problems of gradient-based optimization techniques, namely their poor performance in regions with very at gradients. It also appears that the particle swarm optimizer performs well in regions where the error surface is highly perturbed, as is the case with product unit networks. It is important to note that the PSO cannot guarantee a good solution, partly due to its stochastic nature. When compared to a second order optimization algorithm, e.g. scaled conjugate gradient descent, the PSO will typically have both the lower minimum error and a larger maximum error, when a number of experiments are performed.

3.1 Two Steps Forward, One Step Back


Consider an n-dimensional vector v, constrained to the error surface by the function to be minimized. Somewhere in this vector three components, va vb vc , 1 a b c n, will now be examined. Assume that the optimal solution to the problem requires that all three components must have a value of say 20. Now consider a particle swarm containing, in particular, two vectors v j and vk . Lets assume that v j is currently the global best solution in the swarm, so that vk will be drawn towards it in the next epoch. Assume that the three components v j a v j b v j c have the values (17, 2, 17), respectively. The equivalent components in vector vk , namely vk a vk b vk c , have the values (5, 20, 5), respectively. In the next epoch, the vector vk might be updated so that it now contains the sub-vector (15, 5, 15) at positions vk a vk b vk c . If it is assumed that the function at the new vk yields a smaller error value, this will be considered 2

3 Cooperative Learning
The driving force in a population of solutions in a genetic algorithm is the competition between the different members of the population. During crossover, two solutions cooperate to produce new solutions, but this serves only to create new members who will compete for the top spot.

an improvement. However, the valuable information contained in the middle component (vk b ) has been lost, as the optimal solution requires a sub-vector of (20,20,20). The reason for this behavior is that the error function is computed only after all the components in the vector have been updated to their new values. This means an improvement in two components (two steps forward) will overrule a potentially good value for a single component (one step back). One way to overcome this behavior is to evaluate the error function more frequently, for example, once for every time a component in the vector has been updated. A problem still remains with this approach: evaluation of the error function is only possible using a complete n-dimensional vector. Thus, after updating component vk b , n 1 values for the other components of the vector still have to be chosen. A method for doing just this is presented below.

above), and the one chosen for the implementation used to obtain the results presented here. This algorithm optimizes each variable separately within each epoch, but cooperates with other variables between epochs.

4 Implementation
The optimization method described in Section 3 involves a trade-off: increasing the number of swarms leads to a directly proportional increase in the number of error function evaluations, as one function evaluation is required for each particle in each swarm during each epoch. The total number of error function evaluations can thus be summarized as: Nf Ns N p E (1)

3.2 N Steps Forward


At the one extreme a single swarm consisted of a number of n-dimensional vectors. The other extreme would thus be to have n different swarms, each representing a single component of an n-dimensional vector. As mentioned above, the error function can only be evaluated using an n-dimensional vector, so representatives from each swarm must be combined to form such a vector so that the function can be evaluated. The simplest way to construct such a vector is to use the best particle from each swarm for a particular component, as outlined in Figure 1. Create and initialize n 1-dimensional PSOs S1 Sn . Repeat: Select best particle b1 bn from each of S1 Sn resp. For k 1 M i 1 n : Select p = kth particle from swarm Si Construct a vector v b1 b2 p bn E v = error function at v Set tness of particle k in swarm Si to E v Update best tness of b1 bn if necessary Perform normal PSO updates on S1 Sn Until stopping criterion is met Figure 1: Pseudo code for split swarm approach, assuming M particles per swarm Note that the algorithm presented in Figure 1 now keeps track of the optimal value for each component of the n-dimensional vector in a separate swarm. This implies that the error with respect to each component will be a non-increasing function of the epoch number, so a good value for a specic component is never forgotten. Care must be taken to keep track of the global minimum error. Whenever a combined vector (v) with a lower error is found, the tness values of the best particle in all the swarms are updated to this value, as all these particles participated in the creation of this improved vector. This is one solution to the credit assignment problem (mentioned 3

where N f is the number of error function evaluations, Ns the number of swarms used, N p the number of particles per swarm and E the number of epochs allowed for training. An extreme example illustrates this point: Consider that a single swarm using n-dimensional vectors is allowed to train for 2n epochs. To keep the number of error function evaluations constant (thus the execution time), a system using n 1-dimensional swarms will only be allowed to train for 2 epochs. Clearly the swarm optimizer will require more than 2 epochs to nd a good solution (assuming that the single-swarm set-up was able to nd a good solution in 2n epochs). One way to increase the number of allowed epochs is to reduce the splitting factor. In the above example, splitting the n-dimensional vector across only two swarms will allow n epochs for optimization, which is more likely to succeed than only two epochs!

4.1 Neural Network Architectures


The neural network is trained by minimizing an error function in W D 1 M M 1 C-dimensional space (See Figure 2). PSOs generate possible solutions in this W dimensional space using a forward propagation throught the neural network to obtain the value of the error function for each particle in the swarm. This error value is used directly as the particles tness, so that constructing a particle with a lower error value is synonymous with learning in the network. A forward propagation through the network is a computationally expensive task, so the aim is to nd the best possible solution using a limited number of forward propagations through the network. To test the effectiveness of cooperative swarm congurations, a number of tests were performed on 2-layer feedforward neural networks. A network as depicted in Figure 2 was used, with D input units, M sigmoidal hidden units and C linear output units. The following architectures were selected for testing:

for E epochs, the Nsplit architecture was allowed only E M C epochs.

X1

Z1

Y1

4.2 Function Minimization Architectures


The following two architectures were used to test the vector splitting approach in a function minimization environment: 4.2.1 Plain

XD

ZM

YC

Figure 2: Neural network architecture 4.1.1 Plain The plain architecture takes the weights from the different layers and serializes them into a D 1 M M 1 C-dimensional vector, which is optimized using a single particle swarm. This architecture is used as the baseline for the benchmarks. 4.1.2 Lsplit The Lsplit, or Layer-split architecture splits the network weights between two swarms. The rst swarm contains the D 1 M weights from the rst layer, while the second swarm optimizes the M 1 C second-layer weights. This architecture was chosen on the assumption that the relative scales of the weights in these two layers differ. The number of allowed forward propagations, N f , is kept constant by halving E, the number of allowed epochs during training. 4.1.3 Esplit The Esplit, or Even-split architecture takes the serialized D 1 M M 1 C-dimensional vector (as used in the plain architecture) and splits it in half. The assumption here was that splitting the vector into a 90%/10% conguration (as sometimes happens with Lsplit) does not leave the 90% part with sufcient freedom to maneuver, thus an even split would result in the optimal 2-way split. The number of epochs are halved again to keep the number of error function evaluations constant. 4.1.4 Nsplit The Nsplit architecture attempts to build on the Lsplit approach by splitting the weights according to function. For each hidden and output unit, a swarm is created containing all the weights entering that unit. This results in M D 1-dimensional vectors plus C M 1-dimensional vectors, for a total of M C swarms. By applying (1) it becomes clear that a large number of hidden or output units would put this architecture to a severe disadvantage, as the number of allowed epochs will be signicantly reduced. Thus, if the plain architecture was allowed to train 4

As was the case with the neural network experiments, the plain architecture makes use of only a single swarm consisting of n-dimensional vectors. This again serves as the baseline for comparison. 4.2.2 Dsplit The Dsplit architecture is the extreme opposite of the plain architecture, splitting the n-dimensional vector across n 1dimensional swarms. It is called Dsplit (in lieu of Nsplit) to avoid confusion with the denition of Nsplit given above.

5 Case Studies
5.1 Neural Network Experiments
The data sets used in this section were obtained from the UCI machine learning repository [2]. All experiments were performed using 500 runs per architecture. The tables appearing in this section show the classication error, expressed as a percentage, followed by the 95% condence interval width. Each table lists the total error (train + test) as well as the training error. Note that the Training error is the more signicant of the two, as no attempt was made to prevent over-tting. 5.1.1 Iris The Iris classication problem consists of 150 patterns falling into three classes. This is considered to be a simple classication example, as the three classes are (almost) linearly separable. A 4-input, 3-hidden and 3-output network architecture was used. From Table 1 it is clear that the 2-way split architectures (Lsplit and Esplit) perform signicantly better than the plain architecture. Early on during training the difference is in the order of 267%, settling to 241% later on. Figure 3 presents the training errors graphically. It is interesting to note that the slope (rate of improvement caused by more iterations) of the split architectures seem to be less steep than the plain approach. The Nsplit architecture is leading by a small amount. Table 2 shows the results of a convergence test. Here, a training session is said to converge when it fails to improve for 200 consecutive epochs. The second column in this table lists the average number of error function evaluations (forward propagations in the network) used during training, until the network converged. The 2-way split

Type plain Lsplit Esplit Nsplit plain Lsplit Esplit Nsplit

Epochs 500 250 250 83 1000 500 500 167

Error (Total) 6.00 2.81 2.59 2.39 3.99 2.02 2.07 1.88

95% dev. 0.43 0.23 0.25 0.24 0.34 0.17 0.23 0.23

Error (Train) 5.52 2.32 2.06 1.73 3.69 1.55 1.53 1.49

95% dev. 0.41 0.23 0.24 0.22 0.32 0.17 0.22 0.23

Table 1: Iris plain vs split architectures Type plain Lsplit Esplit Nsplit Fwd. Props. 27468 64165 62573 300000 Error (Total) 5.03 1.24 1.10 1.24 95% dev. 0.56 0.16 0.07 0.08 Error (Train) 4.50 0.72 0.59 0.64 95% dev. 0.57 0.16 0.06 0.07

Table 2: Iris convergence tests


6 5.5 5 4.5 classification error (%) 4 3.5 3 2.5 2 1.5 1 0.5 4 5 6 7 8 Forward propagations (k) 9 10 11 classification error (%) plain Lsplit Esplit Nsplit 0.7 0.65 0.6 0.55 0.5 0.45 0.4 0.35 0.3 0.25 0.2 8 10 12 14 16 Forward propagations (k) 18 20 22 plain Lsplit Esplit Nsplit

Figure 3: Iris classication error (train) architectures used roughly twice as many function evaluations as the plain method, but much lower classication errors result. The Nsplit architecture completely failed to converge, which probably means that the threshold used in the convergence test is too sensitive. Besides the failure to converge, the error after 300000 epochs still exceeds the Esplit approach. 5.1.2 Ionosphere The ionosphere problem has a much higher input dimension than the Iris problem, but only two output classes. This results in a 34-input, 5-hidden and 2-output network architecture. Due to the high input dimension (versus the output dimension), the Lsplit architecture becomes highly unbalanced at a 175:12 ratio, or a 94%/6% split. Having to train a 175-dimensional vector using 500 epochs (Lsplit) vs 187-dimensional vector using 1000 epochs (plain) explains why the Lsplit architecture performs worse than the plain method. The Esplit method regains this ground by (narrowly) beating the plain architecture. Figure 4 graphi5

Figure 4: Iono classication error (train) cally illustrates this fact. The Nsplit architecture falls somewhere between the Lsplit and the plain architectures. This is probably caused by the higher interdependancy between the variables, i.e. some weights in the network may be correlated.

5.1.3 Glass The glass problem is difcult to solve, possibly because of its severely skewed class distribution. An 8-input, 6hidden, 6-output unit network was used to perform the experiments. Table 4 shows that the Esplit method again outperformed all the other architectures. The Lsplit architecture is relatively well balanced, at a 54:42 ratio, so that it performs similarly to its sibling, Esplit. Nsplit seems to perform well, beating the other architectures at low epoch counts and drawing even at high counts. Figure 5 shows how dramatically the plain network improved by going from 500 epochs to 1000 epochs.

Type plain Lsplit Esplit Nsplit plain Lsplit Esplit Nsplit

Epochs 1000 500 500 143 2000 1000 1000 286

Error (Total) 0.91 1.17 1.09 1.07 0.86 1.06 1.01 1.03

95% dev. 0.05 0.08 0.06 0.06 0.04 0.06 0.06 0.08

Error (Train) 0.42 0.63 0.39 0.48 0.32 0.42 0.29 0.35

95% dev. 0.04 0.07 0.04 0.04 0.03 0.04 0.04 0.04

Table 3: Iono plain vs split architectures Type plain Lsplit Esplit Nsplit plain Lsplit Esplit Nsplit plain Lsplit Esplit Nsplit Epochs 500 250 250 42 1000 500 500 83 2000 1000 1000 167 Error (Total) 13.79 9.87 9.41 8.57 11.08 9.33 8.90 8.94 9.83 8.87 8.74 8.70 95% dev. 0.74 0.39 0.36 0.35 0.53 0.26 0.25 0.22 0.36 0.21 0.23 0.24 Error (Train) 13.78 9.86 9.40 8.58 11.08 9.31 8.90 8.93 9.83 8.87 8.72 8.72 95% dev. 0.74 0.39 0.36 0.35 0.53 0.26 0.25 0.23 0.36 0.21 0.23 0.24

Table 4: Glass plain vs split architectures Type plain Dsplit plain Dsplit Epochs 500 25 1000 50 Error (Total) 8.46e+00 5.73e-05 7.43e+00 1.75e-10 95% dev.

2.26e-01 4.23e-06 2.02e-01 3.32e-11

Table 5: Rastrigin function plain vs split architectures


15 plain Lsplit Esplit Nsplit

5.2 Function Minimization Experiments


By including two pure function minimization problems it is possible to investigate the properties of split swarms in more detail, as more information is available on the nature of the error function as opposed to the error function of a neural network classication problem. 5.2.1 Rastrigin Function

14

13 classification error (%)

12

11

10

7 4 6 8 10 12 14 Forward propagations (k) 16 18 20 22

The Rastrigin function is quite easy to optimize, as it is basically an n-dimensional parabola with some cosine bumps on its surface. Formally, it is given by f x Figure 5: Glass classication error (train) 3 0n x2 3 0 cos2xi i
i 1 n

(2)

where n 20 and 5 12 xi 5 12, with its global minimum at 0. Since there are no cross-component products, it is possible to minimize this function component-wise. As can be seen in Table 5, the Dsplit architecture is decidedly superior, yielding errors that are several orders of magnitude smaller than the plain architecture. Another 6

Type plain Dsplit plain Dsplit

Epochs 2500 250 5000 500

Error (Total) 9.06e-02 4.68e-02 8.08e-02 4.43e-02

2.44e-03 1.15e-03 3.17e-03 1.23e-03

95% dev.

Table 6: Griewangk function plain vs split architectures benet of the Dsplit approach is the reduced execution time. Note that although the number of error function evaluations were the same for both approaches, the overhead between error function evaluations is reduced in the Dsplit architecture as it deals with smaller vectors. This results in an effective speed-up factor of 2.5 for the same number of error function evaluations. 5.2.2 Griewangk Function The Griewangk function cannot easily be minimized component-wise, as it contains a product term: f x 1
i n n x2 i cos i 1 1 4000

lems tested so far. One possible explanation for this phenomenon is that the splitting leads to ner-grained credit assignment, reducing the chance of neglecting a possibly good solution for a specic component of the vector. Although the Esplit method ousted the plain architecture on all the neural network tests, it was not convincing in its victory on the Ionosphere problem. This is most likely due to a high degree of interdependency between the weights in the network, as the function minimization experiments shows that interdependent variables tend to reduce the effectiveness of the split approach.

7 Future Work
It has been observed in the results above that the split architectures are sensitive to the degree of inter-dependency between the variables, compared to the plain architecture. By using both an unsplit and a group of split swarms in conjunction, it might be possible to gain the best of both worlds. The split swarm would continue as usual, but during each epoch the vector formed by combining the best of all the split swarms could be injected into the unsplit swarm, serving as a new attractor especially if it has a lower error value than the previous best particle in the unsplit swarm. Another direction might be to nd the critical splitting factor (the number of disjoint swarms) where the reduction in the allowed number of epochs (for a xed error function evaluation budget) outweighs the benet of having more independent sub-vectors. It would be interesting to see if there is a universal constant governing this behavior. Lastly, different methods of addressing the credit assignment or the selection problems will be investigated.

x i

(3)

where n 10 and 600 xi 600, with a global minimum at 0. Table 6 shows that the Griewangk function, having more inter-dependencies between its component variables, benets less from the Dsplit architecture, although the reduction is still by a factor of two. This example serves to characterize the split-swarm approach to optimization: Problems that have a low interdependency between vector components lend themselves well to split-swarm techniques, while problems with high inter-dependency tend to show very little gain.

6 Conclusion
The improved performance can be cast into the framework of a sociologial phenomenon as follows: Let us assume that the societies under consideration require the skills of a patent lawyer. In society A it is desirable to be an engineer, thus all students strive to become engineers. Unfortunately, this society also requires other skills for other professions, e.g. lawyers or artists. Since all students are aiming at engineering, the do not develop their artistic or law skills, thus they will never develop to be successful patent lawyers. Society B might consist of several homogeneous factions, i.e. engineers, computer scientists, lawyers, artists, etc. A new student can now learn skills from more than one faction, e.g. law and engineering, and end up becoming a patent lawyer. Therefore society B tends to develop a solution to the problem more quickly by using the best specimens (best engineer, best lawyer) to train the new student, whereas society A only has good engineers to train their students. Splitting a vector to be optimized across multiple swarms, be it for neural network training or function minimization, appears to improve performance on all the prob7

References
[1] P.J. Angeline. Using Selection to Improve Particle Swarm Optimization. In Proceedings of IJCNN99, pages 8489, Washington, USA, July 1999. [2] C. Blake, E. Keogh, and C.J. Merz. UCI repository of machine learning databases, 1998. www.ics.uci.edu/ mlearn/MLRepository.html. [3] Russ C. Eberhart and J. Kennedy. A new optimizer using particle swarm theory. In Proc. Sixth Intl. Symposium on Micro Machine and Human Science, pages 3943, Nagoya, Japan, 1995. IEEE Service Center. [4] Russ C. Eberhart and Y. Shi. Comparing inertia weights and constriction factors in particle swarm optimization. In Proceedings of the 2000 Congress on Evolutionary Computing, pages 8489, 2000. [5] Russ C. Eberhart, Pat Simpson, and Roy Dobbins. Computational Intelligence PC Tools, chapter 6, pages 212226. AP Professional, 1996.

[6] AP Engelbrecht and A Ismail. Training product unit neural networks. Stability and Control: Theory and Applications, 2(12):5974, 1999. [7] J. Kennedy. Stereotyping: Improving Particle Swarm Performance With Cluster Analysis. In Proceedings of the 2000 Congress on Evolutionary Computing, pages 15071512, 2000. [8] Mitchell A. Potter and Kenneth A. De Jong. A Cooperative Coevolutionary Approach to Function Optimization. In The Third Parallel Problem Solving from Nature, pages 249257, Jerusalem, Israel, 1994. Springer-Verlag. [9] Y. Shi and Russ C. Eberhart. A Modied Particle Swarm Optimizer. In Proceedings of IJCNN99, pages 6973, Washington, USA, July 1999. [10] F. van den Bergh. Particle Swarm Weight Initialization in Multi-layer Perceptron Articial Neural Networks. In Development and Practice of Articial Intelligence Techniques, pages 4145, Durban, South Africa, September 1999.

You might also like