当前位置: 首页
AI教程
CSS 3D从布局到立方体的完整教程

CSS 3D从布局到立方体的完整教程

热心网友 时间:2026-07-01
转载

CSS 3D:从布局到立方体

初学CSS 3D时,很多人误以为它只是一堆旋转和透视的炫技。但深入实践后会发现,CSS不仅能胜任2D布局,还能借助一系列3D属性,在浏览器中直接构建立体空间。

CSS 3D:从布局到立方体

CSS 3D的真正价值远不止“做出3D效果”——它还能触发GPU加速实现性能优化。有时,即便是一个纯2D界面,开发者也会刻意加上translateZ(0)来强制启用硬件加速,让动画更顺滑。这一思路与Canvas里getContext('webgl')调用显卡如出一辙:浏览器中凡涉及渲染,最终都依赖GPU的表现。

在正式进入3D世界之前,必须先扎实掌握布局基础知识。原因在于:CSS 3D的本质是在“已经布局好的盒子上叠加空间变换”——布局没有搞清楚,变换就无从下手。

布局:外层负责布局,内层负责内容

这句话值得反复强调:

听起来像一句废话,但在实际开发中,它是非常重要的拆分原则。例如下面这个3D立方体的HTML结构:


<div class="box-wrap">
  <div class="box">
    <div class="face front">div>
    <div class="face back">div>
    <div class="face left">div>
    <div class="face right">div>
    <div class="face top">div>
    <div class="face bottom">div>
  div>
div>
  • .box-wrap 是外层,决定“立方体放在页面中的位置”;
  • .box 是中层,决定“立方体整体的旋转”;
  • .face 是内层,决定“每个面的样式和内容”。

这种“布局与内容分离”的思路在后文中会反复出现。编写CSS时如果不分开,很容易出现“修改一处样式却把整个结构打乱”的情况。

水平垂直居中:先有视口,再谈居中

布局中最常见的需求就是“水平垂直居中”。我们从最基础的视口单位讲起。

vh / vw:视口单位

让一个元素铺满全屏,最直接的做法如下:

html, body {
  width: 100%;  /* 块级元素宽度默认 100% */
  height: 100vh; /* CSS3 新增的视口单位 */
}

vh 是 viewport-height,vw 是 viewport-width。它们将整个屏幕(PC端、移动端等)等比例分为100份,从而实现移动端适配。

但移动端存在一个典型问题:在Safari等浏览器上,100vh 有时会包含地址栏和工具栏的高度,导致元素超出预期。此时可以考虑使用 100dvh(动态视口高度)作为更精准的替代方案——当地址栏滑出时高度会自动调整。

flex 实现水平垂直居中

有了全屏视口之后,居中交给flex即可:

html, body {
  display: flex;
  flex-direction: column; /* 定义主轴方向,剩余方向即为次轴 */
  justify-content: center; /* 主轴对齐方式 */
  align-items: center;     /* 次轴对齐方式 */
}

三个关键点:

  1. display: flex 会在当前盒子内开启弹性格式化上下文;
  2. flex-direction 决定主轴方向,另一方向即为次轴;
  3. justify-content 控制主轴,align-items 控制次轴。

flex 是移动端视窗大小多变情况下最常用的布局方案。后续所有的居中场景,几乎都是这种“父容器flex + 子元素居中”的模式。

行内 / 块级元素:display 属性的本质

布局搞懂之后,下一个容易混淆的点是 display 属性。HTML元素本身分为两类:

  • 块级元素(divul 等)
    • display 默认值为 block
    • 独占一行
    • 可以设置宽高
    • 用于构建盒子结构
  • 行内元素(span 等)
    • display 默认值为 inline
    • 不独占一行
    • 无法设置宽高
    • 用于包裹文字、超链接、图片等内容

浏览器会给某些元素默认的 display 行为,但我们可以通过 display 手动切换 inline / block,将块级元素改为行内元素,或反之。

flex:父与子的布局关系

当设置 display: flex 时,会在当前盒子(即flex容器)内开启一个弹性格式化上下文。弹性布局描述的是父元素与子元素之间的布局关系,子元素默认会沿主轴对齐,受到父元素的约束。

示例 3.html 清晰地体现了这一点:

<div class="box">
  <div class="item">1div>
  <div class="item">2div>
  <div class="item">3div>
  <div class="item">4div>
