16- C++多态-4 (C++)

news2025/1/21 6:38:43

第五章 多态

5.1 多态的引入

思考:在之前实现的英雄模型中,假如实现某个接口可以传入一个英雄,在该接口中可以对英雄的力量、敏捷和智力进行加强,请问该接口的参数该如何设计?

以上解决办法利用了C++中的多态,接下来让我们来了解一下 C++中的多态

多态:一个函数有多种形态。

多态的分类:静态联编 动态联编

通常来说联编就是将模块或者函数合并在一起生成可执行代码的处理过程,同时对每个模块或者函数调用分配内存地址,并且对外部访问也分配正确的内存地址,它是计算机程序彼此关联的过程。按照联编所进行的阶段不同,可分为两种不同的联编方法:静态联编动态联编

5.2 静态联编

5.2.1 静态联编的概念

静态联编是指联编工作在编译阶段完成的,这种联编过程是在程序运行之前完成的,又称为早期联编。要实现静态联编,在编译阶段就必须确定程序中的操作调用(如函数调用)与执行该操作代码间的关系,确定这种关系称为束定,在编译时的束定称为静态束定。静态联编对函数的选择是基于指向对象的指针或者引用的类型。其优点是效率高,但灵活性差

5.2.2 静态联编的体现

1、隐藏

2、函数的的重载

3、运算符重载

4、泛型编程

5.3 运算符重载

5.3.1 运算符重载概述

1、运算符重载,就是对已有的运算符重新进行定义,赋予其另一种功能,以适应不同的数据类型。(运算符重载不能改变本来寓意,不能改变基础类型寓意)

2、运算符重载(operator overloading)只是一种 语法上的方便 ,也就是它只是另一种函数调用的方式。

3、在c++中,可以定义一个处理类的新运算符。这种定义很像一个普通的函数定义,只是函数的名字由关键字operator及其紧跟的运算符组成。差别仅此而已。它像任何其他函数一样也是一个函数,当编译器遇到适当的模式时,就会调用这个函数。

