C++基础入门:掌握核心概念(超全!)

news2025/1/18 19:03:52

C++作为一门广泛使用的编程语言,以其高性能和灵活性在软件开发领域占据重要地位。无论是游戏开发、系统编程还是实时应用,C++都是一个不可或缺的工具。本博客旨在为初学者提供C++编程语言的核心概念,帮助你建立坚实的基础。

C++关键字

C++关键字是编程语言中预定义的保留词,用于执行特定的编程功能。了解这些关键字是理解C++编程的第一步。例如,intreturnvoidif等,都是执行基础操作的关键字。一个好的起点是熟悉这些关键字及其用法,这有助于你编写更有效的C++代码。

函数重载

为什么 C++ 支持函数重载,而 C 语言不支持函数重载呢?
C/C++ 中,一个程序要运行起来,需要经历以下几个阶段: 预处理、编译、汇编、链接

 

1. 实际项目通常是由多个头文件和多个源文件构成,而通过 C 语言阶段学习的编译链接,我们
可以知道,【当前 a.cpp 中调用了 b.cpp 中定义的 Add 函数时】,编译后链接前, a.o 的目标
文件中没有 Add 的函数地址,因为 Add 是在 b.cpp 中定义的,所以 Add 的地址在 b.o 中。那么
怎么办呢?
2. 所以链接阶段就是专门处理这种问题, 链接器看到 a.o 调用 Add ,但是没有 Add 的地址,就
会到 b.o 的符号表中找 Add 的地址,然后链接到一起 ( 老师要带同学们回顾一下 )
3. 那么链接时,面对 Add 函数,链接接器会使用哪个名字去找呢?这里每个编译器都有自己的
函数名修饰规则。
4. 由于 Windows vs 的修饰规则过于复杂,而 Linux g++ 的修饰规则简单易懂,下面我们使
用了 g++ 演示了这个修饰后的名字。
5. 通过下面我们可以看出 gcc 的函数修饰后名字不变。而 g++ 的函数修饰后变成【 _Z+ 函数长度
+ 函数名 + 类型首字母

 

结论:linux下,采用gcc编译完成后,函数名字的修饰没有发生改变。 

函数重载允许你使用相同的函数名进行不同的操作,前提是它们的参数列表不同。这是一种提高程序可读性和功能性的有效方式。通过函数重载,你可以根据不同的参数类型或数量来执行不同的任务。

函数重载是C++中一种允许多个同名函数共存的特性,只要这些函数的参数列表不同。这意味着重载的函数可以有不同的参数类型、数量或者两者都不同。函数重载使得函数命名更加直观,避免了为每个不同操作版本的函数起不同名字的麻烦。

函数重载的规则

为了成功实现函数重载,需要遵循以下几个规则:

  1. 不同的参数列表:重载的函数必须在参数类型、数量或者两者都不同。仅仅返回类型的不同是不足以重载函数的。
  2. 作用域:只有在相同的作用域内,函数才能被重载。
  3. 调用的明确性:任何函数调用时,编译器都必须能够明确地确定应该使用哪个函数版本,否则将报错。
示例代码

以下是函数重载的一个简单示例,展示了如何根据不同的参数类型和数量来重载函数:

#include <iostream>
using namespace std;

// 打印整数
void print(int i) {
    cout << "Printing int: " << i << endl;
}

// 打印浮点数
void print(double f) {
    cout << "Printing float: " << f << endl;
}

// 打印字符数组
void print(const char* c) {
    cout << "Printing character: " << c << endl;
}

int main() {
    print(5);    // 调用打印整数的函数
    print(500.263); // 调用打印浮点数的函数
    print("Hello C++"); // 调用打印字符数组的函数
    return 0;
}

在这个例子中,print函数被重载三次,每个函数接受不同类型的参数。当print函数被调用时,编译器通过传递给它的参数类型和数量来决定使用哪个版本的函数。

