首页 > 编程语言 >十四天学会C++之第二天(函数和库)

十四天学会C++之第二天(函数和库)

时间:2023-10-03 20:01:02浏览次数:46  
标签:std 十四天 函数 int C++ numbers 返回值 include

十四天学会C++之第二天(函数和库)_编码

1. 函数的定义和调用

在C++中,函数是组织和结构化代码的关键工具之一。它们允许您将一段代码封装成一个可重复使用的模块,这有助于提高代码的可读性和维护性。

为什么使用函数?

函数在编程中的作用不可小觑。它们有以下几个重要用途:

  • 模块化编程: 函数允许将代码划分为小的、独立的单元,使得代码更易于理解和管理。
  • 代码重用: 一次编写,多次使用。您可以在程序的不同地方调用同一个函数,而不必重复编写相同的代码。
  • 提高可读性: 通过将代码分解为函数,您可以为每个函数取一个描述性的名字,使代码更具可读性。

定义函数

在C++中,函数的定义通常包括以下几个部分:

// 函数声明(函数原型)
返回类型 函数名(参数列表);

// 函数定义
返回类型 函数名(参数列表) {
    // 函数体
    // 执行一些操作
    return 返回值; // 如果有返回值的话
}
  • 返回类型: 函数可以返回一个值,这个值的类型由返回类型指定。如果函数不返回任何值,可以使用 void 关键字表示。
  • 函数名: 函数的名称,是函数的标识符。
  • 参数列表: 函数可以接受零个或多个参数,这些参数在圆括号内列出,并用逗号分隔。
  • 函数体: 包含函数执行的实际代码部分。
  • 返回值: 如果函数有返回值,使用 return 语句返回该值。

函数定义:

// 函数声明
int add(int a, int b);

// 函数定义
int add(int a, int b) {
    int result = a + b;
    return result;
}

调用函数

调用函数意味着执行函数内的代码。要调用函数,只需使用函数名和合适的参数列表。

int main() {
    int num1 = 5;
    int num2 = 3;
    int sum = add(num1, num2); // 调用add函数
    cout << "Sum: " << sum << endl;
    return 0;
}

示例中,add 函数被调用来计算 num1num2 的和,并将结果存储在 sum 变量中。

2. 参数传递

在C++中,参数传递是函数与外部世界进行数据交换的重要方式之一。它可以通过不同的方式实现,包括按值传递和按引用传递。

按值传递 vs. 按引用传递

按值传递

参数按值传递给函数时,函数会创建参数的一个副本,这意味着在函数内部对参数的更改不会影响外部的原始数据。

void modifyValue(int x) {
    x = 10; // 在函数内部修改副本
}

int main() {
    int value = 5;
    modifyValue(value);
    cout << "Value after function call: " << value << endl; // 仍然是5
    return 0;
}
按引用传递

按引用传递参数时,函数操作的是原始数据的引用,这意味着对参数的更改会影响外部的原始数据。

void modifyValue(int &x) {
    x = 10; // 直接修改原始数据
}

int main() {
    int value = 5;
    modifyValue(value);
    cout << "Value after function call: " << value << endl; // 现在是10
    return 0;
}

函数参数的默认值

函数的参数提供默认值,这意味着在调用函数时,可以省略某些参数,让编译器使用默认值。

void printMessage(string message = "Hello, World!") {
    cout << message << endl;
}

int main() {
    printMessage(); // 使用默认消息
    printMessage("Custom message"); // 使用自定义消息
    return 0;
}

函数重载

函数重载允许在同一范围内定义多个具有相同名称但不同参数列表的函数。编译器根据函数调用的参数来选择正确的函数。

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

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

示例

参数传递的不同方式和默认值的影响:

void modify(int x) {
    x = 10;
}

void modify(double &y) {
    y = 3.14;
}

int main() {
    int num = 5;
    double pi = 3.14159265359;

    modify(num); // 传值,num不变
    modify(pi);  // 传引用,pi被修改

    cout << "Modified num: " << num << endl;
    cout << "Modified pi: " << pi << endl;

    return 0;
}

在示例中,modify 函数分别按值和按引用传递参数,从而导致了不同的行为。

3. 函数的返回值

函数的返回值是函数执行后向调用者提供的结果。在C++中,您可以指定函数的返回值类型,并使用return语句从函数中返回值。

返回值类型

每个C++函数都有一个返回值类型,它指定了函数返回的数据类型。返回值类型在函数声明和定义中都必须指定。

int add(int a, int b) { // 返回值类型为int
    return a + b;
}

