游戏引擎学习第42天

仓库: https://gitee.com/mrxiao_com/2d_game

简介

目前我们正在研究的内容是如何构建一个基本的游戏引擎。我们将深入了解游戏开发的每一个环节,从最基础的技术实现到高级的游戏编程。

角色移动代码

我们主要讨论的是角色的移动代码。我一直希望能够使用一些基本的数学和游戏开发的基本矢量数学技术。然而,直到昨晚,我才为大家做了一个数学概述,介绍了在游戏开发中涉及的各种类型的数学。今晚,我们将进行第一次的实验,展示如何使用一些这些数学原理来实现角色的移动。通过使用两个二维矢量映射,我将展示它们是如何运作的,以及我们如何利用它们来实现角色在游戏中的移动。这将是我们工作的起点,帮助我们在游戏中获得一些良好的角色动作。

回顾

让我们从这里开始,看看这些东西。我决定继续使用我正在使用的平板电脑,即使我可能需要用painter来画一些图和线条,因为我觉得这可能会帮助我更好地理解和解释一些概念。虽然使用钢笔可能会给我带来一些困难,但我们要试一试,也许至少在这一集里,我可以适应它。

回忆一下,如果你记得的话,这是我们所停留的地方。我们做了很多工作,关于一些基础的东西。我们测试了滚动,允许自己创造了一个巨大的地图。所以,你知道,我们做了稀疏样式的地图存储,也做了许多其他的事情。但我们现在必须开始考虑如何让我们的角色进行移动。我们现在的角色移动代码是最糟糕的,是最基本的,最令人讨厌的移动代码。我甚至没有提到他还没有进行动画,因为这也是我们要解决的问题,但我说的只是比这更低层次的东西,即使只是让角色进行基本的移动。

当前角色移动代码的问题

你可能会注意到,角色的移动感觉并不好。一个主要问题是角色没有任何动量。现在,基本上所有的事情都发生在你按下按键的时候,他只是随着你按住键移动。一旦你松开了按键,他就停了。此外,还存在一些奇怪的现象。例如,如果我以某个速度向下移动,我的速度是固定的。但如果我同时向右和向下移动,我实际上会以两者速度的结合下移动。这种现象在对角线上显得尤为奇怪,因为角色移动的速度会加快。这也是一种不太令人满意的现象,因为如果我向上移动,角色在穿越一个虚拟门后加速,去到下一个图层。

这些现象最终导致了很多问题。角色非常容易被各种小障碍物卡住,很难穿过门。任何微小的偏差都会导致角色完全停止。这些问题使得玩家的移动感觉很糟糕,这就是我们希望改进的地方。

我们想要做的事情

所以除了这些问题,我们还希望做的就是让玩家的体验更有趣。当他们在游戏中移动角色时,这个动作应该更具吸引力,因为在这样的游戏中,玩家需要不断移动角色,躲避敌人等。他们的角色始终在屏幕上移动,从左到右,他们一直在进行这些操作。因为移动角色是他们游戏中的主要动作之一,所以我们希望提供更多的操作感和互动,让玩家觉得他们的角色更有趣、更充实。

什么使得游戏具有吸引力

所以,视觉和触觉反馈对于游戏而言是非常重要的,因为它们使游戏变得有趣。当玩家在游戏中移动时,他们需要能够感受到这种反馈,这会让他们觉得自己真正参与其中。视频游戏不同于传统的棋盘游戏,它通过音频视觉反馈来吸引玩家。这些反馈不仅仅是表面的,它们让玩家感受到动作的真实性和游戏的吸引力。一个好的游戏不仅仅是机制本身,还包括了让玩家有兴趣继续玩下去的那些细节。这些细节包括移动感、跳跃的感觉等,确保玩家在每个操作时都能体验到愉悦和成就感。我们希望在接下来的时间里,通过优化角色的移动方式来实现这种吸引力。

为什么需要向量数学

