C++初阶之模板深化讲解

news2025/1/9 1:04:40

在这里插入图片描述

模板深化讲解

  • 非类型模板
  • 模板的特化
    • 1.函数模板特化
    • 2.类模板特化
  • 模板分离编译
    • 1.什么是分离编译
    • 2.模板的分离编译
  • 模板总结

非类型模板

在这里插入图片描述

非类型模板(Non-Type Template)是 C++ 中的一种模板形式,它允许你在模板中传递除了类型以外的其他值,比如整数、枚举、指针等。这些参数可以在编译时被解析,用于生成模板的实例化版本。

非类型模板参数(Non-Type Template Parameter)是在模板声明中,作为参数的一部分,而不是类型的一部分。它们可以是常量表达式,例如整数常量、枚举、指针、引用等。非类型模板参数的值必须在编译时是已知的,因为它们用于生成模板的特定实例。

模板参数分类类型形参与非类型形参。

类型形参:出现在模板参数列表中,跟在class或者typename之类的参数类型名称。
非类型形参:就是用一个常量作为类(函数)模板的一个参数,在类(函数)模板中可将该参数当成常量来使用。
使用非类型模板参数的一个典型示例是实现一个固定大小的数组容器,其中数组的大小在编译时由模板参数指定。

例如,以下是一个使用非类型模板参数的示例:

template <typename T, int Size>
class FixedSizeArray {
private:
    T data[Size];
public:
    // ...
};

// 实例化一个大小为 10 的 FixedSizeArray,存储 int 类型
FixedSizeArray<int, 10> intArray;

在这个示例中,int Size 就是一个非类型模板参数,它确定了 FixedSizeArray 的数组大小。在编译时,intArray 的大小将被确定为 10,因为在实例化时传递了 10Size 参数。

注意:

  1. 浮点数、类对象以及字符串是不允许作为非类型模板参数的
  2. 非类型的模板参数必须在编译期就能确认结果。

模板的特化

模板的特化Template Specialization)是 C++ 中一种允许你为特定类型或特定条件提供定制化实现的机制。模板特化允许你在针对某些特定类型或条件的情况下,提供一个专门的模板实现,以覆盖通用模板的行为。

在一般的模板中,你可能会为不同类型的数据提供一个通用的模板实现。然而,在某些情况下,特定类型可能需要特殊处理或定制化的行为。这时,你可以通过模板特化来为这些特定类型提供特殊实现。

模板特化分为两种类型:

类模板特化(Class Template Specialization):针对特定类型的特殊实现。
函数模板特化(Function Template Specialization):针对特定类型的特殊实现。

在某些情况下,如果没有使用模板的特化,可能会导致编译器选择错误的实现,从而产生不明确的错误或意外的行为。一个常见的情况是函数模板的参数匹配不明确,编译器无法确定应该使用哪个模板实现。这种情况下,特化可以提供明确的信息,帮助编译器正确选择实现。

以下是一个示例,展示在没有模板特化的情况下可能会发生的问题:

#include <iostream>

template <typename T>
void printType(T value) {
    std::cout << "Generic template" << std::endl;
}

// 模板特化版本,针对 int 类型
template <>
void printType<int>(int value) {
    std::cout << "Specialized template for int" << std::endl;
}

int main() {
    int num = 5;
    printType(num);  // 编译错误,不明确的调用

    return 0;
}

在上面的示例中,没有模板特化的情况下,编译器无法确定应该调用哪个模板实现,因为 printType 函数可以接受任何类型的参数。这会导致编译错误,因为编译器无法根据上下文来选择正确的实现。

通过为 printType<int> 提供特化版本,我们可以明确告诉编译器在处理 int 类型参数时应该使用哪个实现。这种情况下,特化可以解决不明确调用的问题,确保编译器选择正确的实现。

1.函数模板特化

函数模板的特化步骤:

  1. 必须要先有一个基础的函数模板
  2. 关键字template后面接一对空的尖括号<>
  3. 函数名后跟一对尖括号,尖括号中指定需要特化的类型
  4. 函数形参表: 必须要和模板函数的基础参数类型完全相同,如果不同编译器可能会报一些奇怪的错误。

函数模板特化的示例:

#include <iostream>

// 通用的函数模板
template <typename T>
T add(T a, T b) {
    return a + b;
}

// 函数模板的特化版本,针对 int 类型
template <>
int add(int a, int b) {
    std::cout << "Specialized version for int" << std::endl;
    return a + b + 10;
}

