C++指针(四)万字图文详解!

news2024/11/18 20:35:44

个人主页:PingdiGuo_guo

收录专栏:C++干货专栏

前言

相关文章:C++指针(一)、C++指针(二)、C++指针(三)

本篇博客是介绍函数指针、函数指针数组、回调函数、指针函数的。

点赞破六十,更新下一期哟!

文章目录

前言

1.函数指针

1.1函数指针的概念

1.2函数指针的作用

1.3函数指针的使用

1.3.1定义

1.3.2初始化

1.3.2调用函数

1.3.3比较函数指针

1.4函数指针的练习

2.函数指针数组

2.1函数指针数组的作用

2.2函数指针数组的使用

2.2.1 定义

2.2.2 初始化

2.2.3 调用

2.2.4 传参

2.3函数指针二维数组

2.4函数指针数组的练习

3.回调函数与回调指针

3.1回调函数的用处

3.2回调函数的使用

3.2.1 定义函数原型

3.2.2 声明变量

3.2.3定义

3.2.4 地址传递

3.2.5调用

3.3回调指针

3.3.1回调指针的使用

3.3.2回调函数与回调指针的关系

4.指针函数

4.1指针函数的作用

4.2指针函数的操作

4.2.1 声明

4.2.2 定义

4.2.3 调用

4.2.4返回

4.3指针函数的练习

总结


1.函数指针

大家可能在学习了指针后有这样的疑惑:哎,既然数组有地址,变量有地址,那函数会不会也有地址呀?是的,函数也有地址,请看下面一段代码:

​
#include <bits/stdc++.h>
using namespace std;
void te()
{
	printf("hello!\n");
}
int main()
{
	printf("%p\n", te);
	printf("%p\n", &te);
	return 0;
}

​

输出结果:

函数地址:函数名、&函数名

可要储存怎么办呢?别急,接下来我们要讲的函数指针就是用来存储函数的地址的。

1.1函数指针的概念

函数指针是指可以指向函数的变量。它存储了函数的地址,可以通过函数指针调用对应的函数。在C和C++等编程语言中,函数被编译为一段机器码,该机器码位于内存的某个地址上。函数指针就是存储这个地址的变量,它允许我们通过指针来直接调用对应的函数。

1.2函数指针的作用

函数指针可以用于以下几个方面:

1.回调函数:当一个函数需要在某个事件发生时调用另一个函数时,可以使用函数指针来实现回调函数。比如,当一个操作完成时,可以调用预先设置的回调函数来处理操作结果。

2.排序和搜索算法:函数指针可以用于排序和搜索算法中,通过传递一个比较函数给算法,实现不同的排序顺序或搜索条件。

3.动态库加载:函数指针可以用于加载动态库中的函数。通过函数指针,可以在运行时动态地加载和使用库中的函数。

4.多态性:函数指针可以用于实现多态性,允许在运行时确定要调用的具体函数。

总而言之,函数指针可以增强程序的灵活性,使其能够根据特定的条件或需求选择不同的函数进行调用。这在设计可扩展和可重用的代码时非常有用。

1.3函数指针的使用

1.3.1定义

在C++中,可以通过以下两种方式定义函数指针:

1.使用typedef关键字定义函数指针类型,然后使用该类型定义函数指针变量。例如:

typedef void (*FuncPtr)(int);  // 定义函数指针类型
FuncPtr fnPtr;  // 定义函数指针变量

2.直接使用函数指针的语法定义函数指针变量。例如:

void (*fnPtr)(int);  // 定义函数指针变量

3.使用using关键字来定义。例如:

using FnPtr = void (*)(int);  // 定义函数指针类型FnPtr

以上三种方式都定义了一个指向形参为int类型,返回类型为void的函数指针。

注:这里我们需要加上两个括号,它们表示fnPtr是一个函数指针变量,指向一个参数为int类型的函数。

1.3.2初始化

关于函数指针的初始化,我们可以直接赋值给一个已有函数的地址。例如:
 

