Python 规则30元胞自动机的行生成
规则30是一种一维细胞自动机,当前一代只考虑上一代的细胞。单元格可以处于两种状态:Python 规则30元胞自动机的行生成,python,python-3.x,numpy,numpy-ndarray,cellular-automata,Python,Python 3.x,Numpy,Numpy Ndarray,Cellular Automata,规则30是一种一维细胞自动机,当前一代只考虑上一代的细胞。单元格可以处于两种状态:1或0。创建下一代的规则在下一行中表示,并取决于当前单元正上方的单元及其近邻 细胞自动机按以下规则应用(使用位运算符): 该规则构成下表: 现在,我尝试使用numpy将这些规则实现到Python中。我定义了一个初始状态,它接受了宽度>代码>作为参数,并在中间产生了一个零点为1的初始行。 def initial_state(width): initial = np.zeros((1, width), dty
1
或0
。创建下一代的规则在下一行中表示,并取决于当前单元正上方的单元及其近邻
细胞自动机按以下规则应用(使用位运算符):
该规则构成下表:
现在,我尝试使用numpy将这些规则实现到Python中。我定义了一个初始状态,它接受了<代码>宽度>代码>作为参数,并在中间产生了一个零点为1的初始行。
def initial_state(width):
initial = np.zeros((1, width), dtype=int)
if width % 2 == 0:
initial = np.insert(initial, int(width / 2), values=0, axis=1)
initial[0, int(width / 2)] = 1
return initial
else:
initial[0, int(width / 2)] = 1
return initial
下面的函数只生成给定初始行的第二代。如何创建一个for循环,使其不断生成新的代,直到最后一行的第一个元素变为1
def rule30(array):
row1 = np.pad(array,[(0,0), (1,1)], mode='constant')
next_row = array.copy()
for x in range(1, array.shape[0]+1):
for y in range(1, array.shape[1]+1):
if row1[x-1][y-1] == 1 ^ (row1[x-1][y] == 1 or row1[x-1][y+1] == 1):
next_row[x - 1, y - 1] = 1
else:
next_row[x - 1, y - 1] = 0
return np.concatenate((array, next_row))
例如,如果输入是
A = [0, 0, 0, 1, 0, 0, 0]
输出应该是
>>> print(rule30(A))
[[0, 0, 0, 1, 0, 0, 0],
[0, 0, 1, 1, 1, 0, 0],
[0, 1, 1, 0, 0, 1, 0],
[1, 1, 0, 1, 1, 1, 1]]
方法1-Numpy
您可以通过对当前代码进行以下轻微修改来实现这一点-将rule30
的返回值更改为return np.array(下一行)
。然后可以使用以下功能:
def apply_rule(n):
rv = initial_state(n)
while rv[-1][0] == 0:
rv = np.append(rv, rule30(rv[-1].reshape(1,-1)), axis=0)
return rv
用法:
>>> apply_rule(7)
array([[0, 0, 0, 1, 0, 0, 0],
[0, 0, 1, 1, 1, 0, 0],
[0, 1, 1, 0, 0, 1, 0],
[1, 1, 0, 1, 1, 1, 1]])
def rule(t):
return t[0] ^ (t[1] or t[2])
def initial_state(width):
initial = [0]*width
if width%2:
initial[width // 2] = 1
else:
initial.insert(width//2, 1)
return initial
def get_triples(l):
return zip(l,l[1:],l[2:])
def rule30(l):
return [rule(t) for t in get_triples([0] + l + [0])]
def apply_rule(width):
rv = [initial_state(width)]
while not rv[-1][0]:
rv.append(rule30(rv[-1]))
return rv
>>> apply_rule(7)
[[0, 0, 0, 1, 0, 0, 0],
[0, 0, 1, 1, 1, 0, 0],
[0, 1, 1, 1, 0, 1, 0],
[1, 1, 1, 0, 0, 1, 1]]
>>> [''.join(str(y) for y in x) for x in apply_rule(7)]
['0001000',
'0011100',
'0111010',
'1110011']
或绘制:
>>> plt.imshow(apply_rule(7), cmap='hot')
方法2-清单
或者,您可以在不使用numpy的情况下使用以下解决方案,该解决方案使用一些函数在每个填充列表中的每个三元组上应用规则30逻辑,直到满足停止条件
代码:
>>> apply_rule(7)
array([[0, 0, 0, 1, 0, 0, 0],
[0, 0, 1, 1, 1, 0, 0],
[0, 1, 1, 0, 0, 1, 0],
[1, 1, 0, 1, 1, 1, 1]])
def rule(t):
return t[0] ^ (t[1] or t[2])
def initial_state(width):
initial = [0]*width
if width%2:
initial[width // 2] = 1
else:
initial.insert(width//2, 1)
return initial
def get_triples(l):
return zip(l,l[1:],l[2:])
def rule30(l):
return [rule(t) for t in get_triples([0] + l + [0])]
def apply_rule(width):
rv = [initial_state(width)]
while not rv[-1][0]:
rv.append(rule30(rv[-1]))
return rv
>>> apply_rule(7)
[[0, 0, 0, 1, 0, 0, 0],
[0, 0, 1, 1, 1, 0, 0],
[0, 1, 1, 1, 0, 1, 0],
[1, 1, 1, 0, 0, 1, 1]]
>>> [''.join(str(y) for y in x) for x in apply_rule(7)]
['0001000',
'0011100',
'0111010',
'1110011']
用法:
>>> apply_rule(7)
array([[0, 0, 0, 1, 0, 0, 0],
[0, 0, 1, 1, 1, 0, 0],
[0, 1, 1, 0, 0, 1, 0],
[1, 1, 0, 1, 1, 1, 1]])
def rule(t):
return t[0] ^ (t[1] or t[2])
def initial_state(width):
initial = [0]*width
if width%2:
initial[width // 2] = 1
else:
initial.insert(width//2, 1)
return initial
def get_triples(l):
return zip(l,l[1:],l[2:])
def rule30(l):
return [rule(t) for t in get_triples([0] + l + [0])]
def apply_rule(width):
rv = [initial_state(width)]
while not rv[-1][0]:
rv.append(rule30(rv[-1]))
return rv
>>> apply_rule(7)
[[0, 0, 0, 1, 0, 0, 0],
[0, 0, 1, 1, 1, 0, 0],
[0, 1, 1, 1, 0, 1, 0],
[1, 1, 1, 0, 0, 1, 1]]
>>> [''.join(str(y) for y in x) for x in apply_rule(7)]
['0001000',
'0011100',
'0111010',
'1110011']
Matplotlib可视化(使用任一方法):
以下是基于字符串表示和查找的代码。它确实使用了上面评论中的一些想法。除此之外,我还添加了处理边缘单元格的填充物——具体情况尚不清楚。还要注意,您建议的模式表是不对称的。比较“110”和“011”的新状态
def rule30(a):
patterns = {'111': '0', '110': '0', '101': '0', '100': '1',
'011': '1', '010': '1', '001': '1', '000': '0', }
a = '0' + a + '0' # padding
return ''.join([patterns[a[i:i+3]] for i in range(len(a)-2)])
a = '0001000'
result = [list(map (int, a))]
while a[0] != '1':
a = rule30(a)
result.append (list(map (int, a)))
print (result) # list of lists
print (np.array(result)) # np.array
名单:
[[0, 0, 0, 1, 0, 0, 0], [0, 0, 1, 1, 1, 0, 0], [0, 1, 1, 0, 0, 1, 0], [1, 1, 0, 1, 1, 1, 1]]
np.array:
array([[0, 0, 0, 1, 0, 0, 0],
[0, 0, 1, 1, 1, 0, 0],
[0, 1, 1, 0, 0, 1, 0],
[1, 1, 0, 1, 1, 1, 1]])
如果将
rule30
实现为一个函数,该函数接收一个状态并返回下一个状态,而不是填充现有的二维数组以完成操作,那么这可能更简单。最简单的解决方案可能是根据规则预计算一个将三位映射到一位的查找表。或者,更简单的是,只需将三个位打包成一个从0到7的数字n,并使用规则编号的第n位(例如30)作为输出位,因为。因此,函数现在可以工作了,我更新了问题中的代码。它产生下一代。现在,我的问题是-如何创建一个for循环,在初始状态行的“下方”追加下一代?for循环应该停止生成新行,直到最后一行的第一个元素变为1。