GESP四级 - 第一章 - 第3节 - 作用域

1. 作用域

1.1 什么是作用域

作用域是程序中定义变量的区域,它决定了变量的可见性和生命周期。在C++中,每个变量都有自己的作用域,变量只能在其作用域内访问和使用。

C++中有三种主要的作用域:

  1. 局部作用域(Local Scope):

    • 在函数内部或代码块内部定义的变量具有局部作用域。
    • 局部变量只能在其所在的函数或代码块内部访问。
    • 局部变量在函数或代码块执行完毕后被销毁。
  2. 全局作用域(Global Scope):

    • 在所有函数和代码块之外定义的变量具有全局作用域。
    • 全局变量可以在程序的任何地方访问,包括其他函数和代码块内部。
    • 全局变量在程序的整个生命周期内都存在,直到程序终止。
  3. 类作用域(Class Scope):

    • 在类定义中声明的变量(成员变量)具有类作用域。
    • 类成员变量可以在类的成员函数中访问。
    • 类成员变量的生命周期与类的实例(对象)相关联。

作用域的目的是控制变量的可见性和访问范围,避免命名冲突和不必要的副作用。通过合理地使用作用域,可以提高程序的可读性、可维护性和安全性。

以下是一个简单的示例,演示了局部作用域和全局作用域:

#include <iostream>
using namespace std;

int globalVar = 10;  // 全局变量

void func() {
    int localVar = 20;  // 局部变量
    cout << "Local variable: " << localVar << endl;
    cout << "Global variable: " << globalVar << endl;
}

int main() {
    cout << "Global variable: " << globalVar << endl;
    func();
    // cout << localVar << endl;  // 错误,localVar在此处不可见
    return 0;
}

在上面的示例中,globalVar是一个全局变量,可以在main函数和func函数中访问。而localVarfunc函数内部的局部变量,只能在func函数内部访问,在main函数中不可见。

理解作用域对于编写正确、清晰和安全的C++程序非常重要。通过合理地选择变量的作用域,可以避免命名冲突,控制变量的生命周期,并提高代码的可读性和可维护性。

1.2 局部作用域

局部作用域是指在函数内部或代码块内部定义的变量的作用范围。具有局部作用域的变量称为局部变量。局部变量有以下特点:

  1. 定义位置:

    • 局部变量在函数内部或代码块内部定义,通常在需要使用变量的地方进行声明和初始化。
    • 局部变量的定义不能放在函数或代码块的外部。
  2. 可见性:

    • 局部变量只能在其所在的函数或代码块内部访问。
    • 在函数或代码块外部,局部变量是不可见的,无法访问或使用。
    • 不同函数或代码块中可以定义同名的局部变量,它们互不影响。
  3. 生命周期:

    • 局部变量在函数或代码块执行时被创建,在函数或代码块执行完毕后被销毁。
    • 每次调用函数或执行代码块时,局部变量都会重新创建和初始化。
    • 局部变量的生命周期与函数或代码块的执行期间相同。
  4. 存储位置:

    • 局部变量通常存储在栈(Stack)中,在函数或代码块执行完毕后自动释放。
    • 每个函数或代码块的局部变量独立存储,互不干扰。

以下是一个示例,演示了局部作用域的使用:

#include <iostream>
using namespace std;

void func1() {
    int x = 10;
    cout << "In func1, x = " << x << endl;
}

void func2() {
    int x = 20;
    cout << "In func2, x = " << x << endl;
}

int main() {
    int x = 30;
    cout << "In main, x = " << x << endl;

    {
        int x = 40;
        cout << "In block, x = " << x << endl;
    }

    func1();
    func2();

    return 0;
}

输出结果:

In main, x = 30
In block, x = 40
In func1, x = 10
In func2, x = 20

在上面的示例中,main函数、func1函数、func2函数以及main函数中的代码块都有自己的局部变量x。它们在各自的作用域内定义和使用,互不干扰。每个局部变量只在其所在的函数或代码块内部可见,在外部无法访问。

局部作用域的使用可以提高代码的可读性和可维护性,避免了命名冲突和不必要的副作用。通过合理地使用局部变量,可以使代码更加模块化和清晰,同时也提高了程序的安全性。

1.3 全局作用域

