It wasn’t. It was evolution.
Dawkins’s ‘weasel’ program isn’t really appropriate for testing evolution of irreducibly complex systems, because there is only one possible target, and the components of it are independent of each other. Nor does ‘weasel’ include the concept of non-functionality status, since the only possible metric is difference from the target string.
You haven’t said how you modified ‘weasel’ to include irreducible complexity, but you’ve hinted that all you’ve done is change the weighting mechanism so that some substring(s) don’t count towards matching the target unless all of the letters in them are correct. If that’s the case, you haven’t demonstrated anything about irreducible complexity, you’ve only made some parts of the string less likely to be found by random mutation (1 in 27n instead of 1 in 27) and shown that this makes ‘weasel’ take longer to finish.[1] That’s neither unsurprising nor new. Nor is it relevant to how evolution is expected to produce irreducibly complex systems, which is by producing a fully functional system that isn’t irreducibly complex, then optimising or streamlining it without losing that functionality.
If you really want to model the evolution of irreducibly complex systems, you’d be better of modifying a program such as Dave Thomas’s Steiner tree algorithm which not only includes a criterion for functional vs non-functional (all nodes connected) but also doesn’t have a fixed target, or even a known one.
By changing the initial population generation parameters you can ensure that the starting population contains no functional systems at all, yet it can not only evolve functional systems, but also irreducible complex functional systems.
I’ve appended the Python code of one of my implementations of the algorithm (which runs multiple three-dimensional Steiner simulations back-to-back, all starting with completely non-functional systems), as well as the results of one run which includes some outcomes that are irreducibly complex, and some that are not.[2]
So this is a demonstration of how evolutionary algorithms can produce systems that are irreducibly complex, by indirect routes that involved finding a reducible functional system and then optimising it, just as is expected to happen in nature.
That so-called ‘direct’ Darwinian paths might not be able to generate irreducibly complex systems is completely irrelevant in a world where the vast majority of evolutionary paths (like that of penguin forelimbs) are indirect, and the vast majority of biological systems are both multifunctional and not irreducibly complex.
Python code:
"""
Simple evolutionary algorithm to generate a Steiner tree connecting a set
of defined nodes using some number of additional intermediate nodes.
Based on Dave Thomas's original Fortran code.
The data structure for a tree is
- 3n fixed values, copied from the initial list and never changed [x,y,z]
- 3(n-1) values for variable nodes [x,y,z]
- (2n-1)*(2n-1) booleans representing connectivity, of which only the values where the first index is lower than the second one are used
- 1 boolean representing its connectivity
- one number representing its Steiner length
Update: in order to stimulate finding steiner trees instead of spanning graphs,
fixed nodes are never connected, and all paths between them must be via variable nodes.
"""
import math
import random
# point parameters - change these to look for a different Steiner trees
fixed_points_x = [100,200,100,300,200,400,300,400]
fixed_points_y = [200,100,300,100,400,200,400,300]
fixed_points_z = [200,300,200,300,200,300,200,300]
max_dim_x = max(fixed_points_x)+50
min_dim_x = min(fixed_points_x)-50
max_dim_y = max(fixed_points_y)+50
min_dim_y = min(fixed_points_y)-50
max_dim_z = max(fixed_points_z)+50
min_dim_z = min(fixed_points_z)-50
num_of_fixed_points = len(fixed_points_x)
num_of_nodes = num_of_fixed_points - 1
total_points = num_of_fixed_points + num_of_nodes
# population and mutation parameters - change these to alter the performance of the simulation
population_size = 1000
number_of_generations = 1000
unconnected_length = 10000000
node_mutate = 150
link_mutate = 100
# indices into structure
nodes_index = num_of_fixed_points * 3
connections_index = total_points*3
connected_index = total_points*3 + total_points*total_points
steiner_length_index = total_points*3 + total_points*total_points + 1
"""
determine connectivity of steiner tree
"""
def connected(tree) :
# initialise list of connected points
connected_points = [True] # first node is always connected
for a in range(total_points - 1) :
connected_points.append(False)
points_connected = 1 # first node is always connected
new_points_added = True
# loop through connections until either (i) number of connected points = total number, or (ii) no new points have been added
while new_points_added:
last_connected_count = points_connected
for a in range(total_points-1):
for b in range(a,total_points):
if tree[connections_index + (total_points*a) + b] :
if (connected_points[a] != connected_points[b]) :
connected_points[a] = True
connected_points[b] = True
points_connected += 1
if last_connected_count == points_connected :
new_points_added = False
# only need to check fixed points
fixed_points_connected = True
for a in range(num_of_fixed_points) :
if connected_points[a] == False :
fixed_points_connected = False
return fixed_points_connected
"""
determine total length of steiner tree
"""
def steiner_length(tree) :
total_length = 0
for a in range(total_points-1):
for b in range(a,total_points):
if tree[connections_index + (total_points*a) + b] :
x_diff = tree[3*a] - tree[3*b]
y_diff = tree[3*a+1] - tree[3*b+1]
z_diff = tree[3*a+2] - tree[3*b+2]
total_length += math.sqrt(x_diff*x_diff + y_diff*y_diff + z_diff*z_diff)
return total_length
"""
create a mutated copy of an existing tree
"""
def mutate(tree) :
new_tree = list(tree)
for x in range (num_of_nodes) :
if random.randint(0,node_mutate) == 0 : # chance to change the x position of a variable node
new_tree[nodes_index + 3*x] = random.randint(min_dim_x,max_dim_x)
if random.randint(0,node_mutate) == 0 : # chance to change the y position of a variable node
new_tree[nodes_index + 3*x +1] = random.randint(min_dim_y,max_dim_y)
if random.randint(0,node_mutate) == 0 : # chance to change the z position of a variable node
new_tree[nodes_index + 3*x +2] = random.randint(min_dim_z,max_dim_z)
for x in range (total_points) :
for y in range (total_points) :
if not( (x < num_of_fixed_points) and (y < num_of_fixed_points) ) : # don't change state of connectivity between fixed nodes
if random.randint(0,link_mutate) == 0 : # chance to flip a connection
if new_tree[connections_index + (x * total_points) + y]:
new_tree[connections_index + (x * total_points) + y] = False
else :
new_tree[connections_index + (x * total_points) + y] = True
new_tree[connected_index] = connected(new_tree)
if new_tree[connected_index] :
new_tree[steiner_length_index] = steiner_length(new_tree)
else :
new_tree[steiner_length_index] = unconnected_length
return new_tree
def single_run() :
population = []
# generate initial population
for x in range(population_size) :
tree = []
for y in range(num_of_fixed_points):
tree.append(fixed_points_x[y])
tree.append(fixed_points_y[y])
tree.append(fixed_points_z[y])
for y in range(num_of_nodes):
tree.append(random.randint(min_dim_x,max_dim_x)) # x co-ordinate of variable node
tree.append(random.randint(min_dim_y,max_dim_y)) # y co-ordinate of variable node
tree.append(random.randint(min_dim_z,max_dim_z)) # z co-ordinate of variable node
for y in range(total_points):
for z in range(total_points):
tree.append(False) # start with no connections
tree.append(False) # they're always unconnected
tree.append(unconnected_length)
population.append(tree)
for gen in range (number_of_generations) :
# generate next generation
for x in range (population_size) :
# compare each member of the population to a random other one,
# and replace it with a mutated copy of the one that is shorter
competitor = random.randint(0,population_size-1) # it might be competing against itself (it'll win!)
if population[x][steiner_length_index] > population[competitor][steiner_length_index] :
population[x] = mutate(population[competitor])
else :
population[x] = mutate(population[x])
# find shortest tree
shortest_nodes = ""
current_steiner_record = unconnected_length+1 # one more than default unconnected value, to ensure there is always a winner
for x in range (population_size) :
if population[x][steiner_length_index] < current_steiner_record :
shortest_nodes = ""
current_steiner_record = population[x][steiner_length_index]
for a in range(total_points-1):
for b in range(a,total_points):
if population[x][connections_index + (total_points*a) + b] :
if ((population[x][3*a] != population[x][3*b]) or (population[x][3*a+1] != population[x][3*b+1]) or (population[x][3*a+2] != population[x][3*b+2])) :
shortest_nodes = shortest_nodes + "(" + str(population[x][3*a]) + "," + str(population[x][3*a+1])+ "," + str(population[x][3*a+2]) + ")-("+ str(population[x][3*b]) + "," + str(population[x][3*b+1])+ "," + str(population[x][3*b+2]) + "); "
retval = []
retval.append(current_steiner_record)
retval.append(shortest_nodes)
return retval
# batch run
number_of_runs = 10
results = []
for runs in range (number_of_runs) :
print ("Run " + str(runs+1))
results.append(single_run())
results.sort()
for runs in range (number_of_runs) :
print (str(runs+1) + " L: " + str(results[runs][0]) + " " + results[runs][1])
Results (top 3 only):
1 L: 1182.403350618639 (100,200,200)-(158,248,240); (200,100,300)-(235,89,301); (100,300,200)-(198,322,190); (300,100,300)-(353,187,301); (300,100,300)-(428,112,331); (300,100,300)-(235,89,301); (200,400,200)-(223,382,208); (400,200,300)-(353,187,301); (300,400,200)-(223,382,208); (400,300,300)-(353,187,301); (198,322,190)-(223,382,208); (198,322,190)-(158,248,240); (353,187,301)-(158,248,240);
2 L: 1195.145521457145 (100,200,200)-(138,296,224); (200,100,300)-(226,136,286); (200,100,300)-(138,296,224); (100,300,200)-(194,330,237); (100,300,200)-(138,296,224); (300,100,300)-(277,109,258); (200,400,200)-(194,330,237); (400,200,300)-(277,109,258); (400,200,300)-(365,323,241); (300,400,200)-(365,323,241); (400,300,300)-(365,323,241); (277,109,258)-(226,136,286);
3 L: 1196.697508957284 (100,200,200)-(98,214,217); (200,100,300)-(272,198,315); (100,300,200)-(289,213,294); (100,300,200)-(98,214,217); (300,100,300)-(289,213,294); (200,400,200)-(307,399,152); (400,200,300)-(289,213,294); (300,400,200)-(307,399,152); (300,400,200)-(342,258,283); (400,300,300)-(342,258,283); (289,213,294)-(272,198,315); (289,213,294)-(342,258,283);
You can confirm for yourself which ones are irreducibly complex, i.e. will cease to be functional if any of the connecting segments are removed.
I’m assuming you’ve implemented your modifications correctly, though it’s possible you have accidentally done something else; see here for an instance of some-one trying to demonstrate a point using a ‘weasel’ implementation and instead only demonstrating their own lack of understanding. ↩︎
I set a low number of generations for the runs, so the results are nowhere near optimal, but much better results can be achieved with more generations. ↩︎