我们正在学习如何使用基本的矢量数学来改善角色的移动。通过这种方式,我们可以使角色在游戏中的运动更加流畅、更加符合玩家的需求,这样玩家在操作角色时能够感受到视觉和触觉上的满足感。这种方式的运动设计不仅能够让角色移动起来感觉很好,还能够激发玩家的兴趣,使他们更想操作这个角色。这就是我们目前所要达到的目标,即让角色的移动既有吸引力又令人满意,使玩家在每次操作时都感到愉悦。

应该开始修复的内容

我们要解决角色在移动时的动量问题,让他不会瞬间停下来,然后再开始,同时也解决他在对角线方向移动时速度加快的问题。通过学习基本的向量数学,我们可以轻松地修正这些问题。这将使角色在所有方向上移动更加一致,改善玩家的体验。让我们开始尝试解决这些问题吧。

向量数学简介

我们正在讨论矢量在游戏开发中的应用。向量不仅帮助我们处理诸如玩家的速度和方向等变量,还使得它们之间的关系变得更加紧密。例如,在处理角色的移动时,如果我们只看水平或垂直方向,它们就像独立的变量一样,但实际上它们是相互关联的。在实际操作中,角色的对角线移动涉及到两者的结合。这种耦合使得处理起来更直观、更高效,因为数学家早已为我们定义了所有可以对向量进行的基本操作。通过这些操作,我们可以更好地控制和优化游戏的体验。

草图

我们在讨论一个对象的运动时,特别是在游戏开发中,如何理解玩家在屏幕上的移动。我们使用勾股定理来解释这个问题。玩家在游戏中被视为在一个二维空间中移动,通过结合x和y轴的移动来创建一个对角线路径。这种对角线路径并不是玩家实际移动的两个单独的方向,而是这两个方向的综合表现。因此,尽管玩家在x和y轴上移动的距离分别为d和d,最终的路径长度却是勾股定理的平方根。

这解释了为什么玩家可能会看到比实际移动更长的路径,这被称为"幻影远距"。它并不是两个距离的简单相加,而是形成了一个更长的路径,这是因为它在两者之间形成了一个对角线。因此,尽管它们的移动速度可能是相同的,玩家看到的路径却是它们的勾股定理计算得出的更长的距离。

在游戏开发中,这一理解是重要的,因为它帮助我们优化运动轨迹,使得玩家感觉到的游戏体验更加真实。

移动标量问题

在对角线移动时,我们可以使用勾股定理(a² + b² = c²)来计算目标距离(d)。我们首先通过平方根将这个距离除以2来得到一个比例因子(v)。然后,基于该因子,我们可以在移动过程中将它们相乘来确定实际的移动量。这一过程允许我们计算在对角线方向上进行相等移动的需要的速度或距离。

v 2 + v 2 = d 2 d 2 = 2 v 2 v = d 2 2 v = d 2 2 v = d 2 2 v = 0.7071067811865475 ∗ d v^2 + v^2 = d^2 \\[1em] d^2 = 2v^2 \\[1em] v = \sqrt{\frac{d^2}{2}} \\[1em] v = d\frac{\sqrt{2}}{2} \\[1em] v = d\frac{\sqrt{2}}{2} \\[1em] v = 0.7071067811865475*d v2+v2=d2d2=2v2v=2d2 v=d22 v=d22 v=0.7071067811865475∗d

转向向量

我们讨论了矢量如何帮助我们简化问题。矢量实际上是多个信息片段的结合,允许我们以直观的方式操作这些片段,而无需关注低级数学细节。例如,当我们使用矢量时,不再需要思考逐步运动,而是可以直接从原点到目的地的直接移动。矢量概念在多维空间中同样适用,无论是二维、三维,甚至更多维度。它们帮助我们在更高层次上思考和处理问题,使得游戏开发中的各种计算变得更加高效和简洁。矢量不仅仅是简单的数学工具,它们在我们的工作中赋予了深刻的概念和语义。

转向向量(direction vector)是表示一个向量在几何空间中的方向的向量。它通常通过两个或多个点之间的差异来定义。例如,给定两个点 A(x_1, y_1, z_1)B(x_2, y_2, z_2),其转向向量 \\vec{D} 可以通过以下公式计算:

