Граничное условие Робина + Ньюмана на фениках
Я использую fenics на Ubuntu 18.04.
Я пытаюсь адаптировать то, что описано здесь: https://fenicsproject.org/pub/tutorial/sphinx1/._ftut1005.html
Я успешно сделал это в прямоугольной области, с условием Ньюмана $\frac{\left{\ частный u\right}}{\ частный n}=h$ вверху, и условиями Дирихле $u=g$ на остальной части граница.
Теперь я пробую ту же самую прямоугольную область, с тем же условием Ньюмана наверху, но с условием Робина на остальной части: $\alpha u+\beta\frac{\partual u}{\partal n}=h$
Для начала, я пытаюсь с $g=h=0$0 и дельтой фа дирака. Но получил эту ошибку:
/usr/local/lib/python3.6/dist-packages/matplotlib/colors.py:504: RuntimeWarning: invalid value encountered in less
xa[xa < 0] = -1
Вот мой код:
from fenics import *
from dolfin import *
import matplotlib.pyplot as plt
# plot = lambda *args, **kwargs: None
import numpy as np
#parameters
x0 = 0.0#Constant(0)
xN = 8.0#Constant(8)
y0 = -4.0#Constant(-4)
yM = 0.0#Constant(0)
nx = 50#Constant(10)
ny = 15#Constant(10)
hx = (xN-x0)/(nx-1)
hy = (yM-y0)/(ny-1)
# fuente y salida
in_x = 4.0
in_y = 0.0
out_x = 7.0
out_y = 0.0
d=1
#mesh
mesh = RectangleMesh(Point(x0,y0), Point(xN,yM), nx, ny, "right")
boundary_markers = MeshFunction('size_t', mesh, mesh.topology().dim()-1)
V = FunctionSpace(mesh, 'Lagrange', d)
#boundaries
class BoundaryX0(SubDomain):
tol = 1E-14
def inside(self, x, on_boundary):
return on_boundary and near(x[0], x0, self.tol)
class BoundaryXN(SubDomain):
tol = 1E-14
def inside(self, x, on_boundary):
return on_boundary and near(x[0], xN, self.tol)
class BoundaryY0(SubDomain):
tol = 1E-14
def inside(self, x, on_boundary):
return on_boundary and near(x[1], y0, self.tol)
class BoundaryYM(SubDomain):
tol = 1E-14
def inside(self, x, on_boundary):
return on_boundary and near(x[1], yM, self.tol)
bx0 = BoundaryX0()
bx0.mark(boundary_markers, 4)
bx1 = BoundaryXN()
bx1.mark(boundary_markers, 1)
by0 = BoundaryY0()
by0.mark(boundary_markers, 2)
byM = BoundaryYM()
byM.mark(boundary_markers, 3)
# dirichlet
h = Constant(0) #Expression("100*cos(x[0]*x[1])", degree=d) #Constant(0)
# newmann
g = Constant(0) #Expression("cos(x[0])", degree=1) #Constant(0)
boundary_conditions = {4: {'Robin': h},
1: {'Robin': h},
2: {'Robin': h},
3: {'Neumann': g}}
ds = Measure('ds', domain=mesh, subdomain_data=boundary_markers)
# Define variational problem
u = TrialFunction(V)
v = TestFunction(V)
p = Expression("0.015+0.005*cos(3.5+x[1])", degree=d) #Constant(0.01) # #
I = 1.0
DV = hx*hy
M = I/DV
alpha = Expression("x[0]/(x[0]*x[0]+x[1]*x[1])", degree=d)
beta = Constant(1.0)
# source
f = Constant(0) #Expression("cos(x[0]+x[1])", degree=d) #Constant(0) #source term
integrals_R_L = []
integrals_R_a = []
for i in boundary_conditions:
if 'Robin' in boundary_conditions[i]:
if boundary_conditions[i]['Robin'] != 0:
hi = boundary_conditions[i]['Robin']
integrals_R_L.append((p*hi*v/beta)*ds(i))
integrals_R_a.append((alpha*p*u*v/beta)*ds(i))
a = p*inner(nabla_grad(u), nabla_grad(v))*dx + sum(integrals_R_a)
L = f*v*dx + p*g*v*ds(3) + sum(integrals_R_L)
# Compute solution
A, b = assemble_system(a, L)
# Add delta
delta1 = PointSource(V, Point(in_x, in_y), M)
delta1.apply(b)
# delta2 = PointSource(V, Point(out_x, out_y), -M)
# delta2.apply(b)
u = Function(V)
solve(A, u.vector(), b)
#plot y=k
y = 0.0
meshx = IntervalMesh(nx, x0, xN)
W = FunctionSpace(meshx,"Lagrange",d)
w = Function(W)
coor = meshx.coordinates()
w_array = w.vector()#.array()
if meshx.num_vertices() == len(w_array):
for i in range(meshx.num_vertices()):
w_array[i] = u(coor[i][0],y)
w.vector().set_local(w_array)
# plots
plt.subplot(2,3,1)
plot(f, mesh=mesh, title = "f: source term")
# plt.savefig('vc_poisson/f.png')
plt.subplot(2,3,2)
plot(h, mesh=mesh, title = "h: Dirichlet boundary condition")
plt.subplot(2,3,3)
plot(g, mesh=mesh, title = "g: Newmman boundary condition")
# plt.savefig('vc_poisson/g.png')
plt.subplot(2,3,4)
plot(p, mesh=mesh, title = "\sigma: resistivity distribution")
# plt.savefig('vc_poisson/g.png')
plt.subplot(2,3,5)
plot(u, title = "numerical solution")
# plt.savefig('vc_poisson/sol.png')
# plt.subplot(2,3,5)
# plot(mesh)
# plot(u, interactive=True, title = "numerical solution + mesh")
plt.subplot(2,3,6)
plot(w, title="numerical solution u(x,0)")
plt.savefig('vc_poisson/solution.png')
plt.figure()
plot(u.root_node(), mode='warp', title="numerical solution surface")
plt.savefig('vc_poisson/solution_surface.png')
# Save solution to file in VTK format
vtkfile = File('vc_poisson/solution.pvd')
vtkfile << u
plt.show()
Похоже, ошибка в операторе a
на интеграле по границе робина sum(integrals_R_a)
, Если я это прокомментирую, код запускается, но очевидно, что решение не то, что я ожидаю. Я должен добавить, что мне 2 недели в фенике
Итак, что я делаю не так?