【CPP】CPP经典面试题

文章目录

    • 引言
    • 1. C++ 基础
      • 1.1 C++ 中的 `const` 关键字
      • 1.2 C++ 中的 `static` 关键字
    • 2. 内存管理
      • 2.1 C++ 中的 `new` 和 `delete`
      • 2.2 内存泄漏
    • 3. 面向对象编程
      • 3.1 继承和多态
      • 3.2 多重继承
    • 4. 模板和泛型编程
      • 4.1 函数模板
      • 4.2 类模板
    • 5. STL 和标准库
      • 5.1 容器
      • 5.2 迭代器
    • 6. 高级特性
      • 6.1 移动语义和右值引用
      • 6.2 Lambda 表达式
    • 7. 设计模式
      • 7.1 单例模式
      • 7.2 工厂模式
    • 8. 性能优化
      • 8.1 内联函数
      • 8.2 缓存友好性
    • 9. 并发编程
      • 9.1 线程
      • 9.2 条件变量
    • 10. 异常处理
      • 10.1 异常机制
    • 11. C++17 和 C++20 新特性
      • 11.1 C++17 新特性
      • 11.2 C++20 新特性
    • 12. 实际应用
      • 12.1 智能指针
      • 12.2 RAII 原则
    • 13. 调试和测试
      • 13.1 调试技巧
      • 13.2 单元测试
      • 13.3 性能分析
    • 14. C++ 编码规范
      • 14.1 命名规范
      • 14.2 代码格式化
    • 15. C++ 项目构建
      • 15.1 Makefile
      • 15.2 CMake
    • 16. C++ 未来发展趋势
      • 16.1 C++23 新特性
      • 16.2 C++ 的未来
    • 结语

在这里插入图片描述

引言

C++ 是一门强大且复杂的编程语言,广泛应用于系统编程、游戏开发、嵌入式系统和高性能计算等领域。由于其灵活性和性能优势,C++ 程序员在面试中常常会遇到各种深入的问题。本文将探讨一些经典的 C++ 面试题,涵盖从基础语法到高级特性的多个方面,帮助读者更好地准备面试。

1. C++ 基础

1.1 C++ 中的 const 关键字

const 是 C++ 中用于定义常量的关键字。它可以用于修饰变量、函数参数、函数返回值以及成员函数。

问题: const#define 有什么区别?

答案:

  • const 是类型安全的,编译器会进行类型检查,而 #define 是宏定义,只是简单的文本替换。
  • const 定义的常量在编译时分配内存,而 #define 不分配内存。
  • const 可以用于修饰类的成员函数,表示该函数不会修改类的成员变量。

问题: const 成员函数的作用是什么?

答案:
const 成员函数表示该函数不会修改类的成员变量。它可以被 const 对象调用,而非 const 对象既可以调用 const 成员函数,也可以调用非 const 成员函数。

1.2 C++ 中的 static 关键字

static 关键字在 C++ 中有多种用途,包括修饰局部变量、全局变量、类成员变量和类成员函数。

问题: static 局部变量和普通局部变量有什么区别?

答案:

  • static 局部变量的生命周期贯穿整个程序运行期间,即使函数调用结束,static 局部变量也不会被销毁。
  • 普通局部变量的生命周期仅限于函数调用期间,函数调用结束后,局部变量会被销毁。

问题: static 成员函数和普通成员函数有什么区别?

答案:

  • static 成员函数不依赖于类的实例,可以直接通过类名调用。
  • static 成员函数不能访问类的非静态成员变量和非静态成员函数,因为它们没有 this 指针。

2. 内存管理

2.1 C++ 中的 newdelete

newdelete 是 C++ 中用于动态内存分配和释放的操作符。

问题: newmalloc 有什么区别?