4、可以重载的运算符

 5、不可重载的运算符:

  • 成员访问运算符(.)

  • 成员指针访问运算符(->)

  • 域运算符(::)

  • 长度运算符(sizeof)

  • 条件运算符(: ?)

  • 预处理符号(#)

5.3.2 运算符重载实例

#include <iostream>
#include <string>
#include <cstring>
#include <cstdio>

using namespace std;

class Test
{
public:
    int data;
    char *ptr;

    // 默认构造函数
    Test()
    {
        std::cout << "Test() " << std::endl;
    }

    // 带参数的构造函数
    Test(int data, const char *src)
    {
        this->data = data;
        if (src)
        {
            std::cout << "struct test(data)" << std::endl;
            this->ptr = new char[std::strlen(src) + 1];
            std::strcpy(this->ptr, src);
        }
        else
        {
            ptr = new char[10];
        }
    }

    //拷贝构造函数,调用时间:使用一个构造好的对象初始化一个新的对象
    Test(const Test &t)
    {
        cout << "Test(const Test &t)" << endl;
        //实现深拷贝
        this->data = t.data;
        //this->ptr = t.ptr; //浅拷贝
        if (strlen(t.ptr))
        {
            this->ptr = new char[strlen(t.ptr)+1];
            strcpy(this->ptr, t.ptr);
        }
        else
            ptr = new char[10];
    }

    //该运算符重载函数由 左操作数调用,右操作数当做实参传递给该函数,触发:t1 + t3 -> t1.operator + (t3)
    Test operator + (const Test &t)  //operator 操作符
    {
        cout << "Test operator + (Test &t)" << endl;
        Test val;
        val.data = this->data + t.data;
        val.ptr = new char[strlen(this->ptr) + strlen(t.ptr) + 1];
        //将分配的堆空间初始化为/0
        memset(val.ptr, 0, strlen(this->ptr) + strlen(t.ptr) + 1);

        strcat(val.ptr, this->ptr);
        strcat(val.ptr, t.ptr);
        return val;
    }

    bool operator > (const Test &t)
    {
        cout << "bool operator > (const Test &t)" << endl;
        if (strcmp(this->ptr, t.ptr) > 0)
            return true;
        return false;
    }

    char operator [](int index)
    {
        cout << "char operator [](int index)" << endl;
        if (index < 0 || index >= strlen(ptr))
            return '\0';
        return ptr[index];
    }

    Test &operator = (const Test &t)
    {
        cout << "Test &operator =(const Test &t)" << endl;
        delete[] this->ptr;
        this->ptr = new char[strlen(t.ptr) + 1];
        strcpy(this->ptr, t.ptr);

        data = t.data;
        return *this;
    }

    //前置++
    Test &operator++()
    {
        cout << "Test &operator ++(int)" << endl;
        ++data;
        return *this;
    }

    //后置++
    Test operator++(int)
    {
        cout << "Test operator ++(int)" << endl;
        Test tmp = *this;
        data++;
        return tmp;
    }

    //<<重载
    friend ostream &operator << (ostream &os, const Test &t)
    {
        os << t.data << endl;
        os << t.ptr << endl;
        return os;
    }

    // 析构函数
    ~Test()
    {
        std::cout << "~Test() delete test()" << std::endl;
        if (ptr)
            delete[] ptr;
    }
};

int main() {
#if 0
    // + 运算符重载
    Test t1(10, "hello");
    Test t2 = t1;
    cout << t2.data << endl;
    cout << t2.ptr << endl;

    Test t3(23, "world");
    cout << "********** input t3" << endl;
    Test t4 = t1 + t3;
    cout << t4.data << endl;
    cout << t4.ptr << endl;
#endif

#if 0
    // > 运算符的重载
    Test t1(10, "hello");
    Test t2(20, "hello");
    if (t1 > t2)
        cout << "t1 > t2" << endl;
    else
        cout << "t1 < t2" << endl;

    // [] 运算符的重载
    cout << "t1.ptr:" << t1.ptr << endl;
    cout << "t1[4]: " << t1[4] << endl;

    // = 符号的重载
    t1 = t2;
    cout << "-------------- t1.ptr: " << t1.ptr << endl;
    printf("t1.ptr: %p, t2.ptr: %p\n", t1.ptr, t2.ptr);  //地址
    printf("t1.ptr: %s, t2.ptr: %s\n", t1.ptr, t2.ptr);

    ++t1;
#endif
#if 0
    //前置++的重载
    Test t1(10, "hello");
    Test t2;
    t2 = ++t1;
    cout << "t2.data: " << t2.data << endl;
    cout << "t1.data: " << t1.data << endl;
    cout << "t1.ptr: " << t1.ptr << endl;

    //后置++的重载
    //Test t3;
    //t3 = t1++;
    //cout << t3.data << endl;
    //cout << t1.data << endl;

    string s("hello");
    cout << "s: " << s << endl;
    cout << "t1: " << t1 << endl;
#endif
    Test t1(10, "hello");
    cout << "++++++++++++++ t1: " << t1 << endl;
    return 0;
}

5.3.3 前置++和后置++

class A
{
private:
    int a;
public:
    A& operator++()
    {
        ++a;
        return *this;
    }
    A operator++(int)
    {
        A a = *this;
        ++*this;
        return a;
    }
};

因为后置++在实现的时候构造了一个临时对象,临时对象的构造和销毁都需要消耗一定的系统资源,所以后置++的效率比前置++的效率低

5.4 友元

5.4.1 友元函数

友元函数是可以直接访问类的私有成员的非成员函数。它是定义在类外的普通函数,它不属于任何类,但需要在类的定义中加以声明,声明时只需在友元的名称前加上关键字friend,其格式如下:

 friend 类型 函数名(形式参数);

友元函数 的使用:

1、友元函数的声明可以放在类的私有部分,也可以放在公有部分,它们是没有区别的,都说明是该类的一个友元函数。

2、一个函数可以是多个类的友元函数,只需要在各个类中分别声明。

3、友元函数的调用与一般函数的调用方式和原理一致。

4、友元函数中没有this指针

5、两个类要共享数据的时候可以使用友元函数,比如:类A中的函数需要访问类B中的成员,那

么类A中该函数要是类B的友元函数。

6、运算符重载的某些场合需要使用友元函数,例如 << 的重载

#include <iostream>

using namespace std;
class CCar;
class CDriver
{
public:
    void ModifyCar(CCar* pCar);
};
class CCar
{
private:
    int price;
    friend int MostExpensiveCar(CCar cars[], int total);
    friend void CDriver::ModifyCar(CCar *pCar);
};
void CDriver::ModifyCar(CCar *pCar) {
    pCar->price += 1000;
}

int MostExpensiveCar(CCar cars[], int total)
{
    int tmpMax = -1;
    for (int i = 0; i < total; ++i)
        if (cars[i].price > tmpMax)
            tmpMax = cars[i].price;
        return tmpMax;
}

int main() {
    return 0;
}

5.4.2 友元类

友元类的所有成员函数都是另一个类的友元函数,都可以访问另一个类中的隐藏信息(包括私有成员和保护成员)。

当希望一个类可以访问另一个类的私有成员时,可以将该类声明为另一类的友元类。定义友元类的语句格式如下:

  friend class 类名;

其中:friend和class是关键字,类名必须是程序中的一个已定义过的类。

#include <iostream>

using namespace std;

class CCar
{
private:
    int price;
    friend class CDriver;
};

class CDriver
{
public:
    CCar myCar;
    void ModifyCar()
    {
        myCar.price += 1000;
    }
};

int main() {
    return 0;
}

使用友元时注意:

  1、友元关系不能被继承

  2、友元关系是单向的,不具有交换性。若类B是类A的友元,类A不一定是类B的友元,要看在类中是否有相应的声明。

  3、友元关系具有非传递性。若类B是类A的友元,类C是B的友元,类C不一定是类A的友元,同样要看类中是否有相应的申明。

注意:友元的作用是提高了程序的运行效率(即减少了类型检查和安全性检查等都需要时间开销),但它破坏了类的封装性和隐藏性,使得非成员函数可以访问类的私有成员,不建议使用!

5.4.3 为什么输出运算符的重载需要用friend修饰呢?

如果是 重载双目操作符(即为类的成员函数),就只要设置一个参数作为右侧运算量,而左侧运算量就是对象本身。而 >> 或<< 左侧运算量是 cin或cout 而不是对象本身,所以不满足后面一点,就只能申明为友元函数了。。。

如果一定要声明为成员函数,只能成为如下的形式:

ostream & operator << (ostream &output)
{
    output << this->x << endl;
    return output;
}

5.5 动态多态

1、动态多态(动态绑定):即运行时的多态,在程序执行期间(非编译期)判断所引用对象的实际类型,根据其实际类型调用相应的方法。

2、有的工程师认为真正的多态是动态多态

3、动态多态(多态)满足的三个条件

  • 有继承关系

  • 有虚函数

  • 基类指针指向派生类对象或者基类的引用变量引用了派生类对象

5.6 重载、重写(覆盖)、隐藏

  1. 重载:overload 重载只发生在同一个作用域中,比如 一个类中的多个成员函数函数名相同,但是形参数据类型或或者个数或者顺序不相同,那么我们就称这些函数是重载。
  2. 重写:override 也叫做覆盖 重写发生在不同的作用域中(发生在基类和派生类中),而且派生类中的成员函数的名字和基类中的虚函数的名字相同,并且返回值相同,形参列表也相同!!!
  3. 隐藏隐藏发生在不同的作用域中(发生在基类和派生类中),派生类中的成员函数的名字、返回值、形参列表与基类中的普通函数完全相同, 派生类中的成员函数的名字和基类中的成员函数的名字。
  4. 相同但是形参列表不同或者返回值不同这也叫做隐藏,此时基类中的那个函数不论是普通函数还是虚函数都会被派生类中的函数隐藏。
#include <iostream>
using namespace std;

class Hero
{
public:
    virtual void huicheng()
    {cout << "Base::huicheng" << endl;}

    void func2()
    {cout << "Base::func2" << endl;}
};

class Libai : public Hero
{
public:
    void func2()
    {cout << "A::func2" << endl;}

    void func2(int x)
    {}

    void huicheng()
    {cout << "Libai::huicheng" << endl;}
    void huicheng(int x){}
};

class Caocao : public Hero
{
public:
    void func2()
    {cout << "A::func2" << endl;}

    void huicheng()
    {cout << "Caocao::huicheng" << endl;}
};

class Houyi : public Hero
{};

void goback(Hero &h)
{
    h.huicheng();
    h.func2();
}

int main() {
    Libai libai;
    Caocao caocao;
    Houyi houyi;

    goback(libai);
    goback(caocao);
    goback(houyi);

    cout << " +++++++++++++++++++ output caocao a:" << endl;
    Caocao a;
    a.func2();
    a.huicheng();

    cout << " +++++++++++++++++++ output hero *p:" << endl;
    Hero *p;
    p = &a;
    p->func2();
    p->huicheng();   //指向派生类
    return 0;
}

5.7 虚函数

5.7.1 虚函数的基本使用

C++中的虚函数的作用主要是实现了多态的机制。基类定义虚函数,子类可以重写该函数。

虚函数的定义:

virtual 函数类型 函数名(形参列表);

#include <iostream>
using namespace std;

class Base
{
public:
    virtual void func()
    {cout << "Base func: " << endl;}
    virtual void func2()
    {cout << "Base func2: " << endl;}
};

class Derived:public Base
{
public:
    void func()
    {cout << "Derived func: " << endl;}
};

int main() {
    Derived d;
    d.func();
    return 0;
}

虚函数的实现原理:虚函数的实现是由两个部分组成的,虚函数指针与虚函数表

5.7.2 虚函数指针

虚函数指针 (virtual function pointer, vptr) 从本质上来说就只是一个指向函数的指针,与普通的指针并无区别。它指向用户所定义的虚函数,具体是在子类里的实现,当子类调用虚函数的时候,实际上是通过调用该虚函数指针从而找到接口。

只有拥有虚函数的类才会拥有虚函数指针,每一个虚函数也都会对应一个虚函数指针。

5.7.3 虚函数表

存放虚函数指针的数组我们称之为 虚函数表(virtual function table, vtbl)。

每个包含了虚函数的类都包含一个虚表。当一个类(A)继承另一个类(B)时,类 A 会继承类 B 的函数。所以如果一个基类包含了虚函数,那么其派生类也可调用这些虚函数,换句话说,一个类继承了包含虚函数的基类,那么这个类也拥有自己的虚表。

我们来看以下的代码。类 A 包含虚函数vfunc1,vfunc2,由于类 A 包含虚函数,故类 A 拥有一个虚表。

class A{
public:
    virtual void vfunc1(){};
    virtual void vfunc2(){};
    void func1(){};
    void func2(){};
private:
    int m_data1, m_data2;
};

5.7.4 虚表指针(虚函数表指针)

虚表是属于类的,而不是属于某个具体的对象,一个类只需要一个虚表即可。同一个类的所有对象都使用同一个虚表。

为了指定对象的虚表,对象内部包含一个虚表的指针,来指向自己所使用的虚表。为了让每个包含虚表的类的对象都拥有一个虚表指针,编译器在类中添加了一个指针,*__vfptr,用来指向虚表。这样,当类的对象在创建时便拥有了这个指针,且这个指针的值会自动被设置为指向类的虚表。

虚表指针存在于每一个被实例化的对象中,前提是该对象中有虚函数,它总是被存放在该对象的地址首位,这种做法的目的是为了保证运行的快速性。

动态绑定, C++ 是如何利用虚表和虚表指针来实现动态绑定的。

class A{
public:
    virtual void vfunc1(){};
    virtual void vfunc2(){};
    void func1(){};
    void func2(){};
private:
    int m_data1, m_data2;
};

class B : public A
{
public:
    void vfunc1();
    void func1();
private:
    int m_data3;
};

class C: public B
{
public:
    virtual void vfunc2();
    void func2();
private:
    int m_data1, m_data4;
};

类 A 是基类,类 B 继承类 A,类 C 又继承类 B。类 A,类 B,类 C。

由于这三个类都有虚函数,故编译器为每个类都创建了一个虚表,即类 A 的虚表(A vtbl),类 B 的虚表(B vtbl),类 C 的虚表(C vtbl)。类 A,类 B,类 C 的对象都拥有一个虚表指针,*__vfptr,用来指向自己所属类的虚表。

类 A 包括两个虚函数,故 A vtbl 包含两个指针,分别指向A::vfunc1()和A::vfunc2()。

类 B 继承于类 A,故类 B 可以调用类 A 的函数,但由于类 B 重写了B::vfunc1()函数,故 B vtbl 的两个指针分别指向B::vfunc1()和A::vfunc2()。

类 C 继承于类 B,故类 C 可以调用类 B 的函数,但由于类 C 重写了C::vfunc2()函数,故 C vtbl 的两个指针分别指向B::vfunc1()(指向继承的最近的一个类的函数)和C::vfunc2()。

非虚函数的调用不用经过虚表,故不需要虚表中的指针指向这些函数。

#include <iostream>
using namespace std;

class A{
public:
    virtual void vfunc1()
    {cout << "A::virtual void vfunc1()" << endl;};
    virtual void vfunc2()
    {cout << "A::virtual void vfunc2()" << endl;};
    void func1()
    {cout << "A::void func1()" << endl;};
    void func2()
    {cout << "A::void func2()" << endl;};
};

class B : public A
{
public:
    void vfunc1()
    {{cout << "B::virtual void vfunc1()" << endl;}};
    void func1()
    {{cout << "B::void func1()" << endl;}};
};

int main() {
    B b;
    b.vfunc1();
    b.vfunc2();
    b.func1();
    b.func2();

    A *p;
    p = &b;
    p->vfunc1();
    p->vfunc2();
    p->func1();
    p->func2();
    return 0;
}

5.7.6 虚表指针、虚函数的访问

#include <iostream>
#include <Cstdio>
using namespace std;
 
typedef void(*Fun)(void);

class Base {
public:
    int x;
    virtual void f() { cout << "Base::f" << endl; }
    virtual void g() { cout << "Base::g" << endl; }
    virtual void h() { cout << "Base::h" << endl; }
};

int main() {
    Base b;
    Fun pFun = NULL;

    //虚函数表指针是对象中的第一个元素,虚函数表指针的地址就是对象的首地址
    printf("%p\n",(int*)(&b)); // &_vfptr
    //(*(int*)(&b)) : 虚函数表指针的值/虚函数表的地址
    printf("%p\n", (int*)(*(int*)(&b))); //_vfptr
    printf("%p\n", (int*)(*(int*)(&b))+2);
    printf("%p\n", (int*)(*(int*)(&b))+4);

    printf("%p\n", *(int*)(*(int*)(&b)));
    printf("%p\n", *(int*)(*(int*)(&b))+2);
    printf("%p\n", *(int*)(*(int*)(&b))+4);

    pFun = (Fun)(*((int*)(*(int*)(&b))+4));
    pFun();
}

5.8 纯虚函数和抽象基类

5.8.1 纯虚函数

1、一般来说,许多时候基类并不能确定函数的实现方法,只能确定函数的功能。但是函数调用的时候必须要用到该函数。这种情况下,C++提供了一种机制,成为纯虚函数,属于虚函数的一种,体现了面向对象的多态性。

2、定义纯虚函数是为了实现一个接口,起到一个规范的作用,规范继承这个类的程序员必须实现这个函数。

3、虚函数的语法格式如下:

virtual 返回值类型 函数名 (函数参数) = 0;

4、纯虚函数没有函数体,只有函数声明,在虚函数声明的结尾加上=0,表明此函数为纯虚函数。

最后的=0并不表示函数返回值为0,它只起形式上的作用,告诉编译系统“这是纯虚函数”。

= 0;告诉编译器在vtable中为函数保留一个位置,但在这个特定位置不放地址

#include <iostream>

using namespace std;

//游戏中所有英雄的基类
class Hero
{
public:
    virtual void huicheng() = 0;//纯虚函数 英雄的回城功能
};

class Libai:public Hero
{
public:
    void huicheng() //重写
    {
        cout << "Libai::huicheng" << endl;
    }
};

class Caocao:public Hero
{
public:
    void huicheng()
    {
        cout << "Caocao::huicheng" << endl;
    }
};

int main()
{
    Libai h1;
    Caocao h2;

    h1.huicheng();
    h2.huicheng();

    Hero *p;
    p = &h1;
    p->huicheng();

    p = &h2;
    p->huicheng();

    return 0;
}

5.8.2 抽象基类

1、包含纯虚函数的类称为抽象类(Abstract Class)。之所以说它抽象,是因为它无法实例化,也就是无法创建对象。原因很明显,纯虚函数没有函数体,不是完整的函数,无法调用,也无法为其分配内存空间。

2、抽象类通常是作为基类,让派生类去实现纯虚函数。派生类必须实现纯虚函数才能被实例化。

#include <iostream>

using namespace std;

//游戏中所有英雄的基类
//抽象类:规定了某一类事物的特征
class Hero
{
public:
    virtual void huicheng() = 0;//纯虚函数 英雄的回城功能
    virtual void attack() = 0;
};

//假设这是对Libai类进行声明 (libai.h)
class Libai:public Hero
{
public:
    void huicheng(); //对派生类中重写基类中的纯虚函数的声明
    void attack();
};

//对Libai类的实现 (libai.cpp)
void Libai::huicheng() {
     cout << "Libai::huicheng" << endl;
}

void Libai::attack() {

}

class Caocao:public Hero
{
public:
    void huicheng()
    {
        cout << "Caocao::huicheng" << endl;
    }
};

int main()
{
#if 0
    Libai h1;
    Caocao h2;

    h1.huicheng();
    h2.huicheng();

    Hero *p;
    p = &h1;
    p->huicheng();

    p = &h2;
    p->huicheng();

  //Hero h; //抽象类不能实例化对象
#endif
    return 0;
}

抽象类实例:设计一个Shape类可以计算各种形状的面积、周长

#include <iostream>
using namespace std;
typedef unsigned int u32_t;

class Shape
{
public:
    virtual double getPermiter() = 0;
    virtual double getArea() = 0;
};

class Trangle : public Shape
{
public:
    Trangle():_a(10), _b(10), _c(10)
    {}
    Trangle(u32_t a, u32_t b, u32_t c):_a(a), _b(b), _c(c)
    {}

    double getPermiter()
    {return double(_a + _b + _c);}

    double getArea()
    {return 10000;}

private:
    u32_t _a, _b, _c;
    u32_t _permiter;
    double _area;
};

class Circle : public Shape
{
private:
    double _r;
    double _permiter;
    double _area;
    static double pi;

public:
    Circle():_r(10)
    {}
    double getPermiter();
    double getArea();
};

double Circle::pi = 3.14;
double Circle::getPermiter() {
    return 2 * pi * _r;
}

double Circle::getArea() {
    return pi * _r * _r;
}

int main() {
    Trangle x;
    cout << x.getPermiter() << endl;

    Circle x2;
    cout << x2.getPermiter() << endl;
    cout << x2.getArea() << endl;

    Shape *shape[2];
    shape[0] = &x;
    shape[1] = &x2;

    for (int i = 0; i < 2; i++)
    {
        cout << shape[i]->getArea() << endl;
        cout << shape[i]->getPermiter() << endl;
    }
    return 0;
}

5.9 虚析构函数

虚析构函数是为了解决基类的指针指向派生类对象,通过基类的指针删除派生类对象

#include <iostream>

using namespace std;

class Base
{
	public:
		virtual void func()
		{
			cout << "Base func"  << endl;
		}
		virtual ~Base()
		{
			cout << "~Base()" << endl;
		}
};

class Derived:public Base
{
	public:
		void func()
		{
			cout << "Derived func"  << endl;
                    cout << "访问基类的虚函数: ";
                    Base::func();
		}

		~Derived()
		{
			cout << "~Derived" << endl; 
		}
};

int main(void)
{
	Base *p = new Derived;
	delete p;
	return 0;
}

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

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

相关文章

51单片机定时器/计数器

目录 1、定时器/计数器0/1介绍 1.1 定时器介绍 1.2 单片机定时/计数器原理 2、定时器/计数器0和1的相关寄存器 2.1 定时器/计数器控制寄存器TCON 2.2 定时器/计数器工作模式寄存器TMOD 2.3 定时器/计数器工作模式 2.3.1 模式0(13位定时器/计数器) 2.3.2 模式1(16位定…

CSDN如何输入公式

方法分三步&#xff1a; 1&#xff09;预先设置MathType的复制剪切选项 2&#xff09;将MathType已经编写好的公式复制到CSDN 3&#xff09;把复制的公式文本&#xff0c;首尾的“\[”和“\]”符号替换成“$$”和“$$” 1&#xff09;预先设置MathType的复制剪切选项 2&#x…

海量数据存储组件Hbase

hdfs hbase NoSQL数据库 支持海量数据的增删改查 基于Rowkey查询效率特别高 kudu 介于hdfs和hbase之间 hbase依赖hadoopzookeeper&#xff0c;同时整合框架phoenix(擅长读写),hive&#xff08;分析数据&#xff09; k&#xff0c;v 储存结构 稀疏的&#xff08;为空的不存…

Qt实现思维导图锦集

序号简述文章导航1思维导图树形结构、不重叠且均匀分布、支持折叠和展开核心树2菜单按钮风格、菜单提示风格、侧滑菜单、侧滑功能窗口UI设计3支持JPEG、PNG、XML、JSON、PDF、SVG格式文件数据导入导出4支持撤销回撤功能、显示节点操作流程、点击可跳转历史撤销回撤5思维导图横向…

哈工大计算机网络课程网络安全基本原理详解之:消息完整性与数字签名

哈工大计算机网络课程网络安全基本原理详解之&#xff1a;消息完整性与数字签名 这一小节&#xff0c;我们继续介绍网络完全中的另一个重要内容&#xff0c;就是消息完整性&#xff0c;也为后面的数字签名打下基础。 报文完整性 首先来看一下什么是报文完整性。 报文完整性…

C++模拟实现反向迭代器

1.代码实现 1.有了解正向迭代器的应该知道&#xff0c;比如list的正向迭代器其实本质是一个类&#xff0c;而有些人想模拟实现反向迭代器&#xff0c;依旧想再创建一个类&#xff0c;但是库里面想要的是&#xff0c;你给我一个迭代器&#xff0c;我就能给你反馈一个反向迭代器…

信号槽中的函数重载

信号槽中的函数重载 QT4的方式QT5的方式函数指针重载函数QT5信号函数重载解决方案 总结 QT4的方式 Qt4中声明槽函数必须要使用 slots 关键字, 不能省略。 信号函数&#xff1a; 槽函数&#xff1a; mainwondow: cpp文件&#xff1a; #include "mainwindow.h"…

C/C++多线程操作

文章目录 多线程C创建线程join 和detachthis_thread线程操作锁lock_guardunique_lock 条件变量 condition_variablewaitwaitfor C语言线程创建线程同步 参考 多线程 传统的C&#xff08;C11标准之前&#xff09;中并没有引入线程这个概念&#xff0c;在C11出来之前&#xff0c…

C语言:反转一个单链表

Lei宝啊&#xff1a;个人主页 题目&#xff1a; 描述&#xff1a; 给你单链表的头节点 head &#xff0c;请你反转链表&#xff0c;并返回反转后的链表。 接口&#xff1a; struct ListNode* reverseList(struct ListNode* head){} 示例&#xff1a; 输入&#xff1a; head [1…

一起学算法(冒泡排序篇)

1.概念 冒泡排序&#xff08;Bubble Sort&#xff09;又称泡式排序&#xff0c;是一种简单的排序算法 核心思想&#xff1a;它重复地走访过要排列的次数&#xff0c;一次比较两个元素&#xff0c;如果它们的顺序错误就把它们交换过来&#xff0c;走访数列的工作是重复地进行交…

【Datawhale夏令营】任务三学习笔记

任务一笔记回顾 任务二笔记回顾 目录 一&#xff1a;竞赛上分流程 1.1问题建模1.2数据分析 1.3数据清洗1.4特征工程 1.5模型训练与验证 二&#xff1a;任务总结与心得 一&#xff1a;竞赛上分流程 问题建模——>数据分析 ——>数据清洗——>特征工程——>模型…

ERROR: AddressSanitizer: heap-buffer-overflow on address 0x602000000

leetcode上的一道题&#xff0c;当 s “a” 这样的单字符的时候&#xff0c;使用memset会出错&#xff0c;ERROR: AddressSanitizer: heap-buffer-overflow on address 0x602000000。 有些奇怪&#xff0c; 记录一下&#xff0c;双字符以及以上的时候不会报错&#xff0c;

Spring注解开发,bean的作用范围及生命周期、Spring注解开发依赖注入

&#x1f40c;个人主页&#xff1a; &#x1f40c; 叶落闲庭 &#x1f4a8;我的专栏&#xff1a;&#x1f4a8; c语言 数据结构 javaweb 石可破也&#xff0c;而不可夺坚&#xff1b;丹可磨也&#xff0c;而不可夺赤。 Spring注解开发 一、注解开发定义Bean二、纯注解开发Bean三…

MacOS Sonoma 14.0 (23A5301g) Beta4 带 OC 0.9.3 and FirPE 双分区原版黑苹果镜像

7月26日苹果发布macOS Sonoma Beta 4预览版更新&#xff0c;但部分用户在升级后遇到了各种问题。这一测试版目前还处于开发阶段&#xff0c;因此出现各种问题并不意外。 一、镜像下载&#xff1a; 1.微信公众号&#xff1a;MacOS Sonoma 14.0 (23A5301g) Beta4 带 OC 引导双分…

视频爬虫:解析m3u8文件 python m3u8库,m3u8文件中.ts视频流的解密下载

一、引用的库 这里需要引用的库是&#xff1a;from Crypto.Cipher import AES 有坑哈&#xff0c;python3.0之后直接安装crypto你会发现不管怎么着都会报错。 经过查找资料找到了原因&#xff0c;原来是20年之后crypto已经被pycryptohome替换掉啦&#xff0c; 如果之前安装过…

暴力破解(DVWA和pikachu)

目录 前言暴力破解模式一.pikachu靶场1.基于表单的暴力破解2. 验证码绕过2. token防爆破 二. DVWA1.low,Medium2.High3. Impossible 前言 渗透测试中暴力破解方法解释&#xff1a;通过尝试所有可能的字符组合来猜测密码的方法。 这种攻击方法需要大量计算资源和时间&#xff0…

人类文明进入下个纪元奇点:UFO听证会-恒温超导发现-GPT大模型

今年以来&#xff0c;科技领域出圈的事件频繁发生&#xff0c;每一个事件都意味着一个领域的重大突破的可能。这些事件是UFO听证会、恒温超导LK99的论文、GPT类大模型的广泛应用&#xff0c;我常将这些事件串在一起思考&#xff0c;细思极恐&#xff0c;一种”火鸡与农场主“的…

Vue(待续)

概念 一套用于构建用户界面的渐进式JavaScript框架 Vue可以自底向上逐层的应用&#xff1a; 简单应用:只需一个轻量小巧的核心库。 复杂应用:可以引入各式各样的Vue插件。 1.采用组件化模式&#xff0c;提高代码复用率、且让代码更好维护。 2.声明式编码&#xff0c;让编码人员…

使用Appuploader工具将IPA上传到App Store的最新流程和步骤

​ 苹果官方提供的工具xcode上架ipa非常复杂麻烦。用appuploader 可以在 mac 和windows 上制作管理 证书 &#xff0c;无需钥匙串工具 条件&#xff1a;1.以Windows为例&#xff0c;创建app打包ios需要的证书和描述文件 2.准备好一个苹果开发者账号&#xff08;如果没有到苹果…

C++ | unordered_map与unordered_set的用法指南

目录 前言 一、unordered_set 1、简介 2、构造相关函数 3、容量相关函数 4、修改与查找相关接口 5、迭代器 二、unordered_map 1、简介 2、构造相关函数 3、容量相关接口 4、迭代器、查找与修改相关接口 5、方括号接口 三、红黑树系列与哈希系列对比 前言 unorde…