函数重载的好处

  • 提高代码可读性:通过函数重载,可以为执行相似功能的函数使用统一的名称,使得代码更加直观。
  • 增加程序的灵活性:能够根据不同的数据类型或参数数量执行不同的操作,从而使程序更加灵活。
  • 类型安全:与使用泛型编程相比,函数重载提供了一种类型安全的方式来处理不同类型的数据。

注意事项

  • 避免歧义:确保重载的函数之间在调用时能够被编译器明确区分,否则会导致编译错误。
  • 不要仅依赖于返回类型:由于函数调用时不考虑返回类型,因此不能仅通过返回类型来重载函数。
  • 使用默认参数时的额外注意:函数重载与默认参数同时使用时要特别小心,以避免调用时的歧义。

 

 

命名空间

命名空间是C++中用于解决名称冲突的一种机制。最常见的命名空间是std,它包含了标准库的所有功能。通过使用using namespace std;声明,你可以避免在调用标准库函数时重复使用std::前缀。理解并正确使用命名空间,对于编写组织良好的代码非常重要。

命名空间的声明和使用

命名空间通过namespace关键字声明,其基本语法如下:

namespace NamespaceName {
    // code declarations
}

 例如,可以创建一个名为myNamespace的命名空间,其中包含一个函数和一个变量:

namespace myNamespace {
    void myFunction() {
        std::cout << "Inside myNamespace" << std::endl;
    }
    int myVariable = 42;
}

为了使用命名空间中的成员,你需要使用作用域解析运算符::来指定成员所属的命名空间,如下所示: 

myNamespace::myFunction();
std::cout << myNamespace::myVariable << std::endl;

 

using namespace myNamespace;
myFunction(); // 直接调用,不需要前缀

使用using声明

如果你不希望每次调用命名空间中的成员时都输入完整的命名空间,可以使用using声明。这可以让你在特定范围内不必使用命名空间前缀,直接访问其中的成员。例如:

 

using namespace myNamespace;
myFunction(); // 直接调用,不需要前缀

 需要注意的是,滥用using声明可能会导致命名冲突,特别是当两个命名空间包含同名的成员时。因此,推荐仅在确信不会引发冲突的情况下使用using声明,或者只针对特定的命名空间成员使用using指令,如:

using myNamespace::myFunction;
myFunction(); // 直接调用myFunction,但其它myNamespace成员仍需前缀

命名空间的嵌套

C++允许命名空间的嵌套使用,即在一个命名空间内部定义另一个命名空间。这种结构有助于进一步组织和封装代码,例如:

 

namespace outer {
    int x = 10;
    namespace inner {
        void display() {
            std::cout << "Value: " << x << std::endl; // 访问外部命名空间的x
        }
    }
}

 在这个例子中,inner命名空间嵌套在outer命名空间内部。要访问inner命名空间中的display函数,可以使用:

outer::inner::display();

无名(匿名)命名空间

无名(或匿名)命名空间是一个特殊的命名空间,它不具有名称。在同一翻译单元(通常是一个源文件)中,无名命名空间的成员可以直接访问,就好像它们被声明在全局作用域一样,但在其他翻译单元中不可见。这提供了一种限制访问范围的方法,用于替代静态全局变量:

namespace {
    void privateFunction() {
        std::cout << "This is a private function." << std::endl;
    }
}
// 在同一源文件中可以直接调用
privateFunction();

 

C++输入&输出

C++使用cincout来处理标准输入输出,它们分别用于从键盘读取输入和向屏幕输出文本。掌握这些基本的I/O操作对于与用户交互的程序来说至关重要。例如,使用cout << "Hello, World!" << endl;可以在屏幕上打印一条消息。

C++标准库中的输入和输出操作主要通过iostream库实现,它包括用于输入的cin、用于输出的cout、用于处理错误输出的cerr,以及用于日志输出的clog对象。