答案:

  • new 是 C++ 的操作符,而 malloc 是 C 标准库函数。
  • new 会自动调用对象的构造函数,malloc 不会。
  • new 返回的是对象类型的指针,malloc 返回的是 void*,需要显式类型转换。
  • new 分配内存失败时会抛出 std::bad_alloc 异常,malloc 失败时返回 NULL

问题: deletefree 有什么区别?

答案:

  • delete 是 C++ 的操作符,而 free 是 C 标准库函数。
  • delete 会自动调用对象的析构函数,free 不会。
  • delete 用于释放 new 分配的内存,free 用于释放 malloc 分配的内存。

2.2 内存泄漏

内存泄漏是指程序在动态分配内存后,未能正确释放该内存,导致内存占用不断增加。

问题: 如何避免内存泄漏?

答案:

  • 使用智能指针(如 std::unique_ptrstd::shared_ptr)来管理动态内存。
  • 确保每次 new 操作都有对应的 delete 操作。
  • 使用 RAII(Resource Acquisition Is Initialization)原则,将资源的生命周期与对象的生命周期绑定。

3. 面向对象编程

3.1 继承和多态

继承和多态是面向对象编程中的核心概念。

问题: 什么是虚函数?为什么需要虚函数?

答案:
虚函数是用于实现多态的机制。通过在基类中声明虚函数,派生类可以重写该函数,从而实现运行时多态。当通过基类指针或引用调用虚函数时,实际调用的是派生类的重写函数。

问题: 虚函数表(vtable)是什么?

答案:
虚函数表是编译器为每个包含虚函数的类生成的一个表,表中存储了虚函数的地址。每个对象在内存中都有一个指向虚函数表的指针(vptr),通过这个指针可以在运行时确定调用哪个虚函数。

3.2 多重继承

多重继承是指一个类可以从多个基类继承。

问题: 多重继承会带来什么问题?如何解决?

答案:
多重继承可能导致菱形继承问题(Diamond Problem),即一个类从两个基类继承,而这两个基类又共同继承自同一个基类,导致派生类中包含多个相同的基类子对象。可以通过虚继承(virtual inheritance)来解决这个问题。

4. 模板和泛型编程

4.1 函数模板

函数模板允许编写通用的函数,可以处理不同类型的参数。

问题: 函数模板和函数重载有什么区别?

答案:

  • 函数模板通过参数类型推导生成具体的函数实例,适用于不同类型的数据。
  • 函数重载是通过定义多个同名函数,每个函数处理不同类型的参数。

问题: 如何特化一个函数模板?

答案:
可以通过显式特化或部分特化来为特定类型提供特殊的实现。显式特化是为特定类型提供完全不同的实现,而部分特化是为特定类型的一部分提供不同的实现。

4.2 类模板

类模板允许编写通用的类,可以处理不同类型的数据。

问题: 类模板和模板类的区别是什么?

答案:

  • 类模板是模板的定义,尚未实例化。
  • 模板类是类模板实例化后的具体类。

问题: 如何特化一个类模板?

答案:
可以通过显式特化或部分特化来为特定类型提供特殊的实现。显式特化是为特定类型提供完全不同的实现,而部分特化是为特定类型的一部分提供不同的实现。

5. STL 和标准库

5.1 容器

STL 提供了多种容器,如 vectorlistmap 等。

问题: vectorlist 有什么区别?

答案:

  • vector 是动态数组,支持随机访问,插入和删除操作在尾部高效,但在中间或头部效率较低。
  • list 是双向链表,不支持随机访问,插入和删除操作在任何位置都高效。

问题: mapunordered_map 有什么区别?

答案:

  • map 是基于红黑树实现的,元素按键值有序存储,查找、插入和删除操作的时间复杂度为 O(log n)。
  • unordered_map 是基于哈希表实现的,元素无序存储,查找、插入和删除操作的平均时间复杂度为 O(1)。

5.2 迭代器

迭代器是用于遍历容器中元素的对象。

问题: 迭代器的种类有哪些?

