【问题标题】:How to static cast 2D vector size() (size_t to int) in C++?如何在 C++ 中静态转换 2D 向量大小()(size_t 到 int)?
【发布时间】:2018-01-06 19:33:27
【问题描述】:

我还没有找到任何其他回答这个问题的帖子。这是一个 C++ 程序。我收到一条错误消息:

int entryval = static_cast<int>(database[0].size());

错误提示:线程 1:EXC_BAD_ACCESS(代码 = 1,地址 = 0x8) 它将我的注意力引向矢量文件中的 size() 文件,指向以下行:

{return static_cast<size_type>(this->__end_ - this->__begin_);}

我的类型转换格式不正确吗?

//Species ID,Kingdom,Phylum,Class,Order,Family,Genus,Species,Authority,Infraspecific rank,Infraspecific name,Infraspecific authority,Stock/subpopulation,Synonyms,Common names (Eng),Common names (Fre),Common names (Spa),Red List status,Red List criteria,Red List criteria version,Year assessed,Population trend,Petitioned
//2055,ANIMALIA,CHORDATA,MAMMALIA,CARNIVORA,OTARIIDAE,Arctocephalus,australis,"(Zimmermann, 1783)",,,,,Arctophoca australis,South American Fur Seal,Otarie  fourrure Australe,Oso Marino Austral,LC,,3.1,2016,increasing,N
//41664,ANIMALIA,CHORDATA,MAMMALIA,CARNIVORA,OTARIIDAE,Arctocephalus,forsteri,"(Lesson, 1828)",,,,,Arctocephalus australis subspecies forsteri|Arctophoca australis subspecies forsteri,"New Zealand Fur Seal, Antipodean Fur Seal, Australasian Fur Seal, Black Fur Seal, Long-nosed Fur Seal, South Australian Fur Seal",,,LC,,3.1,2015,increasing,N

#include <iostream>
#include <sstream>
#include <fstream>
#include <string>
#include <vector>
#include <algorithm>

int main()
{

std::string entry;
std::vector<std::vector<std::string> > database;
std::ifstream file("C:/Users/mewtwo/Desktop/Programs/Seals.csv");

if (file) {
    while (std::getline(file, entry)) {
        size_t dbsize = database.size();
        database.resize(dbsize + 1);
        std::istringstream ss(entry);
        std::string field, push_field("");
        bool no_quotes = true;

        while (std::getline(ss, field, ',')) {
            if (static_cast<size_t>(std::count(field.begin(), field.end(), '"')) % 2 != 0) {
                no_quotes = !no_quotes;
                field.erase(remove(field.begin(), field.end(), '\"'), field.end());
            }

            push_field += field + (no_quotes ? "" : ",");

            if (no_quotes) {
                database[dbsize].push_back(push_field);
                push_field.resize(0);
            }
        }
    }
}

int dbval =  static_cast<int>(database.size());
int entryval = static_cast<int>(database[0].size());
int animalChoice;
int dataChoice;
int switchChoice;
bool moreData;
bool moreAnimal = true;
while (moreAnimal == true) {

    std::cout << "*********************ENDANGERED ANIMALS*********************" << std::endl;
    std::cout << "Choose an animal! (1-" << dbval - 1 << ")" << std::endl;
    for (int x = 1; x < database.size(); x++) {
        if (database[x][14] != ""){
            std::cout << x << ". " << database[x][14] << std::endl;
        }
        else {
            std::cout << x << ". " << database[x][6] << " " << database[x][7] << std::endl;
        }

    }
    std::cout << "\nAnimal choice:";
    while (!(std::cin >> animalChoice) || animalChoice <= 0 || animalChoice >= dbval ){
        std::cout << "\nYou pressed an incorrect key. Try again1." << std::endl;
        std::cin.clear();
        std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
        std::cout << "\nAnimal choice:";
    }
    moreData = true;
    while (moreData == true) {
        std::cout << "\nWhat data do you want to access about the ";
        if(database[animalChoice][14] != ""){
            std::cout << database[animalChoice][14] << std::endl;
        }
        else{
            std::cout << database[animalChoice][6] << " " << database[animalChoice][7] << std::endl;
        }
        std::cout << "?" << std::endl;
        for (int x = 0; x < entryval; x++) {
            std::cout << x + 1 << "." << database[0][x] << std::endl;
        }
        std::cout << "\nData choice:";
        while (!(std::cin >> dataChoice) || dataChoice <= 0 || dataChoice >= entryval ){
            std::cout << "You pressed an incorrect key. Try again2." << std::endl;
            std::cin.clear();
            std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
            std::cout << "\nData choice:";
        }
        std::cout << database[animalChoice][14] << std::endl;
        std::cout << database[0][dataChoice - 1] << ": " << database[animalChoice][dataChoice - 1] << std::endl;
        std::cout << "\nWould you like to access more data about the " << database[animalChoice][14];
        if(database[animalChoice][14] != ""){
            std::cout << database[animalChoice][14] << std::endl;
        }
        else{
            std::cout << database[animalChoice][6] << " " << database[animalChoice][7] << std::endl;
        }
        std::cout << "?" << std::endl;
        std::cout << "1. Yes" << std::endl;
        std::cout << "2. Access data of a different animal" << std::endl;
        std::cout << "3. Quit" << std::endl;
        std::cout << "\nChoice:";
        while (!(std::cin >> switchChoice) || switchChoice <= 0 || switchChoice >= 4 ){
            std::cout << "You pressed an incorrect key. Try again3." << std::endl;
            std::cin.clear();
            std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
            std::cout << "\nChoice:";
        }
        if (switchChoice == 1) {
            moreData = true;
        }
        else if (switchChoice == 2) {
            moreData = false;
            break;
        }
        else if (switchChoice == 3) {
            exit(1);
        }
        else {
            std::cout << "\nPROGRAM FAILURE" << std::endl;
            exit(1);
        }
    }
}
return 0;
}

另外,我的代码还有其他明显的问题吗?该代码获取IUCN的整个物种数据库,并使用菜单呈现数据。

【问题讨论】:

  • 崩溃与强制转换无关,但database 为空(因此database[0] 超出范围)或database[0] 不是有效的std::vector 对象。使用调试器找出它可能是什么。
  • 另外,你应该有一个类对象的向量来存储所有相关信息,而不是使用字符串向量的向量——这将使代码更具可读性(通过删除用于访问特定字段的幻数)
  • @Someprogrammerdude 就是这样,它解决了我的问题
  • @UnholySheep 非常感谢您的反馈。我一直在寻找一种更简洁的方法来编写这个程序,但我仍然是初学者,所以我没有能力看到它。我要回去使用类对象的向量。

标签: c++ vector type-conversion static-cast size-t


【解决方案1】:

address = 0x8 中的低值暗示代码试图通过空指针访问成员,该空指针来自database[0] 返回的无效“空”引用。

operator[] 返回这样一个损坏的引用的唯一方法是访问超出范围,触发未定义的行为。在这种情况下,看起来您的std::vector 实现只是盲目地通过其数据指针进行索引,当向量为空时该指针为空。

底线:没有任何东西被插入到你的向量中。这可能是由于读取文件时出错引起的。您应该在代码中添加错误检查并干净地报告它们。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-12-08
    • 2016-08-15
    • 2015-09-11
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多