全局作用域是指在所有函数和代码块之外定义的变量的作用范围。具有全局作用域的变量称为全局变量。全局变量有以下特点:

  1. 定义位置:

    • 全局变量在所有函数和代码块之外定义,通常在文件的顶部或者在所有函数之前。
    • 全局变量的定义不能放在函数或代码块内部。
  2. 可见性:

    • 全局变量可以在程序的任何地方访问,包括其他函数和代码块内部。
    • 全局变量的作用域是整个程序,从定义的位置开始,直到程序结束。
    • 在不同的源文件中,可以通过extern关键字声明全局变量,以便在其他文件中访问。
  3. 生命周期:

    • 全局变量在程序开始执行时被创建,在程序终止时被销毁。
    • 全局变量的生命周期与整个程序的生命周期相同。
    • 全局变量在程序运行期间一直存在,并且保持其值不变,除非显式地修改它们。
  4. 存储位置:

    • 全局变量通常存储在内存的数据段或BSS段中。
    • 数据段用于存储初始化的全局变量,BSS段用于存储未初始化的全局变量。
    • 全局变量在程序的整个生命周期内占用内存空间。

以下是一个示例,演示了全局作用域的使用:

#include <iostream>
using namespace std;

int globalVar = 10;  // 全局变量

void func1() {
    cout << "In func1, globalVar = " << globalVar << endl;
    globalVar = 20;
}

void func2() {
    cout << "In func2, globalVar = " << globalVar << endl;
}

int main() {
    cout << "In main, globalVar = " << globalVar << endl;
    func1();
    func2();
    cout << "In main, globalVar = " << globalVar << endl;
    return 0;
}

输出结果:

In main, globalVar = 10
In func1, globalVar = 10
In func2, globalVar = 20
In main, globalVar = 20

在上面的示例中,globalVar是一个全局变量,它在所有函数之外定义。globalVar可以在main函数、func1函数和func2函数中访问和修改。在func1函数中对globalVar的修改影响了其他函数中globalVar的值。

全局变量提供了一种在整个程序范围内共享数据的机制。但是,过度使用全局变量可能会导致代码的可读性和可维护性降低,并增加了命名冲突和不必要的副作用的风险。因此,在使用全局变量时,需要谨慎考虑其必要性和影响范围,并尽可能地使用局部变量和函数参数来传递数据。

1.4 作用域规则

在C++中,作用域规则确定了变量的可见性和访问范围。以下是C++中的主要作用域规则:

  1. 局部作用域规则:

    • 在函数内部或代码块内部定义的变量具有局部作用域。
    • 局部变量只能在其所在的函数或代码块内部访问。
    • 不同函数或代码块中可以定义同名的局部变量,它们互不干扰。
    • 局部作用域遵循"先定义,后使用"的原则,变量只能在定义之后的代码中访问。
  2. 全局作用域规则:

    • 在所有函数和代码块之外定义的变量具有全局作用域。
    • 全局变量可以在程序的任何地方访问,包括其他函数和代码块内部。
    • 全局变量的作用域从定义的位置开始,直到程序结束。
    • 如果局部变量与全局变量同名,在局部作用域内,局部变量会屏蔽(隐藏)全局变量。
  3. 块作用域规则:

    • 在代码块({})内部定义的变量具有块作用域。
    • 块作用域是局部作用域的一种特殊情况,它限定了变量的可见范围在代码块内部。
    • 块作用域内定义的变量只能在该代码块内部访问,在代码块外部不可见。
    • 不同的代码块中可以定义同名的变量,它们互不干扰。
  4. 命名空间作用域规则:

    • 在命名空间内定义的变量具有命名空间作用域。
    • 命名空间用于避免命名冲突,将相关的变量、函数和类组织在一起。
    • 命名空间内的变量可以通过命名空间名称和作用域解析运算符(::):来访问。
    • 如果使用using指令或using声明,可以在当前作用域内直接访问命名空间内的变量。
  5. 类作用域规则:

    • 在类定义中声明的变量(成员变量)具有类作用域。
    • 类成员变量可以在类的成员函数中访问。
    • 类成员变量的作用域限定在类的内部,在类的外部需要通过对象或指针来访问。
    • 不同的类可以定义同名的成员变量,它们属于不同的类作用域。

以下是一个示例,演示了局部作用域和全局作用域的作用域规则:

#include <iostream>
using namespace std;