D ⃗ = ( x 2 − x 1 , y 2 − y 1 , z 2 − z 1 ) \vec{D} = (x_2 - x_1, y_2 - y_1, z_2 - z_1) D =(x2−x1,y2−y1,z2−z1)

这种转向向量的计算方式反映了从点 A 到点 B 的方向和大小。

在更广泛的数学和物理学中,转向向量可以应用于描述运动方向,速度矢量,力矢量等。例如,在物理学中,一个物体的运动速度可以用一个转向向量表示,这个向量指示物体的运动方向和速度的大小。

举例:

假设点 A(2, 3, 4) 和点 B(5, 7, 10),我们可以计算它们之间的转向向量:

D ⃗ = ( 5 − 2 , 7 − 3 , 10 − 4 ) = ( 3 , 4 , 6 ) \vec{D} = (5 - 2, 7 - 3, 10 - 4) = (3, 4, 6) D =(5−2,7−3,10−4)=(3,4,6)

这个向量 (3, 4, 6) 表示从点 A 到点 B 的方向,以及从 AB 的距离(大小)。

向量符号

我们在讨论矢量时,主要是在谈论如何组合多个信息片段。矢量可以通过竖直排列的方式来书写,这与传统的水平排列不同。这种书写方式反映了矢量在多维空间中的位置和方向。例如,一个向量可以被视为一个二维或三维的矩阵,其列或行代表特定方向上的量。而这种书写方式的选择影响了我们对矢量进行操作时的理解和处理方式。因此,决定如何书写这些对象非常重要,特别是在保持数学一致性的前提下。尽管代码编辑器不允许在中间插入垂直排列的矢量,但这种写法仍然是描述和理解矢量的直观方式。在代码中,我们通常会以这种方式来书写向量,尽管它可能在代码编辑器中有些不便。总体来说,理解和使用矢量的这种方式有助于我们在处理多维数据时更高效。

向量和矩阵之间有着紧密的关系,特别是在数学和计算机科学领域中。理解它们的关系可以帮助我们更好地理解和操作多维数据。

向量

一个向量是由一个或多个数值组成的对象,它在数学中被用来表示方向和大小。例如,一个二维向量可以表示一个点在平面上的位置,而一个三维向量可以表示一个点在三维空间中的坐标。

  • 二维向量: \\vec{v} = \\begin{pmatrix} x \\ y \\end{pmatrix}
  • 三维向量: \\vec{v} = \\begin{pmatrix} x \\ y \\ z \\end{pmatrix}

矩阵

矩阵是一组由数字(元素)组成的表格,可以用来表示多个向量的线性组合。矩阵的元素通常用来描述系统的线性关系。例如,一个二维矩阵可以表示一个线性变换,如旋转或缩放。

  • 二维矩阵 :
    A = ( a 11 a 12 a 21 a 22 ) A = \begin{pmatrix} a_{11} & a_{12} \\ a_{21} & a_{22} \end{pmatrix} A=(a11a21a12a22)
  • 三维矩阵 :
    B = ( b 11 b 12 b 13 b 21 b 22 b 23 b 31 b 32 b 33 ) B = \begin{pmatrix} b_{11} & b_{12} & b_{13} \\ b_{21} & b_{22} & b_{23} \\ b_{31} & b_{32} & b_{33} \end{pmatrix} B= b11b21b31b12b22b32b13b23b33

行列关系

  1. 向量可以看作是矩阵的"列":

    • 一个向量 \\vec{v} 可以被看作是一个 n \\times 1 矩阵。
    • 例如,二维向量 \\vec{v} = \\begin{pmatrix} x \\ y \\end{pmatrix} 是一个 2 \\times 1 矩阵。
  2. 矩阵的乘法:

    • 当一个向量乘以矩阵时,结果是另一个向量。
    • 如果矩阵 A 的列数等于向量的大小,矩阵乘法的结果将是一个新的向量。
  3. 矩阵的行和列:

    • 矩阵的每一行可以看作是一个向量。
    • 矩阵的每一列也可以视为一个向量。
    • 例如,对于一个 2 \\times 3 矩阵:
      A = ( 1 2 3 4 5 6 ) A = \begin{pmatrix} 1 & 2 & 3 \\ 4 & 5 & 6 \end{pmatrix} A=(142536)
      每一列 \\begin{pmatrix} 1 \\ 4 \\end{pmatrix}\\begin{pmatrix} 2 \\ 5 \\end{pmatrix} 都是向量。
  4. 转置矩阵:

    • 矩阵的转置操作(将矩阵的行转变为列)常用于向量和矩阵之间的操作。例如,矩阵的转置 A\^TA 中的每个列变成行。

