1. 1 : import { FuzzySet } from '../FuzzySet.js';
  2. 2 :
  3. 3 : /**
  4. 4 : * Class for representing a fuzzy set that has a normal distribution shape. It can be defined
  5. 5 : * by the mean and standard deviation.
  6. 6 : *
  7. 7 : * @author {@link https://github.com/robp94|robp94}
  8. 8 : * @augments FuzzySet
  9. 9 : */
  10. 10 : class NormalDistFuzzySet extends FuzzySet {
  11. 11 :
  12. 12 : /**
  13. 13 : * Constructs a new triangular fuzzy set with the given values.
  14. 14 : *
  15. 15 : * @param {Number} left - Represents the left border of this fuzzy set.
  16. 16 : * @param {Number} midpoint - Mean or expectation of the normal distribution.
  17. 17 : * @param {Number} right - Represents the right border of this fuzzy set.
  18. 18 : * @param {Number} standardDeviation - Standard deviation of the normal distribution.
  19. 19 : */
  20. 20 : constructor( left = 0, midpoint = 0, right = 0, standardDeviation = 0 ) {
  21. 21 :
  22. 22 : super( midpoint );
  23. 23 :
  24. 24 : /**
  25. 25 : * Represents the left border of this fuzzy set.
  26. 26 : * @type {Number}
  27. 27 : * @default 0
  28. 28 : */
  29. 29 : this.left = left;
  30. 30 :
  31. 31 : /**
  32. 32 : * Represents the peak value of this fuzzy set.
  33. 33 : * @type {Number}
  34. 34 : * @default 0
  35. 35 : */
  36. 36 : this.midpoint = midpoint;
  37. 37 :
  38. 38 : /**
  39. 39 : * Represents the right border of this fuzzy set.
  40. 40 : * @type {Number}
  41. 41 : * @default 0
  42. 42 : */
  43. 43 : this.right = right;
  44. 44 :
  45. 45 : /**
  46. 46 : * Represents the standard deviation of this fuzzy set.
  47. 47 : * @type {Number}
  48. 48 : * @default 0
  49. 49 : */
  50. 50 : this.standardDeviation = standardDeviation;
  51. 51 :
  52. 52 : //
  53. 53 :
  54. 54 : this._cache = {};
  55. 55 :
  56. 56 : }
  57. 57 :
  58. 58 : /**
  59. 59 : * Computes the degree of membership for the given value.
  60. 60 : *
  61. 61 : * @param {Number} value - The value used to calculate the degree of membership.
  62. 62 : * @return {Number} The degree of membership.
  63. 63 : */
  64. 64 : computeDegreeOfMembership( value ) {
  65. 65 :
  66. 66 : this._updateCache();
  67. 67 :
  68. 68 : if ( value >= this.right || value <= this.left ) return 0;
  69. 69 :
  70. 70 : return probabilityDensity( value, this.midpoint, this._cache.variance ) / this._cache.normalizationFactor;
  71. 71 :
  72. 72 : }
  73. 73 :
  74. 74 : /**
  75. 75 : * Transforms this instance into a JSON object.
  76. 76 : *
  77. 77 : * @return {Object} The JSON object.
  78. 78 : */
  79. 79 : toJSON() {
  80. 80 :
  81. 81 : const json = super.toJSON();
  82. 82 :
  83. 83 : json.midpoint = this.midpoint;
  84. 84 : json.standardDeviation = this.standardDeviation;
  85. 85 :
  86. 86 : return json;
  87. 87 :
  88. 88 : }
  89. 89 :
  90. 90 : /**
  91. 91 : * Restores this instance from the given JSON object.
  92. 92 : *
  93. 93 : * @param {Object} json - The JSON object.
  94. 94 : * @return {NormalDistFuzzySet} A reference to this fuzzy set.
  95. 95 : */
  96. 96 : fromJSON( json ) {
  97. 97 :
  98. 98 : super.fromJSON( json );
  99. 99 :
  100. 100 : this.midpoint = json.midpoint;
  101. 101 : this.standardDeviation = json.standardDeviation;
  102. 102 :
  103. 103 : return this;
  104. 104 :
  105. 105 : }
  106. 106 :
  107. 107 : //
  108. 108 :
  109. 109 : _updateCache() {
  110. 110 :
  111. 111 : const cache = this._cache;
  112. 112 : const midpoint = this.midpoint;
  113. 113 : const standardDeviation = this.standardDeviation;
  114. 114 :
  115. 115 : if ( midpoint !== cache.midpoint || standardDeviation !== cache.standardDeviation ) {
  116. 116 :
  117. 117 : const variance = standardDeviation * standardDeviation;
  118. 118 :
  119. 119 : cache.midpoint = midpoint;
  120. 120 : cache.standardDeviation = standardDeviation;
  121. 121 : cache.variance = variance;
  122. 122 :
  123. 123 : // this value is used to ensure the DOM lies in the range of [0,1]
  124. 124 :
  125. 125 : cache.normalizationFactor = probabilityDensity( midpoint, midpoint, variance );
  126. 126 :
  127. 127 : }
  128. 128 :
  129. 129 : return this;
  130. 130 :
  131. 131 : }
  132. 132 :
  133. 133 : }
  134. 134 :
  135. 135 : //
  136. 136 :
  137. 137 : function probabilityDensity( x, mean, variance ) {
  138. 138 :
  139. 139 : return ( 1 / Math.sqrt( 2 * Math.PI * variance ) ) * Math.exp( - ( Math.pow( ( x - mean ), 2 ) ) / ( 2 * variance ) );
  140. 140 :
  141. 141 : }
  142. 142 :
  143. 143 : export { NormalDistFuzzySet };