上一篇地址:持续总结中!2024年面试必问 100 道 Java基础面试题(三十九)-CSDN博客
七十九、Java泛型中的T、R、K、V、E分别指什么?
在Java泛型中,T、R、K、V 和 E 是类型参数的常见占位符,它们用来表示泛型类型中的类型变量。这些占位符通常用于泛型类、泛型接口或泛型方法的定义中,它们分别代表不同的类型,但在泛型的上下文中,它们是等价的,可以互换使用。以下是每个占位符的常见用法:
-  
T(Type):T是最常用的泛型类型参数占位符,代表任意类型。它通常用于表示泛型类或方法中的类型,可以是任何类或接口。
public class GenericClass<T> {private T value;public GenericClass(T value) {this.value = value;}public T getValue() {return value;} } -  
R(Return Type):R通常用于泛型方法的返回类型。它代表方法返回值的类型,这个类型可以由方法的泛型参数决定。
public class UtilityClass {public static <R> R convertType(Object obj, Class<R> clazz) {// 转换obj到Class<R>指定的类型return null;} } -  
K和V:K和V常用于表示键(Key)和值(Value)的类型,特别是在定义关联数组或映射(如Map)时。
public class Pair<K, V> {private final K key;private final V value;public Pair(K key, V value) {this.key = key;this.value = value;}// getters and other methods } -  
E(Element):E通常用于表示元素类型,比如在定义一个可以存储任意类型元素的集合类时。
public class ElementList<E> {private List<E> elements;public void addElement(E element) {elements.add(element);}// 其他方法 } 
这些泛型类型参数占位符的使用约定并非强制性的,它们只是常见的用法,目的是为了让代码更具可读性。开发者也可以使用其他任何有效的标识符作为泛型类型参数的名称,例如MyType、InputType、OutputType 等。
示例代码
public class Example<T, V> {private T id;private V name;public Example(T id, V name) {this.id = id;this.name = name;}public static <K, V> Map<K, V> createMap(K key, V value) {Map<K, V> map = new HashMap<>();map.put(key, value);return map;}public static void main(String[] args) {Example<Integer, String> example = new Example<>(1, "Example");Map<String, Integer> map = createMap("one", 1);}
} 
在这个示例中,Example 类使用了 T 和 V 作为泛型类型参数,而 createMap 静态方法使用了 K 和 V 来创建一个 Map 对象。
总结来说,泛型类型参数的名称可以根据上下文和开发者的偏好来选择,关键是保持代码的清晰和一致性。
八十、Java金额计算怎么避免精度丢失?
在Java中进行金额计算时避免精度丢失是一个重要的考虑因素,因为错误的结果可能会导致财务错误或不准确的报告。以下是一些避免精度丢失的方法:
使用合适的数据类型
-  
BigDecimal:BigDecimal类是Java提供的一个用于进行精确小数运算的类,它允许进行多精度的定点数运算,非常适合用于金额计算。
 -  
double和float:double和float类型是基于IEEE 754标准的浮点数,它们在表示非常大或非常小的数值时可能会有精度问题,因此不推荐用于精确的金额计算。
 
使用 BigDecimal 的最佳实践
 
-  
避免使用构造器:
- 不推荐使用 
BigDecimal的构造器(如BigDecimal(double))从double或float类型创建BigDecimal实例,因为这可能会引入初始的精度问题。 
 - 不推荐使用 
 -  
使用字符串构造器:
- 使用 
BigDecimal(String)构造器从字符串创建BigDecimal对象,这样可以确保数值的精确表示。 
 - 使用 
 -  
链式操作:
- 使用 
BigDecimal的链式方法调用,如add()、subtract()、multiply()和divide()等,来进行运算。 
 - 使用 
 -  
设置小数点后的位数:
- 在进行除法运算时,可以使用 
setScale()方法设置小数点后的位数和舍入模式。 
 - 在进行除法运算时,可以使用 
 
示例代码
import java.math.BigDecimal;
import java.math.RoundingMode;public class MonetaryCalculation {public static void main(String[] args) {// 使用字符串构造BigDecimal对象BigDecimal amount1 = new BigDecimal("100.5");BigDecimal amount2 = new BigDecimal("2.5");// 进行加法和减法运算BigDecimal sum = amount1.add(amount2);BigDecimal difference = amount1.subtract(amount2);// 进行乘法和除法运算,并设置结果的小数点后位数BigDecimal product = amount1.multiply(amount2);BigDecimal quotient = amount1.divide(amount2, 2, RoundingMode.HALF_UP);System.out.println("Sum: " + sum);System.out.println("Difference: " + difference);System.out.println("Product: " + product);System.out.println("Quotient: " + quotient);}
} 
在这个示例中,我们使用 BigDecimal 类进行了金额的加法、减法、乘法和除法运算。注意,我们使用字符串构造器创建 BigDecimal 对象,并在除法运算后设置了小数点后的位数和舍入模式。
注意事项
- 避免类型转换:在进行金额计算时,避免将 
BigDecimal转换为double或float类型。 - 舍入模式:在设置小数点位数时,选择合适的舍入模式非常重要,如 
RoundingMode.HALF_UP是最常用的模式之一。 - 维护精度:在设计财务系统时,始终保持足够的精度以避免舍入误差的累积。
 
总结来说,为了避免Java中的金额计算精度丢失,推荐使用 BigDecimal 类,并遵循最佳实践,如使用字符串构造器创建对象,链式调用运算方法,并在需要时设置小数点后的位数和舍入模式。