double divide(double x, double y) { // 返回值类型为double
    return x / y;
}

返回语句

return语句用于从函数中返回值。可以出现在函数的任何位置,但一旦执行,函数将立即终止,并将控制返回给调用者。

int multiply(int a, int b) {
    int result = a * b;
    return result; // 返回计算结果
}

示例

使用函数的返回值:

int main() {
    int sum = add(5, 3); // 调用add函数并接收返回值
    double quotient = divide(10.0, 2.0); // 调用divide函数并接收返回值

    cout << "Sum: " << sum << endl;
    cout << "Quotient: " << quotient << endl;

    return 0;
}

在上面的示例中,adddivide 函数返回整数和浮点数,分别被存储在 sumquotient 变量中。

返回值在表达式中的应用

函数的返回值可以直接用作表达式的一部分。这使得函数调用非常灵活,可以在数学表达式或其他计算中使用。

int main() {
    int result = multiply(add(2, 3), 4); // 使用函数返回值进行嵌套调用和计算

    cout << "Result: " << result << endl;

    return 0;
}

在示例中,add(2, 3) 的返回值被传递给 multiply 函数,以便进行进一步的计算。

4. 标准C++库介绍

C++作为一门强大的编程语言,拥有丰富的标准库,提供了许多有用的功能和数据结构。

包含头文件

要使用C++标准库中的功能,首先需要包含相应的头文件。头文件包含了库中的类、函数和对象的声明,它们是使用这些库的关键。

#include <iostream>  // 包含iostream头文件,用于输入输出操作
#include <string>    // 包含string头文件,用于字符串操作
#include <vector>    // 包含vector头文件,用于动态数组操作

示例用法

库的示例用法:

使用iostream进行输入和输出
#include <iostream>

int main() {
    // 输出文本到控制台
    std::cout << "Hello, World!" << std::endl;

    // 从用户输入读取数据
    int num;
    std::cout << "Enter a number: ";
    std::cin >> num;

    // 输出读取到的数据
    std::cout << "You entered: " << num << std::endl;

    return 0;
}
使用string进行字符串操作
#include <string>

int main() {
    std::string greeting = "Hello, ";
    std::string name = "John";

    // 字符串拼接
    std::string message = greeting + name;

    // 获取字符串长度
    int length = message.length();

    // 输出结果
    std::cout << message << " (Length: " << length << ")" << std::endl;

    return 0;
}
使用vector创建动态数组
#include <vector>

int main() {
    std::vector<int> numbers;

    // 向vector添加元素
    numbers.push_back(1);
    numbers.push_back(2);
    numbers.push_back(3);

    // 遍历并输出vector的元素
    for (int i = 0; i < numbers.size(); ++i) {
        std::cout << numbers[i] << " ";
    }
    std::cout << std::endl;

    return 0;
}

5. 头文件和命名空间

在C++编程中,头文件和命名空间是非常重要的概念。头文件用于包含声明和定义,而命名空间则用于避免命名冲突。

头文件的作用

头文件通常包含了函数、类和变量的声明,以及必要的函数原型和常量定义。头文件的作用是将这些声明集中在一起,以便在多个源文件中共享。这有助于模块化编程,提高了代码的可维护性。

创建自定义头文件

要创建自定义头文件,只需新建一个以.h.hpp为扩展名的文本文件,并在其中包含所需的声明。例如,以下是一个名为myheader.h的头文件的示例:

#ifndef MYHEADER_H
#define MYHEADER_H

// 在这里添加声明

#endif

命名空间的概念

命名空间是一种将全局作用域划分为不同部分以防止命名冲突的机制。它允许您将相关的函数、类和变量组织到一个命名空间中,以避免与其他代码的命名冲突。

使用命名空间

要使用命名空间,您可以使用namespace关键字定义一个命名空间,然后将相关声明放入其中。例如:

// 定义一个名为mynamespace的命名空间
namespace mynamespace {
    int myVariable;
    void myFunction();
}
// 使用mynamespace中的变量和函数
mynamespace::myVariable = 42;
mynamespace::myFunction();

简化命名空间的使用

为了简化命名空间的使用,可以使用using关键字来声明在命名空间中的特定成员。例如:

// 使用mynamespace中的myVariable
using mynamespace::myVariable;

int main() {
    myVariable = 42; // 不需要指定命名空间
    return 0;
}

6.案例分析

函数重载

#include <iostream>

// 函数重载:处理整数
int add(int a, int b) {
    return a + b;
}

// 函数重载:处理双精度浮点数
double add(double a, double b) {
    return a + b;
}