void myFunction(int x) {
    // 函数体
}

void (*fnPtr)(int);  // 定义函数指针变量

fnPtr = &myFunction;  // 将myFunction函数的地址赋值给fnPtr

在C++中,可以使用取地址运算符&来获取函数的地址,也可以省略取址运算符直接赋值给函数指针变量。另外,使用using关键字定义的函数指针类型可以提高代码的可读性。

对了,在输出函数指针时需要注意,在C++中,函数指针的打印输出不会直接显示函数的地址,而是显示为1。因为C++中的函数指针是一种可调用类型,可以像函数一样进行调用,因此在进行输出时,函数指针会被隐式地转换为一个bool值,而非函数的内存地址。如果想要获得函数的地址,可以使用reinterpret_cast将函数指针转换为void*类型,然后进行打印输出。如下所示:
 

cout << reinterpret_cast<void*>(fnPtr) << endl;  // 输出函数指针fnPtr所指向的函数的地址

这样,我们将得到正确的函数地址的输出。

接下来,我们就可以输出函数的地址了!如下:

1.3.2调用函数

我们可以通过函数指针直接调用相应的函数。例如,假设有一个函数指针fnPtr,可以使用(*fnPtr)(参数列表)的方式调用函数。

void myFunction(int x) {
    // 函数体
}

int main() {
    void (*fnPtr)(int) = myFunction;  // 定义函数指针并初始化

    // 调用函数
    (*fnPtr)(10);

    return 0;
}

1.3.3比较函数指针

我们可以使用函数指针进行比较操作,判断两个函数指针是否相等。函数指针相等表示两个函数指向同一个函数。

#include<bits/stdc++.h>
using namespace std;

void myFunction1(int x) {
    // 函数体
}

void myFunction2(int x) {
    // 函数体
}

int main() {
    void (*fnPtr1)(int) = myFunction1;  // 定义函数指针1并初始化
    void (*fnPtr2)(int) = myFunction2;  // 定义函数指针2并初始化

    // 比较函数指针
    if (fnPtr1 == fnPtr2) {
        cout<<"ok"<<endl;
    } else {
        cout<<"no"<<endl;
    }
    
cout << reinterpret_cast<void*>(fnPtr1) << endl;// 输出函数指针fnPtr所指向的函数的地址
cout << reinterpret_cast<void*>(fnPtr2) << endl; // 输出函数指针fnPtr所指向的函数的地址



    return 0;
}

结果:

1.4函数指针的练习

题目:编写一个程序,计算三角形的面积。利用函数指针来选择计算面积的方法。

步骤:
1. 声明一个函数指针,指向一个计算面积的函数。
2. 编写一个计算面积的函数,该函数接受三个参数表示三角形的边长,并返回计算得到的面积。
3. 编写第二个计算面积的函数,该函数接受两个参数表示三角形的底和高,并返回计算得到的面积。
4. 编写主函数,在其中通过用户输入选择计算面积的方法。
5. 在主函数中,根据用户选择的方法,使用函数指针调用相应的计算面积的函数,并输出结果。

知识点:
1. 函数指针的声明和使用。
2. 函数的定义和调用。
3. 三角形面积的计算方法。

三角形面积的计算方法有两种,一个是三边法,采用的是海伦公式,另一个是底高法,也就是公式面积=底*高/2。

流程图:
 


开始
声明函数指针
定义计算面积函数1
定义计算面积函数2
接收用户输入选择方法
根据选择使用函数指针调用相应函数
输出结果
结束

代码:

#include <iostream>

using namespace std;

// 声明函数指针
typedef double (*AreaFunc)(double, double);

// 计算面积的函数1:三边法
double calcAreaBySide(double a, double b, double c) {
    double s = (a + b + c) / 2;
    return sqrt(s * (s - a) * (s - b) * (s - c));
}

// 计算面积的函数2:底高法
double calcAreaByBaseHeight(double base, double height) {
    return 0.5 * base * height;
}

