C++11中的右值引用以及移动构造等

news2025/1/18 6:42:33

目录

一、右值引用

1.左值引用和右值引用

2.左值引用与右值引用比较

3.右值引用使用场景和意义

1️⃣ 传返回值

2️⃣ STL中的应用

4.完美转发

模板中的&& 万能引用(引用折叠)

二、 新的类功能

1.默认成员函数

2.类成员变量初始化

3.强制生成默认函数的关键字default:

4.禁止生成默认函数的关键字delete:

5.继承和多态中的final与override关键字


一、右值引用

1.左值引用和右值引用

传统的C++语法中就有引用的语法,而C++11中新增了的右值引用语法特性,所以从现在开始我们之前学习的引用就叫做左值引用。无论左值引用还是右值引用,都是给对象取别名

什么是左值?什么是左值引用?

左值是一个表示数据的表达式(如变量名或解引用的指针),我们可以获取它的地址+可以对它赋值,左值可以出现赋值符号的左边,右值不能出现在赋值符号左边。定义时const修饰符后的左值,不能给他赋值,但是可以取它的地址。左值引用就是给左值的引用,给左值取别名。

什么是右值?什么是右值引用?

右值也是一个表示数据的表达式,如:字面常量、表达式返回值,函数返回值(这个不能是左值引用返回)等等,右值可以出现在赋值符号的右边,但是不能出现出现在赋值符号的左边,右值不能取地址。右值引用就是对右值的引用,给右值取别名。

int main()
{
    // 以下的p、b、c、*p都是左值
    // 左值:可以取地址
    int* p = new int(0);
    int b = 1;
    const int c = b;
    *p = 10;
    string s("111111");
    s[0];
    cout << &c << endl;
    cout << &s[0] << endl;

//    // 右值:不能取地址
//    double x = 1.1, y = 2.2;
//    // 以下几个都是常见的右值,常量临时对象,匿名对象
//    10;
//    x + y;
//    fmin(x, y);
//    string("11111");
//
//    cout << &10 << endl;
//    cout << &(x+y) << endl;
//    cout << &(fmin(x, y)) << endl;
//    cout << &string("11111") << endl;

    return 0;
}

需要注意的是右值是不能取地址的,但是给右值取别名后,会导致右值被存储到特定位置,且可以取到该位置的地址,也就是说例如:不能取字面量10的地址,但是rr1引用后,可以对rr1取地址,也可以修改rr1。如果不想rr1被修改,可以用const int&& rr1 去引用,是不是感觉很神奇,这个了解一下实际中右值引用的使用场景并不在于此,这个特性也不重要。

int main()
{
    double x = 1.1, y = 2.2;
    int&& rr1 = 10;
    const double&& rr2 = x + y;
    
    rr1 = 20;
    rr2 = 5.5; // 报错
    
    return 0;
}

注:无论是左值引用还是右值引用,在字面上不占空间,在汇编、底层上是开空间的

2.左值引用与右值引用比较

左值引用总结:

✸ 左值引用只能引用左值,不能引用右值。

✸ 但是const左值引用既可引用左值,也可引用右值

int main(){
    // 左值引用只能引用左值,不能引用右值
    int a = 10;
    int& ra1 = a;// ra为a的别名
    //int& ra2 = 10; 编译失败,因为10是右值
    
    // const左值引用既可引用左值,也可引用右值。
    const int& ra3 = 10;
    const int& ra4 = a;
    return 0;
}

右值引用总结:

✸ 右值引用只能右值,不能引用左值。

✸ 但是右值引用可以move以后的左值。

✸ move的本质是强制类型转换

int main()
{
    // 右值引用只能右值,不能引用左值。
    int&& r1 = 10;
    
    // error C2440: “初始化”: 无法从“int”转换为“int &&”
    // message : 无法将左值绑定到右值引用
    int a = 10;
//    int&& r2 = a;
    
    // 右值引用可以引用move以后的左值
    int&& r3 = std::move(a);
    
    
    string s("111111");//s是左值 “111111”是右值
    string&& rrx4 = move(s);//左值move后给右值引用
    // move的本质是强制类型转换
    string&& rrx5 = (string&&)s;//与上句等同
    return 0;
}

 

3.右值引用使用场景和意义

1️⃣ 传返回值

前面我们可以看到左值引用既可以引用左值和又可以引用右值,那为什么C++11还要提出右值引用呢?是不是化蛇添足呢?下面我们来看看左值引用的短板,右值引用是如何补齐这个短板的!