行列操作示例

  • 向量和矩阵的加法:

    • \\vec{v} = \\begin{pmatrix} x \\ y \\end{pmatrix}\\vec{w} = \\begin{pmatrix} u \\ v \\end{pmatrix} 是两个向量。
    • \\vec{v} + \\vec{w} = \\begin{pmatrix} x + u \\ y + v \\end{pmatrix}
  • 向量和矩阵的乘法:

    • 设矩阵 A 是一个 2 \\times 3 矩阵,\\vec{v} = \\begin{pmatrix} x \\ y \\end{pmatrix} 是一个 2 \\times 1 向量。
    • 矩阵乘法得到的向量为:
      A v ⃗ = ( 1 2 4 5 ) ( x y ) = ( 1 x + 2 y 4 x + 5 y ) A\vec{v} = \begin{pmatrix} 1 & 2 \\ 4 & 5 \end{pmatrix} \begin{pmatrix} x \\ y \end{pmatrix} = \begin{pmatrix} 1x + 2y \\ 4x + 5y \end{pmatrix} Av =(1425)(xy)=(1x+2y4x+5y)

这种行列关系使得向量和矩阵的结合成为处理多维数据和进行各种线性变换的基础工具。

将向量例程添加到代码中

这是一个非常详细的讲解,说明了如何在编程和数学中讨论向量乘法、运算符重载,以及代码实现时的实用性。以下是这段话的重点总结和主要思想:


1. 关于向量乘法

  • 点积和叉积: 向量乘法有多种形式,点积和叉积是其中最常用的两种。它们有具体的几何和实际意义,比如用于计算投影或方向。
  • 逐分量乘法: 这是简单的将对应分量相乘,这种操作在某些领域有用,例如处理颜色或掩码,但对表示空间关系的操作来说意义不大。

2. 运算符重载的实用性

  • 为什么在向量中有用 : 在实现向量时,运算符重载(如 +-)可以让代码看起来更像数学表达式。
  • 局限性: 尽管在向量中方便,但在其他场景中,运算符重载未必同样有效。

3. 实现向量操作

作者的目标是:

  1. 创建一个简单的向量结构,包含 xy 分量。
  2. 提供既能按名称访问(如 .x)又能按索引访问(如 [0][1])的能力。
  3. 为常见操作定义运算符重载,如加法、减法和取反。

4. 示例实现代码

以下是这段思想的可能代码实现示例(假设用 C++ 实现):

cpp 复制代码
#include <iostream>

struct Vector2 {
    float x, y;

    // 通过索引访问
    float& operator[](int index) {
        return (index == 0) ? x : y;
    }

    // 运算符重载:加法
    Vector2 operator+(const Vector2& other) const {
        return {x + other.x, y + other.y};
    }

    // 运算符重载:减法
    Vector2 operator-(const Vector2& other) const {
        return {x - other.x, y - other.y};
    }

    // 运算符重载:取反
    Vector2 operator-() const {
        return {-x, -y};
    }
};

int main() {
    Vector2 v1 = {1.0f, 2.0f};
    Vector2 v2 = {3.0f, 4.0f};

    Vector2 sum = v1 + v2;
    Vector2 diff = v1 - v2;
    Vector2 neg = -v1;

    std::cout << "Sum: (" << sum.x << ", " << sum.y << ")\n";
    std::cout << "Difference: (" << diff.x << ", " << diff.y << ")\n";
    std::cout << "Negation: (" << neg.x << ", " << neg.y << ")\n";

    return 0;
}

5. 学习的关键点

  • 理解向量操作的几何意义: 比如点积计算投影长度,叉积计算面积等。
  • 运算符重载的设计: 保持代码的可读性,同时兼顾功能性。
  • 逐步实现复杂功能: 从基本操作(加、减、取反)开始,再扩展到更高级的矩阵运算和向量操作。