int main() {
    int choice;
    double a, b, c, base, height;
    AreaFunc areaFunc;

    cout<<"选择方法:"<<endl;
    cin >> choice;

    switch (choice) {
        case 1:
            areaFunc = calcAreaBySide;
            cout << "三角形的三边长度:" << endl;
            cin >> a >> b >> c;
            cout << "三角形的面积为:" << areaFunc(a, b, c) << endl;
            break;
        case 2:
            areaFunc = calcAreaByBaseHeight;
            cout << "三角形的底和高:" << endl;
            cin >> base >> height;
            cout << "三角形的面积为:" << areaFunc(base, height) << endl;
            break;
        default:
            cout << "无效" << endl;
            break;
    }

    return 0;
}

本示例中,通过函数指针 AreaFunc 来选择使用不同的计算面积的函数。用户可以通过输入选择三边法或底高法来计算三角形的面积,程序会根据选择来调用相应的计算函数并输出结果。

2.函数指针数组

既然有函数指针,那肯定也有函数指针数组啦!函数指针数组是一个数组,每个成员都是一个函数指针。

2.1函数指针数组的作用

函数指针数组的作用主要有以下几个方面:

1. 调用不同的函数:函数指针数组可以存储多个函数的指针,通过数组的索引可以选择调用不同的函数。这样可以在运行时根据需要动态地选择不同的函数来执行特定的操作。

2. 函数回调:函数指针数组可以用于实现函数回调的机制。回调函数是指在特定的条件或事件发生时,通过函数指针来调用事先定义好的函数。通过函数指针数组,可以在需要的时候选择合适的回调函数进行调用。

3. 状态机:函数指针数组可以用于实现状态机的机制。状态机是一种表示状态和状态转换的模型,通过函数指针数组可以将每个状态定义为一个函数,使用数组索引进行状态转换。

4. 函数指针数组作为参数传递:函数指针数组可以作为参数传递给其他函数,以便在函数内部使用。这样可以将一组相关的函数作为整体传递,方便对这组函数进行统一的操作或处理。

总之,函数指针数组提供了一种灵活、动态和可扩展的方式来管理和调用函数,使得程序可以根据需要选择不同的函数进行执行,增强了程序的灵活性和可维护性。

2.2函数指针数组的使用

在C++中,函数指针数组的操作与C类似,但可以使用更具有面向对象特性的方式。

2.2.1 定义

使用typedef关键字可以简化函数指针类型的定义,使代码更具可读性。
 

typedef void (*FuncPtr)(int);  // 定义函数指针类型

FuncPtr func_ptr_array[N];  // 定义大小为N的函数指针数组

2.2.2 初始化

可以直接在定义函数指针数组时给出初始值,或通过循环赋值操作为数组元素赋予特定的函数指针。这里介绍在定义函数指针数组时给出初始值和数组元素赋予特定的函数指针这两种方法。

void func1(int);
void func2(int);

FuncPtr func_ptr_array[] = {func1, func2};  // 初始化函数指针数组

func_ptr_array[0] = func1;  // 为函数指针数组赋值
func_ptr_array[1] = func2;

2.2.3 调用

这里我们可以使用数组的索引来选择并调用函数指针数组中的函数。

int index = 0;//索引
func_ptr_array[index](10);  // 调用函数指针数组中索引为0的函数,并传入参数10


2.2.4 传参

这里同样可以将函数指针数组作为参数传递给其他函数,在函数内部使用。

void process_func_array(FuncPtr array[], int size) {
    //函数体......
}

process_func_array(func_ptr_array, 2);  // 将函数指针数组作为参数传递给函数

在C++中,可以使用函数指针数组来实现更加灵活的函数回调机制、策略模式等。此外,C++还提供了更强大的工具,如函数对象、Lambda表达式等,可以替代函数指针数组,在处理函数选择和回调方面更加方便和易用。

2.3函数指针二维数组

