【问题标题】:c++ template: 'Instantiation of variable required here, but no definition is available'c ++模板:'此处需要变量的实例化,但没有可用的定义'
【发布时间】:2017-04-16 17:15:22
【问题描述】:

我为T 类型的矩阵创建了一个矩阵模板类。我收到一个链接器错误和一个警告,上面写着:Instantiation of variable 'ndmatrix<unsigned int>::Subscriptable::_data' required here, but no definition is available

模板在单独的标头中定义:

ndmatrix.h:
#ifndef ndmatrix_h
#define ndmatrix_h
#include <vector>
#include <functional>
#include <algorithm>

//Generic matrix class to hold the table levels of type T
template<class T>
class ndmatrix{

public:

    size_t R; //rows
    size_t C; //cols

    //Hack to use [][] operator on std::vector
    class Subscriptable{
    public:

        //Default constructor
        Subscriptable(){};

        Subscriptable(int idx, size_t R, size_t C) : _idx(idx), _R(R), _C(C) {
            for (int i = 0; i<_C*_R; i++) {
                _data.push_back(0);
            }
        };

        Subscriptable setIdx(int i){
            this->_idx = i;
            return *this;
        }

        //This is [] operator for ndmatrix::Subscriptable
        T operator[](int index) {
            return _data[_C *_idx + index];
        }

        auto addElement(T e, int i, int j){
            return _data.insert( _data.begin() + _C*i + j, e);
        };

        auto data() { return _data; };
    private:
        size_t _R; //rows
        size_t _C; //cols
        int _idx;
        static std::vector<T> _data; <== Warning: Instantiation of variable 'ndmatrix<unsigned int>::Subscriptable::_data' required here, but no definition is available.
    };

    Subscriptable _s;

    //This is [] operator for ndmatrix
    Subscriptable operator[](int idx){
        return _s.setIdx(idx);
    };

    //Constructors
    ndmatrix();
    ndmatrix(int __C, int __R);

    //Member functions
    void addRow(std::vector<T> row);
    void print_to_console( std::function<void(T)> printer ); //printer is used to print one element of type T to the console.
};

template <class T>
ndmatrix<T>::ndmatrix(){
    _s = Subscriptable();
};

template <class T>
ndmatrix<T>::ndmatrix(int __C, int __R){
    _s = Subscriptable();
    for (int i=0; i<__C*__R; i++) {
        _s.push_back(0);
    }
};

template <class T>
void ndmatrix<T>::addRow(std::vector<T> row){
    this->_s.data().insert( _s.data().end() , row.begin(), row.end() );
};

template <class T>
void ndmatrix<T>::print_to_console( std::function<void(T)>printer ) {
    for(int i=0; i<R; i++){
        std::cout << std::endl;
        for (int j=0; j<C; j++) {
            printer( (*this)[i][j] );
        }
    }
}

#endif /* ndmatrix_h */

然后我尝试在main.cpp: 中使用它

#include <iostream>
#include <fstream>
#include <sstream>
#include "ndmatrix.h"

typedef unsigned int level_t;


//Read table of levels from file into a matrix
void read_matrix(ndmatrix<level_t>* const mat, const char* filename){
    std::ifstream fileInput(filename);

    int rows = 0;
    while (fileInput) {
        std::string line;
        std::getline(fileInput, line, '\n');

        std::stringstream strstream(line);
        std::vector<level_t> row;
        level_t level;
        int cols=0;
        while ( strstream >> level ) {
            row.push_back(level);
            cols++;
        }

        mat->addRow(row);
        rows++;

        mat->C = cols;
        mat->R = rows;

        //debug
        std::cout << line << std::endl;
    }
};

int main(int argc, const char * argv[]) {

    //tests
    ndmatrix<level_t> matrix;
    read_matrix(&matrix, "table.dat");

    // insert code here...
    std::cout << "Hello, World!\n";
    return 0;
}

在尝试构建时,我收到了上面的警告,以及链接器错误:

clang: error: linker command failed with exit code 1 (use -v to see invocation)

【问题讨论】:

    标签: c++ templates


    【解决方案1】:

    您已经编写了静态成员变量_data 的声明,但跳过了定义。

    template< typename T > typename ::std::vector< T > ndmatrix< T >::Subscriptable::_data;
    

    更新:在 C++17 中,可以使用 inline 静态变量,这些变量可以在类内初始化并且不需要在类外单独定义:

    static inline std::vector<T> _data{};
    

    【讨论】:

    • 如果我添加 template&lt; typename T &gt; typename ::std::vector&lt; T &gt; ndmatrix&lt; T &gt;:: Subscriptable ::_data; 那么它就可以了。 (你忘记了 ::Subscriptable :))。但是我真的不明白,我为什么需要这个?
    • 你需要这个,因为类定义只能包含静态变量声明(除了 const enum 或 const integer)。为了使用这个变量,您需要将定义放在类范围之外的某个地方。这些规则很奇怪。
    猜你喜欢
    • 2018-06-21
    • 2019-02-22
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多