Class: Musa::Darwin::Darwin
Overview
Evolutionary selector for population-based optimization.
Evaluates population using measures and weights, returning sorted population by fitness score.
Defined Under Namespace
Classes: MainContext, Measure, MeasuresEvalContext
Instance Method Summary collapse
-
#evaluate_weights(measure_a, measure_b) ⇒ Integer
private
Compares two measures by their weighted fitness.
-
#initialize { ... } ⇒ void
constructor
Creates Darwin selector with evaluation rules.
-
#select(population) ⇒ Array
Selects and ranks population by fitness.
Constructor Details
#initialize { ... } ⇒ void
Creates Darwin selector with evaluation rules.
109 110 111 112 113 114 115 116 |
# File 'lib/musa-dsl/generative/darwin.rb', line 109 def initialize(&block) raise ArgumentError, 'block is needed' unless block main_context = MainContext.new &block @measures = main_context._measures @weights = main_context._weights end |
Instance Method Details
#evaluate_weights(measure_a, measure_b) ⇒ Integer
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Compares two measures by their weighted fitness.
188 189 190 |
# File 'lib/musa-dsl/generative/darwin.rb', line 188 def evaluate_weights(measure_a, measure_b) measure_b.evaluate_weight(@weights) <=> measure_a.evaluate_weight(@weights) end |
#select(population) ⇒ Array
Selects and ranks population by fitness.
Evaluates each object with measures, normalizes dimensions across population, applies weights, and returns population sorted by fitness (highest first). Objects marked as died are excluded.
134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 |
# File 'lib/musa-dsl/generative/darwin.rb', line 134 def select(population) measured_objects = [] population.each do |object| context = MeasuresEvalContext.new context.with object, **{}, &@measures measure = context._measure measured_objects << { object: object, measure: context._measure } unless measure.died? end limits = {} measured_objects.each do |measured_object| measure = measured_object[:measure] measure.dimensions.each do |measure_name, value| limit = limits[measure_name] ||= { min: nil, max: nil } limit[:min] = value.to_f if limit[:min].nil? || limit[:min] > value limit[:max] = value.to_f if limit[:max].nil? || limit[:max] < value limit[:range] = limit[:max] - limit[:min] end end # warn "Darwin.select: weights #{@weights}" measured_objects.each do |measured_object| measure = measured_object[:measure] measure.dimensions.each do |dimension_name, value| limit = limits[dimension_name] measure.normalized_dimensions[dimension_name] = limit[:range].zero? ? 0.5 : (value - limit[:min]) / limit[:range] end # warn "Darwin.select: #{measured_object[:object]} #{measured_object[:measure]} weight=#{measured_object[:measure].evaluate_weight(@weights).round(2)}" end measured_objects.sort! { |a, b| evaluate_weights a[:measure], b[:measure] } measured_objects.collect { |measured_object| measured_object[:object] } end |