int globalVar = 10;  // 全局变量

void func() {
    int localVar = 20;  // 局部变量
    cout << "在函数内部, 局部变量 localVar = " << localVar << endl;
    cout << "在函数内部, 全局变量 globalVar = " << globalVar << endl;
}

int main() {
    int localVar = 30;  // 局部变量
    cout << "在main函数内部, 局部变量 localVar = " << localVar << endl;
    cout << "在main函数内部, 全局变量 globalVar = " << globalVar << endl;
    func();
    return 0;
}

输出结果:

在main函数内部, 局部变量 localVar = 30
在main函数内部, 全局变量 globalVar = 10
在函数内部, 局部变量 localVar = 20
在函数内部, 全局变量 globalVar = 10

在上面的示例中,globalVar是全局变量,可以在main函数和func函数中访问。localVarmain函数和func函数中都有定义,但它们属于不同的局部作用域,互不干扰。在func函数内部,局部变量localVar屏蔽了全局变量globalVar

理解和遵循作用域规则可以帮助我们编写清晰、可读性强的代码,并避免变量命名冲突和意外的副作用。

1.5 变量的生命周期

变量的生命周期是指变量从创建到销毁的整个过程。在C++中,变量的生命周期取决于其作用域和存储类型。以下是几种常见的变量生命周期:

  1. 局部变量的生命周期:

    • 局部变量在函数或代码块执行时被创建,在函数或代码块执行完毕后被销毁。
    • 每次调用函数或执行代码块时,局部变量都会重新创建和初始化。
    • 局部变量的生命周期与函数或代码块的执行期间相同。
    • 示例:
      void func() {
          int localVar = 10;  // 局部变量在函数开始时创建
          // 使用局部变量
      }  // 局部变量在函数结束时销毁
      
  2. 全局变量的生命周期:

    • 全局变量在程序开始执行时被创建,在程序终止时被销毁。
    • 全局变量的生命周期与整个程序的生命周期相同。
    • 全局变量在程序的任何地方都可以访问和修改。
    • 示例:
      int globalVar;  // 全局变量在程序开始时创建
      
      int main() {
          // 使用全局变量
          return 0;
      }  // 全局变量在程序结束时销毁
      
  3. 静态局部变量的生命周期:

    • 静态局部变量在函数第一次被调用时创建,在程序终止时被销毁。
    • 静态局部变量只会被初始化一次,之后的函数调用会保留其值。
    • 静态局部变量的生命周期与程序的生命周期相同,但其作用域仍然限定在函数内部。
    • 示例:
      void func() {
          static int staticVar = 0;  // 静态局部变量在第一次调用时创建
          staticVar++;
          cout << "staticVar = " << staticVar << endl;
      }  // 静态局部变量在程序结束时销毁
      
  4. 动态分配变量的生命周期:

    • 使用new运算符动态分配的变量,其生命周期由程序员手动控制。
    • 动态分配的变量在使用new运算符分配内存时被创建,在使用delete运算符释放内存时被销毁。
    • 动态分配的变量的生命周期与程序员的内存管理有关,需要手动释放内存以避免内存泄漏。
    • 示例:
      int* ptr = new int;  // 动态分配变量被创建
      // 使用动态分配的变量
      delete ptr;  // 动态分配变量被销毁
      

理解变量的生命周期对于正确管理内存和避免内存相关的错误非常重要。根据变量的作用域和存储类型,合理地选择变量的生命周期,可以提高程序的效率和可靠性。同时,对于动态分配的变量,要注意手动释放内存,以避免内存泄漏等问题。

2. 函数的返回值

2.1 return语句