int main() {
    int result1 = add(5, 3);    // 调用通用版本
    std::cout << "Result 1: " << result1 << std::endl;  // 输出: 8

    int result2 = add<int>(5, 3);  // 调用特化版本
    std::cout << "Result 2: " << result2 << std::endl;  // 输出: Specialized version for int
                                                        // Result 2: 18

    return 0;
}

在上面的示例中,add 是一个函数模板,用于计算两个数的和。通过为 add<int> 提供特化版本,我们覆盖了默认的行为,并在特定类型下进行了定制化的操作。在 main 函数中,我们展示了如何调用通用版本和特化版本,以及它们的输出结果。

注意一般情况下如果函数模板遇到不能处理或者处理有误的类型,为了实现简单通常都是将该函数直接给出。

2.类模板特化

在 C++ 中,类模板特化是指为特定类型或条件提供定制化的类模板实现。有两种类型的类模板特化:全特化(Full Specialization)和偏特化(Partial Specialization)。

全特化Full Specialization):全特化是针对特定类型的完整特化。当你为某个特定类型提供了一个完整的特化版本,它将会覆盖通用的类模板定义。全特化的语法是在模板名后面添加 <> 并指定特化的类型。

示例:

// 通用的类模板
template <typename T>
class MyTemplate {
public:
    void print() {
        std::cout << "Generic Template" << std::endl;
    }
};

// 类模板的全特化版本,针对 int 类型
template <>
class MyTemplate<int> {
public:
    void print() {
        std::cout << "Specialized Template for int" << std::endl;
    }
};

偏特化Partial Specialization):偏特化是在某些条件下对模板参数进行特化,通常用于更细粒度的定制。偏特化允许你为某些特定情况提供特化版本,而不是为每个类型提供完整的特化版本。偏特化的语法是在模板名后面添加 <>,并在尖括号中指定要特化的参数。

示例:

// 通用的类模板
template <typename T, typename U>
class Pair {
public:
    Pair(T first, U second) : first_(first), second_(second) {}
    void print() {
        std::cout << "Generic Pair: " << first_ << ", " << second_ << std::endl;
    }
private:
    T first_;
    U second_;
};

// 类模板的偏特化版本,针对两个相同类型的参数
template <typename T>
class Pair<T, T> {
public:
    Pair(T first, T second) : first_(first), second_(second) {}
    void print() {
        std::cout << "Specialized Pair for same type: " << first_ << ", " << second_ << std::endl;
    }
private:
    T first_;
    T second_;
};

在上述示例中,全特化针对整型类型提供了特化版本,而偏特化则针对两个相同类型的参数提供了特化版本。

总结:

全特化:为特定类型提供完整特化版本。
偏特化:为特定情况或条件提供特化版本。

通过使用模板特化,你可以在需要定制化行为的情况下,为类模板提供精确的实现,增强了模板的灵活性和适用性。

类模板允许我们为不同的数据类型提供相同的代码结构,以适应多种类型的需求。然而,在某些情况下,编译器可能无法推断出如何比较不同类型的对象。这就是为什么在下面的示例中,如果不使用类模板的特化,编译器可能无法正确排序日期类的原因。

在默认情况下,std::sort 函数使用元素的<运算符来比较元素的大小。对于基本类型(如整数)或支持 < 运算符的类型,这是没有问题的。但对于自定义类型(如日期类),编译器无法知道如何执行比较。

通过使用类模板的特化,我们为不同类型的日期类提供了明确的比较逻辑,即 operator< 运算符的重载。这告诉编译器如何在特定情况下比较日期对象,使得 std::sort 函数能够正确工作。

特化类模板允许我们在需要的地方为特定类型提供定制的实现,从而解决编译器无法推断的问题,确保程序能够正确运行。

下面是一个关于日期类的示例,使用类模板特化来比较大小,并通过 std::sort 函数对日期进行排序
在示例中,我们定义了一个 Date 类模板,然后为 Date 类提供了特化版本,用于实现日期的比较操作:

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

// 通用的类模板
template <typename T>
class Date {
public:
    Date(T year, T month, T day) : year_(year), month_(month), day_(day) {}

    // 比较运算符
    bool operator<(const Date& other) const {
        if (year_ != other.year_) return year_ < other.year_;
        if (month_ != other.month_) return month_ < other.month_;
        return day_ < other.day_;
    }