答案:

  • 输入迭代器:只能读取元素,单向移动。
  • 输出迭代器:只能写入元素,单向移动。
  • 前向迭代器:可以读取和写入元素,单向移动。
  • 双向迭代器:可以读取和写入元素,双向移动。
  • 随机访问迭代器:可以读取和写入元素,支持随机访问。

6. 高级特性

6.1 移动语义和右值引用

移动语义和右值引用是 C++11 引入的重要特性,用于提高性能。

问题: 什么是右值引用?为什么需要右值引用?

答案:
右值引用是用于绑定临时对象(右值)的引用类型,通过 && 表示。右值引用允许将资源(如动态内存)从一个对象“移动”到另一个对象,避免不必要的拷贝操作,从而提高性能。

问题: 什么是移动构造函数和移动赋值运算符?

答案:
移动构造函数和移动赋值运算符是用于实现移动语义的特殊成员函数。移动构造函数接受一个右值引用参数,将资源从源对象移动到目标对象。移动赋值运算符也接受一个右值引用参数,将资源从源对象移动到目标对象,并释放目标对象原有的资源。

6.2 Lambda 表达式

Lambda 表达式是 C++11 引入的匿名函数,可以方便地定义和使用函数对象。

问题: Lambda 表达式的语法是什么?

答案:
Lambda 表达式的语法为:

[捕获列表](参数列表) -> 返回类型 { 函数体 }

捕获列表用于指定 Lambda 表达式可以访问的外部变量,参数列表和返回类型与普通函数类似。

问题: Lambda 表达式的捕获列表有哪些方式?

答案:

  • [&]:以引用方式捕获所有外部变量。
  • [=]:以值方式捕获所有外部变量。
  • [&x, =y]:以引用方式捕获 x,以值方式捕获 y
  • [this]:捕获当前对象的 this 指针。

7. 设计模式

7.1 单例模式

单例模式是一种创建型设计模式,确保一个类只有一个实例,并提供全局访问点。

问题: 如何实现线程安全的单例模式?

答案:
可以通过双重检查锁定(Double-Checked Locking)或使用局部静态变量来实现线程安全的单例模式。

class Singleton {
public:
    static Singleton& getInstance() {
        static Singleton instance;
        return instance;
    }

private:
    Singleton() {}
    Singleton(const Singleton&) = delete;
    Singleton& operator=(const Singleton&) = delete;
};

7.2 工厂模式

工厂模式是一种创建型设计模式,用于创建对象而不指定具体的类。

问题: 工厂模式和抽象工厂模式有什么区别?

答案:

  • 工厂模式定义一个创建对象的接口,但由子类决定实例化哪个类。
  • 抽象工厂模式提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。

8. 性能优化

8.1 内联函数

内联函数是一种优化技术,通过在调用点展开函数体来减少函数调用的开销。

问题: 内联函数有什么优缺点?

答案:

  • 优点:减少函数调用的开销,提高执行效率。
  • 缺点:增加代码体积,可能导致缓存不命中,影响性能。

问题: 如何定义内联函数?

答案:
可以通过 inline 关键字定义内联函数,或者在类定义中直接定义成员函数。

inline int add(int a, int b) {
    return a + b;
}

8.2 缓存友好性

缓存友好性是指程序在访问内存时能够充分利用 CPU 缓存,减少缓存未命中的次数。

问题: 如何编写缓存友好的代码?

答案:

  • 尽量使用连续内存访问模式,如数组遍历。
  • 避免频繁的内存分配和释放,减少内存碎片。
  • 使用适当的数据结构和算法,减少不必要的内存访问。

9. 并发编程

9.1 线程

C++11 引入了多线程支持,提供了 std::thread 类。

问题: 如何创建和启动一个线程?

答案:
可以通过 std::thread 类创建和启动一个线程,线程函数可以是普通函数、Lambda 表达式或成员函数。

#include <iostream>
#include <thread>

