十六进制转十进制problem.Also,如何将char数转换为实际的int数

Hexadecimal to decimal conversion problem.Also, how to convert a char number to an actual int number

请帮我找出这个程序中的错误,对我来说它看起来是正确的,我已经检查过了,但是它给出了错误的答案。 在这个程序中,我明确检查了 A、B、C、D、E、F,并根据它们各自的值。

[已编辑]:此外,这个问题与如何将字符数转换为实际整数有关。

#include<iostream>
#include<cmath>
#include<bits/stdc++.h>
using namespace std;
void convert(string num)
{
   long int last_digit;
    int s=num.length();
    int i;
    long long int result=0;
    reverse(num.begin(),num.end());                 
    for(i=0;i<s;i++)
    {
        if(num[i]=='a' || num[i]=='A')
        {
            last_digit=10;
            result+=last_digit*pow(16,i);
        }
        else if(num[i]=='b'|| num[i]=='B')
        {
            last_digit=11;
            result+=last_digit*pow(16,i);
        }
        else if(num[i]=='c' || num[i]=='C')
        {
            last_digit=12;
            result+=last_digit*pow(16,i);
        }
        else if(num[i]=='d'|| num[i]=='D' )
        {
            last_digit=13;
            result+=last_digit*pow(16,i);
        }
        else if(num[i]=='e'|| num[i]=='E' )
        {
            last_digit=14;
            result+=last_digit*pow(16,i);
        }
        else if(num[i]=='f' || num[i]=='F')
        {
            last_digit=15;
            result+=last_digit*pow(16,i);
        }
        else {
            last_digit=num[i];
        result+=last_digit*pow(16,i);
        }
    }
    cout<<result;
}
int main()
{
    string hexa;
    cout<<"Enter the hexadecimal number:";
    getline(cin,hexa);
    convert(hexa);
}

您的问题出在 else 的情况下,您将 num[i]char 转换为它的 ascii 等价物。因此,例如,如果您尝试转换 A0,则 0 会转换为 48 而不是 0。 要更正,您应该将 num[i] 转换为等效整数(不是 asci)。

为此,替换:

else {
            last_digit=num[i];
        result+=last_digit*pow(16,i);

  else {
            last_digit = num[i]-'0';
            result+=last_digit*pow(16,i);
    }

新行中,last_digit = num[i]-'0';等同于last_digit = (int)num[i]-(int)'0';[=24=的表示代码减去num[i]中任意一位数的表示代码]

之所以有效,是因为 C++ 标准保证 10 位十进制数字的数字表示是连续的且按递增顺序(official ref iso-cpp 并在 chapter 2.3paragraph 3

因此,如果您采用任何一位数的表示形式(例如 ascii 代码)num[i] 并用表示代码“0”(在 ascii 中为 48)减去它,您直接获得数字本身作为整数值。

更正后的执行示例将给出:

A0
160

F5
245

一点代码回顾: 你在用很多 result+=last_digit*pow(16,i); 重复你自己。您只能在循环结束时执行一次。但那是另一回事了。

你的代码非常复杂而且错误。

你可能想要这个:

void int convert(string num)
{
  long int last_digit;
  int s = num.length();
  int i;
  long long int result = 0;

  for (i = 0; i < s; i++)
  {
    result <<= 4;                     // multiply by 16, using pow is overkill
    auto digit = toupper(num[i]);     // convert to upper case

    if (digit >= 'A' && digit <= 'F')
      last_digit = digit - 'A' + 10;   // digit is in range 'A'..'F'
    else
      last_digit = digit - '0';        // digit is (hopefully) in range '0'..'9'

    result += last_digit;
  }

  cout << result;
}

但这还是不太好:

  • 函数应该return long long int 而不是打印结果
  • 其他一些事情可以更优雅地完成

所以更好的版本是这样的:

#include <iostream>
#include <string>

using namespace std;

long long int convert(const string & num)  // always pass objects as const & if possible
{
  long long int result = 0;

  for (const auto & ch : num)      // use range based for loops whenever possible
  {
    result <<= 4;
    auto digit = toupper(ch);

    long int last_digit;           // declare local variables in the inner most scope

    if (digit >= 'A' && digit <= 'F')
      last_digit = digit - 'A' + 10;
    else
      last_digit = digit - '0';

    result += last_digit;
  }

  return result;
}

int main()
{
  string hexa;
  cout << "Enter the hexadecimal number:";
  getline(cin, hexa);
  cout << convert(hexa);
}

上面的代码假定要转换的字符串仅包含十六进制字符,因此仍有更多改进的余地。理想情况下,应该以某种方式检查无效字符。我把它留作练习。

last_digit = digit - 'A' + 10;假定字母A到F的代码是连续的,理论上可能不是这种情况。但是,您遇到不是这种情况的编码方案的可能性接近于零。当今使用的绝大多数计算机系统都使用 ASCII 编码方案,有些使用 EBCDIC,但是在这两种编码方案中,字母 A 到 F 的字符代码都是连续的。我不知道今天使用的任何其他编码方案。

你使问题变得比你需要的更复杂(std::pow 也有点慢)。 std::stoul 可以取一个数字基数并自动为您转换为整数:

#include <string>
#include <iostream>

std::size_t char_count{0u};
std::string hexa{};
std::getline(std::cin, hexa);
hexa = "0x" + hexa;
unsigned long value_uint = std::stoul(hexa, &char_count, 16);