文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Java 泛型详解:从基础到实战

2024-11-28 16:40

关注

一、泛型基础

泛型的核心思想是将类型参数化。这意味着我们可以编写适用于多种类型的代码,而不需要为每种类型都编写一个版本。

基本语法:

public class ClassName {
    private T field;
    
    public void setField(T field) {
        this.field = field;
    }
    
    public T getField() {
        return field;
    }
}

这里的 T 是类型参数,它可以在类的定义中被当作类型使用。

二、泛型类

泛型类是最常见的泛型用法之一。让我们通过一个例子来理解泛型类:

public class Box {
    private T content;

    public void put(T item) {
        this.content = item;
    }

    public T get() {
        return content;
    }
}

// 使用泛型类
Box stringBox = new Box<>();
stringBox.put("Hello, Generics!");
String message = stringBox.get();
System.out.println(message);  // 输出: Hello, Generics!

Box intBox = new Box<>();
intBox.put(42);
int number = intBox.get();
System.out.println(number);  // 输出: 42

在这个例子中,Box 类可以存储任何类型的对象,类型安全由编译器保证。

三、泛型方法

泛型方法允许在方法级别引入类型参数,即使它们所在的类不是泛型类。

public class Utilities {
    public static  void swapElements(T[] array, int i, int j) {
        T temp = array[i];
        array[i] = array[j];
        array[j] = temp;
    }
}

// 使用泛型方法
String[] names = {"Alice", "Bob", "Charlie"};
Utilities.swapElements(names, 0, 2);
System.out.println(Arrays.toString(names));  // 输出: [Charlie, Bob, Alice]

在这个例子中,swapElements 方法可以交换任何类型数组的元素。

四、泛型接口

泛型接口允许我们定义可以被不同类型实现的契约。

public interface Repository {
    T findById(ID id);
    void save(T entity);
    void delete(T entity);
}

public class UserRepository implements Repository {
    @Override
    public User findById(Long id) {
        // 实现查找用户的逻辑
    }

    @Override
    public void save(User user) {
        // 实现保存用户的逻辑
    }

    @Override
    public void delete(User user) {
        // 实现删除用户的逻辑
    }
}

这个例子展示了一个通用的 Repository 接口,它可以被不同的实体类型实现。

五、泛型通配符

通配符提供了更灵活的类型匹配。有三种主要的通配符用法:

public class WildcardExample {
    // 无界通配符
    public static void printList(List list) {
        for (Object item : list) {
            System.out.println(item);
        }
    }

    // 上界通配符
    public static double sumOfList(List list) {
        double sum = 0.0;
        for (Number n : list) {
            sum += n.doubleValue();
        }
        return sum;
    }

    // 下界通配符
    public static void addNumbers(List list) {
        for (int i = 1; i <= 10; i++) {
            list.add(i);
        }
    }
}

这些例子展示了不同通配符的用法,允许更灵活地处理泛型类型。

六、类型擦除

Java 的泛型是通过类型擦除实现的,这意味着泛型信息在运行时是不可用的。编译器会将泛型类型替换为它们的上界(通常是 Object)。

public class ErasureExample {
    public static void main(String[] args) {
        List stringList = new ArrayList<>();
        List intList = new ArrayList<>();

        System.out.println(stringList.getClass() == intList.getClass());  // 输出: true
    }
}

这个例子说明了 List 和 List 在运行时是相同的类型。

七、泛型约束