void threadFunc() {
    std::cout << "Hello from thread!" << std::endl;
}

int main() {
    std::thread t(threadFunc);
    t.join();
    return 0;
}

问题: 如何避免数据竞争?

答案:
可以通过互斥锁(std::mutex)、原子操作(std::atomic)或其他同步机制来避免数据竞争。

9.2 条件变量

条件变量是用于线程间同步的机制,允许线程等待某个条件成立。

问题: 如何使用条件变量?

答案:
可以通过 std::condition_variablestd::mutex 来实现线程间的条件等待和通知。

#include <iostream>
#include <thread>
#include <mutex>
#include <condition_variable>

std::mutex mtx;
std::condition_variable cv;
bool ready = false;

void waitForReady() {
    std::unique_lock<std::mutex> lock(mtx);
    cv.wait(lock, []{ return ready; });
    std::cout << "Ready!" << std::endl;
}

void setReady() {
    std::this_thread::sleep_for(std::chrono::seconds(1));
    {
        std::lock_guard<std::mutex> lock(mtx);
        ready = true;
    }
    cv.notify_all();
}

int main() {
    std::thread t1(waitForReady);
    std::thread t2(setReady);
    t1.join();
    t2.join();
    return 0;
}

10. 异常处理

10.1 异常机制

C++ 提供了异常处理机制,允许程序在运行时处理错误。

问题: trycatchthrow 的作用是什么?

答案:

  • try 块用于包含可能抛出异常的代码。
  • catch 块用于捕获并处理异常。
  • throw 用于抛出异常。

问题: 如何自定义异常类?

答案:
可以通过继承 std::exception 类来定义自定义异常类,并重写 what() 方法以提供异常描述。

#include <exception>
#include <string>

class MyException : public std::exception {
public:
    MyException(const std::string& msg) : msg(msg) {}
    const char* what() const noexcept override {
        return msg.c_str();
    }

private:
    std::string msg;
};

11. C++17 和 C++20 新特性

11.1 C++17 新特性

C++17 引入了许多新特性,如结构化绑定、std::optionalstd::variant 等。

问题: 什么是结构化绑定?

答案:
结构化绑定允许将结构体或数组的元素绑定到变量上,简化代码。

#include <iostream>
#include <tuple>

int main() {
    std::tuple<int, double, std::string> t(1, 2.0, "hello");
    auto [a, b, c] = t;
    std::cout << a << ", " << b << ", " << c << std::endl;
    return 0;
}

11.2 C++20 新特性

C++20 引入了更多新特性,如概念(Concepts)、范围(Ranges)、协程(Coroutines)等。

问题: 什么是概念(Concepts)?

答案:
概念是用于约束模板参数的机制,可以在编译时检查模板参数是否满足特定要求。

#include <concepts>
#include <iostream>

template<typename T>
requires std::integral<T>
void print(T value) {
    std::cout << value << std::endl;
}

int main() {
    print(42);  // OK
    // print(3.14);  // Error: does not satisfy std::integral
    return 0;
}

12. 实际应用

12.1 智能指针

智能指针是 C++11 引入的用于自动管理动态内存的工具。

问题: std::unique_ptrstd::shared_ptr 有什么区别?

答案:

  • std::unique_ptr 是独占所有权的智能指针,不能复制,只能移动。
  • std::shared_ptr 是共享所有权的智能指针,通过引用计数管理资源,可以复制和移动。

问题: 如何使用 std::make_shared

答案:
std::make_shared 是用于创建 std::shared_ptr 的工厂函数,可以一次性分配内存并创建对象。

#include <memory>
#include <iostream>

int main() {
    auto ptr = std::make_shared<int>(42);
    std::cout << *ptr << std::endl;
    return 0;
}

12.2 RAII 原则

RAII(Resource Acquisition Is Initialization)是 C++ 中的重要原则,用于管理资源。

问题: 什么是 RAII 原则?