基础的输入输出

  • 输出cout用于向标准输出设备(通常是终端或屏幕)输出数据。使用插入操作符<<cout传递数据。
  • 输入cin用于从标准输入设备(通常是键盘)读取数据。使用提取操作符>>cin接收数据。
  • 示例代码
    #include <iostream>
    
    int main() {
        int number;
        std::cout << "Enter an integer: ";
        std::cin >> number;
        std::cout << "You entered " << number << std::endl;
        return 0;
    }
    

    这段代码演示了如何从用户那里获取一个整数并打印出来

格式化输出

C++提供了多种方式来格式化输出,例如设置宽度、填充字符和精度等。

设置宽度和填充 

使用setw来设置下一个输出项的最小宽度,setfill来设置用于填充额外空间的字符。

 

#include <iostream>
#include <iomanip> // For setw and setfill

int main() {
    std::cout << std::setw(10) << std::setfill('*') << 123 << std::endl;
    return 0;
}

输出:*******123

设置精度

使用setprecision来设置浮点数的输出精度。

 

#include <iostream>
#include <iomanip> // For setprecision

int main() {
    double pi = 3.14159;
    std::cout << std::setprecision(4) << pi << std::endl;
    return 0;
}

 输出:3.142

处理文件输入输出

C++使用fstream库中的ifstream(输入文件流)和ofstream(输出文件流)来处理文件输入输出。

文件写入示例
#include <fstream>
#include <iostream>

int main() {
    std::ofstream file("example.txt");
    if (file.is_open()) {
        file << "Hello, file IO!" << std::endl;
        file.close();
    } else {
        std::cout << "Unable to open file";
    }
    return 0;
}

 文件读取示例

#include <fstream>
#include <iostream>
#include <string>

int main() {
    std::ifstream file("example.txt");
    std::string line;
    if (file.is_open()) {
        while (getline(file, line)) {
            std::cout << line << '\n';
        }
        file.close();
    } else {
        std::cout << "Unable to open file";
    }
    return 0;
}

这些例子展示了如何写入和读取文件。在处理文件时,总是检查文件是否成功打开,以及在操作完成后关闭文件是一个好习惯。 

缺省参数

缺省参数允许你在函数声明时指定默认值。这意味着在调用函数时,如果没有提供足够的参数,C++会自动使用这些默认值。这使得函数调用更加灵活,同时减少了代码的冗余。

缺省参数(也称为默认参数)是C++函数定义中的一项特性,它允许函数调用时省略某些参数。如果调用函数时没有提供足够的实参,C++编译器会自动使用定义函数时指定的默认值来填充缺失的参数。这一特性可以使函数调用更加灵活,减少重载函数的需要,同时也让代码更简洁。

使用缺省参数的规则和示例

缺省参数的设置非常简单,只需在函数声明或定义时,为参数指定一个默认值即可。但是,有几个规则需要遵守:

  1. 缺省参数从右向左:如果函数的某个参数有默认值,那么它右边的所有参数也必须有默认值。
  2. 声明与定义:如果函数在头文件中声明,在源文件中定义,那么缺省参数应该在声明时给出,而在定义时则不应重复。
  3. 调用时省略参数:调用函数时,可以省略具有默认值的参数,但是必须从最右边的参数开始省略。
示例代码

考虑以下函数,它用于显示一条消息,并可以指定消息前的空格数和是否换行。

#include <iostream>
using namespace std;

void displayMessage(string message, int indent = 0, bool newline = true) {
    for (int i = 0; i < indent; ++i) {
        cout << " ";
    }
    cout << message;
    if (newline) cout << endl;
}

int main() {
    displayMessage("Hello, World!"); // 使用默认参数
    displayMessage("Hello, C++!", 5); // 指定缩进,使用默认的换行参数
    displayMessage("Welcome to programming!", 0, false); // 指定所有参数
    return 0;
}

在这个例子中,displayMessage函数有两个缺省参数:indentnewline。这使得函数调用非常灵活,你可以只提供必要的参数。

缺省参数的好处与限制

