湖南省网站建设_网站建设公司_Java_seo优化
2025/12/23 12:38:43 网站建设 项目流程

随机短纤维(线型)和圆形骨料细观模型,骨料纤维之间不干涉,可调节半径,长度,也可随机半径长度,纤维可修改为3维圆柱,代码可修改性好。

直接上代码吧,咱们今天要搞个能批量生成随机纤维的脚本。先看这个核心函数,保证纤维之间绝对不打架,还能自定义各种参数。这玩意儿改吧改吧直接扔到有限元软件里用都没问题。

先来点硬核的——生成参数的函数。这个generatefiberparams能吐出来一堆随机参数,注意看里面的np.random.uniform,这里玩了个小花招让短纤维长度和半径关联:

def generate_fiber_params(num_fibers, max_radius=0.5, area_size=10): params = [] for _ in range(num_fibers): radius = np.random.uniform(0.1, max_radius) length = np.random.uniform(2*radius, 4*radius) # 长度至少是直径的两倍 angle = np.random.uniform(0, 2*np.pi) params.append((radius, length, angle)) return params

重点是这个长度随机范围,2倍半径起步,防止生成牙签似的畸形纤维。angle参数留着后面计算方向向量用,这样纤维朝向就不会全是一个德性了。

接下来是重头戏——碰撞检测。这个check_overlap函数用了个空间划分的骚操作,把区域划分成网格来加速检测。直接两层循环检测所有纤维会要命,特别是数量多的时候:

def check_overlap(new_fiber, existing_fibers, grid, cell_size): x, y, r = new_fiber[0], new_fiber[1], new_fiber[3] grid_x = int(x / cell_size) grid_y = int(y / cell_size) for i in range(-1, 2): for j in range(-1, 2): if (grid_x+i, grid_y+j) in grid: for fiber in grid[(grid_x+i, grid_y+j)]: dx = x - fiber[0] dy = y - fiber[1] if dx**2 + dy**2 < (r + fiber[3])**2: return True return False

注意这里判断距离用的是平方比较,比开根号快不止一个量级。网格划分的cell_size建议设成最大纤维直径的1.5倍,这样基本不会漏检。

三维扩展其实特简单,把z坐标加进来,判断距离时多加个dz²就行。不过可视化得用Mayavi或者PyVista了,matplotlib的3D功能太弱鸡。比如这样改造成圆柱体:

class Fiber3D: def __init__(self, start, end, radius): self.start = np.array(start) self.end = np.array(end) self.radius = radius self.axis = self.end - self.start self.length = np.linalg.norm(self.axis) def to_mesh(self): # 返回三角面片数据,用于导出或可视化 cylinder = pv.Cylinder(self.start, self.axis, self.radius, self.length) return cylinder

参数化设计才是精髓。在脚本开头定义这几个全局变量,想怎么改就怎么改:

CONFIG = { 'domain_size': (50, 50, 50), # 三维区域尺寸 'radius_range': (0.2, 1.0), # 半径随机范围 'aspect_ratio': (3, 8), # 长径比范围 'max_attempts': 100 # 单个纤维的最大尝试次数 }

遇到生成失败的情况别慌,加个尝试次数限制。当连续100次都找不到合适位置时,自动停止生成,避免死循环。实测在纤维体积占比30%以下时基本都能成功生成。

最后说下怎么导出数据。建议保存成CSV或者JSON,这样其他软件好读取。比如:

def save_to_csv(fibers, filename): with open(filename, 'w') as f: f.write("x,y,z,dx,dy,dz,radius\n") for fiber in fibers: line = f"{fiber.start[0]},{fiber.start[1]},{fiber.start[2]},"\ f"{fiber.axis[0]},{fiber.axis[1]},{fiber.axis[2]},"\ f"{fiber.radius}\n" f.write(line)

这个格式直接把纤维起点和方向向量都记录下来了,拿到ABAQUS或者COMSOL里用矢量数据直接生成圆柱体,爽歪歪。

需要专业的网站建设服务?

联系我们获取免费的网站建设咨询和方案报价,让我们帮助您实现业务目标

立即咨询