Печать сетки данных из вектора структуры
Я создал код, который представляет таблицу данных. Я хотел бы сделать его более универсальным.
Мой код:
#include <algorithm>
#include <iomanip>
#include <iostream>
#include <numeric>
#include <string>
#include <vector>
struct Student
{
std::string name;
unsigned points;
unsigned height;
unsigned weight;
};
namespace console
{
constexpr unsigned max_chars_in_line = 79;
void draw_line(unsigned count = max_chars_in_line, char sign = '-')
{
std::cout << std::string(count, sign) << "\n";
}
}
int main()
{
std::cout << std::fixed << std::setprecision(2);
std::vector<Student> students
{
{ "John", 30, 200, 300 },
{ "Peter", 80, 160, 80 },
{ "Albert", 200, 175, 1230 },
{ "Hans", 100, 178, 130 }
};
console::draw_line(45);
std::cout
<< std::right
<< std::setw(10) << "name"
<< std::setw(10) << "points"
<< std::setw(10) << "height"
<< std::setw(10) << "weight"
<< "\n";
console::draw_line(45);
for (auto& student : students)
{
std::cout
<< std::setw(10) << student.name
<< std::setw(10) << student.points
<< std::setw(10) << student.height
<< std::setw(10) << student.weight
<< "\n";
}
console::draw_line(45);
std::cout << "\n\n";
auto total_weight = std::accumulate(students.begin(), students.end(), 0u, [](unsigned sum, const Student& student) {return sum += student.weight; });
std::sort(students.begin(), students.end(), [](const Student& a, const Student& b) {return a.weight > b.weight; });
console::draw_line(40);
std::cout
<< std::setw(10) << "name"
<< std::setw(10) << "weight"
<< std::setw(15) << "percent (%)"
<< "\n";
console::draw_line(40);
for (auto& student : students)
{
std::cout
<< std::setw(10) << student.name
<< std::setw(10) << student.weight
<< std::setw(15) << 100 * static_cast<double>(student.weight) / total_weight
<< "\n";
}
console::draw_line(40);
std::cout
<< std::setw(10) << "total"
<< std::setw(10) << total_weight
<< std::setw(15) << 100.
<< "\n";
console::draw_line(40);
}
Как бы я хотел, чтобы код выглядел:
#include <algorithm>
#include <iomanip>
#include <iostream>
#include <functional>
#include <fstream>
#include <numeric>
#include <string>
#include <vector>
struct Student
{
std::string name;
unsigned points;
unsigned height;
unsigned weight;
};
class GridView { /*...*/ };
int main()
{
std::cout << std::fixed << std::setprecision(2);
std::vector<Student> students
{
{ "John", 30, 200, 300 },
{ "Peter", 80, 160, 80 },
{ "Albert", 200, 175, 1230 },
{ "Hans", 100, 178, 130 }
};
GridView<Student> diagram1(&students); // pointer?
diagram1.addCol("name", &Student::name);
diagram1.addCol("points", &Student::points);
diagram1.addCol("height", &Student::height);
diagram1.addCol("weight", &Student::weight);
diagram1.print(std::cout); // or |std::cout << diagram1;|
std::cout << "\n\n";
auto total_weight = std::accumulate(students.begin(), students.end(), 0u, [](unsigned sum, const Student& student) {return sum += student.weight; });
std::sort(students.begin(), students.end(), [](const Student& a, const Student& b) {return a.weight > b.weight; });
std::ofstream file("grid.txt");
if (!file.is_open())
{
return -1;
}
GridView<Student> diagram2(&students);
diagram2.addCol("name", &Student::name, 10);
diagram2.addCol("weight", &Student::weight, 10);
diagram2.addCol("percent (%)", [=]() {return 100 * static_cast<double>(Student::weight) / total_weight; }, 15); // bind? how to do it?
diagram2.addFooter("total", total_weight, 100.);
diagram2.print(file);
}
Как мне начать писать класс GridView? Каким должно быть добавление столбца с процентами? Может быть, я могу сделать это по-другому (лучше)? Похоже, мой пост в основном кодовый; Я извиняюсь за слишком мало деталей.
Редактировать: Код:
#include <algorithm>
#include <iomanip>
#include <iostream>
#include <functional>
#include <fstream>
#include <numeric>
#include <string>
#include <vector>
#include <variant>
struct Student
{
std::string name;
unsigned points;
unsigned height;
unsigned weight;
};
template <typename DataType>
class GridView
{
std::vector<DataType>* data_ptr;
using variant_cell_type = std::variant<int, float, double, unsigned, std::string>;
struct output_visitor
{
std::ostream& out;
output_visitor(std::ostream& out)
:out(out)
{
}
template <typename T>
void operator() (const T& value) const
{
out << value;
}
};
struct Col
{
std::string name;
std::function<variant_cell_type(DataType&)> access_data;
unsigned width;
};
std::vector<Col> cols;
unsigned max_chars_in_line;
std::vector<variant_cell_type> footer;
public:
GridView(std::vector<DataType>* data_ptr)
:data_ptr(data_ptr)
{
max_chars_in_line = 5;
}
template <class Magic, class Type>
void addCol(const std::string& colName, Magic Type::* ceilType, unsigned width = 10)
{
cols.push_back({colName, std::mem_fn(ceilType), width});
max_chars_in_line += width;
}
void addFooter(const std::vector<variant_cell_type>& footer)
{
this->footer = footer;
}
void print(std::ostream& out)
{
using std::string;
out << std::right;
out << string(max_chars_in_line, '-') << "\n";
for (auto& col : cols)
{
out << std::setw(col.width) << col.name;
}
out << "\n";
out << string(max_chars_in_line, '-') << "\n";
for (auto student : *data_ptr)
{
for (auto& col : cols)
{
out << std::setw(col.width);
std::visit(output_visitor{out}, col.access_data(student));
}
out << "\n";
}
if (footer.size() != 0)
{
if (footer.size() != cols.size())
{
throw std::invalid_argument("Footer must have " + std::to_string(cols.size()) + " arguments.");
}
out << string(max_chars_in_line, '-') << "\n";
auto foo = std::begin(footer);
auto col = std::begin(cols);
for (; foo != footer.end(); ++foo, ++col)
{
out << std::setw(col->width);
std::visit(output_visitor{ out }, *foo);
}
out << "\n";
}
out << string(max_chars_in_line, '-') << "\n";
}
};
int main()
{
std::cout << std::fixed << std::setprecision(2);
std::vector<Student> students
{
{ "John", 30, 200, 300 },
{ "Peter", 80, 160, 80 },
{ "Albert", 200, 175, 1230 },
{ "Hans", 100, 178, 130 }
};
try
{
GridView<Student> diagram1(&students);
diagram1.addCol("name", &Student::name);
diagram1.addCol("points", &Student::points);
diagram1.addCol("height", &Student::height);
diagram1.addCol("weight", &Student::weight);
diagram1.print(std::cout);
std::cout << "\n\n";
auto total_weight = std::accumulate(students.begin(), students.end(), 0u, [](unsigned sum, const Student& student) {return sum += student.weight; });
std::sort(students.begin(), students.end(), [](const Student& a, const Student& b) {return a.weight > b.weight; });
GridView<Student> diagram2(&students);
diagram2.addCol("name", &Student::name, 10);
diagram2.addCol("weight", &Student::weight, 10);
//auto calculate_weight = [=](Student student) {return 100 * static_cast<double>(student.weight) / total_weight; };
//diagram2.addCol("percent (%)", calculate_weight, 15); // bind? how to do it?
diagram2.addFooter({ "total", total_weight/*, 100.*/ });
diagram2.print(std::cout);
}
catch (const std::exception& e)
{
std::cerr << e.what() << "\n";
}
}
Что я должен сделать, чтобы закомментированный фрагмент работал?