//构造函数
string(const char* str = "")
        :_size(strlen(str))
        , _capacity(_size)
    {
        cout << "string(char* str)" << endl;
        _str = new char[_capacity + 1];
        strcpy(_str, str);
    }
// 赋值重载
    string& operator=(const string& s)
    {
        cout << "string& operator=(const string& s) -- 深拷贝" << endl;
        if (this != &s)
        {
            _str[0] = '\0';
            _size = 0;

            reserve(s._capacity);//开空间,深拷贝
            for (auto ch : s)
            {
                push_back(ch);
            }
        }

        return *this;
    }

bit::string to_string(int value)
{
    bool flag = true;
    if (value < 0)
    {
        flag = false;
        value = 0 - value;
    }
    bit::string str;//临时对象

    while (value > 0)
    {
        int x = value % 10;
        value /= 10;
        str += ('0' + x);
    }

    if (flag == false)
    {
        str += '-';
    }

    std::reverse(str.begin(), str.end());

    return str;
}
}

int main()
{
    bit::string s1;
    s1 = bit::to_string(1234);

    return 0;
}

输出结果:
string(char* str)		//构造 bit::string s1;
string(char* str)		//构造内部str对象
string& operator=(const string& s) -- 深拷贝  //赋值重载

 对于返回值,编译器会生成临时对象,再对临时对象进行拷贝构造,对于临时对象进行拷贝构造这不就浪费了吗?为了更进一步优化,C++11新引入移动构造,以及移动赋值

    // 移动构造
    // 临时创建的对象,不能取地址,用完就要消亡
    // 深拷贝的类,移动构造才有意义
    string(string&& s)
    {
        cout << "string(string&& s) -- 移动拷贝" << endl;
        swap(s);
    }
    // 移动赋值
    string& operator=(string&& s)
    {
        cout << "string& operator=(string&& s) -- 移动拷贝" << endl;

        swap(s);//转移资源
        return *this;
    }
int main()
{
    bit::string s1;
    s1 = bit::to_string(1234);

    return 0;
}
输出结果:
string(char* str)
string(char* str)
string& operator=(string&& s) -- 移动拷贝

我们先看编译器做的几种优化

情况一:string 没有移动构造,只有拷贝构造也就是C++11之前的

优化前:

✸ 传值返回会生成临时对象

✸ 传值返回会导致至少1次拷贝构造(如果是一些旧一点的编译器可能是两次拷贝构造)。

优化后

✸ 编译器认为两次拷贝太浪费了,直接优化临时对象,发生一次拷贝


情况二:string有拷贝构造,也有移动构造C++11之后 

优化前: ✸ str是一个左值(可以取地址)

✸ 临时对象是一个右值

✸ 把一次拷贝构造,变为移动构造

优化后:

✸ 没有临时对象

✸ 理应来说str是一个左值,应进行拷贝构造,但是这里把str隐式转换为右值,直接进行移动构造

✸ 相当于传值返回,没有拷贝了

更激进的优化

 

int main()
{
    bit::string s1 = bit::to_string(1234);
    return 0;
}
输出结果:
string(char* str) 直接构造,没有移动

ps:这两种情况是不一样的

//编译器再厉害也不行,这种情况还是要靠移动赋值与移动构造
int main()
{
    bit::string s1;
    s1 = bit::to_string(1234);

    return 0;
}

int main()
{
    bit::string s1 = bit::to_string(1234);
 //   s1 = bit::to_string(1234);

    return 0;
}

我们之前使用的一直都是C++11/C++11以后的编译器,所以我们以前也并未提起相关知识,而是直接用,但我们现在学习了之后,发现不影响使用,该随便用就随便用,本次也只是介绍如果没有C++11,而是以前的,我们对于这种情况就很麻烦处理

2️⃣ STL中的应用

对于库中list 

 

int main()
{
    cout << "-------1-------" << endl;
    list<bit::string> lt;
    
    cout << "-------2-------" << endl;
    bit::string s1("111111111111111111111");

    cout << "-------3-------" << endl;
    lt.push_back(s1);

    cout << "-------4-------" << endl;
    lt.push_back(bit::string("22222222222222222222222222222"));

    cout << "-------5-------" << endl;
    lt.push_back("3333333333333333333333333333");

    cout << "-------6-------" << endl;
    lt.push_back(move(s1));
    
    return 0;
}

