gpt4 book ai didi

c++ - 错误 : redefinition of function template (or C2995)

转载 作者:行者123 更新时间:2023-11-30 02:19:58 26 4
gpt4 key购买 nike

GCC 和 MSVC 在矩阵与模板相乘的方法上抛出一个编译错误,而 Clang 编译成功没有任何错误,我不明白为什么。有人可以提出什么问题吗?

我认为实例化 operator*(matrix, matrix) 存在问题。

注意:(错误:函数模板重定义,C2995:函数模板已经定义)[声明和定义在一个.hpp中,有header guards]

我在 gcc 7.2.0 及更高版本和 MSVC 19.14.26428.1 及更高版本上对此进行了测试。我也使用 C++17 标准。

有问题的方法:

template <std::size_t Rows_lhs, std::size_t Columns_lhs,
std::size_t Rows_rhs, std::size_t Columns_rhs>
friend constexpr matrix<value_type, Rows_lhs, Columns_rhs> operator*(
const matrix<value_type, Rows_lhs, Columns_lhs>& lhs,
const matrix<value_type, Rows_rhs, Columns_rhs>& rhs)
{
static_assert(Columns_lhs == Rows_rhs, "Incorrect matrix for product!");

matrix<value_type, Rows_lhs, Columns_rhs> result{};
container<value_type, Rows_rhs> thatColumn{};

for (size_type j = 0; j < Columns_rhs; ++j)
{
for (size_type k = 0; k < Rows_rhs; ++k)
{
thatColumn.at(k) = rhs(k, j);
}

for (size_type i = 0; i < Rows_lhs; ++i)
{
const auto thisRow = lhs(i);
value_type summand{};
for (size_type k = 0; k < Rows_rhs; ++k)
{
summand += thisRow.at(k) * thatColumn.at(k);
}
result(i, j) = summand;
}
}
return result;
}

完整代码:

#include <iostream>
#include <string>
#include <cmath>
#include <iterator>
#include <algorithm>
#include <array>
#include <initializer_list>


namespace vv
{
template <class Type = double, std::size_t Rows = 1, std::size_t Columns = 1>
class matrix
{
public:
using value_type = Type;
using size_type = std::size_t;

template <class Type = value_type, std::size_t N = Rows>
using container = std::array<Type, N>;
using row_container = container<value_type, Columns>;
using row_container_reference = container<value_type, Columns>&;
using const_row_container_reference = const container<value_type, Columns>&;

using reference = value_type&;
using const_reference = const value_type&;

using std_matrix = matrix<value_type, Rows, Columns>;


static constexpr value_type EPS = static_cast<value_type>(1e-10);

static_assert(std::is_arithmetic_v<value_type>, "Matrix elements type has to be arithmetic!");
static_assert(Rows > 0 && Columns > 0, "Incorrect size parameters!");

constexpr matrix() = default;

constexpr matrix(const std::initializer_list<value_type> list)
: _data()
{
size_type row_counter = 0;
size_type col_counter = 0;
for (const auto elem : list)
{
_data.at(row_counter).at(col_counter) = elem;
++col_counter;
if (row_counter == Rows && col_counter == Columns)
{
break;
}
if (col_counter == Columns)
{
col_counter = 0;
++row_counter;
}
}
}

std::string get_dimension() const noexcept
{
return std::to_string(Rows) + std::string("x")
+ std::to_string(Columns);
}

constexpr const_reference operator()(const size_type i, const size_type j) const
{
return _data.at(i).at(j);
}

constexpr reference operator()(const size_type i, const size_type j)
{
return _data.at(i).at(j);
}

constexpr const_row_container_reference& operator()(const size_type i) const
{
return _data.at(i);
}

constexpr row_container_reference& operator()(const size_type i)
{
return _data.at(i);
}

constexpr std_matrix& operator*=(const value_type num) noexcept
{
for (auto& row : _data)
{
for (auto& elem : row)
{
elem *= num;
}
}
return *this;
}

friend constexpr std_matrix operator*(const std_matrix& mat, const value_type num) noexcept
{
std_matrix temp(mat);
return (temp *= num);
}

friend constexpr std_matrix operator*(const value_type num, const std_matrix& mat) noexcept
{
return (mat * num);
}

friend std::ostream& operator<<(std::ostream& os, const std_matrix& mat)
{
os << "[" << mat.get_dimension() << "]\n";
for (const auto& row : mat._data)
{
std::copy(std::begin(row), std::end(row),
std::ostream_iterator<value_type>(os, " "));
os << '\n';
}
return os;
}

// PROBLEMS HERE BEGIN
template <std::size_t Rows_lhs, std::size_t Columns_lhs,
std::size_t Rows_rhs, std::size_t Columns_rhs>
friend constexpr matrix<value_type, Rows_lhs, Columns_rhs> operator*(
const matrix<value_type, Rows_lhs, Columns_lhs>& lhs,
const matrix<value_type, Rows_rhs, Columns_rhs>& rhs)
{
static_assert(Columns_lhs == Rows_rhs, "Incorrect matrix for product!");

matrix<value_type, Rows_lhs, Columns_rhs> result{};
container<value_type, Rows_rhs> thatColumn{};

for (size_type j = 0; j < Columns_rhs; ++j)
{
for (size_type k = 0; k < Rows_rhs; ++k)
{
thatColumn.at(k) = rhs(k, j);
}

for (size_type i = 0; i < Rows_lhs; ++i)
{
const auto thisRow = lhs(i);
value_type summand{};
for (size_type k = 0; k < Rows_rhs; ++k)
{
summand += thisRow.at(k) * thatColumn.at(k);
}
result(i, j) = summand;
}
}
return result;
}
// END

private:
container<container<value_type, Columns>, Rows> _data;
};

} // namespace vv