答案:
RAII 原则是指将资源的生命周期与对象的生命周期绑定,通过对象的构造函数获取资源,通过析构函数释放资源,确保资源在对象销毁时自动释放。

问题: 如何实现 RAII?

答案:
可以通过类的构造函数和析构函数来实现 RAII。例如,使用智能指针管理动态内存,使用 std::fstream 管理文件资源等。

#include <fstream>
#include <iostream>

class FileHandler {
public:
    FileHandler(const std::string& filename) : file(filename) {
        if (!file.is_open()) {
            throw std::runtime_error("Failed to open file");
        }
    }

    ~FileHandler() {
        file.close();
    }

    void write(const std::string& data) {
        file << data;
    }

private:
    std::ofstream file;
};

int main() {
    try {
        FileHandler fh("test.txt");
        fh.write("Hello, RAII!");
    } catch (const std::exception& e) {
        std::cerr << e.what() << std::endl;
    }
    return 0;
}

13. 调试和测试

13.1 调试技巧

调试是程序开发中的重要环节,掌握调试技巧可以提高开发效率。

问题: 如何使用 GDB 调试 C++ 程序?

答案:

  • 编译时添加 -g 选项生成调试信息。
  • 使用 gdb 启动程序,设置断点,单步执行,查看变量值等。
g++ -g -o my_program my_program.cpp
gdb ./my_program

问题: 如何使用 assert 进行调试?

答案:
assert 是用于检查条件的宏,如果条件为假,程序会终止并输出错误信息。

#include <cassert>
#include <iostream>

int main() {
    int x = 5;
    assert(x == 5);
    std::cout << "Assertion passed" << std::endl;
    return 0;
}

13.2 单元测试

单元测试是软件开发中的重要实践,用于验证代码的各个单元(如函数、类)是否按预期工作。

问题: 什么是单元测试?为什么需要单元测试?

答案:

  • 单元测试是对代码的最小可测试单元(如函数、方法)进行测试的过程。
  • 单元测试可以帮助开发者尽早发现代码中的错误,提高代码质量,减少回归问题的发生。

问题: 如何使用 Google Test 进行单元测试?

答案:
Google Test 是一个流行的 C++ 单元测试框架。以下是一个简单的示例:

#include <gtest/gtest.h>

int add(int a, int b) {
    return a + b;
}

TEST(AddTest, HandlesPositiveInput) {
    EXPECT_EQ(add(1, 2), 3);
    EXPECT_EQ(add(10, 20), 30);
}

TEST(AddTest, HandlesNegativeInput) {
    EXPECT_EQ(add(-1, -2), -3);
    EXPECT_EQ(add(-10, -20), -30);
}

int main(int argc, char **argv) {
    ::testing::InitGoogleTest(&argc, argv);
    return RUN_ALL_TESTS();
}

编译并运行测试:

g++ -std=c++11 -isystem /path/to/gtest/include -pthread test.cpp /path/to/gtest/libgtest.a /path/to/gtest/libgtest_main.a -o test
./test

13.3 性能分析

性能分析是优化代码性能的关键步骤,帮助开发者找到代码中的性能瓶颈。

问题: 如何使用 gprof 进行性能分析?

答案:
gprof 是一个常用的性能分析工具,可以生成函数调用图和执行时间统计。

  1. 编译时添加 -pg 选项:

    g++ -pg -o my_program my_program.cpp
    
  2. 运行程序生成性能数据:

    ./my_program
    
  3. 使用 gprof 分析性能数据:

    gprof my_program gmon.out > analysis.txt
    

问题: 如何使用 valgrind 进行内存分析?

答案:
valgrind 是一个强大的工具,用于检测内存泄漏和内存错误。

  1. 安装 valgrind

    sudo apt-get install valgrind
    
  2. 使用 valgrind 运行程序:

    valgrind --leak-check=full ./my_program
    
  3. 查看输出,分析内存泄漏和错误。