输出结果:
-------1-------
-------2-------
string(char* str)
-------3-------
string(const string& s) -- 深拷贝
-------4-------
string(char* str)
string(string&& s) -- 移动拷贝
-------5-------
string(char* str)
string(string&& s) -- 移动拷贝
-------6-------
string(string&& s) -- 移动拷贝

对于我们之前手搓链表我们加上移动版本push_back 和 insert

template<class T>
struct ListNode
{
    ListNode<T>* _next;
    ListNode<T>* _prev;
    
    T _data;
    
    ListNode(const T& data = T())
    :_next(nullptr)
    ,_prev(nullptr)
    ,_data(data)
    {}
    
    ListNode(T&& data)
    :_next(nullptr)
    , _prev(nullptr)
    , _data(move(data))
    {}
};

		void push_back(const T& x)
    {
        insert(end(), x);
    }
    
    void push_back(T&& x)
    {
        insert(end(), move(x));
    }

    iterator insert(iterator pos, const T& x)
    {
        Node* cur = pos._node;
        Node* newnode = new Node(x);
        Node* prev = cur->_prev;
        
        // prev  newnode  cur
        prev->_next = newnode;
        newnode->_prev = prev;
        newnode->_next = cur;
        cur->_prev = newnode;
        
        return iterator(newnode);
    }
    
    iterator insert(iterator pos, T&& x)
    {
        Node* cur = pos._node;
        Node* newnode = new Node(move(x));
        Node* prev = cur->_prev;
        
        // prev  newnode  cur
        prev->_next = newnode;
        newnode->_prev = prev;
        newnode->_next = cur;
        cur->_prev = newnode;
        
        return iterator(newnode);
    }

 

int main()
{
    cout << "-------1-------" << endl;
    bit::list<bit::string> lt;
    //这里会由于创造哨兵位头节点 发生构造+拷贝构造
    
    cout << "-------2-------" << endl;
    //左值
    bit::string s1("111111111111111111111");

    cout << "-------3-------" << endl;
    //左值
    lt.push_back(s1);

    cout << "-------4-------" << endl;
    //右值
    lt.push_back(bit::string("22222222222222222222222222222"));

    cout << "-------5-------" << endl;
    //右值 产生临时对象
    lt.push_back("3333333333333333333333333333");
    
    //右值
    cout << "-------6-------" << endl;
    lt.push_back(move(s1));
    
    cout << "-------7-------" << endl;
    bit::string&& r1 = bit::string("22222222222222222222222222222");
    // r1(右值引用本身)的属性是左值还是右值?-> 左值

    return 0;
}

输出结果:
-------1-------
string(char* str)
string(const string& s) -- 深拷贝
-------2-------
string(char* str)
-------3-------
string(const string& s) -- 深拷贝
-------4-------
string(char* str)
string(string&& s) -- 移动拷贝
-------5-------
string(char* str)
string(string&& s) -- 移动拷贝
-------6-------
string(string&& s) -- 移动拷贝
-------7-------
string(char* str)

 

注意:由于move以后本身的属性还是左值,所以只要使用到右值,我们都需要写一个右值版本,少一个都不行,比如这里的节点构造、push_back、insert

编译器是能过自动匹配右值还是左值

void func(const bit::string& s)
{
    cout << "void func(bit::string& s)" << endl;
}

void func(bit::string&& s)
{
    cout << "void func(bit::string&& s)" << endl;
}


int main()
{
    cout << "左值" << endl;
    bit::string s1("1111111");
    func(s1);

    cout << "右值" << endl;
    func((bit::string&&)s1);

    cout << "右值" << endl;
    func(bit::string("1111111"));
    
    //强转成左值
   // func((bit::string&)bit::string("1111111"));

    return 0;
}
输出结果
左值
string(char* str)
void func(bit::string& s)
右值
void func(bit::string&& s)
右值
string(char* str)
void func(bit::string&& s)

4.完美转发

比如我们写一个Fun()函数,我们既要写它的左值版本,又要写它的右值版本,这是不是不太合适?所以C++引入完美转发

模板中的&& 万能引用(引用折叠)

### 

```cpp
//函数模版中
template <class T>
void Func(T&& x)
{
    
}
```

对于void Func(T&& x)中的T&& x,如果我们进行传值

✸ 如果传的是一个右值类型,那么T&& x就是一个右值引用

✸ 如果传的是左值类型,那么T&& x就是一个左值引用

使用关键字forward可以保留原先数据属性,传什么,就保留什么属性,解决了需要传左值右值不需要写太多版本的问题

