第四篇 C++元模板编程以及模板类的具体实现

模板类

模板编程可以带来更加灵活、通用、高效和安全的代码,尤其在 C++ 中,模板编程被广泛应用于实现通用的数据结构、算法库以及提供高性能的代码

主要有以下优点:
  • 通用性和灵活性:模板编程使得代码可以编写成通用的形式,从而能够处理多种不同类型的数据。通过模板编程,可以编写出适用于不同数据类型的通用算法和数据结构,提高了代码的通用性和灵活性。
  • 性能优化:模板编程可以在编译期间进行很多工作,包括优化代码、内联函数、消除冗余代码等,从而提高程序的性能。通过模板特化和内联展开等技术,可以生成高效的代码。
  • 类型安全:使用模板可以在编译期对类型进行检查,从而避免一些运行时的类型错误。模板能够提供更好的类型安全性,使得代码更加健壮和可靠。
  • 代码复用:通过模板编程,可以编写出高度复用的代码,避免重复的劳动,并且可以确保所有实例化的代码都是经过严格测试和验证的。
  • 元编程:模板编程还可以用于进行元编程,即在编译期间生成代码。这样可以实现一些在运行期无法完成的任务,如计算、条件判断、类型处理等,从而提高程序的灵活性和性能

模板函数的实现

// 定义一个模板结构体 IsPointer,用于判断类型是否为指针类型
template <typename T>
struct IsPointer {
    static const bool value = false;
};

template <typename T>
struct IsPointer<T*> {
    static const bool value = true;
};

// 利用 IsPointer 类型结构体判断是否为指针类型,并输出结果
template <typename T>
void CheckIfPointer() {
    if (IsPointer<T>::value) {
        std::cout << "类型 " << typeid(T).name() << " 是指针类型\n";
    } else {
        std::cout << "类型 " << typeid(T).name() << " 不是指针类型\n";
    }
}


// 假设有一个需要管理的类
class MyClass {
public:
    void greet() {
        std::cout << "Hello from MyClass!" << std::endl;
    }
};

//测试代码
void test(){
	CheckIfPointer<int>();      // 检查 int 类型
    CheckIfPointer<float*>();   // 检查 float* 类型
    CheckIfPointer<std::string>();  // 检查 std::string 类型
}

智能指针模板类

是 C++ 中的一个重要概念,用于管理动态分配的内存,可以帮助避免内存泄漏和悬空指针等问题。下面是智能指针的一些好处:

  • 自动内存管理:智能指针可以自动管理内存的生命周期,在对象不再需要时自动释放内存,避免了手动调用 delete 或 free 的繁琐操作。
  • 避免内存泄漏:由于智能指针会在其作用域结束时自动释放内存,因此可以有效避免因忘记释放内存而导致的内存泄漏问题。
  • 避免悬空指针:智能指针会在对象销毁后将指针置为 nullptr,避免了悬空指针的问题,使得程序更加健壮。
  • 方便拷贝和赋值:智能指针通常实现了拷贝构造函数和赋值运算符重载,使得对指针的使用更加方便和安全。
  • 提高代码可维护性:使用智能指针可以减少手动管理内存的复杂性,降低出错的可能性,从而提高代码的可维护性和可读性。
  • 支持多线程:一些智能指针(如 std::shared_ptr)提供了引用计数的机制,可以在多线程环境下安全地共享对象所有权。

智能指针模板类的实现

//TODO: 实现一个智能指针模板类
//原理:利用类在构造时初始化,析构时销毁的特性,
//把一个类的内存管理委托到模板类上面,实现内存自动管理
template <class T>
class SmartPointer {
private:
    T* ptr;

public:
    SmartPointer(T* p) : ptr(p) {}

    ~SmartPointer() {
        delete ptr;
    }

    T& operator*() {
        return *ptr;
    }

    T* operator->() {
        return ptr;
    }
};

SmartPointer 是一个模板类,它接受一个模板参数 T,表示指针指向的类型。在类中包含一个私有指针 ptr,用于管理动态分配的内存。构造函数接受一个指向 T 类型对象的指针,并将其存储在 ptr 中。析构函数负责释放 ptr 指向的内存。此外,重载了 * 和 -> 运算符,以便通过智能指针访问指向的对象。