函数指针二维数组用于存储多个函数指针,每个函数指针可以指向一个函数。它的定义和使用方式与普通二维数组类似,只是数组元素是函数指针。

函数指针二维数组的定义如下:

typedef returnType (*functionName)(parameter1, parameter2, ...);

functionName arrayName[rowSize][colSize];

其中,returnType 是函数的返回类型,functionName 是函数指针类型的名称,parameter1, parameter2, ... 是函数的参数列表。arrayName 是函数指针二维数组的名称,rowSize 是数组的行数,colSize 是数组的列数。

2.4函数指针数组的练习

题目:编写一个程序,根据输入的年份判断是否为闰年。

步骤:
1. 定义一个函数指针数组,用于存储判断闰年的函数地址。
2. 定义多个判断闰年的函数,如根据公历规则判断、根据儒略历规则判断等。
3. 在主函数中,输入一个年份。
4. 遍历函数指针数组,依次调用判断闰年的函数,并将输入的年份作为参数传递给该函数。
5. 根据函数的返回结果判断是否为闰年,如果是则输出“该年份是闰年”,否则输出“该年份不是闰年”。

知识点:
1. 函数指针数组的定义和使用。
2. 根据输入的年份调用对应的判断闰年函数。
3. 函数的参数传递和返回值。

流程图:

输入一个年份
 |
V
遍历函数指针数组
 |
V
调用判断闰年的函数,将年份作为参数传递
 |
V
根据返回结果判断是否为闰年
 |
V
输出结果

代码:
 

在运行时,程序会要求输入一个年份,然后遍历函数指针数组,依次调用判

#include <iostream>

typedef bool (*leapYearChecker)(int); // 定义函数指针类型

// 根据公历规则判断是否为闰年
bool isLeapYearByGregorian(int year) {
    if ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0) {
        return true;
    } else {
        return false;
    }
}

// 根据儒略历规则判断是否为闰年
bool isLeapYearByJulian(int year) {
    if (year % 4 == 0) {
        return true;
    } else {
        return false;
    }
}

int main() {
    int year;
    
    std::cout << "请输入一个年份:";
    std::cin >> year;
    
    leapYearChecker leapYearCheckers[] = {isLeapYearByGregorian, isLeapYearByJulian}; // 函数指针数组
    
    int numCheckers = sizeof(leapYearCheckers) / sizeof(leapYearCheckers[0]); // 计算数组长度
    
    bool isLeapYear = false;
    
    for (int i = 0; i < numCheckers; i++) {
        isLeapYear = leapYearCheckers[i](year); // 调用判断闰年的函数
        
        if (isLeapYear) {
            std::cout << "该年份是闰年" << std::endl;
            break;
        }
    }
    
    if (!isLeapYear) {
        std::cout << "该年份不是闰年" << std::endl;
    }
    
    return 0;
}

断闰年的函数,并将输入的年份作为参数传递给该函数。根据函数的返回结果判断是否为闰年,并输出相应的结果。

3.回调函数与回调指针

回调函数(callback function)是指在某个函数执行完毕后自动调用的函数。它的作用是将一个函数传递给另一个函数,以便在适当的时候被调用。

3.1回调函数的用处

回调函数的作用主要体现在以下几个方面:

1.异步操作:通过传递回调函数作为参数,可以在某个操作完成后立即执行回调函数。这样可以避免阻塞主线程,提高程序的并发性和响应性。

2.事件处理:通过注册回调函数作为事件处理函数,可以实现对事件的响应和处理。这样可以增加程序的可扩展性和可维护性。

3.抽象接口:通过回调函数,可以将具体的实现逻辑与调用逻辑分离。这样可以提高代码的复用性和可读性。

总结而言,回调函数是一种常用的编程技术,可以实现异步操作、事件处理和抽象接口等功能。它可以提高程序的灵活性和可扩展性,使程序更加模块化和可维护。

3.2回调函数的使用

回调指针的使用包括以下几个步骤:

3.2.1 定义函数原型