虽然 Java 不支持直接的泛型约束(如 C# 中的 where 子句),但我们可以通过上界通配符来实现类似的效果:

public class ConstraintExample> {
    public T findMax(List list) {
        if (list.isEmpty()) {
            return null;
        }
        T max = list.get(0);
        for (T item : list) {
            if (item.compareTo(max) > 0) {
                max = item;
            }
        }
        return max;
    }
}

这个例子中,T 被约束为必须实现 Comparable 接口。

八、Spring Boot 中的泛型应用

Spring Boot 广泛使用泛型来提供灵活和可重用的组件。以下是一些常见的应用:

1.泛型 Repository

@Repository
public interface UserRepository extends JpaRepository {
    List findByLastName(String lastName);
}

这里 JpaRepository 使用了泛型来指定实体类型和主键类型。

2.泛型 Service

@Service
public class GenericService {
    @Autowired
    private JpaRepository repository;

    public T findById(ID id) {
        return repository.findById(id).orElse(null);
    }

    public List findAll() {
        return repository.findAll();
    }

    public T save(T entity) {
        return repository.save(entity);
    }
}

@Service
public class UserService extends GenericService {
    // 可以添加特定于 User 的方法
}

这个例子展示了如何创建一个通用的 Service 类,然后针对特定实体进行扩展。

3.泛型 Controller

@RestController
@RequestMapping("/api")
public class GenericController {
    @Autowired
    private GenericService service;

    @GetMapping("/{id}")
    public ResponseEntity getById(@PathVariable ID id) {
        T entity = service.findById(id);
        return entity != null ? ResponseEntity.ok(entity) : ResponseEntity.notFound().build();
    }

    @PostMapping
    public ResponseEntity create(@RequestBody T entity) {
        T savedEntity = service.save(entity);
        return ResponseEntity.ok(savedEntity);
    }
}

@RestController
@RequestMapping("/api/users")
public class UserController extends GenericController {
    // 可以添加特定于 User 的端点
}

这个例子展示了如何创建一个通用的 RESTful Controller,然后针对特定实体进行扩展。

九、泛型最佳实践

十、常见问题和解决方案

1.泛型数组问题

问题:不能直接创建泛型数组。 解决方案:使用 ArrayList 或其他泛型集合,或者使用反射创建数组。

public class GenericArray {
    private T[] array;

    @SuppressWarnings("unchecked")
    public GenericArray(int size) {
        // 使用反射创建泛型数组
        array = (T[]) Array.newInstance(Object.class, size);
    }

    public void set(int index, T item) {
        array[index] = item;
    }

    public T get(int index) {
        return array[index];
    }
}

2.类型擦除导致的问题

问题:由于类型擦除,某些泛型操作在运行时可能会失败。 解决方案:使用类型标记(Type Token)或反射。

public class TypeReference {
    private final Type type;

    protected TypeReference() {
        Type superclass = getClass().getGenericSuperclass();
        if (superclass instanceof ParameterizedType) {
            this.type = ((ParameterizedType) superclass).getActualTypeArguments()[0];
        } else {
            throw new IllegalArgumentException("Invalid TypeReference");
        }
    }

    public Type getType() {
        return type;
    }
}

// 使用示例
TypeReference> typeRef = new TypeReference>() {};
Type listStringType = typeRef.getType();

3.泛型和重载

问题:由于类型擦除,某些泛型方法重载可能会导致编译错误。 解决方案:使用不同的方法名或添加额外的参数来区分方法。

public class OverloadingExample {
    // 这两个方法在编译后会产生冲突
    // public void process(List list) { }
    // public void process(List list) { }

    // 解决方案
    public void processStrings(List list) { }
    public void processIntegers(List list) { }
}

结语

Java 泛型是一个强大的语言特性,它提供了类型安全、代码重用和API设计灵活性。通过本文,我们详细探讨了泛型的基本概念、各种用法以及在 Spring Boot 中的应用。掌握泛型不仅能让你写出更安全、更灵活的代码,还能帮助你更好地理解和使用Java生态系统中的各种框架和库。

需要注意,正确使用泛型可以显著提高代码质量和可维护性。然而,过度使用泛型可能会导致代码复杂度增加。因此,在实际应用中,需要权衡使用泛型带来的好处和可能的复杂性。

通过不断实践和深入理解泛型的工作原理,你将能够更加得心应手地在各种场景下运用泛型,从而编写出更加健壮和灵活的 Java 应用程序。

来源:源话编程内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