//TODO: 实现共享指针,在智能指针的基础上加上引用计数,
//通过适当增加和减少引用计数,可以确保正确地共享资源并在不再需要时释放资源
template <class T>
class SharedPointer {
private:
    T* ptr;
    int* refCount;

public:
    SharedPointer(T* p) : ptr(p), refCount(new int(1)) {}

    SharedPointer(const SharedPointer<T>& other) : ptr(other.ptr), refCount(other.refCount) {
        (*refCount)++;
    }

    ~SharedPointer() {
        (*refCount)--;
        std::cout << "~SharedPointer()" << std::endl;
        if (*refCount == 0) {
            delete ptr;
            delete refCount;
        }
    }

    SharedPointer<T>& operator=(const SharedPointer<T>& other) {
        if (this != &other) {
            (*refCount)--;
            if (*refCount == 0) {
                delete ptr;
                delete refCount;
            }

            ptr = other.ptr;
            refCount = other.refCount;
            (*refCount)++;
        }
        return *this;
    }

    T& operator*() {
        return *ptr;
    }

    T* operator->() {
        return ptr;
    }
};
//测试代码
// 假设有一个需要管理的类
class MyClass {
public:
    void greet() {
        std::cout << "Hello from MyClass!" << std::endl;
    }
};

typedef SmartPointer<MyClass> myClassPtr;
typedef SharedPointer<MyClass> myClassSharedPtr;

void test(){
	myClassPtr ptr = new MyClass();
    myClassSharedPtr  sharedPtr1 = new MyClass();
    ptr->greet();
    {
        myClassSharedPtr  sharedPtr2 = sharedPtr1;
    }
}

这里实现了一个简单的 SharedPtr 类模板,其中包含了引用计数 refCount。通过适当增加和减少引用计数,可以确保正确地共享资源并在不再需要时释放资源,确保资源在适当时机被释放。

单例模板类的实现

#include <iostream>

template <typename T>
class Singleton {
public:
    static T& getInstance() {
        static T instance;
        return instance;
    }
};

//测试代码
class MySingleton {
private:
    MySingleton() {} // Private constructor to prevent instantiation

public:
    void doSomething() {
        std::cout << "Doing something in MySingleton" << std::endl;
    }
};

void test() {
	MySingleton& singleton = Singleton<MySingleton>::getInstance();
    singleton.doSomething();
}