首先需要定义回调函数的函数原型,包括返回类型和参数列表。这个函数原型将作为回调函数的类型,后续需要定义回调指针来指向这个函数。

// 定义回调函数的函数原型
typedef void (*CallbackFunction)(int);

3.2.2 声明变量

使用定义的回调函数类型来声明回调指针变量。回调指针变量是一个指向回调函数的指针,可以用来存储回调函数的地址。

// 声明回调指针变量
CallbackFunction callbackPointer;

3.2.3定义

根据之前定义的函数原型,实现回调函数的具体逻辑。回调函数可以在特定事件或条件发生时被调用执行,可以在函数内部进行相应的操作。

// 回调函数
void callbackFunction(int value) {
    std::cout << "Callback Function: " << value << std::endl;
}

// 需要回调的函数
void doSomething(int value, CallbackFunction callback) {
    std::cout << "Doing something with value: " << value << std::endl;
    // 调用回调函数
    callback(value);
}

3.2.4 地址传递

在需要使用回调函数的地方,将回调函数的地址作为参数传递给其他函数。这些函数在特定的时候可以通过回调指针调用回调函数。

// 将回调函数的地址传递给其他函数
    callbackPointer = callbackFunction;

3.2.5调用

在需要触发回调函数的时候,通过回调指针调用回调函数。回调函数将执行其内部的逻辑操作,并返回结果或执行特定的任务。
 

 // 调用需要回调的函数
    doSomething(10, callbackPointer);

3.3回调指针

回调指针(callback pointer)是指一个指针,它指向回调函数的内存地址。通过使用回调指针,可以在适当的时候调用回调函数。

3.3.1回调指针的使用

在C++中,可以通过回调指针在回调函数中调用特定的函数或方法。下面是一个代码示例:
 

#include <iostream>

// 回调函数的定义
void callbackFunc(int arg) {
    std::cout << "Callback called with argument: " << arg << std::endl;
}

// 接受回调函数指针的函数
void doSomething(int data, void (*callback)(int)) {
    // 执行其他操作
    std::cout << "Doing something with data: " << data << std::endl;
    
    // 调用回调函数
    callback(data);
    
    // 执行其他操作
}

int main() {
    int data = 42;
    
    // 调用doSomething函数,并传递回调函数指针
    doSomething(data, callbackFunc);
    
    return 0;
}

在这个例子中,callbackFunc是一个回调函数,它接受一个整数参数,并在函数内部打印该参数的值。doSomething函数接受一个整数参数和一个指向回调函数的指针。它在执行其他操作后调用回调函数,将传入的数据作为参数传递给回调函数。

在main函数中,首先定义了一个整数变量data,然后调用doSomething函数,并将data和callbackFunc作为参数传递给它。