int main() {
    int intResult = add(5, 7);
    double doubleResult = add(3.5, 2.7);

    std::cout << "Integer Result: " << intResult << std::endl;
    std::cout << "Double Result: " << doubleResult << std::endl;

    return 0;
}

运行结果:

十四天学会C++之第二天(函数和库)_编码_02

递归函数

#include <iostream>

int fibonacci(int n) {
    if (n <= 1) {
        return n;
    }
    return fibonacci(n - 1) + fibonacci(n - 2);
}

int main() {
    int n = 10;
    for (int i = 0; i < n; ++i) {
        std::cout << fibonacci(i) << " ";
    }

    return 0;
}

运行结果:

十四天学会C++之第二天(函数和库)_命名空间_03

Lambda表达式

#include <iostream>
#include <vector>
#include <algorithm>

int main() {
    std::vector<int> numbers = {5, 2, 8, 1, 3};

    // 使用Lambda表达式对向量进行排序
    std::sort(numbers.begin(), numbers.end(), [](int a, int b) {
        return a < b;
    });

    // 使用Lambda表达式筛选出偶数
    auto isEven = [](int x) { return x % 2 == 0; };
    std::vector<int> evenNumbers;

    std::copy_if(numbers.begin(), numbers.end(), std::back_inserter(evenNumbers), isEven);

    // 输出排序后的向量
    std::cout << "Sorted Numbers: ";
    for (int num : numbers) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    // 输出筛选后的偶数
    std::cout << "Even Numbers: ";
    for (int num : evenNumbers) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    return 0;
}

运行结果:

十四天学会C++之第二天(函数和库)_头文件_04

字符串处理

#include <iostream>
#include <string>
#include <algorithm>

int main() {
    std::string str = "Hello, World!";
    
    // 反转字符串
    std::reverse(str.begin(), str.end());
    std::cout << "Reversed String: " << str << std::endl;

    // 查找子字符串
    std::string subStr = "World";
    size_t found = str.find(subStr);
    if (found != std::string::npos) {
        std::cout << "Substring found at position: " << found << std::endl;
    } else {
        std::cout << "Substring not found." << std::endl;
    }

    // 将字符串拆分为单词
    std::string sentence = "This is a sample sentence";
    size_t startPos = 0;
    while (startPos < sentence.length()) {
        size_t spacePos = sentence.find(' ', startPos);
        if (spacePos == std::string::npos) {
            spacePos = sentence.length();
        }
        std::string word = sentence.substr(startPos, spacePos - startPos);
        std::cout << "Word: " << word << std::endl;
        startPos = spacePos + 1;
    }

    return 0;
}

运行结果:

十四天学会C++之第二天(函数和库)_命名空间_05

容器操作

#include <iostream>
#include <vector>
#include <algorithm>
#include <map>

int main() {
    // 使用std::vector进行容器操作
    std::vector<int> numbers = {5, 2, 8, 1, 3};

    // 添加元素
    numbers.push_back(7);

    // 删除元素
    numbers.erase(std::remove(numbers.begin(), numbers.end(), 3), numbers.end());

    // 对容器排序
    std::sort(numbers.begin(), numbers.end());

    // 输出容器元素
    std::cout << "Vector Elements: ";
    for (int num : numbers) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    // 使用std::map进行容器操作
    std::map<std::string, int> scores;

    // 添加键值对
    scores["Alice"] = 95;
    scores["Bob"] = 87;
    scores["Charlie"] = 92;

    // 查找元素
    std::string name = "Bob";
    if (scores.find(name) != scores.end()) {
        std::cout << name << "'s Score: " << scores[name] << std::endl;
    } else {
        std::cout << "Name not found." << std::endl;
    }

    return 0;
}

运行结果:

十四天学会C++之第二天(函数和库)_编码_06

多线程和并发

#include <iostream>
#include <thread>
#include <vector>

// 用于计算部分数组总和的函数
void partialSum(const std::vector<int>& arr, size_t start, size_t end, int& result) {
    result = 0;
    for (size_t i = start; i < end; ++i) {
        result += arr[i];
    }
}