14. C++ 编码规范

14.1 命名规范

良好的命名规范可以提高代码的可读性和可维护性。

问题: C++ 中常见的命名规范有哪些?

答案:

  • 变量和函数名使用小写字母和下划线分隔(snake_case)。
  • 类名使用大写字母开头的驼峰命名法(PascalCase)。
  • 常量名使用全大写字母和下划线分隔(UPPER_CASE)。

问题: 为什么命名规范很重要?

答案:
命名规范可以提高代码的可读性,使其他开发者更容易理解代码的意图,减少沟通成本。

14.2 代码格式化

代码格式化是保持代码风格一致的重要手段。

问题: 如何使用 clang-format 格式化代码?

答案:
clang-format 是一个自动格式化 C++ 代码的工具。

  1. 安装 clang-format

    sudo apt-get install clang-format
    
  2. 创建配置文件 .clang-format

    clang-format -style=llvm -dump-config > .clang-format
    
  3. 格式化代码:

    clang-format -i my_program.cpp
    

15. C++ 项目构建

15.1 Makefile

Makefile 是用于自动化构建 C++ 项目的工具。

问题: 如何编写一个简单的 Makefile

答案:
以下是一个简单的 Makefile 示例:

CXX = g++
CXXFLAGS = -std=c++11 -Wall
TARGET = my_program
SRCS = main.cpp utils.cpp
OBJS = $(SRCS:.cpp=.o)

all: $(TARGET)

$(TARGET): $(OBJS)
    $(CXX) $(CXXFLAGS) -o $(TARGET) $(OBJS)

%.o: %.cpp
    $(CXX) $(CXXFLAGS) -c $< -o $@

clean:
    rm -f $(OBJS) $(TARGET)

问题: 如何使用 make 构建项目?

答案:
在项目根目录下运行以下命令:

make

15.2 CMake

CMake 是一个跨平台的构建工具,可以生成 Makefile 或其他构建系统的配置文件。

问题: 如何编写一个简单的 CMakeLists.txt

答案:
以下是一个简单的 CMakeLists.txt 示例:

cmake_minimum_required(VERSION 3.10)
project(MyProgram)

set(CMAKE_CXX_STANDARD 11)

add_executable(my_program main.cpp utils.cpp)

问题: 如何使用 CMake 构建项目?

答案:

  1. 创建构建目录并进入:

    mkdir build
    cd build
    
  2. 运行 cmake 生成构建文件:

    cmake ..
    
  3. 使用 make 构建项目:

    make
    

16. C++ 未来发展趋势

16.1 C++23 新特性

C++23 是 C++ 的下一个版本,预计将引入更多新特性。

问题: C++23 可能引入哪些新特性?

答案:

  • 模块化标准库(Modular Standard Library)。
  • 协程改进(Coroutine Improvements)。
  • 更强大的概念支持(Enhanced Concepts)。

16.2 C++ 的未来

C++ 作为一门历史悠久的语言,仍在不断进化。

问题: C++ 的未来发展方向是什么?

答案:

  • 更加注重性能和安全性。
  • 引入更多现代编程范式,如函数式编程。
  • 提高开发效率,减少语言复杂性。

结语

C++ 是一门强大且复杂的语言,掌握其核心概念和高级特性对于成为一名优秀的 C++ 开发者至关重要。本文涵盖了许多经典的 C++ 面试题,希望能够帮助读者更好地准备面试,并在实际开发中运用这些知识。无论是初学者还是经验丰富的开发者,持续学习和实践都是提升技能的关键。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

深入浅出谈VR(虚拟现实、VR镜头)

1、VR是什么鬼&#xff1f; 近两年VR这次词火遍网上网下&#xff0c;到底什么是VR&#xff1f;VR是“Virtual Reality”&#xff0c;中文名字是虚拟现实&#xff0c;是指采用计算机技术为核心的现代高科技手段生成一种虚拟环境&#xff0c;用户借助特殊的输入/输出设备&#x…