联合类型在C++中的工作原理及其缺点?

C++ 中的 Union(联合体)

Union 的工作原理:

在 C++ 中,union 是一种特殊的数据类型,它允许多个数据成员共享同一块内存。Union 的特性是同一时间只能有一个成员存储有效的数据,因为所有成员共用同一片内存空间。这种机制常用于内存优化或处理底层数据操作。

例如,一个包含 intfloat 成员的联合体,写入一个成员会覆盖另一个成员的值,因为它们的内存地址相同。

示例:

cpp 复制代码
union Data {
    int intValue;
    float floatValue;
};

访问和修改联合体的成员:

cpp 复制代码
Data data;
data.intValue = 5; // 赋值给 intValue
std::cout << data.floatValue; // 读取 floatValue(以 float 格式解释同一内存)

使用 Union 的缺点:

  1. 类型安全性低: 如果访问的是最近没有被赋值的成员,可能会导致未定义行为。这使得联合体的使用需要非常小心。

  2. 初始化复杂: C++ 中的联合体不支持同时对多个成员进行直接初始化。如果需要更复杂的初始化,必须定义构造函数或额外的方法,这会让语法变得不一致。

  3. 兼容性问题: 现代 C++ 的某些特性(如使用大括号的初始化方法或某些运算符重载)在联合体中可能无法直接使用。

  4. 限制现代功能: 联合体不能与某些现代 C++ 特性(如构造函数、析构函数和虚函数)完全兼容,这限制了它在复杂类型中的使用。

  5. 容易出现错误: 联合体中的数据解释依赖于开发者的正确访问,如果预期的数据类型和实际存储的数据类型不匹配,就可能引发难以定位的错误。

实际问题示例:

假设想设计一个联合体,既能存储一对 float 值(如 xy),又能通过数组的方式访问这些值。在 C++ 中,由于联合体初始化和语法限制,无法直接实现这样的灵活访问方式。

一种解决方法是定义自定义运算符来访问联合体成员,但这会让代码看起来与普通的结构体语法不一致,降低可读性和直观性。

总结:

虽然 Union 提供了内存效率高、灵活性强的特性,但其缺点包括类型安全性低、对现代 C++ 特性的支持不足以及复杂的语法要求。这使得 Union 在实际应用中往往显得笨拙。只有在对其行为有深刻理解的前提下,才能合理避免相关陷阱并有效利用其特性。

关于向量的约定

矢量和矩阵约定的标准化

关于矢量和矩阵约定的标准化问题,有人提到,某种约定大约在 1993 年得到了统一,尤其是涉及矢量和列向量在空间点表示中的使用。然而,这一时间点是否准确存在疑问,因为数学早在这之前已经在大量应用中使用列向量来表示空间中的点。

矢量和矩阵在数学中的传统使用
  • 矢量和列向量:在数学和几何中,矢量的使用可以追溯到更早的时期。尤其是在描述点的位置或操作变换时,列向量作为一种标准方式被广泛接受和应用。
  • 矩阵的使用:矩阵在数学中的历史悠久,并在 19 世纪就已经被用于线性变换的表示。列向量的使用通常与这种矩阵操作密切相关。
关于标准化时间点的质疑

1993 年被认为是矢量和矩阵约定标准化的一个时间点,但这种说法可能存在问题:

  1. 数学的传统:矢量的列向量表示形式和其在空间中的几何意义早已被数学家所熟知并广泛采用。线性代数等领域中的列向量概念,显然在1993年之前就已经是数学和物理领域的常识。
  2. 实际应用:计算机图形学、工程力学以及其他科学领域在 20 世纪中叶甚至更早就开始广泛使用矩阵和矢量操作,这意味着标准化的实践在很久以前已经存在。
可能的解释
  • 行业或技术标准化:1993 年的标准化可能指的是某个行业或某种技术的具体约定,而非数学本身。例如,计算机图形学领域在这一时期可能对矩阵和矢量的约定进行了进一步统一,以便跨平台或跨工具的协作。
  • 误解来源:将一个具体领域的标准化时间误认为是整个数学领域的标准化可能导致了这一时间点的混淆。