Singleton 类模板使用了模板元编程技术,通过静态成员函数 getInstance 返回单例对象的引用。这样可以确保只有一个实例被创建,并且在程序的整个生命周期中保持活动。

  • 23
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
鸡啄米的资料,大家可以上官网看。这是我整理的压缩包,大家可以离线看。(这个资源还要大家积分,实在是惭愧,只因积分不够用,抱歉,体谅!) 大家要是觉得不错,可以下载我整理得另一套资源:VS2010+MFC编程入门,也是鸡啄米的,写得不错,通俗易懂! --------------------------完整目录如下------------------- 第一部分:C++编程概述 鸡啄米:C++编程入门系列之前言 鸡啄米:C++编程入门系列之一(进制数) 鸡啄米:C++编程入门系列之二(原码、反码与补码) 第二部分:C++简单程序设计 鸡啄米:C++编程入门系列之三(VS2010的使用介绍) 鸡啄米:C++编程入门系列之四(数据类型) 鸡啄米:C++编程入门系列之五(运算符和表达式) 鸡啄米:C++编程入门系列之六(算法的基本控制结构之选择结构) 鸡啄米:C++编程入门系列之七(算法的基本控制结构之循环结构) 鸡啄米:C++编程入门系列之八(自定义数据类型) 第三部分:函数 鸡啄米:C++编程入门系列之九(函数的定义与调用) 鸡啄米:C++编程入门系列之十(函数的参数传递与内联函数) 鸡啄米:C++编程入门系列之十一(重载函数与函数模板) 第四部分:类与对象 鸡啄米:C++编程入门系列之十二(类与对象:面向对象设计的基本思 想和特点) 鸡啄米:C++编程入门系列之十三(类与对象:类的声明、成员的访问 控制和对象) 鸡啄米:C++编程入门系列之十四(类与对象:构造函数和析构函数) 鸡啄米:C++编程入门系列之十五(类与对象:类的组合) 鸡啄米:C++编程入门系列之十六(类与对象:类模板) 鸡啄米:C++编程入门系列之十七(类与对象:UML简介) 第五部分:C++程序设计必知 鸡啄米:C++编程入门系列之十八(C++程序设计必知:作用域和可见 性) 鸡啄米:C++编程入门系列之十九(C++程序设计必知:生存期) 鸡啄米:C++编程入门系列之二十(C++程序设计必知:数据和函数) 鸡啄米:C++编程入门系列之二十一(C++程序设计必知:类的静态成 员) 鸡啄米:C++编程入门系列之二十二(C++程序设计必知:友) 鸡啄米:C++编程入门系列之二十三(C++程序设计必知:常引用、常对 象和对象的常成员) 鸡啄米:C++编程入门系列之二十四(C++程序设计必知:多文件结构和 编译预处理命令) 第六部分:数组、指针和字符串 鸡啄米:C++编程入门系列之二十五(数组、指针和字符串:数组的声 明和使用) 鸡啄米:C++编程入门系列之二十六(数组、指针和字符串:数组的存 储与初始化、对象数组、数组作为函数参数) 鸡啄米:C++编程入门系列之二十七(数组、指针和字符串:指针变量 的声明、地址相关运算--“*”和“&”) 鸡啄米:C++编程入门系列之二十八(数组、指针和字符串:指针的赋 值和指针运算) 鸡啄米:C++编程入门系列之二十九(数组、指针和字符串:指向数组 素的指针和指针数组) 鸡啄米:C++编程入门系列之三十(数组、指针和字符串:指针用作函 数参数、指针型函数和函数指针) 鸡啄米:C++编程入门系列之三十一(数组、指针和字符串:对象指 针) 鸡啄米:C++编程入门系列之三十二(数组、指针和字符串:动态内存 分配和释放) 鸡啄米:C++编程入门系列之三十三(数组、指针和字符串:用字符数 组存放和处理字符串) 鸡啄米:C++编程入门系列之三十四(数组、指针和字符串:string 类) 第七部分:继承与派生 鸡啄米:C++编程入门系列之三十五(继承与派生:概念介绍与派生类 的声明) 鸡啄米:C++编程入门系列之三十六(继承与派生:派生类从基类继承 的过程) 鸡啄米:C++编程入门系列之三十七(继承与派生:派生类对基类成员 的访问控制之公有继承) 鸡啄米:C++编程入门系列之三十八(继承与派生:派生类对基类成员 的访问控制之保护继承与私有继承) 鸡啄米:C++编程入门系列之三十九(继承与派生:派生类的构造函 数) 鸡啄米:C++编程入门系列之四十(继承与派生:派生类的析构函数) 鸡啄米:C++编程入门系列之四十一(继承与派生:作用域分辨符) 鸡啄米:C++编程入门系列之四十二(继承与派生:虚基类及其派生类 的构造函数) 鸡啄米:C++编程入门系列之四十三(继承与派生:赋值兼容规则) 第八部分:多态性 鸡啄米:C++编程入门系列之四十四(多态性:多态的概念和类型) 鸡啄米:C++编程入门系列之四十五(多态性:运算符重载的概念和规 则) 鸡啄米:C++编程入门系列之四十六(多态性:运算符重载为类的成员 函数) 鸡啄米:C++编程入门系列之四十七(多态性:运算符重载为类的友 函数) 鸡啄米:C++编程入门系列之四十八(多态性:虚函数) 鸡啄米:C++编程入门系列之四十九(多态性:纯虚函数和抽象类) 第九部分:异常处理 鸡啄米:C++编程入门系列之五十(异常处理) 鸡啄米:C++编程入门系列之目录和总结

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值