我一直在尝试将输出格式化为控制台的时间最长,并且没有发生任何事情。我一直在尝试尽可能多地使用iomanip
ofstream&
函数。
void list::displayByName(ostream& out) const
{
node *current_node = headByName;
// I have these outside the loop so I don't write it every time.
out << "Name\t\t" << "\tLocation" << "\tRating " << "Acre" << endl;
out << "----\t\t" << "\t--------" << "\t------ " << "----" << endl;
while (current_node)
{
out << current_node->item.getName() // Equivalent tabs don't work?
<< current_node->item.getLocation()
<< current_node->item.getAcres()
<< current_node->item.getRating()
<< endl;
current_node = current_node->nextByName;
}
// The equivalent tabs do not work because I am writing names,
// each of different length to the console. That explains why they
// are not all evenly spaced apart.
}
他们可以使用的任何东西都可以使它们彼此正确对齐吗? 我所调用的函数是不言自明的,所有长度都不同,所以彼此之间的对齐不好。
我已经尝试了iomanip
中的所有内容。
答案 0 :(得分:15)
把它想象成使用Microsoft Excel :) 您将自己的流视为字段。因此,首先设置字段的宽度,然后在该字段中插入文本。例如:
#include <iostream>
#include <iomanip>
#include <string>
int main()
{
using namespace std;
string firstName = "firstName",
secondName = "SecondName",
n = "Just stupid Text";
size_t fieldWidth = n.size(); // length of longest text
cout << setw(fieldWidth) << left << firstName << endl // left padding
<< setw(fieldWidth) << left << secondName << endl
<< setw(fieldWidth) << left << n << endl;
cout << setw(fieldWidth) << right << firstName << endl // right padding
<< setw(fieldWidth) << right << secondName << endl
<< setw(fieldWidth) << right << n << endl;
}
...
...
字段宽度仅代表text + spaces
的宽度。您可以fill
除空格以外的任何内容:
string name = "My first name";
cout << setfill('_') << setw(name.size() + 10) << left << name;
.....
output::
My first name__________
...
我认为最好的方法是弄清楚你的格式,然后编写一个新的格式化程序来完成你想要的所有工作:
#include <iostream>
#include <iomanip>
#include <string>
std::ostream& field(std::ostream& o)
{
// usually the console is 80-character wide.
// divide the line into four fields.
return o << std::setw(20) << std::right;
}
int main()
{
using namespace std;
string firstName = "firstName",
secondName = "SecondName",
n = "Just stupid Text";
size_t fieldWidth = n.size();
cout << field << firstName << endl
<< field << secondName << endl
<< field << n << endl;
}
如果你开始考虑参数化操纵器,那么只接受一个int
或long
参数很容易实现,如果你不熟悉C++
中的流,其他类型真的很模糊
答案 1 :(得分:6)
Boost有一个格式库,允许您像旧的C printf()一样轻松地格式化输出,但是C ++的类型安全。
请记住,旧的C printf()允许您指定字段宽度。如果输出尺寸过小,则此空间将填充该字段(请注意,它不能处理超大字段)。
#include <iostream>
#include <iomanip>
#include <boost/format.hpp>
struct X
{ // this structure reverse engineered from
// example provided by 'Mikael Jansson' in order to make this a running example
char* name;
double mean;
int sample_count;
};
int main()
{
X stats[] = {{"Plop",5.6,2}};
// nonsense output, just to exemplify
// stdio version
fprintf(stderr, "at %p/%s: mean value %.3f of %4d samples\n",
stats, stats->name, stats->mean, stats->sample_count);
// iostream
std::cerr << "at " << (void*)stats << "/" << stats->name
<< ": mean value " << std::fixed << std::setprecision(3) << stats->mean
<< " of " << std::setw(4) << std::setfill(' ') << stats->sample_count
<< " samples\n";
// iostream with boost::format
std::cerr << boost::format("at %p/%s: mean value %.3f of %4d samples\n")
% stats % stats->name % stats->mean % stats->sample_count;
}
答案 2 :(得分:3)
放弃标签。您应该能够使用io操纵器来设置字段宽度,填充字符和格式标记(以获得左对齐或右对齐)。使用与标题相同的标题值,一切都应该很好。
还要注意你已经在你的例子中改变了评级和英亩。
答案 3 :(得分:2)
您可以编写一个始终在标准输出中打印相同字符数的过程。
类似的东西:
string StringPadding(string original, size_t charCount)
{
original.resize(charCount, ' ');
return original;
}
然后在你的程序中使用这个:
void list::displayByName(ostream& out) const
{
node *current_node = headByName;
out << StringPadding("Name", 30)
<< StringPadding("Location", 10)
<< StringPadding("Rating", 10)
<< StringPadding("Acre", 10) << endl;
out << StringPadding("----", 30)
<< StringPadding("--------", 10)
<< StringPadding("------", 10)
<< StringPadding("----", 10) << endl;
while ( current_node)
{
out << StringPadding(current_node->item.getName(), 30)
<< StringPadding(current_node->item.getLocation(), 10)
<< StringPadding(current_node->item.getRating(), 10)
<< StringPadding(current_node->item.getAcres(), 10)
<< endl;
current_node = current_node->nextByName;
}
}