int main()
{
constexpr vv::matrix<double, 2, 1> a{ 1.0, 2.0 };
constexpr vv::matrix<double, 1, 2> b{ 4.0, 3.0 };

constexpr auto c = a * b; // This code occurs error.
std::cout << c;
return 0;
}

最佳答案

问题是 operator*()在模板内部定义matrix类。

因此,当您定义 matrix对象,说 matrix<double, 1, 2> , 这个函数被定义了;当您定义另一个具有相同类型和不同维度的对象时,比如说 matrix<double, 2, 1> , 重新定义了完全相同的模板函数。

在我看来,没有什么需要该函数是 friendmatrix所以——建议——在类内部删除,在类外部重写,如下

template <class Type, std::size_t N>
using container = std::array<Type, N>;

using size_type = std::size_t;

template <typename value_type, std::size_t Rows_lhs, std::size_t Columns_lhs,
std::size_t Rows_rhs, std::size_t Columns_rhs>
constexpr matrix<value_type, Rows_lhs, Columns_rhs> operator*(
const matrix<value_type, Rows_lhs, Columns_lhs>& lhs,
const matrix<value_type, Rows_rhs, Columns_rhs>& rhs)
{
static_assert(Columns_lhs == Rows_rhs, "Incorrect matrix for product!");

matrix<value_type, Rows_lhs, Columns_rhs> result{};
container<value_type, Rows_rhs> thatColumn{};

for (size_type j = 0; j < Columns_rhs; ++j)
{
for (size_type k = 0; k < Rows_rhs; ++k)
{
thatColumn.at(k) = rhs(k, j);
}

for (size_type i = 0; i < Rows_lhs; ++i)
{
const auto thisRow = lhs(i);
value_type summand{};
for (size_type k = 0; k < Rows_rhs; ++k)
{
summand += thisRow.at(k) * thatColumn.at(k);
}
result(i, j) = summand;
}
}
return result;
}

如果你真的想要,你可以维护它friend但只在类中声明它 matrix

template <typename value_type, std::size_t Rows_lhs, 
std::size_t Columns_lhs, std::size_t Rows_rhs,
std::size_t Columns_rhs>
friend constexpr matrix<value_type, Rows_lhs, Columns_rhs> operator*(
const matrix<value_type, Rows_lhs, Columns_lhs>& lhs,
const matrix<value_type, Rows_rhs, Columns_rhs>& rhs);

奖金(题外话)建议:不需要定义四个矩阵维度并用static_assert()强加。第二个 ( Columns_lhs ) 和第三个 ( Rows_rsh ) 是相等的。

您可以将它们统一在一个模板参数中( midDim ,在以下示例中)

template <typename value_type, std::size_t Rows_lhs, std::size_t midDim,
std::size_t Columns_rhs>
constexpr matrix<value_type, Rows_lhs, Columns_rhs> operator*(
const matrix<value_type, Rows_lhs, midDim>& lhs,
const matrix<value_type, midDim, Columns_rhs>& rhs)
{
matrix<value_type, Rows_lhs, Columns_rhs> result{};
container<value_type, midDim> thatColumn{};

for (size_type j = 0; j < Columns_rhs; ++j)
{
for (size_type k = 0; k < midDim; ++k)
{
thatColumn.at(k) = rhs(k, j);
}

for (size_type i = 0; i < Rows_lhs; ++i)
{
const auto thisRow = lhs(i);
value_type summand{};
for (size_type k = 0; k < midDim; ++k)
{
summand += thisRow.at(k) * thatColumn.at(k);
}
result(i, j) = summand;
}
}
return result;
}

关于c++ - 错误 : redefinition of function template (or C2995),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50193468/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com