С++ наследование Qt проблема qstring

У меня есть следующий код:

template <class T>
bool loadCSV (const QString &filename, map<T,int> &mapping){
    QFile boxfile;
    boxfile.setFileName(filename);
    QString line;
    QStringList list;

    if (!boxfile.open(QIODevice::ReadOnly)){
        cout << "Could not open box data file." << endl;
        return false;
    }

    QTextStream stream2( &boxfile ); 

    while (!stream2.atEnd()){
        line = stream2.readLine();
        list = line.split(',');
        mapping[list.front().toInt()]=list.back().toInt();      
    }

    return true;
}

Он берет файл CSV и вставляет его в:

map<int, int> mapping

структура. Теперь это делается как шаблон, так что я также могу использовать данные в

map<string, int> mapping

структура. Теперь для этого необходимо изменить конечную строку в цикле while, и я не уверен, что это лучший способ достичь этого.

Я мог бы подумать о нескольких путях:

  • некоторые, как обнаружить тип и иметь какую-то условную строку там (я на самом деле не уверен, что это возможно, и если да, то как это сделать.

  • добавьте функцию класса в QStringList, чтобы сделать это.

Я попробовал вариант 2 и сделал следующее:

void QStringList::cInsert(map<int,int> &mapping){
    mapping[this->front().toInt()]=this->back().toInt();
}

void QStringList::cInsert(map<string,int> &mapping){
    mapping[(this->front()).toAscii()]=this->back().toInt();
}

Это не сработало, так как мне также необходимо было определить эти функции в классе QStringList, и поэтому это будет немного беспорядочно. Вместо. Я пытаюсь наследовать от QStringList:

class myQStringList: public QStringList{
    public:
        void cInsert(map<int,int> &mapping);
        void cInsert(map<string,int> &mapping);
};

void myQStringList::cInsert(map<int,int> &mapping){
    mapping[this->front().toInt()]=this->back().toInt();
}

void myQStringList::cInsert(map<string,int> &mapping){
    mapping[(this->front()).toAscii()]=this->back().toInt();
}

И затем измените код:

template <class T>
bool loadCSV (const QString &filename, map<T,int> &mapping){
    QFile boxfile;
    boxfile.setFileName(filename);
    QString line;
    myQStringList list;   

if (!boxfile.open(QIODevice::ReadOnly)){
    cout << "Could not open box data file." << endl;
    return false;
}

QTextStream stream2( &boxfile ); 

while (!stream2.atEnd()){
    line = stream2.readLine();
    list = line.split(',');
        list.cInsert(mapping);
}

return true;}

НО. Я получаю сообщение об ошибке в отношении назначения line.split/list:

main.cpp:123: error: no match for 'operator=' in 'list = QString::split(const QChar&, QString::SplitBehavior, Qt::CaseSensitivity) const(((const QChar&)(&QChar(44))),  KeepEmptyParts,  CaseSensitive)'

Я не уверен, что это за ошибка, и не уверен, что это связано с тем, что оператор присваивания/копирования не наследуется?

И по отношению к фактическому новому классу я получаю эту ошибку:

main.cpp:104: error: no match for 'operator[]' in 'mapping[QString::toAscii() const()]'
c:/qt/mingw/bin/../lib/gcc/mingw32/3.4.2/../../../../include/c++/3.4.2/bits/stl_map.h:332: note: candidates are: _Tp& std::map<_Key, _Tp, _Compare, _Alloc>::operator[](const _Key&) [with _Key = std::string, _Tp = int, _Compare = std::less<std::string>, _Alloc = std::allocator<std::pair<const std::string, int> >]
main.cpp: In function `bool loadCSV(const QString&, std::map<T, int, std::less<_Key>, std::allocator<std::pair<const T,int> > >&) [with T = int]':

Которое я просто не понимаю. Может кто-нибудь объяснить?

Кроме того, я не уверен, правильно ли я это сделаю, и буду благодарен за советы по этому поводу. Спасибо.

Ответ 1

Я бы переместил преобразование из QString, чтобы ввести T вне вашей функции. Поэтому есть что-то вроде:

template< typename T>
struct converter { };

template<>
struct converter< int>
{
static int convert( const QString& source)
{
    return source.toInt();
}
};

template<>
struct converter< std::string>
{
static std::string convert( const QString& source)
{
    return source.toStdString();
}
};

template< typename T>
T convertFromString( const QString& source)
{
    return converter<T>::convert( source);
}

Затем используйте эту функцию преобразования в вашем цикле (внутри while):

Вместо

mapping[list.front().toInt()]=list.back().toInt();   

записи:

mapping[ convert<T>(list.front())]=list.back().toInt();

Примечание: Преобразователь класса (и его две специализации) определены так, что функция преобразования может использоваться как функция шаблона (аналогично static_cast или boost: lexical_cast, если хотите).

Я не пытался скомпилировать код. Это просто идея.

Ответ 2

Когда дело доходит до CSV файлов, Qt обеспечивает четкую функциональность с помощью функции QString:: section. Итак, чтобы узнать нужные столбцы, я обрабатываю заголовок следующим образом:

int AmountInd = (line.left(line.indexOf("Amount")).count(',');

чтобы получить индекс этого столбца. Затем для будущих строк я извлекаю эту подстроку столбца:

QString AmountStr = (line.section(',', AmountInd, AmountInd);
double Amount = AmountStr.toDouble();

Надеюсь, вы найдете полезную функцию раздела; -)

Ответ 3

Я решил проблему, явно нажав:

list = (const myQStringList&) line.split(',');
void myQStringList::cInsert(map<string,int> &mapping){
    mapping[(string)(this->front()).toAscii()]=this->back().toInt();
}