当doSomething函数执行时,它会执行一些操作,然后调用传入的回调函数指针,传递给它data`作为参数。这样,回调函数callbackFunc将被调用,并打印出传入的参数值。

通过回调指针,可以在回调函数中灵活地调用不同的函数或方法,实现不同的功能和逻辑。

3.3.2回调函数与回调指针的关系

回调函数与回调指针之间存在着紧密的关系。下面是一个简单的图表,用于说明它们之间的关系:
      

        +---------------------+
        |      调用者函数       |
        +---------------------+
                 |
                 |   回调指针
                 |
        +---------------------+
        |       回调函数       |
        +---------------------+

解析:在这个图表中,调用者函数是一个函数,它需要在特定的事件或条件发生时执行某些操作。为了实现这个功能,调用者函数会定义一个回调指针,并将需要执行的回调函数的地址赋给这个指针。

回调函数是一个独立的函数,它执行实际的操作。它的原型(参数列表和返回值类型)需要与回调指针指向的函数类型相匹配。当特定事件或条件发生时,调用者函数将使用回调指针调用回调函数,从而执行所需的操作。

通过使用回调指针,调用者函数可以在运行时灵活地指定要执行的回调函数。这使得回调函数的执行可以根据具体的需求动态地确定。

总结:回调函数通过回调指针与调用者函数建立联系,实现在特定事件或条件下执行指定操作的机制,相当于“过渡”的作用。回调指针允许在运行时动态地指定回调函数,从而实现灵活性和可扩展性。

4.指针函数

指针函数是指返回指针的函数。与普通函数返回基本数据类型或对象不同,指针函数返回一个指针,该指针指向内存中的某个地址。

4.1指针函数的作用

指针函数的作用主要有以下几个方面:

1.返回动态分配的内存:指针函数可以用于动态分配内存,并返回指向该内存的指针。这样可以在函数外部使用这些动态分配的内存,并在适当的时候释放。

2.返回局部变量的地址:指针函数可以返回局部变量的地址。虽然局部变量的作用域只限于函数内部,但通过返回指针,可以在函数外部访问并修改局部变量的值。

3.返回数组、链表等数据结构:指针函数可以返回数组、链表等数据结构的指针,这样可以在函数外部通过指针访问和操作这些数据结构。

4.函数指针的使用:指针函数可以返回函数指针,即返回一个指向函数的指针。这样可以在函数外部调用不同的函数,实现程序的动态性。

4.2指针函数的操作

指针函数是一种返回指针的函数。它可以用来返回动态分配的内存、对象的指针或者函数指针。

下面是一些关于指针函数的常见操作:

4.2.1 声明

int* myFunction();  // 返回int类型指针的指针函数

上述代码声明了一个返回int类型指针的指针函数myFunction。

4.2.2 定义

int* myFunction() {
   int* ptr = new int(10);
   return ptr;
}


上述代码定义了一个指针函数myFunction,它动态分配了一个int类型的内存,并返回指向该内存的指针。

4.2.3 调用

int* result = myFunction();


上述代码调用指针函数myFunction,并将返回的指针存储在result变量中。

4.2.4返回

cout << *result << endl;  // 输出指针所指向的值
delete result;  // 释放动态分配的内存


上述代码使用指针函数返回的指针,输出指针所指向的值,并释放动态分配的内存。

4.3指针函数的练习

题目:编写一个指针函数,接受一个整数数组和数组的长度作为参数,返回一个指向数组中最大元素的指针。

步骤:
1. 声明并定义一个指针函数,函数名为`findMax()`,参数为整数数组和数组的长度。
2. 在函数内部,声明一个指针变量`max`,并将其初始化为指向数组的第一个元素。
3. 使用一个循环遍历数组的每个元素:
   - 如果当前元素大于`max`指向的值,则将`max`指向当前元素。
4. 返回指针`max`。

知识点:
- 指针函数的声明和定义
- 指针的初始化和赋值
- 指针的比较

流程图:
 

开始
|
声明并定义指针函数 findMax(arr, length)
|
声明并初始化指针变量 max = arr
|
循环遍历数组的每个元素
|
|   如果当前元素 > *max
|   |
|   将 max 指向当前元素
|
返回 max
|
结束


代码:
 

int* findMax(int* arr, int length) {
    int* max = arr;
    for (int i = 0; i < length; i++) {
        if (arr[i] > *max) {
            max = &arr[i];
        }
    }
    return max;
}

接下来,我们就可以在main函数里直接调用这个函数啦!这个练习可以帮助你熟悉指针函数的声明、定义和使用,以及指针的操作。可以通过测试不同的数组来验证函数的正确性。记得在使用完动态分配的内存后,及时释放它们,以避免内存泄漏。

总结

本篇博客到这里就结束了,感谢大家的支持与观看,如果有好的建议欢迎留言,制作不易,如果对您有帮助,那请给PingdiGuo_guo一个免费的赞,谢谢啦!

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

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

相关文章

【校园安全】支小蜜防校园霸凌语音识别系统的好处

在校园安全领域&#xff0c;防校园霸凌语音识别系统的出现&#xff0c;为预防和应对校园霸凌行为提供了新的技术手段。本文将探讨防校园霸凌语音识别系统的好处&#xff0c;并分析其在校园安全建设中的重要作用。 通过安装在校园各个角落的语音识别设备&#xff0c;系统能够捕…

LeetCode 刷题 [C++] 第3题.无重复字符的最长子串

题目描述 给定一个字符串 s &#xff0c;请你找出其中不含有重复字符的 最长子串 的长度。 题目分析 可以使用滑动窗口加哈希表来实现&#xff1a; 使用start和end两个变脸来表示滑动窗口的头部位置和尾部位置&#xff0c;两者开始均为0&#xff1b;借助哈希表来记录已经遍…

WinSCP下载安装并结合内网穿透实现固定公网TCP地址访问本地服务器

文章目录 1. 简介2. 软件下载安装&#xff1a;3. SSH链接服务器4. WinSCP使用公网TCP地址链接本地服务器5. WinSCP使用固定公网TCP地址访问服务器 1. 简介 ​ Winscp是一个支持SSH(Secure SHell)的可视化SCP(Secure Copy)文件传输软件&#xff0c;它的主要功能是在本地与远程计…

走向审计4.0:内部审计数字化转型的路径与方法【文末送书-34】

文章目录 走向审计4.0&#xff1a;内部审计数字化转型的路径与方法一、内部审计的发展阶段二、内部审计的逻辑架构三、内部审计数字化转型面临的问题四、内部审计数字化转型的框架方法五、内部审计的数字化转型能力体系六、内部审计的数字化转型路径七、内部审计的数字化系统平…

Research Agent:具有解决基于大型文本语料库的复杂问题,具备深度多跳推理的能力

原文地址&#xff1a;https://towardsdatascience.com/the-research-agent-4ef8e6f1b741 2023 年 8 月 29 日 问题简介 在2021年&#xff0c;开始应对基于大量文本回答问题的挑战。在预训练transformers之前的时代&#xff0c;这个问题很难破解。 人工智能和大型预训练tran…

143.和弦是什么?和声是什么?三和弦

内容参考于&#xff1a; 三分钟音乐社 上一个内容&#xff1a;142.音程的构唱练习 和弦的定义&#xff1a; 一个音可以把它称为单音 两个音可以把它称为音程 更多的音&#xff0c;通俗的定义上&#xff0c;三个音或者三个以上的音构成的集体就可以叫做和弦&#xff0c;这些音…

Redis进阶(三):主从复制

为了解决单点问题&#xff0c;实现多服务器部署redis&#xff0c;有几种解决方案可以实现&#xff1a;主从复制&#xff0c;主从哨兵还有集群。 何为主从复制 简单来说有三个服务器分别部署了redis-server程序&#xff0c;选中一个服务器当作主节点&#xff0c;其他的就是从节…

TypeScript 哲学 - Object Types

readonly 修饰对象和数组的 双向可分配性是不同的 Combind types 交叉类型对 值类型可以识别 never 但是 Object 类型识别不了 Tuple

AI时代PPT如何制作?用这10款pptai生成器一键制作!

ppt如何制作&#xff1f; 这可能是很多职场人或大学生日常头疼的问题&#xff0c;职场上随便一个工作汇报、提案展示、团队会议&#xff0c;学校里的小组作业、论文答辩等场景&#xff0c;都会用到ppt。 都说人是视觉动物&#xff0c;在两份文档内容质量一致的情况下&#xf…

Qt插件之输入法插件的构建和使用(二)

文章目录 主键盘搭建Google开源引擎音节分割工具类参考项目下载搭建好各个基础控件之后,就可以开发输入法的主界面和引擎了,这也是输入法的核心。 主键盘搭建 输入法的主界面本质上是一个QStackedWidget容器,将各个类型的输入键盘插入到容器中,然后根据业务需要切换不同的…

通过对话式人工智能打破语言障碍

「AI突破语言障碍」智能人工智能如何让全球交流无障碍 在当今互联的世界中&#xff0c;跨越语言界限进行交流的能力比以往任何时候都更加重要。 对话式人工智能&#xff08;包括聊天机器人和语音助手等技术&#xff09;在打破这些语言障碍方面发挥着关键作用。 在这篇博文中&am…

Java对接海康威视摄像头实现抓图

目录 一、下载SDK 二、拷贝示例代码 三、拷贝库文件 四、运行Demo 五、抓图业务 六、调参 ​七、发布Linux正式环境 一、下载SDK 海康开放平台 二、拷贝示例代码 三、拷贝库文件 这时候直接运行ClientDemo会报错&#xff0c;因为缺失库文件&#xff01; 四、运行Demo …

13 OpenCv自定义线性滤波

文章目录 卷积算子示例 卷积 卷积是图像处理中一个操作&#xff0c;是kernel在图像的每个像素上的操作。Kernel本质上一个固定大小的矩阵数组&#xff0c;其中心点称为锚点(anchor point) 把kernel放到像素数组之上&#xff0c;求锚点周围覆盖的像素乘积之和&#xff08;包括锚…

杨辉三角(附html,Python,c++杨辉三角代码)

1、前言 杨辉三角很早学编程就知道&#xff0c;但不知道杨辉是古人&#xff01; 一次偶然的机会&#xff0c;和杨辉三角对上了眼神——杨辉&#xff01;究竟你是怎么发现杨辉三角的呢&#xff1f; 于是经过了长达3个月锲而不舍的探究。 终究也没发现自己想要的最终结果。 …

内螺旋面的计算

内螺旋面的计算 内螺旋面的计算是机械工程和数学领域中的一个重要问题。它涉及到三维几何、微积分和数值计算等多个方面的知识。内螺旋面的形状和特性对于许多机械装置和工程应用来说至关重要&#xff0c;因此准确计算内螺旋面的几何参数和性质至关重要。内螺旋面是一种特殊的…

warning: #188-D: enumerated type mixed with another type

警告解释&#xff1a;枚举类型混合了其它的数据类型&#xff1b; 解决方法&#xff1a; 1&#xff1a;检查代码&#xff0c;是不是存在混用&#xff1b;&#xff1b; 2&#xff1a;结构体初始化为 0 报warning&#xff0c;不能将结构体的第一个变量&#xff0c;使用枚举类型&am…

Dgraph 入门教程五(Shema)

要想做好图数据库&#xff0c;必须做图数据库设计&#xff0c;想做好Dgraph&#xff0c;就需要设计好Shema。Shema的语法是必须理解&#xff0c;如果不理解&#xff0c;别人写的Shema&#xff0c;我们也看不懂。我这里也是走马观花式的记录&#xff0c;细节还需要在使用的时候去…

如何恢复未保存的 Excel 文件

本周我们将 Office 恢复系列扩展到 Excel 恢复&#xff0c;并提出了最常见的问题&#xff1a;如何恢复 Excel 文件&#xff1f; 与 Office Word 不同&#xff0c;Excel 完全是关于表格和计算的。在处理Excel文件时&#xff0c;您可能会遇到更多问题。与往常一样&#xff0c;我们…

java-抢红包一些简单概念

抢红包&#xff0c;比如微信中抢红包&#xff0c;红包金额分配使用的是二倍均值算法。 二倍均值拆包&#xff1a; 拆包要求:所有人抢到的金额之和等于红包总额&#xff0c;每个人最少抢到 0.01 元&#xff0c;每个人抢到的红包金额不要相差太大二倍均值法:假设红包总金额是X&…

python安装laspy遍历las获取所有点的坐标

python3.12 pip install laspy 读取 LAS 文件中的点坐标&#xff0c;可以使用 laspy 库。这个库允许你轻松地读取和操作 LAS 文件中的数据。下面是一个示例代码&#xff0c;演示了如何使用 laspy 读取 LAS 文件中的坐标&#xff1a; 1、方法一 #version laspy&#xff1a;2.…