diff --git a/Comparison/src/main/java/cz/fidentis/analyst/visitors/face/HausdorffDistancePrioritized.java b/Comparison/src/main/java/cz/fidentis/analyst/visitors/face/HausdorffDistancePrioritized.java
index 4402d79bc6c8b6a203563fcb75206e924390ac1e..1d2ccb68250f1d628c034be348ed6529a64d7855 100644
--- a/Comparison/src/main/java/cz/fidentis/analyst/visitors/face/HausdorffDistancePrioritized.java
+++ b/Comparison/src/main/java/cz/fidentis/analyst/visitors/face/HausdorffDistancePrioritized.java
@@ -29,23 +29,25 @@ import javax.vecmath.Point3d;
  *       points separately</li>
  *   <li>Priorities of the faces' vertices with respect to all the given types
  *       of feature points merged together</li>
- *   <li>Average weighted Hausdorff distance with respect to the given types of feature
- *       points for each of the faces' mesh facets</li>
+ *   <li>Weighted average of Hausdorff distance of all vertices within the priority sphere
+ *       of feature points of the given types for each of the faces' mesh facets</li>
  * </ul>
  * This visitor is instantiated with a single k-d tree (either given as the input
  * parameter, or automatically created from the triangular mesh).
  * When applied to other human faces, it computes the Hausdorff distance from their mesh facets
  * to the instantiated k-d tree.
+ * 
  * <p>
- *   The Hausdorff distance is computed either as absolute or relative. Absolute
- *   represents Euclidean distance (all numbers are positive). On the contrary,
- *   relative distance considers orientation of the visited face's facets (determined by its normal vectors)
- *   and produces positive or negative distances depending on whether the primary
- *   mesh is "in front of" or "behind" the given vertex.
+ * The Hausdorff distance is computed either as absolute or relative. Absolute
+ * represents Euclidean distance (all numbers are positive). On the contrary,
+ * relative distance considers orientation of the visited face's facets (determined by its normal vectors)
+ * and produces positive or negative distances depending on whether the primary
+ * mesh is "in front of" or "behind" the given vertex.
  * </p>
+ * 
  * <p>
- *   This visitor is thread-safe, i.e., a single instance of the visitor can be used
- *   to inspect multiple human faces simultaneously.
+ * This visitor is thread-safe, i.e., a single instance of the visitor can be used
+ * to inspect multiple human faces simultaneously.
  * </p>
  * 
  * @author Daniel Schramm
