Chinaunix首页 | 论坛 | 博客
  • 博客访问: 92427856
  • 博文数量: 19283
  • 博客积分: 9968
  • 博客等级: 上将
  • 技术积分: 196062
  • 用 户 组: 普通用户
  • 注册时间: 2007-02-07 14:28
文章分类

全部博文(19283)

文章存档

2011年(1)

2009年(125)

2008年(19094)

2007年(63)

分类: C/C++

2008-05-18 20:45:36

 来源:


  小数点的位置

  要进行格式化,首先要确定小数点的位置,如果小数位多于decplaces,do_fraction()会删除多余的。

  要定位小数位,可使用string::find(),在STL算法中使用了一个常量来代表"数值未",在字符串中,这个常量为string::npos:

char DECIMAL_POINT='.'; // 欧洲用法为','

size_t n=str.find(DECIMAL_POINT);
if ((n!=string::npos)//是否有小数点呢?
{
//检查小数的位数
}

  如果没有小数点,函数直接返回字符串,否则,函数将继续检查小数位是否多于decplaces。如果是,小数部分将会被截断:

size_t n=str.find(DECIMAL_POINT);
if ((n!=string::npos)//有小数点吗?
&&(str.size()> n+decplaces)) //后面至少还有decplaces位吗?

//在小数decplaces位之后写入nul
str[n+decplaces]='';

  最后一行覆盖了多余的小数位,它使用了常量来截断字符串,要注意,string对象的数据可以包含nul字符;而字符串的实际长度由size()的返回值决定。因此,你不能假定字符串已被正确地格式化,换句话来说,如果在str中原来为"123.4567",在插入常量之后,它变成了"123.457",为把str缩减为"123.45",一般可使用自交换的: str.swap(string(str.c_str()) );//删除nul之后的多余字符

  那它的原理是什么呢?函数string::c_str()返回一个const char *代表此字符串对象,而这个值被用作一个临时string对象的初始化值,接着,临时对象又被用作str.swap()的参数,swap()会把值"123.45"赋给str。一些老一点的编译器不支持默认模板参数,可能不会让swap()通过编译,如果是这样的话,使用手工交换来代替:

string temp=str.c_str();
str=temp;

  代码虽不是很"优美",但能达到目的就行。以下是do_fraction()的完整代码:

string do_fraction(long double value, int decplaces=3)
{
 ostringstream out;
 int prec=
 numeric_limits::digits10; // 18

 out.precision(prec);//覆盖默认精度
 out< string str= out.str(); //从流中取出字符串
 size_t n=str.find(DECIMAL_POINT);
 if ((n!=string::npos) //有小数点吗?
 && (str.size()> n+decplaces)) //后面至少还有decplaces位吗?
 {
  str[n+decplaces]='';//覆盖第一个多余的数
 }

 str.swap(string(str.c_str()));//删除nul之后的多余字符

 return str;
}

  如果不想通过传值返回一个string对象,还可增加一个参数,把str对象以引用传递:

void do_fraction(long double value, string & str, int decplaces=3);

  从个人的角度来讲,还是倾向于让编译器做这样的优化,另外,使用传值返回,还可以让你以下面这种方式使用do_fraction():

cout << funct(123456789.69999001) << 't' << funct(12.011)<

  输出:

  123456789.69 12.01
阅读(388) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~