在C++中,return语句用于结束函数的执行并将控制权返回给调用函数。return语句可以有以下几种用法:

  1. 无返回值的return语句:

    • 对于void类型的函数,可以使用无返回值的return语句来结束函数的执行。
    • 无返回值的return语句可以省略,函数会在执行完最后一条语句后自动返回。
    • 示例:
      void func() {
          // 函数体
          return;  // 无返回值的return语句
      }
      
  2. 有返回值的return语句:

    • 对于有返回值的函数,return语句用于指定函数的返回值。
    • 返回值可以是一个表达式或变量,其类型必须与函数声明的返回类型相同或可转换。
    • return语句执行后,函数会立即结束,并将返回值传递给调用函数。
    • 示例:
      int add(int a, int b) {
          int sum = a + b;
          return sum;  // 返回sum的值
      }
      
  3. 多个return语句:

    • 函数中可以有多个return语句,用于在不同的条件下返回不同的值。
    • 当执行到某个return语句时,函数会立即结束,后面的语句不会被执行。
    • 示例:
      int max(int a, int b) {
          if (a > b) {
              return a;  // 返回a的值
          } else {
              return b;  // 返回b的值
          }
      }
      
  4. 返回引用或指针:

    • return语句可以返回一个引用或指针类型的值。
    • 返回引用或指针时,需要确保引用或指针指向的对象在函数返回后仍然有效。
    • 返回局部变量的引用或指针是危险的,因为局部变量在函数返回后会被销毁。
    • 示例:
      int& getRef(int& num) {
          return num;  // 返回num的引用
      }
      
      int* getPtr(int* ptr) {
          return ptr;  // 返回ptr的值(指针)
      }
      

return语句是函数返回的关键语句,它决定了函数的返回值和控制流的转移。合理地使用return语句可以使函数的逻辑更加清晰,并且可以根据不同的条件返回不同的结果。需要注意的是,函数应该有明确的返回路径,确保在所有可能的执行路径上都有return语句,以避免未定义的行为。

2.2 void函数

在C++中,void函数是一种特殊的函数,它没有返回值。void函数的主要特点如下:

  1. 函数声明和定义:

    • void函数的返回类型为void,表示函数不返回任何值。
    • 函数声明和定义时,在函数名前面加上void关键字。
    • 示例:
      void printMessage();  // 函数声明
      
      void printMessage() {
          cout << "Hello, world!" << endl;
      }  // 函数定义
      
  2. 函数调用:

    • 调用void函数时,不需要接收其返回值,因为它没有返回值。
    • void函数通常用于执行某些操作或任务,而不是计算并返回结果。
    • 示例:
      printMessage();  // 调用void函数
      
  3. 不使用return语句:

    • void函数中,可以省略return语句,因为它没有返回值。
    • 如果需要在某个条件下提前结束函数的执行,可以使用无返回值的return语句。
    • 示例:
      void printNumbers(int n) {
          if (n <= 0) {
              return;  // 提前结束函数的执行
          }
          for (int i = 1; i <= n; i++) {
              cout << i << " ";
          }
          cout << endl;
      }
      
  4. 用途:

    • void函数通常用于执行某些操作或任务,如打印输出、修改全局变量、进行计算等。
    • void函数可以通过参数接收数据,并根据需要对数据进行处理。
    • void函数可以与其他函数配合使用,完成特定的功能。
    • 示例:
      void printArray(int arr[], int size) {
          for (int i = 0; i < size; i++) {
              cout << arr[i] << " ";
          }
          cout << endl;
      }
      
      int main() {
          int numbers[] = {1, 2, 3, 4, 5};
          int size = sizeof(numbers) / sizeof(numbers[0]);
          printArray(numbers, size);  // 调用void函数打印数组
          return 0;
      }
      

void函数在C++中非常常见,它们用于执行一些不需要返回值的操作或任务。虽然void函数没有返回值,但它们可以通过修改全局变量、输出信息、调用其他函数等方式与程序的其他部分进行交互。合理地使用void函数可以使代码更加模块化和可读,并且可以将特定的功能封装在独立的函数中,方便调用和维护。

2.3 返回值类型

