十六进制转十进制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.3
和 paragraph 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);
请帮我找出这个程序中的错误,对我来说它看起来是正确的,我已经检查过了,但是它给出了错误的答案。 在这个程序中,我明确检查了 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.3
和 paragraph 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);