【Redis】安装配置Redis超详细教程 / Linux版

Linux安装配置Redis超详细教程 安装redis依赖安装redis启动redis停止redisredis.conf常见配置设置redis为后台启动修改redis监听地址设置工作目录修改密码监听的端口号数据库数量设置redis最大内存设置日志文件设置redis开机自动启动 学习视频&#xff1a;黑马程序员Redis入门到…

[LeetCode]day16 242.有效的字母异位词

242. 有效的字母异位词 - 力扣&#xff08;LeetCode&#xff09; 题目描述 给定两个字符串 s 和 t &#xff0c;编写一个函数来判断 t 是否是 s 的 字母异位词 示例 1: 输入: s "anagram", t "nagaram" 输出: true示例 2: 输入: s "rat"…

[MoeCTF 2022]baby_file

题目 <html> <title>Heres a secret. Can you find it?</title> <?phpif(isset($_GET[file])){$file $_GET[file];include($file); }else{highlight_file(__FILE__); } ?> </html> 读取flag /?filephp://filter/readconvert.base64-encode…

Centos挂载镜像制作本地yum源,并补装图形界面

内网环境centos7.9安装图形页面内网环境制作本地yum源 上传镜像到服务器目录 创建目录并挂载镜像 #创建目录 cd /mnt/ mkdir iso#挂载 mount -o loop ./CentOS-7-x86_64-DVD-2009.iso ./iso #前面镜像所在目录&#xff0c;后面所挂载得目录#检查 [rootlocalhost mnt]# df -h…

判断您的Mac当前使用的是Zsh还是Bash:echo $SHELL、echo $0

要判断您的Mac当前使用的是Zsh还是Bash&#xff0c;可以使用以下方法&#xff1a; 查看默认Shell: 打开“终端”应用程序&#xff0c;然后输入以下命令&#xff1a; echo $SHELL这将显示当前默认使用的Shell。例如&#xff0c;如果输出是/bin/zsh&#xff0c;则说明您使用的是Z…

python 小游戏:扫雷

目录 1. 前言 2. 准备工作 3. 生成雷区 4. 鼠标点击扫雷 5. 胜利 or 失败 6. 游戏效果展示 7. 完整代码 1. 前言 本文使用 Pygame 实现的简化版扫雷游戏。 如上图所示&#xff0c;游戏包括基本的扫雷功能&#xff1a;生成雷区、左键点击扫雷、右键标记地雷、显示数字提示…

【重新认识C语言----文件管理篇】

目录 ​编辑 -----------------------------------------begin------------------------------------- 引言 1. 文件的基本概念 2. 文件指针 3. 文件的打开与关闭 3.1 打开文件 3.2 关闭文件 4. 文件的读写操作 4.1 读取文件 4.1.1 使用fgetc()读取文件 4.1.2 使用fg…

EasyExcel 导出合并层级单元格

EasyExcel 导出合并层级单元格 一、案例 案例一 1.相同订单号单元格进行合并 合并结果 案例二 1.相同订单号的单元格进行合并2.相同订单号的总数和总金额进行合并 合并结果 案例三 1.相同订单号的单元格进行合并2.相同订单号的商品分类进行合并3.相同订单号的总数和总金额…

WPF 进度条(ProgressBar)示例一

本文讲述&#xff1a;WPF 进度条(ProgressBar)简单的样式修改和使用。 进度显示界面&#xff1a;使用UserControl把ProgressBar和进度值以及要显示的内容全部组装在UserControl界面中&#xff0c;方便其他界面直接进行使用。 <UserControl x:Class"DefProcessBarDemo…

LabVIEW自定义测量参数怎么设置?

以下通过一个温度采集案例&#xff0c;说明在 LabVIEW 中设置自定义测量参数的具体方法&#xff1a; 案例背景 ​ 假设使用 NI USB-6009 数据采集卡 和 热电偶传感器 监测温度&#xff0c;需自定义以下参数&#xff1a; 采样率&#xff1a;1 kHz 输入量程&#xff1a;0~10 V&a…

