如何尽可能有效地进行一维离散碰撞检测?

2024-05-13 07:12:02 发布

您现在位置:Python中文网/ 问答频道 /正文

我有以下情况。在离散域0,1,…,L上有M个独立的随机游走者。我们对N个相同的域这样做。这导致矩阵X,其中X[i, j]是域j上的walker i的位置。为了进行随机步骤,我向矩阵X添加了一个形状相同的矩阵,其中包含random+1和-1。然后我处理边缘。这很有效

但是,我想扩展这个模型,使其具有不能相互通过的固体粒子。这显示在2个案例中

  1. 一个粒子位于i位置,第二个粒子位于i+1位置。第一个粒子向右移动,而第二个粒子向左移动
  2. 一个粒子位于i位置,第二个粒子位于i+2位置。第一个粒子向右移动,而第二个粒子向左移动

如果我独立完成所有步骤,我可以手动检查每个步骤,看它是否合法。然而,这是糟糕的性能。有没有更有效的方法来检测哪些矩阵元素对X[i,j], X[k, j]导致两个粒子相互穿过,最好是以矢量化的方式?这样,我可以使模拟跳过这些步骤


Tags: 方法模型元素情况步骤粒子矩阵random
1条回答
网友
1楼 · 发布于 2024-05-13 07:12:02

我想您必须使用某种形式的循环来实现这一点,但这可能会有所帮助:

import numpy as np
import matplotlib.pyplot as plt

L = 50
N = 30
W = 20
n_steps = 1000

# Initialize all walkers on the left side
wn0 = np.arange(W)[:, np.newaxis].repeat(N, axis=1)

# Set up the plot
fig, ax = plt.subplots()
worlds = np.zeros((N, L))
worlds[np.arange(N)[np.newaxis, :], wn0] = np.arange(W)[:, np.newaxis]
h = ax.imshow(worlds, cmap='gray_r')  # cmap='tab20')
ax.set_xlabel('Distance in 1D World')
ax.set_ylabel('Ensemble of Worlds')

for _ in range(n_steps):

    r = np.where(np.random.random(wn0.shape) < 0.5, 1, -1)

    wn1 = wn0 + r
    wn1 = np.clip(wn1, 0, L-1)

    # Case 1
    rest_mat = np.zeros_like(wn0, dtype=bool)
    for i in range(W):
        for j in range(i+1, W):
            rest_mat[[[i], [j]], np.logical_and(wn0[i] == wn1[j], wn1[i] == wn0[j])] = True
    wn1[rest_mat] = wn0[rest_mat]

    # Case 2, go from 0->W and than from W->0 to make sure all duplicates are gone
    for i in np.hstack((range(W), range(W)[-2::-1])):
        temp = (wn1[i] == wn1).sum(axis=0) > 1
        wn1[i, temp] = wn0[i, temp]

    # for wn1_j in wn1.T:  Check if there are collisions
    #     assert len(np.unique(wn1_j)) == W

    wn0 = wn1

    worlds = np.zeros((N, L))
    worlds[np.arange(N)[np.newaxis, :], wn1] = np.arange(W)[:, np.newaxis]
    h.set_data(worlds)
    plt.pause(0.1)

Random Walk GIF

相关问题 更多 >