好处

  • 减少函数重载数量:通过为参数提供默认值,可以减少需要编写的函数重载数量,使代码更加简洁。
  • 提高函数调用的灵活性:函数调用者可以根据需要提供不同数量的参数。

限制

  • 可能影响代码清晰度:使用缺省参数时,函数的调用可能不那么直观,尤其是当函数有多个参数时,理解每个参数的默认值可能需要查看函数定义。
  • 与函数重载的潜在冲突:如果函数重载与缺省参数结合使用不当,可能会导致调用模糊不清,编译器可能无法确定调用哪个函数版本。

最佳实践

  • 当函数的某些参数大多数情况下都是相同的值时,考虑使用缺省参数。
  • 在设计接口时,仔细考虑哪些参数是真正需要由调用者指定的,哪些可以有合理的默认值。
  • 保持缺省参数的简单性,避免使用复杂的表达式作为默认值。
  • 明确文档化每个参数的默认值,确保函数的使用者能够清楚地理解每个参数的作用和默认行为。

缺省参数是C++提供的一个强大工具,它在简化函数调用和减少代码重复方面非常有用。合理使用缺省参数可以使你的C++代码更加简洁和灵活,但同时也需要注意避免上述提到的潜在陷阱。

 

引用

在C++中,引用是一种复合类型,它允许你创建一个别名,即对另一个变量的引用。引用在传递大型数据结构给函数或返回函数结果时特别有用,因为它们可以避免数据的复制,从而提高效率。

引用在C++中是一个非常强大的特性,它为变量创建了一个别名。通过使用引用,可以创建一个新的名字来代表某个已存在的变量。引用主要用于函数参数传递和返回值,它们使得函数能够直接操作实参,而不是操作其副本。这不仅提高了效率,因为避免了不必要的对象复制,还允许函数修改传递给它的参数。

引用的基本概念

  • 声明引用:声明引用时,需要在类型后面加上&符号和引用的名称。例如,int& ref = x;声明了一个对整数x的引用。
  • 初始化引用:引用必须在声明时被初始化,并且一旦被初始化后,就不能改变引用的目标。这意味着引用一旦绑定到一个变量上,就会一直代表那个变量,不能被重新赋值为其他变量的引用。
  • 引用与指针:引用在某种意义上类似于指针,但是它们更安全、更易于使用。引用总是指向一个有效的内存地址,而且不需要使用解引用操作符*来访问目标变量。

使用引用的场景

  1. 函数参数传递:通过将函数参数声明为引用类型,可以允许函数内部的操作影响到实参。这对于数据结构如字符串和大型对象尤其有用,因为它避免了复制的开销。
  2. 函数返回值:函数可以返回一个引用,这样可以允许函数调用作为左值使用。但是要小心不要返回局部变量的引用,因为局部变量在函数返回后会被销毁。
  3. 范围for循环:在基于范围的for循环中使用引用,可以直接修改容器中的元素。
示例代码
引用作为函数参数
#include <iostream>
using namespace std;

void swap(int& a, int& b) {
    int temp = a;
    a = b;
    b = temp;
}

int main() {
    int x = 10, y = 20;
    swap(x, y);
    cout << "x: " << x << " y: " << y << endl; // 输出:x: 20 y: 10
    return 0;
}

 返回引用

#include <vector>
#include <iostream>
using namespace std;

int& getElement(vector<int>& v, int index) {
    return v[index];
}

int main() {
    vector<int> vec = {1, 2, 3};
    getElement(vec, 1) = 4; // 修改vec中索引为1的元素
    cout << vec[1] << endl; // 输出:4
    return 0;
}

注意事项

  • 避免返回局部变量的引用。
  • 使用引用时,确保引用的对象在引用的生命周期内是有效的。
  • 引用一旦被初始化后,就不能再绑定到另一个对象上。

 

【面试题】

