在编程世界中,代码的可重用性和可维护性是至关重要的。为了实现这些目标,Java 5 引入了一种名为泛型(Generics)的强大功能。本文将详细介绍 Java 泛型的概念、优势和局限性,以及如何在实际编程中应用泛型。泛型是一种允许程序员在类、接口和方法中使用类型参数的编程范式。这意味着,我们可以编写一段具有通用性质的代码,从而减少重复的代码,同时提高代码的可读性和可维护性。泛型允许我们在编译时检查类型安全,减少运行时类型转换错误,使得代码更加健壮。此外,泛型还有助于减少代码中的类型强制转换,从而提高代码质量。在接下来的文章中,我们将深入探讨 Java 泛型的各个方面,包括类型参数、泛型类、泛型接口、泛型方法、类型擦除、泛型的实际应用和最佳实践等。无论您是初学者还是有经验的 Java 开发者,这篇文章都将帮助您更好地理解和掌握 Java 泛型的概念和应用。public class Box<T> {
// ...
}public class Box<T> {
private T content;
public void setContent(T content) {
this.content = content;
}
public T getContent() {
return content;
}
}Box<Integer> integerBox = new Box<>();
integerBox.setContent(42);
Integer value = integerBox.getContent();public interface Comparable<T> {
int compareTo(T other);
}public class Person implements Comparable<Person> {
private String name;
// ...
@Override
public int compareTo(Person other) {
return this.name.compareTo(other.name);
}
}public static <T> void swap(T[] array, int i, int j) {
T temp = array[i];
array[i] = array[j];
array[j] = temp;
}Integer[]public class NumericBox<T extends Number> {
private T content;
// ...
}NumericBox<Integer> integerBox = new NumericBox<>(); // 有效
NumericBox<String> stringBox = new NumericBox<>(); // 编译错误public class PrintableComparableBox<T extends Comparable<T> & Printable> {
private T content;
// ...
}public static void processNumbers(List<? extends Number> numbers) {
// ...
}public class Box<T> {
private T content;
// ...
}
// 保留泛型参数
public class SpecialBox<T> extends Box<T> {
// ...
}
// 为泛型参数指定具体类型
public class IntegerBox extends Box<Integer> {
// ...
}public interface Comparable<T> {
int compareTo(T other);
}
// 保留泛型参数
public class GenericComparable<T> implements Comparable<T> {
// ...
}
// 为泛型参数指定具体类型
public class StringComparable implements Comparable<String> {
// ...
}public class Parent {
public <T> T doSomething(T input) {
// ...
}
}
public class Child extends Parent {
@Override
public <T> T doSomething(T input) {
// ...
}
}public class Box<T> {
private T content;
public void setContent(T content) {
this.content = content;
}
public T getContent() {
return content;
}
}public class Box {
private Object content;
public void setContent(Object content) {
this.content = content;
}
public Object getContent() {
return content;
}
}T[] array = new T[10]; // 编译错误Object[] array = new Object[10];
array[0] = new Integer(42);
T value = (T) array[0];public static void printBoxContent(Box<?> box) {
System.out.println(box.getContent());
}public static void processNumbers(List<? extends Number> numbers) {
// ...
}Box<?> box = new Box<?>(); // 编译错误public static void setBoxContent(Box<?> box, Object content) {
box.setContent(content); // 编译错误
}public static <T> void setBoxContent(Box<T> box, T content) {
box.setContent(content);
}List<String> names = new ArrayList<>();
names.add("Alice");
names.add("Bob");
for (String name : names) {
System.out.println(name);
}public interface Transformer<T, R> {
R transform(T input);
}
public class UpperCaseTransformer implements Transformer<String, String> {
@Override
public String transform(String input) {
return input.toUpperCase();
}
}public static <T> void swap(T[] array, int i, int j) {
T temp = array[i];
array[i] = array[j];
array[j] = temp;
}// 使用泛型类型参数
public class Box<T> {
private T content;
// ...
}
// 不推荐的做法:使用 Object 替代泛型类型参数
public class Box {
private Object content;
// ...
}// 使用有界通配符
public void processBoxes(List<? extends Box> boxes) {
// ...
}
// 不推荐的做法:不使用通配符
public void processBoxes(List<Box> boxes) {
// ...
}public static <T> T getFirst(List<T> list) {
if (!list.isEmpty()) {
return list.get(0);
}
return null;
}// 使用具体类型
List<String> names = new ArrayList<>();
// 使用通配符
List<?> items = new ArrayList<>();
// 不推荐的做法:使用原始类型
List names = new ArrayList();public class KeyValuePair<K, V> {
private K key;
private V value;
public KeyValuePair(K key, V value) {
this.key = key;
this.value = value;
}
// .../**
* This class represents a cache for storing objects of type V, keyed by objects of type K.
*
* @param <K> the type of the keys used to access the cache
* @param <V> the type of the objects stored in the cache
*/
public class Cache<K, V> {
// ...
}很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere
我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"
我可以得到Infinity和NaNn=9.0/0#=>Infinityn.class#=>Floatm=0/0.0#=>NaNm.class#=>Float但是当我想直接访问Infinity或NaN时:Infinity#=>uninitializedconstantInfinity(NameError)NaN#=>uninitializedconstantNaN(NameError)什么是Infinity和NaN?它们是对象、关键字还是其他东西? 最佳答案 您看到打印为Infinity和NaN的只是Float类的两个特殊实例的字符串
我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
两者都可以defsetup(options={})options.reverse_merge:size=>25,:velocity=>10end和defsetup(options={}){:size=>25,:velocity=>10}.merge(options)end在方法的参数中分配默认值。问题是:哪个更好?您更愿意使用哪一个?在性能、代码可读性或其他方面有什么不同吗?编辑:我无意中添加了bang(!)...并不是要询问nobang方法与bang方法之间的区别 最佳答案 我倾向于使用reverse_merge方法:option
我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano
有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