在C++中,函数的返回值类型指定了函数返回的数据类型。返回值类型决定了函数可以返回的数据类型,以及如何处理和使用函数的返回值。以下是关于返回值类型的一些要点:

  1. 基本数据类型:

    • 函数可以返回基本数据类型,如intfloatdoublecharbool等。
    • 返回值类型必须与函数定义中指定的类型匹配。
    • 示例:
      int getSum(int a, int b) {
          return a + b;
      }
      
      double getAverage(double x, double y) {
          return (x + y) / 2;
      }
      
  2. 用户自定义类型:

    • 函数可以返回用户自定义的类型,如结构体、类对象等。
    • 返回用户自定义类型时,需要确保该类型已经被正确定义和实现。
    • 示例:
      struct Point {
          int x;
          int y;
      };
      
      Point getMiddlePoint(Point p1, Point p2) {
          Point middle;
          middle.x = (p1.x + p2.x) / 2;
          middle.y = (p1.y + p2.y) / 2;
          return middle;
      }
      
  3. 指针类型:

    • 函数可以返回指针类型,用于返回内存地址或动态分配的内存。
    • 返回指针时,需要确保指针指向的内存在函数返回后仍然有效。
    • 示例:
      int* createArray(int size) {
          int* arr = new int[size];
          return arr;
      }
      
  4. 引用类型:

    • 函数可以返回引用类型,用于返回变量或对象的引用。
    • 返回引用时,需要确保引用的对象在函数返回后仍然有效。
    • 示例:
      int& getElement(int* arr, int index) {
          return arr[index];
      }
      
  5. void类型:

    • 如果函数不需要返回任何值,可以将返回值类型声明为void
    • void函数不能使用带有值的return语句,但可以使用无返回值的return语句来提前结束函数的执行。
    • 示例:
      void printMessage(const string& message) {
          cout << message << endl;
      }
      

选择合适的返回值类型取决于函数的目的和需求。返回值类型应该与函数的功能相匹配,并提供足够的信息给调用方。同时,需要注意返回值的生命周期和有效性,特别是在返回指针或引用时,要确保返回的对象在函数返回后仍然有效。

合理地选择和使用返回值类型可以使函数的接口更加清晰,提高代码的可读性和可维护性。

2.4 返回引用

在C++中,函数可以返回引用类型,即返回变量或对象的引用。返回引用允许函数直接访问和修改外部的变量或对象,而无需复制整个对象。以下是关于返回引用的一些要点:

  1. 引用类型:

    • 返回引用类型的语法是在函数的返回类型前面加上&符号。
    • 返回引用类型必须与函数定义中指定的类型匹配。
    • 示例:
      int& getRefToValue(int& val) {
          return val;
      }
      
  2. 返回引用的用途:

    • 返回引用允许函数直接访问和修改外部的变量或对象。
    • 通过返回引用,可以避免不必要的复制操作,提高程序的效率。
    • 返回引用可以用于实现函数链式调用或者实现类似于赋值操作符的功能。
    • 示例:
      int value = 10;
      int& ref = getRefToValue(value);
      ref = 20;  // 修改value的值
      cout << value << endl;  // 输出: 20
      
  3. 返回引用的注意事项:

    • 不要返回局部变量的引用,因为局部变量在函数返回后会被销毁,引用就变成了悬空引用。
    • 返回引用时,确保引用的对象在函数返回后仍然有效。
    • 返回引用通常用于返回类的成员变量或全局变量的引用。
    • 示例:
      int& getGlobalValue() {
          static int globalValue = 0;
          return globalValue;
      }
      
  4. 返回常量引用:

    • 如果不希望通过返回的引用修改原始对象,可以返回常量引用。
    • 返回常量引用可以避免意外地修改原始对象,提高程序的安全性。
    • 示例:
      const string& getConstRef(const string& str) {
          return str;
      }
      
  5. 返回引用的性能考虑:

    • 返回引用可以避免不必要的复制操作,提高程序的效率。
    • 对于大型对象或频繁访问的对象,返回引用可以显著提高性能。
    • 但是,返回引用也可能导致悬空引用或意外的修改,需要谨慎使用。
    • 示例:
      class LargeObject {
          // 大型对象的定义
      };
      
      LargeObject& getLargeObject() {
          static LargeObject obj;
          return obj;
      }
      

返回引用是C++中一种强大的特性,它允许函数直接访问和修改外部的变量或对象。通过返回引用,可以避免不必要的复制操作,提高程序的效率。但是,使用返回引用时需要注意一些陷阱,如返回局部变量的引用或意外修改原始对象等。

合理地使用返回引用可以使代码更加简洁、高效,并提供更灵活的功能。但同时也要权衡返回引用的风险和限制,确保程序的正确性和安全性。

示范例子:

  1. 返回数组元素的引用:
int& getElement(int* arr, int index) {
    return arr[index];
}

int main() {
    int numbers[] = {1, 2, 3, 4, 5};
    int& ref = getElement(numbers, 2);
    ref = 10;
    cout << "Modified array: ";
    for (int i = 0; i < 5; i++) {
        cout << numbers[i] << " ";
    }
    cout << endl;
    return 0;
}

