在向量中读取和写入不同类型的内容

Read and write different types in vector

本文关键字:同类型 向量 读取      更新时间:2023-10-16

如何在向量中打包几种不同的原始数据类型以及字符串来检查内容?
(在Java中,这是通过Object [] objects = {1.0, "Hello", - 42, 'b'}进行的)

任务如下:给定以下数组:
[3.0,42, "Monkey", 7.2, b]

此数组将传递给在控制台上输出数组内容的方法。如果是string,字符串的每个字母都应作为 ASCII 值添加到同一变量中,最后在控制台上作为int返回。与char完全相同。

从今天开始,当我创建一个vector时,我知道std::vector<double> numbers = {1.0,2.0};如何编写函数以及如何访问索引numbers[i];以及向量numbers.size()的长度。

我现在该如何解决这个问题?由于不幸的是,我没有为向量中的多种类型找到简单的 <- 解决方案。

提前致谢:)

C++不像Java那样执行类型擦除。要创建一个异构容器(这是您尝试执行的操作的技术术语),您需要广泛使用std::anystd::variant,这是 C++17 引入的新类。

std::vector<std::any> values{1.0, "Hello", -42, 'b'};
for(auto & any : values) {
int * i;
if(val = std::any_cast<int>(&any)) std::cout << "int: " << *i << std::endl;
const char ** s;
if(s = std::any_cast<const char *>(&any)) std::cout << "string-literal: " << *s << std::endl;
double * d;
if(d = std::any_cast<double>(&any)) std::cout << "double: " << *d << std::endl;
char * c;
if(c = std::any_cast<char>(&any)) std::cout << "char: " << *c << std::endl;
}

请注意该代码有多混乱。尤其是因为许多人希望"hello"存储为std::string对象,但除非用户明确指定它,否则无法做到这一点:

std::vector<std::any> values{1.0, std::string{"Hello"}, -42, 'b'};

无论如何,我个人的观点是,使用std::variant会更适合,因为您可以更清楚地说明容器的使用方式,并且可以避免与std::any相关的动态分配:

typedef std::variant<std::string, char, double, int> my_variant;
struct visitor {
void operator()(std::string const& v) const {
std::cout << "std::string: " << v<< std::endl;
}
void operator()(double const& v) const {
std::cout << "double: " << v << std::endl;
}
void operator()(int const& v) const {
std::cout << "int: " << v << std::endl;
}
void operator()(char const& v) const {
std::cout << "char: " << v << std::endl;
}
};
int main() {
std::vector<my_variant> values{1.0, "Hello", -42, 'b'};
for(my_variant & variant : values) {
std::visit(visitor{}, variant);
}
return 0;
}

如果我们不需要知道类型,我们甚至可以使用 lambdaauto使变体版本简单得多:

typedef std::variant<std::string, char, double, int> my_variant;
int main() {
std::vector<my_variant> values{1.0, "Hello", -42, 'b'};
for(my_variant & variant : values) {
std::visit(
[](auto const& val) {std::cout << "Some unknown type: " << val << std::endl;}, 
variant
);
}
return 0;
}

我还没有通过我的编译器运行它,但这应该可以很好地了解如何在C++中完成此类任务。

如果您无法访问 C++17,您可以使用boost.anyboost.variant,我有理由确定它们都是仅标头库,因此很容易导入到您的项目中。