底层就是,编译器很辛苦,生成各种类型的函数(函数重载),本质也是类型转换

完美转发是一个函数模版

void Fun(int& x) { cout << "左值引用" << endl; }
void Fun(const int& x) { cout << "const 左值引用" << endl; }
void Fun(int&& x) { cout << "右值引用" << endl; }
void Fun(const int&& x) { cout << "const 右值引用" << endl; }

// 引用
// 传左值->左值引用
// 传右值->右值引用
template<typename T>
void PerfectForward(T&& t)
{
    // 模版实例化是左值引用,保持属性直接传参给Fun
    // 模版实例化是右值引用,右值引用属性会退化成左值,转换成右值属性再传参给Fun
    // 使用std::forward完美转发参数到另一个函数
    Fun(forward<T>(t));
}

int main()
{
    PerfectForward(10);           // 右值

    int a;
    PerfectForward(a);            // 左值

    PerfectForward(std::move(a)); // 右值

    const int b = 8;
    PerfectForward(b);              // const 左值
    PerfectForward(std::move(b)); // const 右值

    return 0;
}
输出结果:
右值引用
左值引用
右值引用
const 左值引用
const 右值引用

二、 新的类功能

1.默认成员函数

原来C++类中,有6个默认成员函数:

✸ 构造函数

✸ 析构函数

✸ 拷贝构造函数

✸ 拷贝赋值重载

✸ 取地址重载

✸ const 取地址重载

C++11 新增了两个:移动构造函数和移动赋值运算符重载。

针对移动构造函数和移动赋值运算符重载有一些需要注意的点如下:

✸ 如果你没有自己实现移动构造函数,且没有实现析构函数 、拷贝构造、拷贝赋值重载中的任意一个。那么编译器会自动生成一个默认移动构造。默认生成的移动构造函数,对于内置类型成员会执行逐成员按字节拷贝,自定义类型成员,则需要看这个成员是否实现移动构造,如果实现了就调用移动构造,没有实现就调用拷贝构造。

✸ 如果你没有自己实现移动赋值重载函数,且没有实现析构函数 、拷贝构造、拷贝赋值重载中的任意一个,那么编译器会自动生成一个默认移动赋值。默认生成的移动赋值,对于内置类型成员会执行逐成员按字节拷贝,自定义类型成员,则需要看这个成员是否实现移动赋值,如果实现了就调用移动赋值,没有实现就调用拷贝赋值。(默认移动赋值跟上面移动构造完全类似)

✸ 如果你提供了移动构造或者移动赋值,编译器不会自动提供拷贝构造和拷贝赋值。

class Person
{
public:
    Person(const char* name = "111111111111", int age = 0)
        :_name(name)
        , _age(age)
    {}

  	这里我们没有写析构函数 、拷贝构造、拷贝赋值重载
    // 自动生成拷贝构造和移动构造
	
private:
    bit::string _name;
    int _age;
};


int main()
{
    Person s1;
    Person s2 = s1;
    Person s3 = std::move(s1);
  
  移动赋值测试,原理同移动构造
    //Person s4;
   // s4 = std::move(s2);
    return 0;
}

 

如果我们加上析构函数

    ~Person()
    {}

输出结果:
string(char* str)
string(const string& s) -- 深拷贝
string(const string& s) -- 深拷贝

 

注:一个函数如果需要显示写析构函数,说明有资源需要释放

✸ 说明需要显示写拷贝构造和赋值重载

✸ 说明需要显示写移动构造和移动赋值

自动生成的移动构造,对于Date这样的类,其实没有什么意义,因为它和拷贝构造的功能是一样的

自动生成的移动构造,对于Person这样的类是很有意义的,因为Person是右值时,他内部的string也是右值,string就可以走移动构造,提高效率了

2.类成员变量初始化

C++11允许在类定义时给成员变量初始缺省值,默认生成构造函数会使用这些缺省值初始化,这个我们在类和对象默认就讲了,这里就不再细讲了。

3.强制生成默认函数的关键字default:

C++11可以让你更好的控制要使用的默认函数。假设你要使用某个默认的函数,但是因为一些原因这个函数没有默认生成。比如:我们提供了拷贝构造,就不会生成移动构造了,那么我们可以使用default关键字显示指定移动构造生成,也就是算显示生成,因此我们需要额外注意一下默认成员函数的条件

    Person(Person&& p) = default;
    Person& operator=(Person && p) = default;
    Person(const Person& p) = default;
    Person& operator=(const Person& p) = default;