结论

矢量和矩阵的列向量约定作为数学的一部分,其历史可以追溯得更早。1993 年可能仅仅是某一行业或特定技术中实现了标准化,但这一时间点无法否认数学界早期的广泛使用和接受。

在第一个向量版本中,是否只缺少了联合的匿名结构?

联合类型的使用和挑战

在初期的实现中,联合类型代码存在一个问题:缺少匿名结构将字段封装起来,导致字段在内存中重叠的问题。这种实现方式可能并未按照预期工作,因为数据的布局方式并未明确定义。

关于匿名结构的使用
  • 问题发现:在代码中添加匿名结构后,字段的重叠问题得以解决。这是一个好的解决方案,可以正确实现对联合类型的封装。
  • 测试效果:在封装匿名结构后,代码的功能按预期运行,问题似乎得到了修复。这表明匿名结构在这种情况下是一个必要的调整。
匿名结构的兼容性问题
  • 编译器行为差异:匿名结构在不同编译器中的表现可能存在差异。在 Visual Studio 中,这种实现没有问题,但可能在某些其他编译器(如 LLVM 或其他工具链)中引发警告或错误。
  • 具体问题:之前的尝试表明,匿名结构在某些环境中会导致编译器投诉。因此,这种解决方案可能并不具有普遍的适用性。
联合类型的局限性
  • 构造函数问题:联合类型在 C++ 中本身存在诸多局限,尤其是无法支持非平凡类型的构造函数和析构函数。如果试图在联合中使用复杂类型,可能会引发意料之外的问题。
  • 复杂性管理:为了在不同编译器之间保持一致性,需要在实现时尽量避免使用复杂的特性,选择简单、清晰的方式来规避潜在问题。
实践中的权衡

在这种有限的场景中,如果代码在两个主流编译器上都能正确编译且表现一致,则可以考虑接受这种实现。考虑到使用的联合类型不会涉及复杂的操作(如构造函数或析构函数),该实现可能足够安全。

后续建议
  • 测试环境多样性:确保在多种编译器(如 GCC、LLVM)环境下进行充分测试,避免仅依赖 Visual Studio 的表现。
  • 关注维护性:尽量使用明确、清晰的结构组织代码,减少潜在的兼容性问题。
  • 避免复杂操作:在联合类型中,不应引入复杂类型或需要特别管理生命周期的对象,以确保代码的稳定性和跨平台可移植性。

通过匿名结构的调整,可以在一定程度上优化联合类型的使用,但其兼容性和局限性仍需小心应对。

在v2上不使用构造函数是否有性能原因?

关于构造函数的使用与否
  1. 选择不使用构造函数的原因
    • 简化代码:避免引入构造函数可以减少额外的代码编写和维护成本。
    • 语法灵活性:在没有构造函数的情况下,可以完全省略初始化代码,而使用默认的变量声明方式。
    • 避免复杂性:一旦引入构造函数,可能需要编写多个版本,例如默认构造函数、带参数的构造函数等。
  2. 性能影响
    • 构造函数的使用通常不会对性能有显著影响,因为编译器通常能够优化这部分代码。
    • 选择不使用构造函数更多是基于代码风格和偏好,而不是性能考量。
  3. 替代方案
    • 通过直接声明或零初始化,可以实现与构造函数类似的功能,但代码更加简洁。
    • 例如,将变量初始化为零或未初始化的状态,利用编译器的默认行为满足需求。

对象成员的私有或公共性何时会发挥作用?

关于对象成员的公开和私有访问权限

成员访问权限的作用
  1. 公开和私有的区分
    • 公共成员可以在类的外部访问和修改。
    • 私有成员只能由类的内部方法或友元函数访问。
  2. 私有访问的目标
    • 限制代码中可以修改类成员的范围。
    • 确保只有特定部分的代码可以访问某些敏感数据或逻辑。
    • 通过缩小访问范围来减少潜在的错误。
