Conversions of numeric types to narrower types can result in lost or misinterpreted data if the value of the wider type is outside the range of values of the narrower type. Therefore, all narrowing conversions must be guaranteed safe by range-checking the value before conversion.
There are 22 possible narrowing primitive conversions in Java. According to the Java Language Specification, §5.1.3, "Narrowing Primitive Conversions" ,
shorttobyteorcharchartobyteorshortinttobyte,short, orcharlongtobyte,short,char, orintfloattobyte,short,char,int, orlongdoubletobyte,short,char,int,long, orfloat
Narrowing primitive conversions are allowed in cases where the value of the wider type is within the range of the narrower type.
Integer Narrowing
Integer type ranges are defined by the JLS, §4.2.1, "Integral Types and Values", [JLS 2005] and are also described in rule NUM00-J. Detect or prevent integer overflow.
The table below presents the rules for narrowing primitive conversions of integer types. In the table, for an integer type T, n represents the number of bits used to represent the resulting type T (precision).
From |
To |
Description |
Possible Resulting Errors |
|---|---|---|---|
signed integer |
integral type |
Keeps only |
Lost or misinterpreted data |
char |
integral type |
Keeps only |
Magnitude error; negative number even though |
When integers are cast to narrow data types, the magnitude of the numeric value and the corresponding sign can be affected. Consequently, data can be lost or misinterpreted.
Floating-Point to Integer Conversion
Floating-point conversion to an integral type T is a two step procedure:
1. When converting a floating-point value to an int or long and the value is a NaN, an int or long, a zero value is produced. Otherwise, if the value is not infinity, it is rounded towards zero to an integer value V:
- If
Tislong, andVcan be represented as along, thelongvalueVis produced. - If
Vcan be represented as anint, then theintvalueVis produced.
Otherwise
- The value is negative infinity or a value too negative to be represented, and
Integer.MIN_VALUEorLong.MIN_VALUEis produced. - The value is positive infinity or a value too positive to be represented,
Integer.MAX_VALUEorLong.MAX_VALUEis produced.
2. If T is byte, char, or short, the result of the conversion is the result of a narrowing conversion to type T of the result of the first step
See the JLS, §5.1.3, "Narrowing Primitive Conversions" for more information.
Other Conversions
Narrower primitive types can be cast to wider types without affecting the magnitude of numeric values. See the JLS, §5.1.2, Widening Primitive Conversion" for more information. Conversion from int or long to float, or long to double can lead to loss of precision (loss of least significant bits). No runtime exception occurs despite this loss.
Note that conversions from float to double or double to float can also lose information about the overall magnitude of the converted value. See rule NUM06-J. Use the strictfp modifier for floating-point calculation consistency across platforms for additional information.
Noncompliant Code Example (Integer Narrowing)
In this noncompliant code example, a value of type int is converted to a value of type byte without range checking.
class CastAway {
public static void main(String[] args) {
int i = 128;
workWith(i);
}
public static void workWith(int i)
byte b = (byte) i; // b has value -128
// work with b
}
}
The resulting value may be unexpected because the initial value (128) is outside of the range of the resulting type.
Compliant Solution (Integer Narrowing)
This compliant solution validates that the value stored in the wider integer type is within the range of the narrower type before converting to the narrower type.
class CastAway {
public static void workWith(int i)
//check if i is within byte range
if ((i < Byte.MIN_VALUE) || (i > Byte.MAX_VALUE)) {
throw new ArithmeticException("Value is out of range");
}
byte b = (byte) i;
// work with b
}
}
Noncompliant Code Example (Floating-Point Conversion to Integer)
The narrowing primitive conversions in this noncompliant code example suffer from loss in the magnitude of the numeric value, as well as a loss of precision.
float i = Float.MIN_VALUE; float j = Float.MAX_VALUE; short b = (short) i; short c = (short) j;
The minimum and maximum float values are converted to minimum and maximum int values (0x80000000 and 0x7fffffff, respectively). The resulting short values are the lower 16 bits of these values (0x0000 and 0xffff). The resulting final values (0 and -1) could be unexpected.
Compliant Solution (Floating-Point to Integer Conversion)
This compliant solution range checks both the i and j variables before converting to the resulting integer type.
float i = Float.MIN_VALUE;
float j = Float.MAX_VALUE;
if ((i < Short.MIN_VALUE) || (i > Short.MAX_VALUE) ||
(j < Short.MIN_VALUE) || (j > Short.MAX_VALUE)) {
throw new ArithmeticException ("Value is out of range");
}
short b = (short) i;
short c = (short) j;
//other operations
Noncompliant Code Example (double to float Conversion)
The narrowing primitive conversions in this noncompliant code example suffer from a loss in the magnitude of the numeric value, as well as a loss of precision. Because Double.MAX_VALUE is larger than Float.MAX_VALUE, c receives the value infinity and because Double.MIN_VALUE is smaller than Float.MIN_VALUE, d receives the value 0.
double i = Double.MIN_VALUE; double j = Double.MAX_VALUE; float b = (float) i; float c = (float) j;
Compliant Solution (double to float Conversion)
Perform range checks on both i and j variables before proceeding with the conversions.
double i = Double.MIN_VALUE;
double j = Double.MAX_VALUE;
if ((i < Float.MIN_VALUE) || (i > Float.MAX_VALUE) ||
(j < Float.MIN_VALUE) || (j > Float.MAX_VALUE)) {
throw new ArithmeticException ("Value is out of range");
}
float b = (float) i;
float c = (float) j;
//other operations
Exceptions
NUM12-EX0: Java's narrowing conversions are both well-defined and portable. The effects of narrowing on integral types can be easily reproduced in code; however the effects of narrowing on floating-point types, and between floating-point types and integral types cannot be easily represented. Knowledgeable programmers may intentionally apply narrowing conversions involving floating-point types in contexts where their output is both expected and reasonable. Consequently, narrowing conversions are permitted when the code contains comments that document both the use of narrowing conversions and that the potential for truncation has been anticipated. A suitable comment might read: "// Deliberate narrowing cast of i; possible truncation OK"
This exception does not permit narrowing conversions without range-checking among integral types. The following code example demonstrates how to perform explicit narrowing from a long to an int where range-checking is not required.
long value = /* initialize */; int i = (int) (value % 0x100000000); // 2^32
The range-checking is unnecessary because the truncation that is normally implicit in a narrowing conversion is made explicit. The compiler will optimize the operation away; hence it incurs no performance penalty.
Similar operations may be used for converting to other integral types.
Risk Assessment
Casting a numeric value to a narrower type can result in information loss related to the sign and magnitude of the numeric value. As a result, data can be misrepresented or interpreted incorrectly.
Rule |
Severity |
Likelihood |
Remediation Cost |
Priority |
Level |
|---|---|---|---|---|---|
NUM12-J |
low |
unlikely |
medium |
P2 |
L3 |
Automated Detection
Automated detection of narrowing conversions on integral types is straightforward. Determining whether such conversions correctly reflect the intent of the programmer is infeasible in the general case. Heuristic warnings could be useful.
Related Guidelines
INT31-C. Ensure that integer conversions do not result in lost or misinterpreted data |
||||
|
FLP34-C. Ensure that floating point conversions are within range of the new type |
|||
INT31-CPP. Ensure that integer conversions do not result in lost or misinterpreted data |
||||
|
FLP34-CPP. Ensure that floating point conversions are within range of the new type |
|||
<ac:structured-macro ac:name="unmigrated-wiki-markup" ac:schema-version="1" ac:macro-id="a6189b21-c376-410c-b52d-ce68b28645ec"><ac:plain-text-body><![CDATA[ |
[ISO/IEC TR 24772:2010 |
http://www.aitcnet.org/isai/] |
"Numeric Conversion Errors [FLC]" |
]]></ac:plain-text-body></ac:structured-macro> |
CWE ID 681, "Incorrect Conversion between Numeric Types" |
||||
|
CWE ID 197, "Numeric Truncation Error" |
Bibliography
NUM11-J. Do not rely on the default string representation of floating-point values 03. Numeric Types and Operations (NUM) NUM13-J. Convert integers to floating-point for floating-point operations