Это делается через специализацию шаблона. Вот простая демонстрация.
#include <iostream>
#include <type_traits>
template<typename T>
struct foo
{
foo( T* d ) : m_data{ d } { }
~foo() {
if ( m_data )
std::cout << "delete called." << std::endl;
}
private:
T* m_data {};
};
template<typename T>
struct foo<T[]>
{
foo( T* d ) : m_data { d } { }
~foo() {
if ( m_data )
delete[] m_data;
std::cout << "delete[] called." << std::endl;
}
private:
T* m_data {};
};
int main()
{
foo<char> inst_1 { new char };
foo<char []> inst_2 { new char[ 100 ] { 'a' } };
}
онлайн-пример
Для экземпляров T[]
, экземпляров struct foo<T[]>
, поэтому полиморфизм во время компиляции выполняется через специализацию шаблона и smart_ptr
обрабатывает освобождение как массивов, так и нормальных типов с помощью этой функции.