私有成员的使用场景
  1. 预防错误
    • 私有成员可以防止其他程序员或自身在不该修改某些部分时意外更改。
    • 限制外部代码对类内部实现的直接操作。
  2. 代码健壮性
    • 将某些实现细节隐藏在类内部,使代码对用户更加友好且易于维护。
    • 避免意外的状态改变,从而减少调试的复杂性。
对使用私有的不同看法
  1. 不使用私有的观点
    • 一些开发者认为私有成员限制了灵活性,使得外部代码操作类变得更困难。
    • 如果对代码结构有足够的控制,并对潜在错误风险有较强的信心,可能不会优先考虑使用私有。
    • 假如代码从未出现任何错误,则不使用私有访问控制也不会造成问题。
  2. 支持使用私有的观点
    • 其他开发者认为私有可以作为一种安全机制,有效减少意外修改和潜在问题。
    • 在协作开发中,私有访问权限可以为团队成员提供明确的操作边界。
使用私有的取舍
  1. 权衡点
    • 私有成员可以提供额外的保护,但同时也可能增加类的复杂性。
    • 在某些场景下,过度使用私有可能导致需要更多辅助代码(如友元函数或公共接口)来完成简单任务。
  2. 个人选择
    • 是否使用私有访问权限通常取决于开发者的风格和项目需求。
    • 对于一些项目,开放的成员访问可能更实用;而对于另一些注重稳定性和代码安全的项目,私有可能更加合适。
总结
  • 公共和私有成员的区分是为了平衡灵活性和安全性。
  • 私有访问权限主要用于减少潜在的错误风险,通过限制访问范围来增强代码的健壮性。
  • 是否使用私有成员是开发者的选择,根据项目需求和开发风格灵活决定。
  • 对于需要频繁操作类成员的场景,可能更倾向于开放访问权限;而在保护关键数据时,私有权限更具优势。

应该能够在对象外部进行复合赋值。

关于复合赋值操作和成员定义的调整

实现复合赋值的方式
  1. 复合赋值的支持

    • 目标是在对象外部实现复合赋值运算。
    • 通过将相关操作定义为外部函数,可以实现这一目标。
    • 将第一个操作数作为引用传递(reference member)是关键。
  2. 操作位置的调整

    • 可以将复合赋值逻辑定义在外部,而无需嵌入类内部。
    • 这种方式减少了类的复杂性,同时保留了功能的灵活性。
调整过程中的探索
  1. 代码调整

    • 初步尝试是在类内部进行处理,但后续发现可以移至外部。
    • 通过测试和验证确认外部实现是可行的,并进行了必要的修改。
  2. 处理细节的回忆

    • 尽管熟悉基本概念,但在具体实现过程中可能会遗忘某些符号或语法。
    • 例如,与模板相关的语法,在没有频繁使用的情况下可能需要额外时间回忆。
关于模板和符号
  1. 符号和模板的学习曲线

    • 处理模板和符号时,回忆具体语法可能是一个挑战。
    • 即使曾经熟练使用模板,长时间未使用可能导致需要额外时间重新熟悉。
  2. 具体问题的处理

    • 针对每个符号或操作进行测试和调整,例如如何正确处理某些算术操作。
    • 逐步调试并修正错误,确保输出结果与预期一致。
复合赋值示例的确认
  1. 操作逻辑验证

    • 通过实际编写和测试,确认了复合赋值操作逻辑的正确性。
    • 结果显示可以成功在外部实现复合赋值,并返回正确的结果。
  2. 调整和优化

    • 删除了初始测试中定义在类内部的冗余部分。
    • 最终优化了代码结构,将相关逻辑调整为外部实现。
总结
  • 复合赋值操作可以通过外部函数实现,简化了类的定义。
  • 确保第一个操作数作为引用传递是实现这一功能的关键。
  • 长期未使用某些符号或语法可能导致一定的回忆困难,但通过测试和调整可以快速解决问题。
  • 通过这种方式,可以提高代码的可读性和灵活性,同时保持功能的完整性。

如果使用竖线符号表示向量,如何不记事行列式?