@@ -187,12 +189,14 @@ public class HausdorffDistancePrioritized extends HumanFaceVisitor  {
     /**
      * Returns Hausdorff distance of the visited faces' mesh facets to the source mesh facets. 
      * 
+     * <p>
      * Keys in the map contain mesh facets that were measured with the 
-     * source facets.
+     * source facets.<br>
      * For each facet of the visited human face, a list of distances to the source 
      * facets is stored. The order of distances corresponds to the order of vertices
      * in the measured facet, i.e., the i-th value is the distance of the i-th vertex
      * of the visited face's facet.
+     * </p>
      * 
      * @return Hausdorff distance for all points of all the visited human faces' facets
      */
@@ -203,12 +207,14 @@ public class HausdorffDistancePrioritized extends HumanFaceVisitor  {
     /**
      * Returns the nearest points of the visited faces' mesh facets to the source mesh facets.
      * 
+     * <p>
      * Keys in the map contain mesh facets that were measured with the
-     * source facets.
+     * source facets.<br>
      * For each facet of the visited human face, a list of the nearest points to the source
      * facets is stored. The order of points corresponds to the order of vertices
      * in the measured facet, i.e., the i-th point is the nearest point of the i-th vertex
      * of the visited face's facet.
+     * </p>
      *
      * @return The nearest points for all points of all the visited human faces' facets
      */
@@ -221,15 +227,17 @@ public class HausdorffDistancePrioritized extends HumanFaceVisitor  {
      * to the given types of feature points.
      * Priorities are calculated for each type of feature point separately.
      * 
+     * <p>
      * Keys in the map contain human faces whose mesh facets were measured with the
-     * source facets.
-     * For each human face, there is a map of examined types of feature points.
+     * source facets.<br>
+     * For each human face, there is a map of examined types of feature points.<br>
      * Each feature point type then stores a map of priorities of vertices
      * of the face's mesh facets computed with respect to the face's feature point of
      * the corresponding type.
      * The order of priorities in the innermost map's value (list) corresponds to the order of vertices
      * in its corresponding key (facet), i.e., the i-th value in the list is the priority of
      * the i-th vertex of the facet.
+     * </p>
      * 
      * @return Priorities of vertices with respect to the given types of feature points
      */
@@ -238,24 +246,26 @@ public class HausdorffDistancePrioritized extends HumanFaceVisitor  {
     }
 
     /**
-     * For all visited faces' mesh facets, the method returns the average weighted Hausdorff distance
-     * with respect to the given types of feature points (i.e. the sum of weighted distances
-     * of the mesh facet's vertices to the source facets divided by the number of vertices
-     * located within the feature point's priority sphere).
-     * The average weighted distances are calculated for each type of feature point separately.
+     * For all visited faces' mesh facets, the method returns the weighted average of Hausdorff distance
+     * of all vertices within the priority sphere of feature points of the given types
+     * (i.e. the sum of weighted distances of the mesh facet's vertices to the source facets
+     * divided by the sum of their priorities).
+     * The weighted average is calculated for each type of feature point separately.
      * 
+     * <p>
      * Keys in the map contain human faces whose mesh facets were measured with the
-     * source facets.
-     * For each human face, there is a map of examined types of feature points.
+     * source facets.<br>
+     * For each human face, there is a map of examined types of feature points.<br>
      * Each feature point type then stores a map of the face's mesh facets together with
-     * the average weighted distances of their vertices to the source facets.
-     * The average weighted distances are calculated with respect to the face's feature point
-     * of the corresponding type.
+     * the weighted average of distance of their vertices to the source facets.
+     * The weighted average of distance is calculated with respect to the face's feature point
+     * of the corresponding type.<br>
      * <i>If there is no vertex within the priority sphere of a feature point,
-     * the value of the average weighted Hausdorff distance is {@link Double#NaN}</i> 
+     * the value of the average weighted Hausdorff distance is {@link Double#NaN}</i>
+     * </p>
      * 
-     * @return Average weighted Hausdorff distances with respect to the given types
-     *         of feature points
+     * @return Weighted average of Hausdorff distance of all vertices within the priority sphere
+     *         of feature points of the given types
      */
     public Map<HumanFace, Map<FeaturePointType, Map<MeshFacet, Double>>> getFeaturePointWeights() {
         return Collections.unmodifiableMap(featurePointWeights);
@@ -265,14 +275,16 @@ public class HausdorffDistancePrioritized extends HumanFaceVisitor  {
      * Returns priorities of vertices of the visited faces' mesh facets with respect
      * to all the given types of feature points merged together.
      * 
+     * <p>
      * Keys in the map contain human faces whose mesh facets were measured with the
-     * source facets.
+     * source facets.<br>
      * For each human face, there is a map of priorities of vertices of the visited
      * face's mesh facets. The priorities are computed with respect to all the face's
      * feature points of given types together.
      * The order of priorities in the inner map's value (list) corresponds to the order of vertices
      * in its corresponding key (facet), i.e., the i-th value in the list is the priority of
      * the i-th vertex of the facet.
+     * </p>
      * 
      * @return Priorities of vertices with respect to all given types of feature
      *         points merged together
@@ -370,19 +382,24 @@ public class HausdorffDistancePrioritized extends HumanFaceVisitor  {
 
                 synchronized (this) {
                     /*
-                     * Calculate the average weighted Hausdorff distance for the feature point.
+                     * Calculate the weighted average of Hausdorff distance
+                     * of all vertices within the feature point's priority sphere.
                      */
                     final List<Double> facetDistances = hausdorffDistances.get(facet);
                     featurePointWeights.computeIfAbsent(humanFace, face -> new HashMap<>())
                             .computeIfAbsent(featurePointType, fPointType -> new HashMap<>())
                             .put(facet, IntStream.range(0, facetDistances.size())
                                     .filter(i -> facetPriorities.get(i) > 0) // Filter out vertices that are outside of the priority sphere
-                                    .mapToDouble(i -> facetDistances.get(i) * facetPriorities.get(i))
-                                    .average()
-                                    .orElse(Double.NaN)); // If there is no vertex in the priority sphere
+                                    .boxed()
+                                    .collect(WeightedAverageCollector.toWeightedAverage(
+                                            facetDistances::get,
+                                            facetPriorities::get
+                                    ))
+                            );
                     
                     /* 
-                     * Merge priorities computed for the current feature point type with the priorities of already computed feature point types.
+                     * Merge priorities computed for the current feature point type
+                     * with the priorities of already computed feature point types.
                      */
                     final List<Double> storedFacetPriorities = mergedPriorities
                             .computeIfAbsent(humanFace, face -> new HashMap<>())
diff --git a/Comparison/src/main/java/cz/fidentis/analyst/visitors/face/WeightedAverageCollector.java b/Comparison/src/main/java/cz/fidentis/analyst/visitors/face/WeightedAverageCollector.java
new file mode 100644
index 0000000000000000000000000000000000000000..f42241899438a6bec7f06a3148fa2c39c5e2035b
--- /dev/null
+++ b/Comparison/src/main/java/cz/fidentis/analyst/visitors/face/WeightedAverageCollector.java
@@ -0,0 +1,135 @@
+package cz.fidentis.analyst.visitors.face;
+
+import java.util.Set;
+import java.util.function.BiConsumer;
+import java.util.function.BinaryOperator;
+import java.util.function.Function;
+import java.util.function.Supplier;
+import java.util.function.ToDoubleFunction;
+import java.util.stream.Collector;
+
+/**
+ * A collector for calculation of the weighted average, suitable for use in Java 8 streams.
+ *
+ * <p>
+ * A mutable reduction operation that accumulates input elements into a mutable
+ * result container, transforming the accumulated result into a final
+ * representation after all input elements have been processed.
+ * </p>
+ * 
+ * @author Daniel Schramm
+ * @param <T> Data type of the stream elements
+ */
+public class WeightedAverageCollector<T> implements Collector<T, IntemediateResults, Double> {
+    
+    private final ToDoubleFunction<? super T> valueFunction, weightFunction;
+    
+    /**
+     * Constructor.
+     * 
+     * @param valueFunction Function returning the value for a given stream element
+     * @param weightFunction Function returning the weight for a given stream element
+     */
+    public WeightedAverageCollector(
+            ToDoubleFunction<? super T> valueFunction,
+            ToDoubleFunction<? super T> weightFunction) {
+        this.valueFunction = valueFunction;
+        this.weightFunction = weightFunction;
+    }
+    
+    /**
+     * Returns a {@link Collector} interface object used to compute the weighted average.
+     * 
+     * @param <T> Data type of the stream elements
+     * @param valueFunction Function returning the value for a given stream element
+     * @param weightFunction Function returning the weight for a given stream element
+     * @return A Collector interface object used to compute the weighted average
+     */
+    public static <T> Collector<T, ?, Double> toWeightedAverage(
+            ToDoubleFunction<? super T> valueFunction,
+            ToDoubleFunction<? super T> weightFunction) {
+        return new WeightedAverageCollector(valueFunction, weightFunction);
+    }
+    
+    /**
+     * A function that creates and returns a new mutable result container.
+     *
+     * @return A function which returns a new, mutable result container
+     */
+    @Override
+    public Supplier<IntemediateResults> supplier() {
+        return IntemediateResults::new;
+    }
+    
+    /**
+     * A function that folds a value into a mutable result container.
+     *
+     * @return A function which folds a value into a mutable result container
+     */
+    @Override
+    public BiConsumer<IntemediateResults, T> accumulator() {
+        return (iResult, streamElement) -> {
+            iResult.weightedValSum += valueFunction.applyAsDouble(streamElement)
+                    * weightFunction.applyAsDouble(streamElement);
+            iResult.weightSum += weightFunction.applyAsDouble(streamElement);
+        };
+    }
+    
+    /**
+     * A function that accepts two partial results and merges them. The
+     * combiner function may fold state from one argument into the other and
+     * return that, or may return a new result container.
+     *
+     * @return A function which combines two partial results into a combined
+     *         result
+     */
+    @Override
+    public BinaryOperator<IntemediateResults> combiner() {
+        return (iResult1, iResult2) -> {
+            iResult1.weightedValSum += iResult2.weightedValSum;
+            iResult1.weightSum += iResult2.weightSum;
+            
+            return iResult1;
+        };
+    }
+    
+    /**
+     * Perform the final transformation from the intermediate accumulation type
+     * {@link IntemediateResults} to the final result type {@link Double}.
+     *
+     * <p>
+     * If the characteristic {@code IDENTITY_FINISH} is
+     * set, this function may be presumed to be an identity transform with an
+     * unchecked cast from {@link IntemediateResults} to {@link Double}.
+     * </p>
+     *
+     * @return A function which transforms the intermediate result to the final
+     *         result
+     */
+    @Override
+    public Function<IntemediateResults, Double> finisher() {
+        return iResult -> iResult.weightedValSum / iResult.weightSum;
+    }
+    
+    /**
+     * Returns a {@code Set} of {@code Collector.Characteristics} indicating
+     * the characteristics of this Collector.
+     *
+     * @return An immutable set of collector characteristics
+     */
+    @Override
+    public Set<Characteristics> characteristics() {
+        return Set.of(Characteristics.UNORDERED);
+    }
+}
+
+/**
+ * A helper class which stores intermediate results
+ * for the {@link WeightedAverageCollector} class.
+ * 
+ * @author Daniel Schramm
+ */
+class IntemediateResults {
+    double weightedValSum = 0;
+    double weightSum = 0;
+}
diff --git a/Comparison/src/test/java/cz/fidentis/analyst/visitors/face/HausdorffDistancePrioritizedTest.java b/Comparison/src/test/java/cz/fidentis/analyst/visitors/face/HausdorffDistancePrioritizedTest.java
index 90bfef802aefa36c307f1dca9b2894d72e2f8803..9706a4224d06ac9bbda21e6f68e9c8f354b82f85 100644
--- a/Comparison/src/test/java/cz/fidentis/analyst/visitors/face/HausdorffDistancePrioritizedTest.java
+++ b/Comparison/src/test/java/cz/fidentis/analyst/visitors/face/HausdorffDistancePrioritizedTest.java
@@ -45,7 +45,9 @@ public class HausdorffDistancePrioritizedTest {
     
     private final List<Double> prioritiesListFP1 = List.of(1d, 0.9, 0.8, 0.7, 0.6, 0.5, 0.4, 0.3, 0.2, 0.1, 0d, 0d);
     private final double weightedDistanceFP1 = 0 + 0.09 + 0.16 + 0.21 + 0.24 + 0.25 + 0.24 + 0.21 + 0.16 + 0.09 + 0 + 0;
-    private final double averageWeightedDistanceFP1 = weightedDistanceFP1 / 10; // There are 10 vertices inside the priority sphere
+    private final double averageWeightedDistanceFP1 = weightedDistanceFP1 / prioritiesListFP1.stream()
+            .mapToDouble(Double::doubleValue)
+            .sum();
     
     private final Point3d featurePoint1 = new Point3d(0, 0, 0);
     private final FeaturePointType fpType1 = new FeaturePointType(0, null, null, null);
@@ -282,7 +284,8 @@ public class HausdorffDistancePrioritizedTest {
         for (final MeshFacet facet: facets) {
             expectedPrioritiesMapFP2.put(facet, Collections.nCopies(VERTICES_NUM, 0d));
         }
-        expectedPrioritiesMapFP2.put(facet1, List.of(0d, 0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8, 0.9, 1d, 0.9));
+        final List<Double> prioritiesList = List.of(0d, 0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8, 0.9, 1d, 0.9);
+        expectedPrioritiesMapFP2.put(facet1, prioritiesList);
         final Map<FeaturePointType, Map<MeshFacet, List<Double>>> expectedPriorities = Map.of(fpType1, expectedPrioritiesMapFP1,
                 fpType2, expectedPrioritiesMapFP2);
         
@@ -290,7 +293,12 @@ public class HausdorffDistancePrioritizedTest {
         for (final MeshFacet facet: facets) {
             expectedWeightedDistancesMapFP2.put(facet, Double.NaN);
         }
-        expectedWeightedDistancesMapFP2.put(facet1, (0 + 0.01 + 0.04 + 0.09 + 0.16 + 0.25 + 0.36 + 0.49 + 0.64 + 0.81 + 1 + 0.99) / 11);
+        expectedWeightedDistancesMapFP2.put(
+                facet1,
+                (0 + 0.01 + 0.04 + 0.09 + 0.16 + 0.25 + 0.36 + 0.49 + 0.64 + 0.81 + 1 + 0.99) / prioritiesList.stream()
+                        .mapToDouble(Double::doubleValue)
+                        .sum()
+        );
         final Map<FeaturePointType, Map<MeshFacet, Double>> expectedFeaturePointsWeights = Map.of(fpType1, expectedWeightedDistancesMapFP1,
                 fpType2, expectedWeightedDistancesMapFP2);
         
@@ -346,7 +354,9 @@ public class HausdorffDistancePrioritizedTest {
             expectedWeightedDistancesMapFP3.put(facet, Double.NaN);
             expectedWeightedDistancesMapFP4.put(facet, Double.NaN);
         }
-        final double weightedDistance = (0 + 0.01 + 0.04 + 0.09 + 0.16 + 0.25 + 0.36 + 0.49 + 0.64 + 0.81 + 1 + 0.99) / 11;
+        final double weightedDistance = (0 + 0.01 + 0.04 + 0.09 + 0.16 + 0.25 + 0.36 + 0.49 + 0.64 + 0.81 + 1 + 0.99) / prioritiesList.stream()
+                .mapToDouble(Double::doubleValue)
+                .sum();
         expectedWeightedDistancesMapFP2.put(facet1, weightedDistance);
         expectedWeightedDistancesMapFP3.put(facet3, weightedDistance);
         expectedWeightedDistancesMapFP4.put(facet5, -weightedDistance);
@@ -409,7 +419,9 @@ public class HausdorffDistancePrioritizedTest {
             expectedWeightedDistancesMapFP2.put(facet, Double.NaN);
             expectedWeightedDistancesMapFP3.put(facet, Double.NaN);
         }
-        final double weightedDistance = 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 1 * vertexPriority + 0;
+        final double weightedDistance = (0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 1 * vertexPriority + 0) / prioritiesList.stream()
+                .mapToDouble(Double::doubleValue)
+                .sum();
         expectedWeightedDistancesMapFP2.put(facet1, weightedDistance);
         expectedWeightedDistancesMapFP2.put(facet3, weightedDistance);
         expectedWeightedDistancesMapFP3.put(facet1, weightedDistance);
@@ -473,7 +485,9 @@ public class HausdorffDistancePrioritizedTest {
             expectedWeightedDistancesMapFP3.put(facet, Double.NaN);
             expectedWeightedDistancesMapFP4.put(facet, Double.NaN);
         }
-        final double weightedDistance = (0 + 0.01 + 0.04 + 0.09 + 0.16 + 0.25 + 0.36 + 0.49 + 0.64 + 0.81 + 1 + 0.99) / 11;
+        final double weightedDistance = (0 + 0.01 + 0.04 + 0.09 + 0.16 + 0.25 + 0.36 + 0.49 + 0.64 + 0.81 + 1 + 0.99) / prioritiesList.stream()
+                .mapToDouble(Double::doubleValue)
+                .sum();
         expectedWeightedDistancesMapFP2.put(facet1, weightedDistance);
         expectedWeightedDistancesMapFP3.put(facet3, weightedDistance);
         expectedWeightedDistancesMapFP4.put(facet5, -weightedDistance);