java中int与char之间的互相转化
引言
在介绍java中int与char之间的互相转化之前,让我们先简单的回顾一些会涉及到的知识。
数据类型
在介绍int与char的转换之前,我们先来回顾一下java中的基本数据类型:
类型 | 大小 | 包装器类型 |
---|---|---|
boolean | - | Boolean |
char | 16-bit | Character |
byte | 8 bits | Byte |
short | 16 bits | Short |
int | 32 bits | Integer |
long | 64 bits | Long |
float | 32 bits | Float |
double | 64 bits | Double |
ascii码
其次,我们要知道,在计算机中,所有的数据在存储和运算时都要使用二进制数表示,例如,像a、b、c、d这样的字母以及各种符号,还有作为字符的数字,都要使用二进制数字来表示,因此需要一种二进制码与这些字符的对应关系。
而现在普遍遵守的对应关系就是ASCII编码,在这里贴几个字母的ascii码作为我们的测试对象:
ascii码(十进制) | 字符 |
---|---|
97 | ‘a’ |
98 | ‘c’ |
99 | ‘d’ |
100 | ‘e’ |
更多的ascii码的对应关系可以在这个链接里看到:更多ascii码
int与char的互相转换
在引言中,我们可以看到,int
类型是一个32位的数据类型,因为其位有符号数,所以,其取值范围为:-2^31 至 2^31 - 1。
而char
为16位的数据,为无符号数,其范围为:0 至 2 ^ 32 -1,即 0 - 65535,用十六进制码来看,则为:’\u0000’ - ‘\uffff’。
再从前面引言中对于ascii码的描述,我们可以看出,无论是什么字符,在计算机中,其实也是以数字(本质为一个二进制数)的形式存储的,因此,在java中,无论是一个字符,还是一个字符型变量,实则也是一个数字,所以,可以直接将一个(不超过char的范围的)数字赋值给一个char变量,即可输出其结果:
char c1 = 97;
System.out.println(c1);
输出:
a
可以看到,正确输出了97的ascii码所对应的字符:‘a’。
同样的,我们可以将一个字符赋值给一个int变量,因为字符也是一个数字嘛
int num1 = 'a';
System.out.println(num1);
输出:
97
甚至,我们可以将一个数字与一个字符在一个表达式中进行计算:
char c2 = 'a' + 1;
int num2 = 'a' + 1;
System.out.println("c2: " + c2);
System.out.println("num2: " + num2);
System.out.println("(char) num2: " + (char) num2);
输出:
c2: b
num2: 98
(char) num2: b
因为字符’b’所对应的ascii码是98,比’a’所对应的97要大1,所以,将’a’加1之后,得到的便是字符’b’了。同时,将int型的num2直接进行强制转换,也可以得到所期望的’b’。
可能出现的错误
从上面一节的描述,我们可以看出,char与int之间的转换还是相当容易的,但是,也有几点需要注意,否则,会导致程序中的错误。
从引言中的基本数据类型的回顾中,我们可以看出,int是比char范围更大的,因此,这些错误主要便是会出现在从int转到char的过程中。
char的大小过界
由于char是一个16位的无符号的数据类型,所以其大小范围是:0 至 2 ^ 32 -1,即 0 - 65535,用十六进制码来看,则为:’\u0000’ - ‘\uffff’。
虽然我们可以直接以一个数字赋给一个char型的变量,但是,若我们赋的值的大小大于了65536,则会报出错误来,如下代码:
char c3 = 65535;
System.out.println(c3);
char c4 = 65536;
System.out.println(c4);
输出:
Error:(33, 19) java: 不兼容的类型: 从int转换到char可能会有损失
报错是报在了上面代码中的第3行,即char c4 = 65536
,这一行,因此,当对一个char类型赋值的时候,要注意,所赋的值不能超过了65535。
将int变量直接赋给char变量
在上面,我们看到了,可以直接将一个数字赋值给char类型的变量,那是不是我们可以直接将一个int类型的变量赋给一个char类型的变量呢?
答案是否定的,这里,涉及到隐式转换与显式转换的知识。简单地来讲,就是只能从一个范围较小的数字转换到一个范围较大的数字,如32位的int类型到64位的long是可以隐式转换的,即可以直接把一个int类型的变量赋给一个long类型的变量。
类似的,我们可以直接把一个16位的char类型的变量赋给一个32位的int类型的变量,但是,反之便不可以,需要用显式转换。
首先给出一个反例如下:
int num5 = 97;
char c5 = num5;
我这里,本地用的IDE是IDEA,在写下这段代码的时候,就已经标红,报出了错误,提示信息为incompatible types, Required: char, Found: int
,即类型不匹配,而如果我们直接运行,或者用javac来编译的话,都会报出如下的错误:
Error:(34, 19) java: 不兼容的类型: 从int转换到char可能会有损失
而如果我们先初始化一个char型的变量c6
,将这个变量赋给一个int型的变量呢?
char c6 = 97;
int num6 = c6;
System.out.println("c6: " + c6);
System.out.println("num6: " + num6);
输出:
c6: a
num6: 97
是可以正确地输出的,也可以看出从char到int是可以隐式转换的。
如果我们确实需要char型到int型的话,需要显式的强制转换:
int num7 = 97;
char c7 = (char) num7;
System.out.println(c7);
输出:
a
这里,由于是强制转换,将一个较多位数的类型转换成一个较少位数的类型,因此,可能会有损失发生,若int的数字大于65535,则会丢弃16位以外的数字,如下例:
int num8 = 65536 + 97;
char c8 = (char) num8;
System.out.println(c8);
// 打印num8与c8的十六进制输出
System.out.println(Integer.toHexString(num8));
System.out.println(Integer.toHexString(c8));
输出:
a
10061
61
可以看到,num8
的十六进制表示为0x0001 0061
,直接将第一个字节丢弃,变为0x0061
,即97,对应的字符为’a’。
一些技巧
在上面,我们看到了,char其实也是一个数字,所以,可以利用这样的特性,在编程的时候,使用一些小技巧。
比如,有一个字符为'1'
,我们想将其变为数字1,或者,我们想将一个int类型的数字1转换为字符'1'
,这时,我们就可以这样写:
// '1' -> 1
char c9 = '1';
int num9 = c9 - '0';
// 1 -> '1'
int num10 = 1;
char c10 = (char)(num10 + '0');
同样的,如果我们想用1 - 26来表示字母’a’ 到 ‘z’ ,也可以用类似的方法来表示。