输出结果:

Modified array: 1 2 10 4 5
  1. 返回类成员变量的引用:
class Circle {
private:
    double radius;

public:
    Circle(double r) : radius(r) {}

    double& getRadius() {
        return radius;
    }
};

int main() {
    Circle circle(5.0);
    double& radiusRef = circle.getRadius();
    radiusRef = 7.5;
    cout << "Modified radius: " << circle.getRadius() << endl;
    return 0;
}

输出结果:

Modified radius: 7.5
  1. 返回字符串中的字符引用:
char& getCharacter(string& str, int index) {
    return str[index];
}

int main() {
    string message = "Hello, World!";
    char& charRef = getCharacter(message, 7);
    charRef = 'w';
    cout << "Modified string: " << message << endl;
    return 0;
}

输出结果:

Modified string: Hello, world!
  1. 返回结构体成员的引用:
struct Point {
    int x;
    int y;
};

int& getX(Point& point) {
    return point.x;
}

int main() {
    Point p = {3, 4};
    int& xRef = getX(p);
    xRef = 5;
    cout << "Modified point: (" << p.x << ", " << p.y << ")" << endl;
    return 0;
}

输出结果:

Modified point: (5, 4)
  1. 返回vector元素的引用:
vector<int>& getVector() {
    static vector<int> v = {1, 2, 3, 4, 5};
    return v;
}

int main() {
    vector<int>& vecRef = getVector();
    vecRef[2] = 10;
    cout << "Modified vector: ";
    for (int i = 0; i < vecRef.size(); i++) {
        cout << vecRef[i] << " ";
    }
    cout << endl;
    return 0;
}

输出结果:

Modified vector: 1 2 10 4 5

这些示例展示了返回引用的不同应用场景,包括返回数组元素、类成员变量、字符串中的字符、结构体成员以及vector元素的引用。通过返回引用,可以直接访问和修改原始对象,提供了更灵活和高效的操作方式。但是,使用返回引用时需要注意引用的生命周期和有效性,确保引用的对象在函数返回后仍然有效,避免悬空引用等问题。

当函数返回值而不是引用时,函数会返回对象的副本,而不是对象本身。这意味着对返回值的任何修改都不会影响原始对象。下面是两个返回值的示例,并解释了它们与返回引用的区别:

  1. 返回整数值:
int getValue() {
    int value = 10;
    return value;
}

int main() {
    int result = getValue();
    result = 20;
    cout << "Value: " << getValue() << endl;
    return 0;
}

输出结果:

Value: 10

在这个示例中,getValue函数返回一个整数值。在main函数中,我们将getValue的返回值赋给变量result,然后将result修改为20。但是,当我们再次调用getValue函数时,它仍然返回原始值10。这是因为resultgetValue返回值的副本,对result的修改不会影响getValue函数内部的value变量。

  1. 返回对象值:
class Point {
public:
    int x;
    int y;

    Point(int xVal, int yVal) : x(xVal), y(yVal) {}
};

Point getPoint() {
    Point p(3, 4);
    return p;
}

int main() {
    Point point = getPoint();
    point.x = 5;
    cout << "Point: (" << getPoint().x << ", " << getPoint().y << ")" << endl;
    return 0;
}

输出结果:

Point: (3, 4)

在这个示例中,getPoint函数返回一个Point对象。在main函数中,我们将getPoint的返回值赋给变量point,然后将point.x修改为5。但是,当我们再次调用getPoint函数时,它仍然返回原始的Point对象,其中x的值为3。这是因为pointgetPoint返回对象的副本,对point的修改不会影响getPoint函数内部的p对象。

与返回引用不同,返回值会创建对象的副本,对返回值的修改不会影响原始对象。这提供了一定的安全性和隔离性,防止了对原始对象的意外修改。但是,当返回大型对象或频繁返回对象时,返回值可能会导致性能开销,因为需要创建和销毁对象的副本。

在决定是返回值还是返回引用时,需要考虑函数的设计目的、对象的大小、修改原始对象的需求以及性能等因素。如果不需要修改原始对象并且对象较小,返回值通常是一个好的选择。如果需要直接修改原始对象或者对象较大,返回引用可能更合适。

  • 17
    点赞
  • 23
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

天秀信奥编程培训

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值