新能源产业的质量革命:六西格玛培训如何重塑制造竞争力

在新能源行业狂飙突进的今天&#xff0c;企业若想在全球供应链中占据高地&#xff0c;仅靠技术突破已远远不够。制造效率的毫厘之差&#xff0c;可能成为市场话语权的千里之距。某光伏巨头曾因电池片良率低于行业均值1.5%&#xff0c;导致年损失超2.3亿元——这恰恰印证了六西格…

(11)gdb 笔记(4):设置执行方向 set exec-direction,

&#xff08;28&#xff09;引入 record 后&#xff0c;可以 设置执行方向 set exec-direction &#xff1a; 实践&#xff1a; &#xff08;29&#xff09; &#xff08;33&#xff09; 谢谢

redis持久化理论

0 前言 什么是持久化 redis操作都是在内存中&#xff0c;如果出现宕机的话&#xff0c;数据将不复存在&#xff0c;所以持久化是将内存中的数据刷盘到磁盘中&#xff0c;redis可以提供RDB和AOF将数据写入磁盘中。 一 持久化技术 本章节将介绍持久化RDB和AOF两个技术&#xf…

25/2/7 <机器人基础>雅可比矩阵计算 雅可比伪逆

雅可比矩阵计算 雅可比矩阵的定义 假设我们有一个简单的两个关节的平面机器人臂&#xff0c;其末端执行器的位置可以表示为&#xff1a; 其中&#xff1a; L1​ 和 L2 是机器人臂的长度。θ1​ 和 θ2是关节的角度。 计算雅可比矩阵 雅可比矩阵 JJ 的定义是将关节速度与末…

鸿蒙UI(ArkUI-方舟UI框架)- 使用文本

返回主章节 → 鸿蒙UI&#xff08;ArkUI-方舟UI框架&#xff09; 文本使用 文本显示 (Text/Span) Text是文本组件&#xff0c;通常用于展示用户视图&#xff0c;如显示文章的文字内容。Span则用于呈现显示行内文本。 创建文本 string字符串 Text("我是一段文本"…

科技赋能数字内容体验的核心技术探索

内容概要 在数字化时代&#xff0c;科技的迅猛发展为我们的生活和工作带来了深刻的变革。数字内容体验已经成为人们获取信息和娱乐的重要途径&#xff0c;而这背后的技术支持则扮演着至关重要的角色。尤其是在人工智能、虚拟现实和区块链等新兴技术的推动下&#xff0c;数字内…

详细教程 | 如何使用DolphinScheduler调度Flink实时任务

Apache DolphinScheduler 非常适用于实时数据处理场景&#xff0c;尤其是与 Apache Flink 的集成。DolphinScheduler 提供了丰富的功能&#xff0c;包括任务依赖管理、动态调度、实时监控和日志管理&#xff0c;能够有效简化 Flink 实时任务的管理和部署。通过 DolphinSchedule…

棋盘(二维差分)

题目&#xff1a; 5396. 棋盘 题目 提交记录 讨论 题解 视频讲解 小蓝拥有 nnnn 大小的棋盘&#xff0c;一开始棋盘上全都是白子。 小蓝进行了 mm 次操作&#xff0c;每次操作会将棋盘上某个范围内的所有棋子的颜色取反(也就是白色棋子变为黑色&#xff0c;黑色棋子变…

MySQL数据库基础(创建/删除 数据库/表)

一、数据库的操作 1.1 显示当前数据库 语法&#xff1a;show databases&#xff1b; <1>show 是一个关键字&#xff0c;表示要执行的操作类型 <2>databases 是复数&#xff0c;表示显示所有数据库 上面的数据库中&#xff0c;除了java113&#xff0c;其它的数据库…