从二进制文件中读取

时间:2013-09-16 15:22:39

标签: c++

我试图从二进制文件中计算零和一的数量。问题是,我得到了正确的答案。零但是出现等于否。零。 我正在做的是通过char读取char文件。由于最多可以有256个字符,因此我将结果存储在一个临时数组中,用于零和一个字符,并在那里检索是否再次出现字符。

#include<iostream>
#include<cstdio>
#include<cstdlib>
using namespace std;
void func(int* a1 ,int* a2)
{
    for(int i=0;i<256;i++)
    for(int j=0;j<8;j++)
    {
        if( (i & 1) ==1 )
        {
            a1[i]+=1;
        }
        else if( (i & 1) ==0 )
        {
            a2[i]+=1;   
        }
        i>>1;
    }
}
int main()
{
    int zero[256];
    int one[256];
    int tzero[256];
    int tone[256];
    for(int i=0;i<256;i++)
    {
        zero[i]=0;
        one[i]=0;
        tzero[i]=0;
        tone[i]=0;
    }
    func(tone,tzero);
    FILE* input;
    FILE* output;
    output=fopen("ascii.txt","w");
    input=fopen("one.bin","r");
    int c;
    while((c=fgetc(input))!=EOF)
    {
        fprintf(output,"%d\n",c);
        zero[c]+=tzero[c];
        one[c]+=tone[c];
    }
    int zeroes=0;
    int ones=0;
    for(int i=0;i<=255;i++)
    {
        zeroes+=zero[i];
        ones+=one[i];
    }
    cout<<"zeroes:"<<zeroes<<endl;
    cout<<"ones:"<<ones<<endl;
    fclose(input);

    fclose(output);

}

2 个答案:

答案 0 :(得分:1)

通过执行

计算零和1的循环会破坏c的值
c >>= 1;

完成所有八个班次后,c始终为零,因此以下代码会增加错误的计数:

// The value of c is always zero
tzero[c]=z;
tone[c]=o;
one[c]+=tzero[c];
zero[c]+=tzero[c];

你应该在比特计数循环之前保存c的值,并在循环结束后恢复它。

更好的是,预先计算tzero[]tone[]的值,而不必等待它们出现在文件中。这会使你的主循环体变得非常短而干净:

while((c=fgetc(input))!=EOF) {
    one[c] += tzero[c];
    zero[c] += tzero[c];
}

答案 1 :(得分:0)

如果您的目标只是按位计算文件中的10位,则可以通过使用C ++文件流而不使用查找表来大大简化:

#include <iostream>
#include <fstream>

int main(int argc, char** argv)
{
  std::ifstream fpInput("Input.txt");
  unsigned unOnes = 0;
  unsigned unZeros = 0;
  char chTemp;

  chTemp = fpInput.get();
  while (fpInput.good())
  {
    for (unsigned i = 0; i < 8; ++i)
    {
      if (chTemp & 0x1<<i) unOnes++;
      else unZeros++;
    }

    chTemp = fpInput.get();
  }

  fpInput.close();

  std::cout << "Found " << unOnes << " ones." << std::endl;
  std::cout << "Found " << unZeros << " zeros." << std::endl;

  return 0;
}

如果你传递正确的优化标志,一个好的编译器应该扩展中间循环。