在使用向量符号时,在矩阵中使用竖条来表示行列式是一个常见的做法。在我的线性代数中,通常使用方括号来表示矩阵和向量,而竖条则用于表示决定。对于一个矩阵或向量,我会用竖条来表示其行列式。

例如,若 A 是一个矩阵,则其行列式可以表示为 |A|。竖条内的表达式代表着矩阵 A 的行列式值,这种写法非常典型且直观。

这个符号 | 用于表示行列式不仅仅是为了便于书写和阅读,也是因为它们传达了数学运算中的一种明确性。例如,行列式的计算常用于线性变换和矩阵理论的其他方面。

总的来说,行列式在数学运算中并不常被频繁写入,因为它们主要在需要进行矩阵运算时才被考虑。因此,在写作和阅读过程中保持符号的简洁和直观是重要的。对于向量,我更倾向于使用直线表示,这种符号化方式最简洁,直观且不容易引起歧义。

你不能把整个向量结构作为联合吗?

在讨论中,人们提到将整个功能放在union中(即联合体)作为一个整体,而不是使用struct。这种方法的可行性引发了质疑,因为传统上,union并不常用于操作符重载或集成的操作。

在这个情况下,答案被认为是肯定的------即可以在union上实现这些功能,而不会出现问题。尽管从未尝试过这样的操作符重载或功能集成,这种方法看起来似乎是可行的。如果问到是否可以将操作符重载应用于union,回答可能是"我从来没有尝试过,但它应该工作"。

因此,在这种讨论中,union被视为一种类似于指导的工具,可以用来组织和管理操作。在这种情况下,union的结构和功能能够显现出其灵活性,并且在某些情况下,如操作符重载,甚至可能会产生意想不到的效果。整体来看,这种方法会一直被用到有问题为止,直到它不能再适用为止。

有没有向量数学书推荐?

".../book/线性代数(第5版) (Gilbert Strang (吉尔伯特·斯特朗)).pdf"

如何在函数参数中添加 const 引用以避免按值复制?

在C++中,使用常量引用作为函数参数以避免按值复制并不会总是提高性能。尽管有时它可以帮助优化,因为编译器可能会在常量引用上应用一些优化策略,但这种情况并不总是显而易见。通常,成本优化对编译器来说并不是一个经常使用的优化策略,因为它依赖于特定的编译器实现和特定的代码路径。

在实际编程过程中,使用常量引用主要是为了减少内存复制操作,从而避免临时变量的创建和销毁,这对于性能影响不大的情况下是有好处的。然而,在大多数情况,尤其是当涉及到指针和复杂的对象时,直接使用值传递可能更合适,编译器能够根据情况优化代码,而不需要额外的编译时成本。

因此,对于普通的编程任务来说,成本优化可能并不总是值得关注的,因为其帮助有限且依赖于具体情况。在性能关键的代码部分中,确实有可能需要考虑常量引用,以避免不必要的内存拷贝和复制操作。

相关推荐
dal118网工任子仪10 分钟前
66,【6】buuctf web [HarekazeCTF2019]Avatar Uploader 1
笔记·学习
02苏_15 分钟前
2025/1/21 学习Vue的第四天
学习
羊小猪~~38 分钟前
MYSQL学习笔记(四):多表关系、多表查询(交叉连接、内连接、外连接、自连接)、七种JSONS、集合
数据库·笔记·后端·sql·学习·mysql·考研
约定Da于配置41 分钟前
uniapp封装websocket
前端·javascript·vue.js·websocket·网络协议·学习·uni-app
东京老树根2 小时前
Excel 技巧15 - 在Excel中抠图头像,换背景色(★★)
笔记·学习·excel
Ronin-Lotus3 小时前
嵌入式硬件篇---ADC模拟-数字转换
笔记·stm32·单片机·嵌入式硬件·学习·低代码·模块测试
qq_428639613 小时前
虚幻基础1:hello world
游戏引擎·虚幻
编程小猹3 小时前
学习golang语言时遇到的难点语法
学习·golang·xcode
promising-w4 小时前
单片机基础模块学习——数码管
单片机·嵌入式硬件·学习
不爱学英文的码字机器4 小时前
我的2024:创作历程与成长总结
学习·程序人生·交友