如何增加 3D 绘图中轴的尺寸(拉伸)

2025-04-10 09:46:00
admin
原创
17
摘要:问题描述:我目前有这个:x,y,z = data.nonzero() fig = plt.figure() ax = fig.add_subplot(111, projection='3d') ax.scatter(x, y, z, zdir='z', c= 'red') plt.savefig(&q...

问题描述:

我目前有这个:

x,y,z = data.nonzero()    
fig = plt.figure()
ax = fig.add_subplot(111, projection='3d')
ax.scatter(x, y, z, zdir='z', c= 'red')
plt.savefig("plot.png")

这将产生:
在此处输入图片描述

我想做的是将其拉伸,使 Z 轴高出 9 倍,并保持 X 和 Y 不变。但我想保持相同的坐标。

到目前为止我尝试过这个家伙:

fig = plt.figure(figsize=(4.,35.))

但这只是拉长了 plot.png 图像。


解决方案 1:

下面的代码示例提供了一种相对于其他轴缩放每个轴的方法。但是,要做到这一点,您需要修改 Axes3D.get_proj 函数。下面是一个基于 matplot lib 提供的示例:http://matplotlib.org/1.4.0/mpl_toolkits/mplot3d/tutorial.html#line-plots

(本答案末尾有一个较短的版本)

from mpl_toolkits.mplot3d.axes3d import Axes3D
from mpl_toolkits.mplot3d import proj3d

import matplotlib as mpl
import numpy as np
import matplotlib.pyplot as plt

#Make sure these are floating point values:                                                                                                                                                                                              
scale_x = 1.0
scale_y = 2.0
scale_z = 3.0

#Axes are scaled down to fit in scene                                                                                                                                                                                                    
max_scale=max(scale_x, scale_y, scale_z)

scale_x=scale_x/max_scale
scale_y=scale_y/max_scale
scale_z=scale_z/max_scale

#Create scaling matrix                                                                                                                                                                                                                   
scale = np.array([[scale_x,0,0,0],
                  [0,scale_y,0,0],
                  [0,0,scale_z,0],
                  [0,0,0,1]])
print scale

def get_proj_scale(self):
    """                                                                                                                                                                                                                                    
    Create the projection matrix from the current viewing position.                                                                                                                                                                        

    elev stores the elevation angle in the z plane                                                                                                                                                                                         
    azim stores the azimuth angle in the x,y plane                                                                                                                                                                                         

    dist is the distance of the eye viewing point from the object                                                                                                                                                                          
    point.                                                                                                                                                                                                                                 

    """
    relev, razim = np.pi * self.elev/180, np.pi * self.azim/180

    xmin, xmax = self.get_xlim3d()
    ymin, ymax = self.get_ylim3d()
    zmin, zmax = self.get_zlim3d()

    # transform to uniform world coordinates 0-1.0,0-1.0,0-1.0                                                                                                                                                                             
    worldM = proj3d.world_transformation(
        xmin, xmax,
        ymin, ymax,
        zmin, zmax)

    # look into the middle of the new coordinates                                                                                                                                                                                          
    R = np.array([0.5, 0.5, 0.5])

    xp = R[0] + np.cos(razim) * np.cos(relev) * self.dist
    yp = R[1] + np.sin(razim) * np.cos(relev) * self.dist
    zp = R[2] + np.sin(relev) * self.dist
    E = np.array((xp, yp, zp))

    self.eye = E
    self.vvec = R - E
    self.vvec = self.vvec / proj3d.mod(self.vvec)

    if abs(relev) > np.pi/2:
    # upside down                                                                                                                                                                                                                          
      V = np.array((0, 0, -1))
    else:
      V = np.array((0, 0, 1))
    zfront, zback = -self.dist, self.dist

    viewM = proj3d.view_transformation(E, R, V)
    perspM = proj3d.persp_transformation(zfront, zback)
    M0 = np.dot(viewM, worldM)
    M = np.dot(perspM, M0)

    return np.dot(M, scale);

Axes3D.get_proj=get_proj_scale

"""
You need to include all the code above.
From here on you should be able to plot as usual.
"""

mpl.rcParams['legend.fontsize'] = 10

fig = plt.figure(figsize=(5,5))
ax = fig.gca(projection='3d')
theta = np.linspace(-4 * np.pi, 4 * np.pi, 100)
z = np.linspace(-2, 2, 100)
r = z**2 + 1
x = r * np.sin(theta)
y = r * np.cos(theta)
ax.plot(x, y, z, label='parametric curve')
ax.legend()

plt.show()

标准输出:

正常比例

按照 (1, 2, 3) 缩放:

比例尺 x = 1,比例尺 y = 2,比例尺 z = 3

按照 (1, 1, 3) 缩放:

比例尺 x = 1,比例尺 y = 1,比例尺 z = 3

我特别喜欢这种方法的原因是,交换 z 和 x,按(3,1,1)缩放:

交换 z 和 x,scale_x=4

下面是该代码的较短版本。

from mpl_toolkits.mplot3d.axes3d import Axes3D
from mpl_toolkits.mplot3d import proj3d

import matplotlib as mpl
import numpy as np
import matplotlib.pyplot as plt

mpl.rcParams['legend.fontsize'] = 10

fig = plt.figure(figsize=(5,5))
ax = fig.gca(projection='3d')
theta = np.linspace(-4 * np.pi, 4 * np.pi, 100)
z = np.linspace(-2, 2, 100)
r = z**2 + 1
x = r * np.sin(theta)
y = r * np.cos(theta)


"""                                                                                                                                                    
Scaling is done from here...                                                                                                                           
"""
x_scale=1
y_scale=1
z_scale=2

scale=np.diag([x_scale, y_scale, z_scale, 1.0])
scale=scale*(1.0/scale.max())
scale[3,3]=1.0

def short_proj():
  return np.dot(Axes3D.get_proj(ax), scale)

ax.get_proj=short_proj
"""                                                                                                                                                    
to here                                                                                                                                                
"""

ax.plot(z, y, x, label='parametric curve')
ax.legend()

plt.show()

解决方案 2:

请注意,下面的答案简化了补丁,但使用与@ChristianSarofeen 的答案相同的基本原理。

解决方案

正如其他答案中指出的那样,它不是当前在 matplotlib 中实现的功能。但是,由于您请求的只是一个可以应用于 matplotlib 使用的现有投影矩阵的3D 变换,并且得益于 Python 的出色功能,可以使用简单的一行程序解决此问题:

ax.get_proj = lambda: np.dot(Axes3D.get_proj(ax), np.diag([scale_x, scale_y, scale_z, 1]))

其中scale_xscale_yscale_z是从 0 到 1 的值,它们将相应地沿每个轴重新缩放绘图。ax只是 3D 轴,可以通过以下方式获得ax = fig.gca(projection='3d')

解释

解释一下,函数get_projAxes3D当前观察位置生成投影矩阵。将其乘以缩放矩阵:

scale_x, 0,       0
0,       scale_y, 0
0,       0,       scale_z
0,       0,       1

将缩放纳入渲染器使用的投影中。因此,我们在这里所做的是将原始get_proj函数替换为一个表达式,该表达式取原始结果get_proj并将其乘以缩放矩阵。

例子

用标准参数函数示例来说明结果:

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

fig = plt.figure()
ax = fig.gca(projection='3d')
theta = np.linspace(-4 * np.pi, 4 * np.pi, 100)
z = np.linspace(-2, 2, 100)
r = z ** 2 + 1
x = r * np.sin(theta)
y = r * np.cos(theta)

# OUR ONE LINER ADDED HERE:
ax.get_proj = lambda: np.dot(Axes3D.get_proj(ax), np.diag([0.5, 0.5, 1, 1]))

ax.plot(x, y, z)
plt.show()

对于值0.5, 0.5, 1,我们得到:

在此处输入图片描述

而对于值0.2, 1.0, 0.2,我们得到:

在此处输入图片描述

解决方案 3:

在我的例子中,我想将 z 轴拉伸 2 倍,以获得更好的点可见性

from mpl_toolkits import mplot3d
from mpl_toolkits.mplot3d import Axes3D

import matplotlib.pyplot as plt
# plt.rcParams["figure.figsize"] = (10,200)
# plt.rcParams["figure.autolayout"] = True
ax = plt.axes(projection='3d')
ax.set_box_aspect(aspect = (1,1,2))

ax.plot(dataX,dataY,dataZ)

解决方案 4:

默认情况下,mplot3d 会在非常高的绘图的顶部和底部留出相当多的空间。但是,你可以诱使它使用 填充该空间fig.subplots_adjust,并将顶部和底部延伸出正常绘图区域(即top > 1bottom < 0)。对于你的特定绘图,这里可能需要一些反复试验。

我已经为 x、y 和 z 创建了一些随机数组,其限制与您的图类似,并且发现下面的参数(bottom=-0.15top = 1.2)似乎运行正常。

您可能还想更改ax.view_init以设置一个良好的视角。

import matplotlib.pyplot as plt
from mpl_toolkits.mplot3d import axes3d
from numpy import random

# Make some random data with similar limits to the OP's example
x,y,z=random.rand(3,100)
z*=250
y*=800
y+=900
x*=350
x+=1200

fig=plt.figure(figsize=(4,35))

# Set the bottom and top outside the actual figure limits, 
# to stretch the 3D axis
fig.subplots_adjust(bottom=-0.15,top=1.2)

ax = fig.add_subplot(111, projection='3d')

# Change the viewing angle to an agreeable one
ax.view_init(2,None)

ax.scatter(x, y, z, zdir='z', c= 'red')
plt.savefig("plot.png")

IT科技

解决方案 5:

听起来你正在尝试调整绘图的比例。我认为没有办法将线性比例拉伸到用户要求的水平,但你可以使用set_yscale()set_xscale()set_zscale()来相互改变比例。

直观地说set_yscale(log),,,可能会解决您的问题。set_xscale(log)`set_zscale(linear)`

可能更好的选择:指定一个拉伸,将它们全部设置为具有相同对数底数的 symlog,然后使用 kwargs 根据您的规范指定 Z 轴的 symlog 比例linscalex/linscaley

更多内容请见:

http://matplotlib.org/mpl_toolkits/mplot3d/api.html

解决方案 6:

我在搜索类似问题时发现了这一点。经过一些实验,也许我可以在这里分享我的一些初步发现。matplotlib 库非常强大!!(我是新手)。请注意,与这个问题非常相似,我想要的只是“视觉上”拉伸图表而不扭曲它。

背景故事(仅显示关键代码片段,以避免对了解该库的人造成不必要的混乱,如果您想要可运行的代码,请发表评论):我有三个 1-d ndarray,分别代表 X、Y 和 Z 数据点。显然我不能使用 plot_surface(因为它需要每个 dim 的 2d ndarray),所以我选择了非常有用的 plot_trisurf:

fig = plt.figure()
ax = Axes3D(fig)
3d_surf_obj = ax.plot_trisurf(X, Y, Z_defl, cmap=cm.jet,linewidth=0,antialiased=True)

在此处输入图片描述

您可以将情节想象成一艘在波浪中变形的漂浮驳船……正如您所看到的,轴的拉伸使其在视觉上非常具有欺骗性(请注意,x 应该比 y 和 >>>>> z 长 6 倍)。虽然情节要点是正确的,但我至少想要一些在视觉上更“拉伸”的东西。如果可以的话,我正在寻找一个快速解决方案。长话短说,我发现使用“figure.figsize”常规设置(见下面的代码片段)取得了一些成功。

    matplotlib.rcParams.update({'font.serif': 'Times New Roman',
                                'font.size': 10.0,
                                'axes.labelsize': 'Medium',
                                'axes.labelweight': 'normal',
                                'axes.linewidth': 0.8,
                                 ###########################################
                                 # THIS IS THE IMPORTANT ONE FOR STRETCHING
                                 # default is [6,4] but...i changed it to
                                'figure.figsize':[15,5]   # THIS ONE #
                              })

对于 [15,5] 我得到了类似的结果......

在此处输入图片描述

非常整洁!!

所以我开始推动它....并上升到 [20,6] 然后决定在那里安定下来..

在此处输入图片描述

如果您想尝试在视觉上拉伸垂直轴,请尝试使用如下比例...... [7,10],在这种情况下,这给了我...

在此处输入图片描述

不太寒酸!

应该为了视觉能力而这样做。

解决方案 7:

将所有 z 值乘以 9,

ax.scatter(x, y, 9*z, zdir='z', c= 'red')

然后为 z 轴提供自定义绘图标签和间距。

ax.ZTick = [0,-9*50, -9*100, -9*150, -9*200];
ax.ZTickLabel = {'0','-50','-100','-150','-200'};
相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   2482  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1533  
  PLM(产品生命周期管理)项目对于企业优化产品研发流程、提升产品质量以及增强市场竞争力具有至关重要的意义。然而,在项目推进过程中,范围蔓延是一个常见且棘手的问题,它可能导致项目进度延迟、成本超支以及质量下降等一系列不良后果。因此,有效避免PLM项目范围蔓延成为项目成功的关键因素之一。以下将详细阐述三大管控策略,助力企业...
plm系统   0  
  PLM(产品生命周期管理)项目管理在企业产品研发与管理过程中扮演着至关重要的角色。随着市场竞争的加剧和产品复杂度的提升,PLM项目面临着诸多风险。准确量化风险优先级并采取有效措施应对,是确保项目成功的关键。五维评估矩阵作为一种有效的风险评估工具,能帮助项目管理者全面、系统地评估风险,为决策提供有力支持。五维评估矩阵概述...
免费plm软件   0  
  引言PLM(产品生命周期管理)开发流程对于企业产品的全生命周期管控至关重要。它涵盖了从产品概念设计到退役的各个阶段,直接影响着产品质量、开发周期以及企业的市场竞争力。在当今快速发展的科技环境下,客户对产品质量的要求日益提高,市场竞争也愈发激烈,这就使得优化PLM开发流程成为企业的必然选择。缺陷管理工具和六西格玛方法作为...
plm产品全生命周期管理   0  
热门文章
项目管理软件有哪些?
曾咪二维码

扫码咨询,免费领取项目管理大礼包!

云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用