宏的优缺点?
优点:
1. 增强代码的复用性。
2. 提高性能。
缺点:
1. 不方便调试宏。(因为预编译阶段进行了替换)
2. 导致代码可读性差,可维护性差,容易误用。
3. 没有类型安全的检查 。
C++ 有哪些技术替代宏
1. 常量定义 换用 const enum
2. 短小函数定义 换用内联函数

内联函数

内联函数是一种优化技术,它通过在编译时将函数体插入到每个调用点,来减少函数调用的开销。对于小而频繁调用的函数,使用内联函数可以提高程序的执行速度。

内联函数是C++提供的一种特性,旨在减少函数调用时的开销。当函数被声明为内联时,编译器会尝试将函数调用处替换为函数本身的代码。这意味着在每个调用点,编译器会将整个函数体插入,从而避免了函数调用的成本,如参数传递、控制权转移和返回值处理等。内联函数特别适用于代码体积小、调用频繁的函数。

使用内联函数的优势

  1. 性能提升:避免了函数调用的开销,对于小型函数,尤其是在循环中频繁调用的情况下,可以显著提高程序的执行速度。
  2. 代码优化:编译器可以对内联函数进行更多优化,因为函数体直接嵌入到调用处,编译器有更大的自由度来优化这些代码,比如消除临时变量、合并计算等。

如何声明内联函数

内联函数通过在函数声明前加上inline关键字来指定。例如:

inline int max(int a, int b) {
    return a > b ? a : b;
}

注意事项和限制

  • 编译器的自由度:声明一个函数为内联并不意味着编译器一定会内联该函数。编译器会根据函数的复杂性、调用的上下文以及特定的编译器优化设置来决定是否内联。
  • 代码膨胀:过度使用内联函数可能导致二进制文件体积的增加,因为每个调用点都会插入完整的函数体。这可能导致缓存命中率降低,反而降低程序的整体性能。
  • 适用场景:一般而言,只有当函数体较小(如只有几行代码)、调用频繁且执行时间短于函数调用开销时,将函数声明为内联才是有意义的。
  • 递归函数:递归函数通常不适合声明为内联,因为递归调用自身会导致大量的代码膨胀。
  • 虚函数:虚函数也可以被声明为内联,但内联是在编译时决定的,而虚函数的动态绑定则是在运行时进行的。如果虚函数在编译时可以确定调用哪个函数,则它可以被内联;否则,即使声明为内联,也只能按照普通虚函数调用来处理。

示例:内联函数与普通函数的比较

假设有一个计算平方的简单函数:

// 普通函数
int square(int x) {
    return x * x;
}

// 内联函数
inline int squareInline(int x) {
    return x * x;
}

 在编译时,对squareInline的每个调用都有可能被替换为x * x,而对square的调用则保留为函数调用。这种差异在循环或频繁调用的情况下可能会导致显著的性能差异。

auto关键字

C++11引入的auto关键字允许自动类型推导,使得编码更加简洁。使用auto可以避免复杂类型的显式声明,让编译器自动为变量推断出最合适的类型。

auto关键字在C++11及其后续版本中引入,标志着类型推导的一个重要步骤。它允许编译器自动推导变量的类型,基于变量初始化时赋予的表达式。这不仅可以简化代码,减少编写的类型声明,还能增强代码的可读性和可维护性,特别是在处理复杂类型(如STL容器的迭代器)时。

auto关键字的优点

  1. 简化类型声明:当变量类型显而易见时,使用auto可以避免冗长的类型名称,使代码更加简洁。
  2. 提高可维护性:如果表达式类型更改,auto可以自动更新变量类型,减少必须手动更改的地方。
  3. 增强代码的通用性:使用auto可以编写与类型无关的代码,特别是在模板编程和使用lambda表达式时更加明显。