    void print() {
        std::cout << year_ << "-" << month_ << "-" << day_ << std::endl;
    }

private:
    T year_;
    T month_;
    T day_;
};

// 类模板的特化版本,用于 int 类型
template <>
class Date<int> {
public:
    Date(int year, int month, int day) : year_(year), month_(month), day_(day) {}

    bool operator<(const Date<int>& other) const {
        if (year_ != other.year_) return year_ < other.year_;
        if (month_ != other.month_) return month_ < other.month_;
        return day_ < other.day_;
    }

    void print() {
        std::cout << year_ << "-" << month_ << "-" << day_ << std::endl;
    }

private:
    int year_;
    int month_;
    int day_;
};

int main() {
    std::vector<Date<int>> dates = {
        {2022, 8, 15},
        {2021, 12, 25},
        {2022, 1, 1},
        {2022, 3, 20}
    };

    std::cout << "Before sorting:" << std::endl;
    for (const auto& date : dates) {
        date.print();
    }

    std::sort(dates.begin(), dates.end());

    std::cout << "After sorting:" << std::endl;
    for (const auto& date : dates) {
        date.print();
    }

    return 0;
}

在上面的示例中,我们首先定义了一个通用的 Date 类模板,用于存储年、月、日信息。然后,我们为 Date<int> 提供了特化版本,用于实现基于 int 类型的日期比较操作。

main 函数中,我们创建了一个存储日期的 std::vector,然后使用 std::sort 函数对日期进行排序。由于我们为 Date<int> 提供了特化版本,它能够正确比较日期并进行排序。最终,我们分别输出排序前后的日期顺序。

模板分离编译

1.什么是分离编译

分离编译Separate Compilation)是一种软件开发技术,它将一个大型程序分割成多个小的源代码文件,每个文件包含一个或多个相关的函数、类或变量的定义和实现。这些源代码文件可以在不同的编译单元中进行编译,然后在链接阶段将它们组合成一个可执行的程序。

分离编译的主要目标是提高代码的可维护性、编译速度和资源利用率。以下是分离编译的一些优点:

模块化开发:将程序分割成多个模块,每个模块负责一个特定的功能。这样,不同的开发人员可以独立地处理不同的模块,从而提高开发效率。
代码复用:在不同的项目中,可以重新使用已经编写并通过测试的模块,从而减少开发时间和资源。
编译速度:只有修改的模块需要重新编译,其他未修改的模块可以保持不变。这可以显著加快编译时间。
资源利用率:只有需要的模块会被编译,减少了不必要的编译和内存占用。

分离编译的基本流程如下:

编写模块:将程序分割成多个模块,并编写每个模块的定义和实现。
编译模块:分别编译每个模块的源代码,生成目标文件(例如 .obj.o 文件)。
链接模块:将所有目标文件链接在一起,解决引用关系,生成最终的可执行文件。

在分离编译中,头文件(.h 文件)通常用于存放函数和类的声明,而源文件(.cpp 文件)包含函数和类的实现。这种划分可以帮助编译器了解每个模块的接口和实现,从而在不同模块之间建立正确的链接。

分离编译是现代软件开发的重要实践,它有助于组织复杂的项目、提高开发效率,并降低维护成本。

2.模板的分离编译

在C++中,模板的分离编译是指将模板的声明和实现分开放置在不同的文件中。模板的声明通常放在头文件(.h.hpp 文件),而模板的实现则放在源文件(.cpp 文件)中。

模板的分离编译是为了解决链接时的模板实例化问题。C++编译器需要在使用模板的地方对模板进行实例化,但编译器在编译一个源文件时只能看到当前源文件的内容,无法知道其他源文件中模板的实现细节。因此,如果模板的声明和实现都放在头文件中,并且被多个源文件引用,会导致模板被多次实例化,最终在链接阶段会出现多个相同的实例化,引发重定义错误。

模板的分离编译定义的一般做法是:

将模板的声明放在头文件中(例如 .h 文件)。
将模板的实现放在源文件中(例如 .cpp 文件),并在源文件末尾包含模板的实现。

这样做的好处是,每个源文件只会对模板进行一次实例化,避免了重定义问题。
然而,模板的分离定义也可能引发一些问题,例如:

编译错误难以定位:如果模板的实现出现错误,编译器可能无法在使用模板的地方给出详细的错误信息,导致调试困难。
代码维护困难:模板的实现分散在多个源文件中,可能导致代码维护变得更加复杂,需要确保每个源文件的模板实现保持一致。
可读性下降:模板的实现被分离到源文件中,可能会降低代码的可读性和可理解性。

为了避免模板分离定义带来的问题,一些编程实践推荐将模板的声明和实现都放在头文件中,以便在使用模板的地方能够看到完整的实现细节。如果模板的实现较为复杂,可以通过将模板特化的方式来解决分离定义的问题。

举一个C++中的模板的分离定义的例子

这个示例演示了如果模板的声明和实现被分离到不同的文件中,可能会导致重定义错误。

假设我们有以下两个文件:

Stack.h(头文件,包含模板的声明):

#ifndef STACK_H
#define STACK_H

template <typename T>
class Stack {
public:
    Stack();
    void push(const T& value);
    T pop();

private:
    T elements[10];
    int top;
};

#include "Stack.cpp"

#endif

Stack.cpp(源文件,包含模板的实现):

#ifndef STACK_CPP
#define STACK_CPP

template <typename T>
Stack<T>::Stack() : top(-1) {}

template <typename T>
void Stack<T>::push(const T& value) {
    elements[++top] = value;
}

template <typename T>
T Stack<T>::pop() {
    return elements[top--];
}

#endif

这个示例中,我们尝试在头文件中包含了源文件 Stack.cpp。这可能会导致以下问题:

重定义错误:当多个源文件包含同一个头文件时,每个源文件都会包含 Stack.cpp 中的模板实现,从而在链接时引发重定义错误。
解决方法是,将模板的声明和实现都放在头文件中,或者使用模板的显式实例化(explicit instantiation)来避免重定义错误
显式实例化是一种告诉编译器在特定类型上进行模板实例化的方式,可以在源文件中使用以下语法来避免问题:

template class Stack<int>;
template class Stack<double>;
// 等等

这样可以确保模板只会在特定类型上进行一次实例化,避免了重定义错误。

虽然显式实例化可以解决模板的分离定义问题,但它也有一些潜在的弊端:

  1. 维护困难:如果代码中使用了多种不同的类型进行实例化,就需要在源文件中为每种类型都显式实例化一次。这可能会导致代码冗余,增加维护的难度,尤其在模板被广泛使用的大型项目中。
  2. 可读性降低显式实例化的语法相对较为繁琐,可能会降低代码的可读性。程序员需要了解这种特殊的语法并在源文件中进行适当的显式实例化。
  3. 影响编译时间:显式实例化会导致编译器在编译时生成模板的具体实例化代码,从而增加了编译时间。特别是在模板被大量使用的情况下,编译时间可能会显著增加
  4. 局限性显式实例化只适用于那些已知要在特定类型上进行实例化的模板。对于一些可能会在不同类型上使用的通用模板,需要为每个可能的类型都显式实例化,这可能不太实际。

综上所述,虽然显式实例化是解决模板分离定义问题的一种方法,但它可能会引入一些不便之处和潜在的问题。因此,一些项目中更倾向于将模板的声明和实现都放在头文件中,以避免这些问题。

模板总结

优点:

  1. 通用性和重用性模板允许编写通用的代码,适用于多种数据类型和数据结构。这种通用性促进了代码的重用,减少了编写重复代码的需求。
  2. 类型安全: 模板可以在编译时进行类型检查,确保在模板实例化时使用正确的数据类型。这有助于避免运行时的类型错误
  3. 性能优势模板生成的代码是在编译时根据实际类型生成的,因此没有函数调用的开销,可以在一定程度上提高性能。
  4. 泛型算法: C++标准库中的算法和容器都使用模板,使得开发人员能够方便地使用通用的排序、查找、遍历等算法
  5. 抽象和封装模板可以实现抽象数据类型,将数据结构和操作封装在一起,提供更高层次的抽象。
  6. 编译时错误检查模板的错误通常在编译时被检测到,使得开发人员能够及早发现和修复问题。

缺点:

  1. 编译时错误信息难以理解: 模板错误的编译器错误信息可能非常复杂,对于初学者来说可能难以理解。这可能增加了调试的难度
  2. 编译时间增加模板的使用可能导致编译时间增加,特别是在大型项目中。模板的实例化会在编译时生成多个版本的代码,可能导致编译器花费更多时间。
  3. 代码膨胀模板的实例化会导致生成多份相似的代码,可能增加可执行文件的大小。
  4. 可读性下降一些复杂的模板代码可能难以阅读和理解,尤其是涉及元编程技巧的情况。
  5. 维护困难: 当模板的实现被分离到不同的文件中,维护可能会变得困难,特别是涉及到显式实例化等情况。