int main() {
    const int numThreads = 4; // 使用4个线程
    const int arrSize = 1000;
    std::vector<int> numbers(arrSize, 1); // 创建一个包含1000个1的数组

    std::vector<std::thread> threads(numThreads);
    std::vector<int> partialResults(numThreads);

    // 创建并启动线程
    for (int i = 0; i < numThreads; ++i) {
        size_t start = i * (arrSize / numThreads);
        size_t end = (i == numThreads - 1) ? arrSize : (i + 1) * (arrSize / numThreads);
        threads[i] = std::thread(partialSum, std::ref(numbers), start, end, std::ref(partialResults[i]));
    }

    // 等待所有线程完成
    for (int i = 0; i < numThreads; ++i) {
        threads[i].join();
    }

    // 计算总和
    int totalSum = 0;
    for (int i = 0; i < numThreads; ++i) {
        totalSum += partialResults[i];
    }

    std::cout << "Total Sum: " << totalSum << std::endl;

    return 0;
}

运行结果:

十四天学会C++之第二天(函数和库)_#include_07

标签:std,十四天,函数,int,C++,numbers,返回值,include
From: https://blog.51cto.com/u_15747017/7694549

相关文章

  • 第03章 Python的数据结构、函数和文件
    本章讨论Python的内置功能,这些功能本书会用到很多。虽然扩展库,比如pandas和Numpy,使处理大数据集很方便,但它们是和Python的内置数据处理工具一同使用的。我们会从Python最基础的数据结构开始:元组、列表、字典和集合。然后会讨论创建你自己的、可重复使用的Python函数。最后,会学习P......
  • $nextTick函数的用法以及全局事件总线
    全局事件总线的目的是可以实现任意组件之间的通信。这里需要涉及到原型链的知识。在Vue里面,我们知道运行一个完整的项目是由若干个VueComponents组件和一个AppVueComponent组件加上一个Vue的实例对象而vc组件在通过this去获取属性的值时,首先会从vc实例对象里面找,没有的话就去vc......
  • 无法引用函数,它是已删除的函数
    E1776无法引用函数"A::A(constA&)"(已隐式声明)--它是已删除的函数:https://blog.csdn.net/Bit_Coders/article/details/117260363>>c++拷贝构造函数(深拷贝,浅拷贝)详解:https://www.gxlsystem.com/bianchengyuyan-416455.html......
  • Bug与函数递归
    调试(debug)发现程序中存在的问题,然后找到问题,这个找问题的过程叫称为调试Debug和Release的区别Debug调试版本,仅调试,不优化Release给用户使用,在代码大小和运行速度都是最优,不能调试函数递归自己调用自己目的:大事化小必要条件:递归存在限制条件,每次递归调用后越来越接近这个限制......
  • 以下是一个复杂的 C 语言代码示例,展示了如何使用递归函数来计算斐波那契数列: ```c #i
    以下是一个复杂的C语言代码示例,展示了如何使用递归函数来计算斐波那契数列:#include<stdio.h>//递归函数计算斐波那契数列intfibonacci(intn){if(n<=1){returnn;}returnfibonacci(n-1)+fibonacci(n-2);}intmain(){intnum;......
  • C++ Thread 基础使用
    C++11Thread使用基础用法头文件#include<thread>函数初始化threadthread(<function_name>);线程分离thread.detach();线程阻塞thread.join()线程取消this_thread::yield();线程休眠this_thread::sleep_for(chrono::seconds(3));代码#in......
  • 前端 splice函数
    splice()函数是JavaScript中数组对象的一个方法,用于修改数组的内容。它可以实现删除、插入和替换数组中的元素。splice()函数的语法如下:array.splice(start,deleteCount,item1,item2,...)start:指定要修改的起始位置的索引。如果为负数,则从数组末尾开始计算,默认为0。deleteCoun......
  • Angular inlineCriticalCss 和内部函数 walkStyleRules 介绍
    有一个客户启用了AngularServerSideRendering,并且启用了inlineCriticalCss,后来发现在Dynatrace的hotspot里的vendor.js文件有个名叫walkStyleRules的函数,耗时比较多。如下图所示:Angular服务器端渲染(ServerSideRendering)Angular的服务器端渲染是一种技术,允许在......
  • C++特种成员函数生成机制及相关原则
    C++特种成员函数生成机制及相关原则注:默认C++标准是C++11及以后的标准,因为C++11之前的标准定义的默认成员函数不包含移动构造函数和移动赋值运算符1.C++默认成员函数默认成员函数的定义:类中没有显示声明,在需要时由编译器自动生成的函数,包括默认构造函数、默认析构函数、......
  • 基于hash_table对STL unordered系列容器的封装 #C++
    概述本文对hash_table进行封装,以模仿SGISTL对unordered系列容器进行简单实现,旨在加深对C++封装与泛型技法的体会与理解。阅读本文之前,建议先对哈希表进行学习。unordered_map与map一样,unordered_map的所有元素类型都是pair,pair的第一个成员为Key,第二个成员为Value。因为Key在任何......