I have the general formula of an ellipsoid:
A*x**2 + C*y**2 + D*x + E*y + B*x*y + F + G*z**2 = 0
where A,B,C,D,E,F,G are constant factors.
How can I plot this equation as a 3D plot in matplotlib? (A wireframe would be best.)
I saw this example but it is in parametric form and I am not sure how to put the z-coordinates in this code. Is there a way to keep the general form to plot this without the parametric form?
I started to put this in some kind of code like this:
from mpl_toolkits import mplot3d
%matplotlib notebook
import numpy as np
import matplotlib.pyplot as plt
def f(x, y):
return ((A*x**2 + C*y**2 + D*x + E*y + B*x*y + F))
def f(z):
return G*z**2
x = np.linspace(-2200, 1850, 30)
y = np.linspace(-100, 60, 30)
z = np.linspace(-100, 60, 30)
X, Y, Z = np.meshgrid(x, y, z)
fig = plt.figure()
ax = fig.add_subplot(111, projection='3d')
ax.plot_wireframe(X, Y, Z, rstride=10, cstride=10)
ax.set_xlabel('x')
ax.set_ylabel('y')
ax.set_zlabel('z');
I got this error:
---------------------------------------------------------------------------
ValueError Traceback (most recent call last)
<ipython-input-1-95b1296ae6a4> in <module>()
18 fig = plt.figure()
19 ax = fig.add_subplot(111, projection='3d')
---> 20 ax.plot_wireframe(X, Y, Z, rstride=10, cstride=10)
21 ax.set_xlabel('x')
22 ax.set_ylabel('y')
C:\Program Files (x86)\Microsoft Visual Studio\Shared\Anaconda3_64\lib\site-packages\mpl_toolkits\mplot3d\axes3d.py in plot_wireframe(self, X, Y, Z, *args, **kwargs)
1847 had_data = self.has_data()
1848 if Z.ndim != 2:
-> 1849 raise ValueError("Argument Z must be 2-dimensional.")
1850 # FIXME: Support masked arrays
1851 X, Y, Z = np.broadcast_arrays(X, Y, Z)
ValueError: Argument Z must be 2-dimensional.
Side note, but what you have is not the most general equation for a 3d ellipsoid. Your equation can be rewritten as
which means that in effect for each value of
z
you get a different level of a 2d ellipse, and the slices are symmetric with respect to thez = 0
plane. This shows how your ellipsoid is not general, and it helps check the results to make sure that what we get makes sense.Assuming we take a general point
r0 = [x0, y0, z0]
, you havewhere
where
@
stands for matrix-vector or vector-vector product.You could take your function and plot its isosurface, but that would be suboptimal: you would need a gridded approximation for your function which is very expensive to do to sufficient resolution, and you'd have to choose the domain for this sampling wisely.
Instead you can perform a principal axis transformation on your data to generalize the parametric plot of a canonical ellipsoid that you yourself linked.
The first step is to diagonalize
M
asM = V @ D @ V.T
, whereD
is diagonal. Since it's a real symmetric matrix this is always possible andV
is orthogonal. Then we havewhich we can regroup as
which motivates the definition of the auxiliary coordinates
r1 = V.T @ r0
and vectorb1 = b0 @ V
, for which we getSince
D
is a symmetric matrix with the eigenvaluesd1, d2, d3
in its diagonal, the above is the equationwhere
r1 = [x1, x2, x3]
andb1 = [b11, b12, b13]
.What's left is to switch from
r1
tor2
such that we remove the linear terms:So we define
For these we finally have
which is the canonical form of a second-order surface. In order for this to meaningfully correspond to an ellipsoid we must ensure that
d1
,d2
,d3
andc2
are all strictly positive. If this is guaranteed then the semi-major axes of the canonical form aresqrt(c2/d1)
,sqrt(c2/d2)
andsqrt(c2/d3)
.So here's what we do:
[x2, y2, z2]
r2 - r1
) to get[x1, y1, z1]
V
to getr0
, the actual[x, y, z]
coordinates we're interested in.Here's how I'd implement this:
Here's an example with an ellipsoid and proof that it works:
The actual plotting from here is trivial. Using the 3d scaling hack from this answer to preserve equal axes:
Here's how the result looks:
Rotating it around it's nicely visible that the surface is indeed reflection symmetric with respect to the
z = 0
plane, which was evident from the equation.You can change the
n_theta
andn_phi
keyword arguments to the function to generate a grid with a different mesh. The fun thing is that you can take any scattered points lying on the unit sphere and plug it into the definition ofr2
in the functionget_ellipsoid_coordinates
(as long as this array has a first dimension of size 3), and the output coordinates will have the same shape, but they will be transformed onto the actual ellipsoid.You can also use other libraries to visualize the surface, for instance mayavi where you can either plot the surface we just computed, or compare it with an isosurface which is built-in there.