综合考虑,模板是一个强大的工具,可以在很多情况下提供巨大的优势。然而,在使用模板时,开发人员需要权衡其优点和缺点,并根据具体情况做出合适的选择。

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

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

相关文章

BLE 学习小结

GAP 和 GATT https://www.youtube.com/watch?vyKJtnkEjPFI GAP: Generic Access Profile. 定义的是Scanner跟Advertiser的角色。负责连接相关的服务 (设备的搜寻&#xff0c;用来建立连接&#xff0c;连接的管理&#xff0c;等)。 GATT: Generic Attribute Profile. 定义的是…

Java实现DTLS之技术背景原理(一)

文章目录 前言一、DTLS是什么&#xff1f;二、RFC6347标准定义DTLS1.中文翻译 总结感谢 前言 需求&#xff1a;升级服务侧SDK&#xff0c;实现与灯控器之间DTLS加密通信&#xff0c;代替SM4国密。目前通信是采用UDP协议并实现SM4国密加密&#xff0c;为了提升产品竞争力需要实…

5,Lambda

Lambda Lambda https://blog.csdn.net/A1138474382/article/details/111149792 Lambda 捕获列表。在C 规范中也称为Lambda导入器&#xff0c; 捕获列表总是出现在Lambda函数的开始处。实际上&#xff0c;[]是Lambda引出符。编译器根据该引出符判断接下来的代码是否是Lambda函数…

【gogogo专栏】指针

go语言指针 为什么需要指针指针使用实例值传递地址传递多级指针 为什么需要指针 作为一个大学划水&#xff0c;毕业一直写java的程序员来说&#xff0c;多多少少对于指针有点陌生&#xff0c;由于近期需要转go&#xff0c;正好学到指针这里&#xff0c;就来探究下指针的使用场景…

springboot 数据库版本管理升级常用解决方案

目录 一、前言 1.1 单独执行初始化sql 1.2 程序自动执行 二、数据库版本升级管理问题 三、spring 框架sql自动管理机制 3.1 jdbcTemplate 方式 3.1.1 创建数据库 3.1.2 创建 springboot 工程 3.1.3 初始化sql脚本 3.1.4 核心配置类 3.1.5 执行sql初始化 3.2 配置文…

Baklib:企业Wiki 知识库管理有序更高效

什么是Baklib? Baklib是一种企业Wiki知识库管理工具&#xff0c;旨在帮助企业更好地管理和共享知识。它提供了一个集中存储和组织知识的平台&#xff0c;使团队成员可以轻松地查找和共享信息。Baklib具有直观的用户界面和强大的搜索功能&#xff0c;可以提高团队的工作效率和…

leetcode 516. 最长回文子序列(JAVA)题解

题目链接https://leetcode.cn/problems/longest-palindromic-subsequence/description/?utm_sourceLCUS&utm_mediumip_redirect&utm_campaigntransfer2china 目录 题目描述&#xff1a; 暴力递归&#xff1a; 动态规划&#xff1a; 题目描述&#xff1a; 给你一个…

第06天 静态代理和动态代理

✅作者简介&#xff1a;大家好&#xff0c;我是Leo&#xff0c;热爱Java后端开发者&#xff0c;一个想要与大家共同进步的男人&#x1f609;&#x1f609; &#x1f34e;个人主页&#xff1a;Leo的博客 &#x1f49e;当前专栏&#xff1a;每天一个知识点 ✨特色专栏&#xff1a…

运维监控学习笔记7

Zabbix的安装&#xff1a; 1、基础环境准备&#xff1a; 安装zabbix的yum源&#xff0c;阿里的yum源提供了zabbix3.0。 rpm -ivh http://mirrors.aliyun.com/zabbix/zabbix/3.0/rhel/7/x86_64/zabbix-release-3.0-1.el7.noarch.rpm 这个文件就是生成了一个zabbix.repo 2、安…

海量数据迁移,亚马逊云科技云数据库服务为大库治理提供新思路

1.背景 目前&#xff0c;文档型数据库由于灵活的schema和接近关系型数据库的访问特点&#xff0c;被广泛应用&#xff0c;尤其是游戏、互联网金融等行业的客户使用MongoDB构建了大量应用程序&#xff0c;比如游戏客户用来处理玩家的属性信息&#xff1b;又如股票APP用来存储与时…

