显然在我的机器上,float、double 和 long double 各有不同的大小。似乎也没有严格的标准来强制执行每种类型必须有多少字节。
然后,如何将浮点值保存到二进制文件中,然后在大小不同的情况下在不同的系统上正确读取它?例如,我的机器有 8 个字节的 double ,而乔的有 12 个字节的 double 。
无需以文本形式导出(例如“0.3232”),也就是说。看起来比二进制表示要紧凑得多。
最佳答案
你必须定义一个格式,并实现它。通常,大多数 我知道的网络协议(protocol)使用 IEEE float 和 double,输出 big-endian (但其他格式也是可能的)。使用 IEEE 格式的优势 是目前大多数日常机器使用的 内部;如果你在其中一台机器上(以及你的便携性 代码到其他机器,比如大型机,不是问题),你可以 只需将类型双关转换为 unsigned int 即可“转换”为格式 相同的大小,并输出它。因此,例如,您可能有:
obstream&
operator<<( obstream& dest, uint64_t value )
{
dest.put((value >> 56) & 0xFF);
dest.put((value >> 48) & 0xFF);
dest.put((value >> 40) & 0xFF);
dest.put((value >> 32) & 0xFF);
dest.put((value >> 24) & 0xFF);
dest.put((value >> 16) & 0xFF);
dest.put((value >> 8) & 0xFF);
dest.put((value ) & 0xFF);
return dest;
}
obstream&
operator<<( obstream& dest, double value )
{
return dest << reinterpret_cast<uint64_t const&>( value );
}
如果您必须移植到不支持 IEEE 的机器(例如任何 现代大型机),你需要一些更复杂的东西:
obstream&
obstream::operator<<( obstream& dest, double value )
{
bool isNeg = value < 0;
if ( isNeg ) {
value = - value;
}
int exp;
if ( value == 0.0 ) {
exp = 0;
} else {
value = ldexp( frexp( value, &exp ), 53 );
exp += 1022;
}
uint64_t mant = static_cast< uint64_t >( value );
dest.put( (isNeg ? 0x80 : 0x00) | exp >> 4 );
dest.put( ((exp << 4) & 0xF0) | ((mant >> 48) & 0x0F) );
dest.put( mant >> 40 );
dest.put( mant >> 32 );
dest.put( mant >> 24 );
dest.put( mant >> 16 );
dest.put( mant >> 8 );
dest.put( mant );
return dest;
}
(请注意,这不能正确处理 NaN 和无穷大。 就个人而言,我会禁止它们出现在格式中,因为不是所有的都 float 点表示支持它们。但是,没有 float 支持 1E306 的 IBM 大型机上的格式,尽管 你可以用上面的 IEEE double 格式对其进行编码。)
阅读当然是相反的。要么:
ibstream&
operator>>( ibstream& source, uint64_t& results )
{
uint64_t value = (source.get() & 0xFF) << 56;
value |= (source.get() & 0xFF) << 48;
value |= (source.get() & 0xFF) << 40;
value |= (source.get() & 0xFF) << 32;
value |= (source.get() & 0xFF) << 24;
value |= (source.get() & 0xFF) << 16;
value |= (source.get() & 0xFF) << 8;
value |= (source.get() & 0xFF) ;
if ( source )
results = value;
return source;
}
ibstream&
operator>>( ibstream& source, double& results)
{
uint64_t tmp;
source >> tmp;
if ( source )
results = reinterpret_cast<double const&>( tmp );
return source;
}
或者如果您不能指望 IEEE:
ibstream&
ibstream::operator>>( ibstream& source, double& results )
{
uint64_t tmp;
source >> tmp;
if ( source ) {
double f = 0.0;
if ( (tmp & 0x7FFFFFFFFFFFFFFF) != 0 ) {
f = ldexp( ((tmp & 0x000FFFFFFFFFFFFF) | 0x0010000000000000),
static_cast<int>( (tmp & 0x7FF0000000000000) >> 52 )
- 1022 - 53 );
}
if ( (tmp & 0x8000000000000000) != 0 ) {
f = -f;
}
dest = f;
}
return source;
}
(假设输入不是 NaN 或无穷大。)
关于c++ - 正确地(独立于 32 位/64 位)将 float 保存到二进制 ofstream,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7103710/
我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
我正在查看instance_variable_set的文档并看到给出的示例代码是这样做的:obj.instance_variable_set(:@instnc_var,"valuefortheinstancevariable")然后允许您在类的任何实例方法中以@instnc_var的形式访问该变量。我想知道为什么在@instnc_var之前需要一个冒号:。冒号有什么作用? 最佳答案 我的第一直觉是告诉你不要使用instance_variable_set除非你真的知道你用它做什么。它本质上是一种元编程工具或绕过实例变量可见性的黑客攻击
question的一些答案关于redirect_to让我想到了其他一些问题。基本上,我正在使用Rails2.1编写博客应用程序。我一直在尝试自己完成大部分工作(因为我对Rails有所了解),但在需要时会引用Internet上的教程和引用资料。我设法让一个简单的博客正常运行,然后我尝试添加评论。靠我自己,我设法让它进入了可以从script/console添加评论的阶段,但我无法让表单正常工作。我遵循的其中一个教程建议在帖子Controller中创建一个“评论”操作,以添加评论。我的问题是:这是“标准”方式吗?我的另一个问题的答案之一似乎暗示应该有一个CommentsController参
我喜欢使用Textile或Markdown为我的项目编写自述文件,但是当我生成RDoc时,自述文件被解释为RDoc并且看起来非常糟糕。有没有办法让RDoc通过RedCloth或BlueCloth而不是它自己的格式化程序运行文件?它可以配置为自动检测文件后缀的格式吗?(例如README.textile通过RedCloth运行,但README.mdown通过BlueCloth运行) 最佳答案 使用YARD直接代替RDoc将允许您包含Textile或Markdown文件,只要它们的文件后缀是合理的。我经常使用类似于以下Rake任务的东西:
我一直致力于让我们的Rails2.3.8应用程序在JRuby下正确运行。一切正常,直到我启用config.threadsafe!以实现JRuby提供的并发性。这导致lib/中的模块和类不再自动加载。使用config.threadsafe!启用:$rubyscript/runner-eproduction'pSim::Sim200Provisioner'/Users/amchale/.rvm/gems/jruby-1.5.1@web-services/gems/activesupport-2.3.8/lib/active_support/dependencies.rb:105:in`co
我正在尝试在Ruby中复制Convert.ToBase64String()行为。这是我的C#代码:varsha1=newSHA1CryptoServiceProvider();varpasswordBytes=Encoding.UTF8.GetBytes("password");varpasswordHash=sha1.ComputeHash(passwordBytes);returnConvert.ToBase64String(passwordHash);//returns"W6ph5Mm5Pz8GgiULbPgzG37mj9g="当我在Ruby中尝试同样的事情时,我得到了相同sha
我需要一些关于TDD概念的帮助。假设我有以下代码defexecute(command)casecommandwhen"c"create_new_characterwhen"i"display_inventoryendenddefcreate_new_character#dostufftocreatenewcharacterenddefdisplay_inventory#dostufftodisplayinventoryend现在我不确定要为什么编写单元测试。如果我为execute方法编写单元测试,那不是几乎涵盖了我对create_new_character和display_invent
我正在尝试使用Curbgem执行以下POST以解析云curl-XPOST\-H"X-Parse-Application-Id:PARSE_APP_ID"\-H"X-Parse-REST-API-Key:PARSE_API_KEY"\-H"Content-Type:image/jpeg"\--data-binary'@myPicture.jpg'\https://api.parse.com/1/files/pic.jpg用这个:curl=Curl::Easy.new("https://api.parse.com/1/files/lion.jpg")curl.multipart_form_