4.禁止生成默认函数的关键字delete:

如果能想要限制某些默认函数的生成,在C++98中,是该函数设置成private,并且只声明补丁已,这样只要其他人想要调用就会报错。在C++11中更简单,只需在该函数声明加上=delete即可,该语法指示编译器不生成对应函数的默认版本,称=delete修饰的函数为删除函数。

class Person
{
public:
    Person(const char* name = "111111111111", int age = 0)
        :_name(name)
        , _age(age)
    {}

    // 只声明不实现,声明为私有
    // C++98的做法
//private:
//    Person(const Person& p);
//    Person& operator=(const Person & p);

    
    //C++11的做法
    Person(const Person& p) = delete;
    Person& operator=(const Person& p) = delete;
private:
    bit::string _name;
    int _age;
};

5.继承和多态中的final与override关键字

这个我们在继承和多态章节已经进行了详细讲解这里就不再细讲,需要的话去复习继承和多台章节吧。

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

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

相关文章

【找到字符串中所有字母异位词】python刷题记录

R2-滑动窗口篇 滑动窗口哈希表 和之前那道一样 http://t.csdnimg.cn/dpIbt class Solution:def findAnagrams(self, s: str, p: str) -> List[int]:#返回字典记录了每个字符出现的次数counter1collections.Counter(p)#滑动窗口记录counter2即可ret[]num2len(s)num1len(p…

VIM基础配置

1. CTAGS配置 下载 上传虚拟机&#xff0c;解压&#xff0c;进入目录 tar -xzvf ctags-5.8.tar.gz cd ctags-5.8/编译 ./configure sudo make sudo make install查看是否安装成功 ctags --version打印如下 2. 使用Vundle 下载 git clone https://github.com/VundleVim/Vund…

如何将WordPress文章中的外链图片批量导入到本地

在使用采集软件进行内容创作时&#xff0c;很多文章中的图片都是远程链接&#xff0c;这不仅会导致前端加载速度慢&#xff0c;还会在微信小程序和抖音小程序中添加各种域名&#xff0c;造成管理上的麻烦。特别是遇到没有备案的外链&#xff0c;更是让人头疼。因此&#xff0c;…

2024下《系统架构设计师》案例简答题,刷这些就够了!

2024年软考下半年已经越来越近了&#xff0c;不知道今年备考架构的同学们准备得怎么样了呢&#xff1f; 简答题一直是架构拿分的重点区域&#xff0c;对于许多考生来说&#xff0c;也往往是最具挑战性的部分。今天我就把那些重要的案例简答题类型整理汇总给大家&#xff0c;希望…

C++选择题带答案

1&#xff0e;下列关于定义一个指向double型变量的指针&#xff0c;正确的是&#xff08; B &#xff09;。 A&#xff0e;int a(5)&#xff1b;double *pda&#xff1b; B&#xff0e;double d(2.5)&#xff0c;*pd&d&#xff1b; C&#xff0e;dou…

uniapp时间戳转时间