ESP 系列的产品 ULP 协处理器的应用

参考文档&#xff1a; 《ESP32-S2 技术参考手册》 中 “1. 超低功耗协处理器 (ULP)” 章节《ESP32-S3 技术参考手册》 中 “2 超低功耗协处理器 (ULPFSM, ULPRISCV)” 章节《ESP32-C6 技术参考手册》 中 “3 低功耗处理器” 章节ULP 协处理器编程ULP RISC-V 协处理器编程Progr…

leetcode2024. 考试的最大困扰度(java)

考试的最大困扰度 leetcode2024. 考试的最大困扰度题目描述滑动窗口最大值 经典算法 leetcode2024. 考试的最大困扰度 难度 - 中等 原题链接 - 考试的最大困扰度 题目描述 一位老师正在出一场由 n 道判断题构成的考试&#xff0c;每道题的答案为 true &#xff08;用 ‘T’ 表示…

程序设计 堆

✅作者简介&#xff1a;人工智能专业本科在读&#xff0c;喜欢计算机与编程&#xff0c;写博客记录自己的学习历程。 &#x1f34e;个人主页&#xff1a;小嗷犬的个人主页 &#x1f34a;个人网站&#xff1a;小嗷犬的技术小站 &#x1f96d;个人信条&#xff1a;为天地立心&…

代码随想录算法训练营第三十四天 | 1005.K次取反后最大化的数组和,134. 加油站 ,135. 分发糖果

代码随想录算法训练营第三十四天 | 1005.K次取反后最大化的数组和&#xff0c;134. 加油站 &#xff0c;135. 分发糖果 1005.K次取反后最大化的数组和:eyes:题目总结:eyes: 134. 加油站暴力方法贪心算法&#xff08;方法一&#xff09;贪心算法&#xff08;方法二&#xff09;:…

Stable Diffusion+Temporal-kit 半虚半实应用

1.先下载temporal-kit,重启webui 2.下载好ffmpeg,配置好环境,下载Ebsynth 3.准备好你需要的视频,拖到预处理视频位置 4.填写参数,点解保存设置,然后并点击生成,会生成到目标文件夹的input位置 5.然后拉出input文件夹里面你想切换成处理的帧图片,然后填写prompt查看效…

分布式 - 服务器Nginx:一小时入门系列之HTTP反向代理

文章目录 1. 正向代理和反向代理2. 配置代理服务3. proxy_pass 命令解析4. 设置代理请求headers 1. 正向代理和反向代理 正向代理是客户端通过代理服务器访问互联网资源的方式。在这种情况下&#xff0c;客户端向代理服务器发送请求&#xff0c;代理服务器再向互联网上的服务器…

将会计转移到 cloud:远程访问软件的作用

在不断变化的金融和技术格局中&#xff0c;会计行业经历了重大演变。账簿和人工计算在金融界占据主导地位的日子已经一去不复返了。如今&#xff0c;随着企业寻求更高效、更准确、更具协作性的财务管理方式&#xff0c;数字化转变比以往任何时候都更加明显。我们正处于会计复兴…

C# Linq源码分析之Take (一)

概要 在.Net 6 中引入的Take的另一个重载方法&#xff0c;一个基于Range的重载方法。因为该方法中涉及了很多新的概念&#xff0c;所以在分析源码之前&#xff0c;先将这些概念搞清楚。 Take方法基本介绍 public static System.Collections.Generic.IEnumerable Take (this …

jeecgboot table 单元格横向和纵向合并

customRender: (value, row, index) > {const obj {children: &#xffe5;200200火力值,attrs: {}}if (index 0) {// obj.attrs.rowSpan this.dataSource.length // 合并数量 纵向合并obj.attrs.colSpan 3 // 横向合并}if (index > 1) {// obj.attrs.rowSpan 0 // …

2023上半年京东手机行业品牌销售排行榜(京东数据平台)

后疫情时代&#xff0c;不少行业都迎来消费复苏&#xff0c;我国智能手机市场在今年上半年也实现温和的复苏&#xff0c;手机市场的出货量回暖。 根据鲸参谋平台的数据显示&#xff0c;2023年上半年&#xff0c;京东平台上手机的销量为2830万&#xff0c;环比增长约4%&#xf…