在3dp中保持等高宽比的同时减小轴长

2024-05-13 08:14:09 发布

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

我正在尝试用python创建一个3-D图和一个2-D并排绘制。我需要两个图的长宽比相等,我使用这个答案提供的代码进行管理:https://stackoverflow.com/a/31364297/125507。我现在面临的问题是如何有效地“裁剪”三维图,这样它就不会占用太多空白。也就是说,我想减少X轴和Y轴的长度,同时保持与(更长)Z轴相同的比例。以下是示例代码和绘图:

from mpl_toolkits.mplot3d import Axes3D
from matplotlib import cm
import matplotlib.pyplot as plt
import numpy as np

def set_axes_equal(ax):
    '''Make axes of 3D plot have equal scale so that spheres appear as spheres,
    cubes as cubes, etc..  This is one possible solution to Matplotlib's
    ax.set_aspect('equal') and ax.axis('equal') not working for 3D.

    Input
      ax: a matplotlib axis, e.g., as output from plt.gca().
    '''

    x_limits = ax.get_xlim3d()
    y_limits = ax.get_ylim3d()
    z_limits = ax.get_zlim3d()

    x_range = abs(x_limits[1] - x_limits[0])
    x_middle = np.mean(x_limits)
    y_range = abs(y_limits[1] - y_limits[0])
    y_middle = np.mean(y_limits)
    z_range = abs(z_limits[1] - z_limits[0])
    z_middle = np.mean(z_limits)

    # The plot bounding box is a sphere in the sense of the infinity
    # norm, hence I call half the max range the plot radius.
    plot_radius = 0.5*max([x_range, y_range, z_range])

    ax.set_xlim3d([x_middle - plot_radius, x_middle + plot_radius])
    ax.set_ylim3d([y_middle - plot_radius, y_middle + plot_radius])
    ax.set_zlim3d([z_middle - plot_radius, z_middle + plot_radius])

ax  = [None]*2
fig = plt.figure()
ax[0] = fig.add_subplot(121, projection='3d', aspect='equal')
ax[1] = fig.add_subplot(122, aspect='equal')



nn   = 30
phis = np.linspace(0,np.pi,  nn).reshape(1,nn)
psis = np.linspace(0,np.pi*2,nn).reshape(nn,1)
ones = np.ones((nn,1))
el_h = np.linspace(-5, 5, nn).reshape(1,nn)

x_sph = np.sin(phis)*np.cos(psis)
y_sph = np.sin(phis)*np.sin(psis)
z_sph = np.cos(phis)*ones

x_elp = np.sin(phis)*np.cos(psis)*.25
y_elp = np.sin(phis)*np.sin(psis)*.25
z_elp = el_h*ones

ax[0].scatter(x_sph, y_sph, z_sph)
ax[0].scatter(x_elp, y_elp, z_elp)
ax[1].scatter(y_sph, z_sph)
ax[1].scatter(y_elp, z_elp)

for ii in range(2):
    ax[ii].set_xlabel('X')
    ax[ii].set_ylabel('Y')
ax[0].set_zlabel('Z')

set_axes_equal(ax[0])
plt.savefig('SphereElipse.png', dpi=300)

下面是它的图像输出: 3-D and 2-D sphere and ellipse side-by-side

显然,2D绘图在保持比例的同时自动修改轴的长度,但是3D绘图没有,这导致了一个很小的表示,不能很好地利用分配给它的子图的空间。有办法吗?这个问题类似于之前的一个未回答的问题How do I crop an Axes3D plot with square aspect ratio?,只是它增加了多个子批次的规定,这意味着那里提供的答案不起作用。在


Tags: middleplotasnprangennsinequal