时间戳转时间 utils页面 function timestampToTime(time) { const date new Date(time); const year date.getFullYear(); const month String(date.getMonth() 1).padStart(2, 0); // 月份从0开始&#xff0c;所以要加1&#xff0c;并补齐0 const day String(date…

系统架构师考点--系统架构设计(下)

大家好。今天总结一下系统架构设计的最后一部分知识点。 一、软件系统的质量属性 软件系统的质量属性 软件系统的质量属性可分为开发期质量属性和运行期质量属性2个部分。 1、开发期质量属性主要指在软件开发阶段所关注的质量属性&#xff0c;主要包含6个方面&#xff1a; …

联想电脑怎么重装系统_联想电脑U盘重装win10详细图文教程

联想电脑怎么重装系统&#xff1f;在当今科技发展迅猛的时代&#xff0c;联想电脑已经成为了人们生活中不可或缺的一部分。然而&#xff0c;随着时间的推移&#xff0c;我们可能会遇到一些问题&#xff0c;例如系统崩溃或者需要更换操作系统。这时&#xff0c;使用U盘来重新安装…

57页PPT智慧水利数字孪生综合解决方案

实现“全局一盘棋”的智慧水利综合管理&#xff0c;关键在于整合水利大数据、数字孪生与人工智能技术&#xff0c;通过“一图、一库、一平台”的构建&#xff0c;为水利工作提供全面、科学、智能的管理和决策支持。以下是对这一目标的详细解读和实现路径&#xff1a; 知识星球…

C/C++进阶 (8)哈希表(STL)

个人主页&#xff1a;仍有未知等待探索-CSDN博客 专题分栏&#xff1a;C 本文着重于模拟实现哈希表&#xff0c;并非是哈希表的使用。 实现的哈希表的底层用的是线性探测法&#xff0c;并非是哈希桶。 目录 一、标准库中的哈希表 1、unordered_map 2、unordered_set 二、模…

【计算机毕设论文】基于SpringBoot线上学习平台的设计与实现

&#x1f497;博主介绍&#xff1a;✌全平台粉丝5W,高级大厂开发程序员&#x1f603;&#xff0c;博客之星、掘金/知乎/华为云/阿里云等平台优质作者。 【源码获取】关注并且私信我 感兴趣的可以先收藏起来&#xff0c;同学门有不懂的毕设选题&#xff0c;项目以及论文编写等相…

「数组」实现动态数组的功能(C++)

概述 动态数组&#xff0c;顾名思议即可变长度的数组。数组这种数据结构的实现是在栈空间或堆空间申请一段连续的可操作区域。 实现可变长度的动态数组结构&#xff0c;应该有以下操作&#xff1a;申请一段足够长的空间&#xff0c;如果数据的存入导致空间已满&#xff0c;则…

CentOS7安装最新版vim;vim自动补齐配置

想练习一会vim&#xff0c;结果发现敲代码没有空号自动补齐和缩进很难受&#xff0c;所以想配置一下。 配置vim&#xff1a; 可以通过 vim ~/.vimrc 来给 vim 加启动的设定&#xff08;比如set nu&#xff0c;这样就会在每次启动的时候都加上行号。当然过程中可以在底行模式输…

基于SpringBoot+Vue的学生考勤管理系统(带1w+文档)

基于SpringBootVue的学生考勤管理系统(带1w文档) 系统为了数据库结构的灵活性选择MySQL来设计&#xff0c;而java技术&#xff0c;B/S架构则保证了较高的平台适应性。本文主要介绍了系统开发背景&#xff0c;需要完成的功能与开发过程&#xff0c;说明系统设计重点与设计思想。…

如何在linux系统中用conda安装R环境及R包

一、miniconda3的安装不再赘述 二、安装R环境 1. 提前准备好conda的R单独环境 conda env list #查看已有环境 查看R的最新版本&#xff1a;r-project ##创建环境和激活环境 conda create -n R4.4.1 conda activate R4.4.1 备注&#xff1a;激活环境Linux&#xff0c;OS X…

企业如何保证公司内网安全

1. 加强网络安全防护 部署防火墙和入侵检测系统&#xff1a;作为内网安全的第一道防线&#xff0c;防火墙和入侵检测系统能够有效阻止外部攻击和恶意软件的入侵。 数据加密&#xff1a;采用先进的加密技术保护敏感数据&#xff0c;确保数据在传输和存储过程中的安全性。 访问…

【Android面试八股文】荣耀面试算法题: 输出一个给定的字符串的最长回文子序列及其长度!

文章目录 一、真题链接二、如何解决2.1算法思路2.2 算法步骤2.3 Java算法实现 一、真题链接 还好我以前刷过这道题&#xff0c; 其实题目就是LeetCode的 516.最长回文子序列&#xff0c; 地址&#xff1a;https://leetcode.cn/problems/longest-palindromic-subsequence/des…

哈希表专题

题解之前&#xff1a; 1.有关unordered_map的count功能&#xff1a;查询key&#xff01; Leetcode 1.两数之和 解题思路&#xff1a; class Solution { public:vector<int> twoSum(vector<int>& nums, int target) {vector<int> res;// key:具体的数值(便…

AI需求海量涌现,Scaleway如何用Spectrum-X 网络从容应对?

“没有好网络&#xff0c;别玩AIGC。” 如今&#xff0c;随着AI需求的大量涌现&#xff0c;越来越多用户意识到网络在AI集群中的重要性。一个超大规模、超高带宽、超强可靠的网络&#xff0c;可以为AI训练提供强有力支撑&#xff0c;从而节约训练成本、缩短训练时间&#xff0…

VBA快速对比数据行

实例需求&#xff1a;对于存在多行数据&#xff08;示例中为双行&#xff09;的项目&#xff0c;对比同一个项目的每列数据&#xff0c;高亮显示数据不同的单元格。 示例代码如下。 Function GetDiff(ByRef rng1 As Range, ByRef rng2 As Range) As RangeDim i As LongFor i …