使用auto的场景

  • 局部变量初始化:最常用于局部变量的初始化,特别是当右侧表达式的类型复杂或难以直接表达时。

    auto x = 5; // x 被推导为 int
    auto pi = 3.14159; // pi 被推导为 double
    

    迭代器和Lambda表达式:在使用STL容器时,迭代器类型经常使用auto来简化

    std::vector<int> vec = {1, 2, 3, 4};
    auto it = vec.begin(); // 自动推导为 std::vector<int>::iterator
    

    范围基于的循环:在C++11中引入的范围基于的循环中使用auto,可以使代码更加清晰

    for (auto& value : vec) {
        std::cout << value << std::endl;
    }
    

    函数返回类型推导:C++14扩展了auto的使用,允许用于函数返回类型的推导。

    auto sum(int x, int y) -> int {
        return x + y;
    }
    

    注意事项

  • 初始化必须:使用auto声明的变量必须在声明时初始化,因为编译器需要初始化表达式来推导类型。

  • 顶层const和引用auto会忽略掉初始化表达式的顶层const,但底层const会被保留。如果需要顶层const或引用,需要显式指定。

    const auto x = 5; // x 是一个 const int
    auto& y = x; // y 是对 const int 的引用
    

    不可用于函数参数auto不能用作函数参数类型,这种情况下仍需明确指定参数类型或使用模板。

基于范围的for循环

基于范围的for循环是C++11标准引入的一项特性,它提供了一种更简洁、更安全的方式来遍历容器(如数组、向量等)或任何序列范围。这种循环方式自动迭代容器或范围的所有元素,无需手动管理迭代器或索引,从而减少了编码错误的可能性并提高了代码的可读性。

基本语法

基于范围的for循环的基本语法如下:

for (declaration : range) {
    // loop body
}

 

其中,declaration声明了一个变量,该变量将被用来访问序列中的每个元素。range是要迭代的序列或容器。

示例

假设有一个整数数组,要遍历数组中的每个元素:

int arr[] = {1, 2, 3, 4, 5};
for (int elem : arr) {
    std::cout << elem << std::endl;
}

使用auto

在基于范围的for循环中,可以使用auto关键字来自动推导元素的类型,这样做可以简化代码并提高其适应性

for (auto elem : vec) {
    std::cout << elem << std::endl;
}

 

修改容器元素

如果想要在循环中修改容器中的元素,需要在循环变量声明中使用引用(&):

for (auto& elem : vec) {
    elem *= 2; // Doubles the value of each element in the vector
}

 

避免不必要的复制

对于包含大型对象的容器,为了避免在每次迭代时复制对象,应该使用引用来遍历元素,特别是当不需要修改元素时,应使用常量引用:

for (const auto& elem : vec) {
    std::cout << elem << std::endl;
}

优点和局限

  • 优点

    • 简化了遍历容器和序列的语法。
    • 减少了与迭代器和索引管理相关的错误。
    • 自动推导元素类型,使代码更简洁。
  • 局限

    • 不适用于需要访问当前迭代元素索引的场景。
    • 当需要在循环中删除或插入元素时,可能不是最佳选择,因为这可能需要直接访问容器的迭代器。

 

结语

C++是一门功能强大的编程语言,拥有广泛的应用领域。通过掌握上述核心概念,你已经迈出了成为C++程序员的第一步。记住,学习编程是一个逐步探索和不断实践的过程。不断练习,寻找资源,并尝试自己编写代码,你将会发现C++的强大之处。祝你旅途愉快!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1447498.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SPP改进(多窗口池化)

论文创新点汇总&#xff1a;人工智能论文通用创新点(持续更新中...)-CSDN博客 原来的模型 15年提出 本质&#xff1a; 多个不同大小的池化窗口进行池化 池化窗口越大得到的特征越少 之后再将不同池化窗口得到的特征拼接起来 现在的改进 实现代码 class SPPCSPC(nn.Modul…

【python】网络爬虫与信息提取--Beautiful Soup库

Beautiful Soup网站&#xff1a;https://www.crummy.com/software/BeautifulSoup/ 作用&#xff1a;它能够对HTML.xml格式进行解析&#xff0c;并且提取其中的相关信息。它可以对我们提供的任何格式进行相关的爬取&#xff0c;并且可以进行树形解析。 使用原理&#xff1a;它能…

