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’ ,也可以用类似的方法来表示。

你可能感兴趣的:(java中int与char之间的互相转化)