div>
.box {
  /* 弹性布局中,子元素默认沿主轴对齐,受父元素约束,开启格式化上下文 */
  display: flex;
}
.item {
  flex: 1;
  background-color: #9c1818;
  width: 50%;
  text-align: center;
}

四个 .item 均设置了 flex: 1,它们会平均分配主轴空间,无论 width: 50% 具体是多少。这正是flex“父管布局”的特性。

inline-block 的经典坑:空白字符间隙

display: inline-block 是一个介于行内与块级之间的属性值:

  • 不独占一行
  • 同时可以设置宽高

但它有一个著名的坑——默认的空格符会占据一定大小。HTML源码中的换行、回车、空格,都会被浏览器渲染成一个空白字符,导致两个设置 50% 宽度的盒子总宽度超过 100%,第二个盒子被挤到下一行。

2.html 就是对这个坑的复现:

<div class="box">1div>
<div class="box">2div>
.box {
  background-color: #9c1818;
  display: inline-block;
  width: 50%;
  /* 两个 50% 宽度盒子 + 间隙 > 100%,第二个会被挤到下一行 */
}

解决方法有几种:

  • 将HTML标签紧挨着写,不留空白;
  • 父元素设置 font-size: 0,子元素再单独设置字号;
  • 直接使用 flex 布局,完全避开 inline-block 的这个坑。

实际开发中,能用flex就别用inline-block做布局——这是从实践中总结出的重要经验。

定位:relative 与 absolute

讲完display,下一个基础是定位。CSS 3D中六面立方体的每个面都需要用绝对定位叠加在一起,因此必须先理清这个知识点。

position: relative; /* 相对定位 */
position: absolute; /* 绝对定位 */
  • relative:相对于自身原有位置偏移,仍然占据文档流;
  • absolute:脱离文档流,相对于最近的非static定位祖先元素进行偏移。

在立方体例子中,.box 设置为 position: relative,作为定位上下文;六个 .face 都是 position: absolute,全部叠加在 .box 的左上角,然后通过 translate 将它们各自移到对应方向。这正是“外层relative + 内层absolute”的经典组合。

CSS 3D 核心:perspective 与 transform-style

补完布局基础后,正式进入3D世界。CSS 3D的核心实际上只有两个属性。

perspective:视距

perspective 定义了观察者到z=0平面的距离,单位是px。它决定了3D效果的“透视强度”——值越小,透视越夸张(近大远小越明显);值越大,越接近正交投影。

.box-wrap {
  width: 200px;
  height: 200px;
  perspective: 600px; /* 3D 核心:视距 */
}

注意:perspective 必须写在需要被透视的元素的父元素上,而不是写在元素本身。这是一个非常容易踩的坑。

transform-style: preserve-3d

光有 perspective 还不够。默认情况下,子元素会被“压平”在父元素的平面上;要构建3D立方体,必须让父元素保留子元素的3D空间:

.box {
  width: 200px;
  height: 200px;
  position: relative;
  transform-style: preserve-3d; /* 保留子元素的 3D 空间 */
  animation: rotate 6s linear infinite;
}

transform-style: preserve-3d 是实现3D立方体的关键语句。缺少它,六个面会被压成一个平面,无论怎么 translateZ 都没有效果。

六面立方体:translate + rotate 的组合

理解了以上两个核心属性后,构建立方体就变成了一个几何问题。一个200×200的立方体,每个面都需要从原点(左上角)移动到对应的方位。

先把每个面叠在原点

六个面共用一组基础样式,先用绝对定位将它们叠在 .box 的左上角:

.face {
  width: 200px;
  height: 200px;
  left: -50px;  /* (100 - 200) / 2,让面相对外层 100x100 居中 */
  top: -50px;
  position: absolute;
  display: flex;
  justify-content: center;
  align-items: center;
  font-size: 30px;
  color: #b41c1c;
  opacity: 0.8;
}

这里的 left: -50pxtop: -50px 是为了将200×200的面相对外层100×100的容器居中偏移((100-200)/2 = -50)。

沿三根轴将每个面推出去

立方体的六个面对应三根轴的正负方向。每个面的操作都是“先沿轴平移100px,再旋转到对应朝向”:

.front  { background: #429911; transform: translateZ(100px); }                     /* 朝前,沿 z 轴正方向 */
.back   { background: #114299; transform: translateZ(-100px) rotateY(180deg); }   /* 朝后,先后退再翻转 */
.left   { background: #994211; transform: translateX(-100px) rotateY(-90deg); } /* 逆时针为负 */
.right  { background: #429911; transform: translateX(100px) rotateY(90deg); }   /* 顺时针为正 */
.top    { background: #994211; transform: translateY(-100px) rotateX(90deg); }
.bottom { background: #429911; transform: translateY(100px) rotateX(-90deg); }

这里有一个容易记混的地方:旋转方向。

  • rotateY(90deg):绕Y轴顺时针旋转90度(从+Y轴方向看向原点);
  • rotateY(-90deg):逆时针旋转。

为什么 left 面要先 translateX(-100px)rotateY(-90deg)?因为如果先旋转再平移,平移方向会跟着旋转矩阵变化,容易算错。采用先平移到位置再旋转朝向的顺序,是更不容易出错的处理方式。

顺序很重要:translate 在前,rotate 在后

CSS的 transform 是从右往左执行的,但当我们写在一起时,习惯上把“想先做的”放在右边。因此:

transform: translateX(-100px) rotateY(-90deg);

实际执行顺序是:先执行 rotateY(-90deg) 将面旋转到朝左,再执行 translateX(-100px) 沿旋转后的x轴推出100px。这听起来和上面“先平移再旋转”矛盾——但实际上不矛盾,因为CSS里 translateX 在字符串中写在前面,意味着它作用于“已被后面rotate改变过的坐标系”。建议将它作为一个约定来记忆:

记住这个公式,六个面都能直接写出来。

旋转动画:@keyframes

立方体构建完成后,最后一步是让它旋转起来。CSS动画的核心是 @keyframes 配合 animation 属性。

.box {
  animation: rotate 6s linear infinite;
}
@keyframes rotate {
  0%   { transform: rotateX(0deg)   rotateY(0deg); }
  25%  { transform: rotateX(0deg)   rotateY(90deg); }
  50%  { transform: rotateX(0deg)   rotateY(180deg); }
  75%  { transform: rotateX(0deg)   rotateY(270deg); }
  100% { transform: rotateX(360deg) rotateY(360deg); }
}

animation 是一个简写属性,包含四个关键信息:

  • 动画名称(自定义):rotate
  • 动画持续时间(单次时长):6s
  • 动画曲线(变化速率):linear,匀速
  • 无限循环:infinite

@keyframes 定义动画的关键帧。这里前75%只绕Y轴旋转,最后25%才加入X轴翻转,整体效果如同“先转一圈观察四面,再翻转观察顶底”。

注意:transform 写在动画中时,每一帧都是完整的变换值,而不是增量。也就是说,50% 那一帧的 rotateY(180deg) 并非“在25%的基础上再增加90度”,而是直接设置为 rotateY(180deg)。这与Canvas中“每帧 += speed”的增量式动画有明显区别。

课后复习:几个容易混淆的知识点

将课程中的几个关键问题整理如下。

1. perspective 写在哪个元素上

写在"被透视元素的父元素"上。写在自身则不生效,写在更远的祖先上也会失效。

2. 为什么必须设置 transform-style

默认 transform-style 为 flat,子元素会被压平在父元素平面。preserve-3d 才能保留子元素的 3D 空间,使六个面真正立体堆叠。

3. 立方体六面公式

front : translateZ( d)
back  : translateZ(-d) rotateY(180deg)
left  : translateX(-d) rotateY(-90deg)
right : translateX( d) rotateY( 90deg)
top   : translateY(-d) rotateX( 90deg)
bottom: translateY( d) rotateX(-90deg)

其中 d 是面到中心的距离,对于200×200的立方体,d = 100

4. 自测题

  1. vhvw 分别代表什么?为什么移动端推荐使用 dvh
  2. inline-block 的“空白间隙”是如何产生的?有哪些解决方法?
  3. perspective 应该写在立方体自身还是父元素上?原因是什么?
  4. 如果不写 transform-style: preserve-3d,立方体会呈现怎样的效果?
  5. 立方体的 left 面为什么使用 translateX(-100px) rotateY(-90deg),而不是反过来?

现在如何理解

写完这篇文章后,对CSS的“绘制”能力有了全新的认识。

CSS 3D并非一门新语言,而是将“布局 + 定位 + 变换”三件事叠加在一起。perspective 提供视距、preserve-3d 保留空间、translate3d / rotate3d 执行变换——三个属性就能把一个2D盒子变成六面立方体。但要让这个立方体真正“正确”,必须打好布局基础:外层与内层的拆分、display 属性的本质、relative + absolute 的组合、flex居中的逻辑。3D是建立在2D布局之上的,缺少2D基础,3D便是空中楼阁。

此外,这节课也再次让人认识到GPU加速的重要性。CSS 3D不仅为了“炫”,更为了“快”——translateZ(0) 这种看似无意义的变换,实际上是在主动调用显卡。这一点与Canvas中的 getContext('webgl') 思路一致:浏览器中凡是需要绘制的内容,最终都要看显卡的性能表现。

inline-block 的空白间隙、100vh 在移动端的坑、perspective 写错位置不生效——这些看似微小的问题,都是真实开发中反复遇到的陷阱。将它们与六面公式一同记下,以后编写3D代码前先回顾一遍,能少走不少弯路。

如果后续要继续往3D方向深入,自然的延伸是three.js——上一节Canvas课中也提到过,AI游戏、3D可视化等领域都离不开它。CSS 3D适合卡片翻转、轮播、轻量级动效,而真正复杂的3D场景需要交给WebGL。但无论选择哪条路,“布局是3D的地基”这一认知,将始终适用。

来源:https://juejin.cn/post/7656828565456175119

游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

同类文章
更多
Stable Diffusion WebUI本地模型下载配置与性能优化指南

Stable Diffusion WebUI本地模型下载配置与性能优化指南

StableDiffusionWebUI适合在个人电脑上运行本地绘图模型,关键在于准备显卡环境、正确下载模型、放入对应目录,并通过分辨率、采样器、显存参数等设置提升生成效率与稳定性。

时间:2026-07-04 06:48
Stable Diffusion WebUI插件安装配置教程:浏览器、编辑器或扩展市场

Stable Diffusion WebUI插件安装配置教程:浏览器、编辑器或扩展市场

StableDiffusionWebUI插件可增强模型管理、提示词、图像处理与工作流效率。安装前需确认版本、环境和来源,按内置扩展页、网址安装或本地导入完成配置,并做好备份与兼容性检查。

时间:2026-07-04 06:48
Stable Diffusion WebUI Docker一键部署:镜像拉取端口映射数据目录配置

Stable Diffusion WebUI Docker一键部署:镜像拉取端口映射数据目录配置

使用Docker部署StableDiffusionWebUI可降低环境配置难度,重点在于选择镜像、映射7860端口、挂载模型与输出目录,并提前确认显卡驱动、存储空间和访问权限。

时间:2026-07-04 06:48
Stable Diffusion WebUI API Key 获取与配置教程:账号注册与国内网络设置

Stable Diffusion WebUI API Key 获取与配置教程:账号注册与国内网络设置

围绕StableDiffusionWebUI的APIKey配置,说明账号注册、密钥获取、本地接口认证、国内网络访问设置、验证方法与安全注意事项,适合AI绘画工具初次部署和团队接入使用。

时间:2026-07-04 06:48
Stable Diffusion WebUI Linux服务器部署完整教程:从环境准备到后台运行

Stable Diffusion WebUI Linux服务器部署完整教程:从环境准备到后台运行

StableDiffusionWebUI在Linux服务器部署需先确认GPU、驱动、Python与依赖环境,再拉取项目、配置模型和启动参数。后台运行建议使用tmux、nohup或systemd,并做好访问鉴权、端口限制、资源监控与模型来源校验。

时间:2026-07-04 06:48
热门专题
更多
刀塔传奇破解版无限钻石下载大全 刀塔传奇破解版无限钻石下载大全
洛克王国正式正版手游下载安装大全 洛克王国正式正版手游下载安装大全
思美人手游下载专区 思美人手游下载专区
好玩的阿拉德之怒游戏下载合集 好玩的阿拉德之怒游戏下载合集
不思议迷宫手游下载合集 不思议迷宫手游下载合集
百宝袋汉化组游戏最新合集 百宝袋汉化组游戏最新合集
jsk游戏合集30款游戏大全 jsk游戏合集30款游戏大全
宾果消消消原版下载大全 宾果消消消原版下载大全
  • 日榜
  • 周榜
  • 月榜