Linux_线程

线程与进程 多级页表 线程控制 线程互斥 线程同步 生产者消费者模型 常见概念 下面选取32位系统举例。 一.线程与进程 上图是曾经我们认为进程所占用的资源的集合。 1.1 线程概念 线程是一个执行分支&#xff0c;执行粒度比进程细&#xff0c;调度成本比进程低线程是cpu…

LeetCode、739. 每日温度【中等,单调栈】

文章目录 前言LeetCode、739. 每日温度【中等&#xff0c;单调栈】题目链接及分类思路单调栈 资料获取 前言 博主介绍&#xff1a;✌目前全网粉丝2W&#xff0c;csdn博客专家、Java领域优质创作者&#xff0c;博客之星、阿里云平台优质作者、专注于Java后端技术领域。 涵盖技…

python算法之 Dijkstra 算法

文章目录 基本思想&#xff1a;步骤&#xff1a;复杂度&#xff1a;注意事项&#xff1a;代码实现K 站中转内最便宜的航班 Dijkstra 算法是一种用于解决单源最短路径问题的经典算法。该问题的目标是找到从图中的一个固定顶点&#xff08;称为源点&#xff09;到图中所有其他顶点…

vue三种路由守卫详解

在 Vue 中&#xff0c;可以通过路由守卫来实现路由鉴权。Vue 提供了三种路由守卫&#xff1a;全局前置守卫、全局解析守卫和组件内的守卫。 全局前置守卫 通过 router.beforeEach() 方法实现&#xff0c;可以在路由跳转之前进行权限判断。在这个守卫中&#xff0c;可以根据用…

C++ Qt框架开发 | 基于Qt框架开发实时成绩显示排序系统(2)折线图显示

对上一篇的工作C学习笔记 | 基于Qt框架开发实时成绩显示排序系统1-CSDN博客继续优化&#xff0c;增加一个显示运动员每组成绩的折线图。 1&#xff09;在Qt Creator的项目文件&#xff08;.pro文件&#xff09;中添加对Qt Charts模块的支持&#xff1a; QT charts 2&#xf…

【动态规划】【中位数】【C++算法】1478. 安排邮筒

# 作者推荐 【深度优先搜索】【树】【图论】2973. 树中每个节点放置的金币数目 本文涉及知识点 动态规划汇总 LeetCode1478. 安排邮筒 给你一个房屋数组houses 和一个整数 k &#xff0c;其中 houses[i] 是第 i 栋房子在一条街上的位置&#xff0c;现需要在这条街上安排 k…

Linux_文件系统

假定外部存储设备为磁盘&#xff0c;文件如果没有被使用&#xff0c;那么它静静躺在磁盘上&#xff0c;如果它被使用&#xff0c;则文件将被加载进内存中。故此&#xff0c;可以将文件分为内存文件和磁盘文件。 内存文件 磁盘文件 软、硬链接 一.内存文件 1.1 c语言的文件接口 …

波奇学Linux:文件系统

磁盘认识 磁盘被访问的基本单元是扇区-512字节。 磁盘可以看成多个同心圆&#xff0c;每个同心圆叫做磁道&#xff0c;多个扇区组成同心圆。 我们可以把磁盘看做由无数个扇区构成的存储介质。 要把数据存到磁盘&#xff0c;先定位扇区&#xff0c;用哪一个磁头&#xff0c;…

算法沉淀——链表(leetcode真题剖析)

算法沉淀——链表 01.两数相加02.两两交换链表中的节点03.重排链表04.合并 K 个升序链表05.K个一组翻转链表 链表常用技巧 1、画图->直观形象、便于理解 2、引入虚拟"头节点" 3、要学会定义辅助节点&#xff08;比如双向链表的节点插入&#xff09; 4、快慢双指针…

高效的工作学习方法

1.康奈尔笔记法 在这里插入图片描述 2. 5W2H法 3. 鱼骨图分析法 4.麦肯锡7步分析法 5.使用TODOLIST 6.使用计划模板&#xff08;年月周&#xff09; 7. 高效的学习方法 成年人的学习特点&#xff1a; 快速了解一个领域方法 沉浸式学习方法&#xff1a; 沉浸学习的判据&am…

MATLAB知识点:fibonacci函数(★☆☆☆☆)返回斐波那契数列

​讲解视频&#xff1a;可以在bilibili搜索《MATLAB教程新手入门篇——数学建模清风主讲》。​ MATLAB教程新手入门篇&#xff08;数学建模清风主讲&#xff0c;适合零基础同学观看&#xff09;_哔哩哔哩_bilibili 节选自第3章&#xff1a;课后习题讲解中拓展的函数 在讲解第…

C++ //练习 5.12 修改统计元音字母的程序,使其能统计以下含有两个字符的字符序列的数量:ff、fl和fi。

C Primer&#xff08;第5版&#xff09; 练习 5.12 练习 5.12 修改统计元音字母的程序&#xff0c;使其能统计以下含有两个字符的字符序列的数量&#xff1a;ff、fl和fi。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09; 工具&#xff1a;vim 代码块 /****…

学习总结11

KMP算法 全称Knuth-Morris-Pratt算法&#xff0c;是一种字符串匹配算法。该算法的目的是在一个文本串S内查找一个模式串P的出现位置。 KMP算法的核心思想是利用模式串自身的特性来避免不必要的字符比较。算法通过构建一个部分匹配表&#xff08;也称为next数组&#xff09;&a…

实景剧本杀小程序:创新体验,沉浸式推理乐趣

随着科技的飞速发展&#xff0c;人们对于娱乐方式的追求也在不断升级。传统的桌面剧本杀游戏已经不能满足玩家的需求&#xff0c;他们渴望更加真实、刺激的游戏体验。正是这种需求推动下&#xff0c;实景剧本杀小程序应运而生&#xff0c;为玩家带来前所未有的推理乐趣。 实景…

【51单片机】利用【时间延迟】的原理规避【按键抖动问题】

前言 大家好吖&#xff0c;欢迎来到 YY 滴单片机系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过单片机的老铁 本章是51LCD单片机设计的一个环节&#xff0c;完整可前往相应博客查看完整传送门 欢迎订阅 YY滴C专栏&#xff01;更多干货持续更新&#xff01;以下…

详解结构体内存对齐及结构体如何实现位段~

目录 ​编辑 一&#xff1a;结构体内存对齐 1.1对齐规则 1.2.为什么存在内存对齐 1.3修改默认对齐数 二.结构体实现位段 2.1什么是位段 2.2位段的内存分配 2.3位段的跨平台问题 2.4位段的应用 2.5位段使用的注意事项 三.完结散花 悟已往之不谏&#xff0c;知来者犹可…

幻兽帕鲁Palworld专用服务器CPU内存配置怎么选择?

腾讯云幻兽帕鲁服务器配置怎么选&#xff1f;根据玩家数量选择CPU内存配置&#xff0c;4到8人选择4核16G、10到20人玩家选择8核32G、2到4人选择4核8G、32人选择16核64G配置&#xff0c;腾讯云百科txybk.com来详细说下腾讯云幻兽帕鲁专用服务器CPU内存带宽配置选择方法&#xff…

【Chrono Engine学习总结】5-sensor-5.1-sensor基础并创建一个lidar

由于Chrono的官方教程在一些细节方面解释的并不清楚&#xff0c;自己做了一些尝试&#xff0c;做学习总结。 1、Sensor模块 Sensor模块是附加模块&#xff0c;需要单独安装。参考&#xff1a;【Chrono Engine学习总结】1-安装配置与程序运行 Sensor Module Tutorial Sensor …