包装类和认识泛型

news2024/10/6 22:22:43

目录

一、包装类

        1.1 基本数据类型和对应的包装类

        1.2 装箱和拆箱

        1.3 自动装箱和自动拆箱

二、泛型概念及引出

        2.1 泛型概念

        2.2 泛型引出

三、泛型类的使用

四、裸类型(了解)

五、泛型如何编译

        5.1 擦除机制

        5.2 为什么不能实例化泛型类型的数组

六、泛型的上界

七、泛型方法


一、包装类

        在Java中,基本数据类型不是继承Object,为了在泛型代码中支持基本数据类型,Java给每个基本数据类型都对应有一个包装类型。

        1.1 基本数据类型和对应的包装类

基本数据类型包装类
        byte        Byte
        short        Short
        int        Integer
        long        Long
        float        Float
        double        Double
        char        Character
        boolean        Boolean

                除了Integer和Character,其他基本数据类型的包装类都是首字母大写。

        1.2 装箱和拆箱

int i=10;
//装箱----新建一个Integer类型的对象,将i的值放入对象的某个属性中
Integer i1=Integer.valueOf(i);
Integer i2=new Integer(i);
//拆箱----将Integer对象中的值取出,放入一个基本数据类型中
int I=i1.intValue();

        1.3 自动装箱和自动拆箱

         在使用过程中,装箱和拆箱会会使我们写不少得到代码量,为了减少开发者的负担,Java提供了自动机制,即自动装箱和自动拆箱。

int i=5;
//自动装箱
Integer i1=i;
Integer i2=(Integer) i;
//自动拆箱
int i3=i1;
int i4=(int) i1;

        面试题:下面代码输出是什么?为什么?

public static void main(String[] args) {
    Integer a=127;
    Integer b=127;

    Integer c=128;
    Integer d=128;

    System.out.println(a==b);
    System.out.println(c==d);
}

        首先,Integer是包装类,==比较的是两个操作数存储的地址。

        其次,Integer的范围在-128~127时,value返回一个对应数组下标的值,否则就会new一个新的Integer对象。故a和b存储127不会new一个对象,指的是同一个地址,c和d存储128都会new一个对象,指的是不同的地址。

二、泛型概念及引出

        2.1 泛型概念

        一般的类和方法,只能使用具体的类型: 要么是基本类型,要么是自定义的类。如果要编写可以应用于多种类型的代码,这种刻板的限制对代码的束缚就会很大,于是在JDK1.5引入新的语法——泛型:适用于许多类型,从代码上看,实现了类型的参数化。

        2.2 泛型引出

        实现一个类,类中包含一个数组成员,使数组中可以存放任何类型的数据,也可以根据成员方法返回数组中某个下标的值。

        使用Object定义数组类型,使数组可以存任何类型的数据。

class MyArray{
    private Object[] array=new Object[10];
    public Object getValue(int pos){
        return this.array[pos];
    }
    public void  setValue(int pos,Object obj){
        this.array[pos]=obj;
    }
}
 public class MyArrayTextDemo {
     public static void main(String[] args) {
         MyArray myArray =new MyArray();
         myArray.setValue(0,10);
         myArray.setValue(1,"hello");
         String ret=myArray.getValue(1);//编译报错
         System.out.println(ret);
     }
}

        

        由以上代码可以发现:任何数据类型都可以存放,但在获取下标对应数据时编译报错,需要强制类型转化。

        

        虽然在这种情况下,数组可以存放任何类型的数据,但是,更多情况下,我们还是希望他只持有一种数据类型而不是同时持有这么多类型。所以,泛型的主要目的就是指定当前容器要持有什么类型的数据,让编译器进行检查。这时,就要把类型作为参数传递,需要什么类型就传什么类型。

        语法

class 泛型类名称<类型参数列表>{

}

class ClassName<T1,T2,T3,...,Tn>{

}

class 泛型类名称<类型形参列表> extends 继承类{

}
class ClassName<T1, T2, ..., Tn> extends ParentClass<T1> {

}

        改进上述代码:

class MyArray<T>{
    private T[] array=(T[]) new Object[10]; //可以使用但不推荐
    public T getValue(int pos){
        return this.array[pos];
    }
    public void  setValue(int pos,T obj){
        this.array[pos]=obj;
    }
}
class MyArray<T>{
    private Object[] array= new Object[10];//推荐使用
    public T getValue(int pos){
        return (T) this.array[pos];
    }
    public void  setValue(int pos,T obj){
        this.array[pos]=obj;
    }
}
public class MyArrayTextDemo {
    public static void main(String[] args) {
        MyArray<Integer> myArray =new MyArray();
        myArray.setValue(0,10);
        myArray.setValue(1,15);
        int ret= myArray.getValue(1);//不用强制转化
        System.out.println(ret);
     }
}

        注意:类名后的<T>代表占位符,表示当前类是一个泛型类。类型形参一般使用第一个大写字母表示,常用的名称有:E ——Element、K——Key、V——Value、N——Number、T——Type、S、U、V等;不能new泛型类型的数组

T[] arr = new T[5];//会报错

        在类型后加<Integer>指定当前存放数据的类型,编译器也会依此检查存放的元素类型是否有误

三、泛型类的使用

        语法

泛型类<类型实参> 变量名;        //定义泛型类引用

new 泛型类<类型实参>(构造方法实参)        //实例化一个泛型类对象

        示例

MyArray<Integer> list=new MyArray<Integer>();

        泛型只能接受类,所有的基本数据类型必须使用包装类

        类型推导

        编译器可以根据上下文推导出类型实参时,可以省略类型实参的填写。

MyArray<Integer> list = new MyArray<>();

四、裸类型(了解)

        裸类型是一个泛型类但没有带着类型实参,例如 MyArray list 就是一个裸类型。

MyArray list=new MyArray();

        注意:我们不要自己去使用裸类型,裸类型是为了兼容老版本的 API 保留的机制。

        小结:泛型是将数据类型参数化,进行传递;使用<T>表示当前类是一个泛型类;泛型类的优点:数据类型参数化,编译时自动进行类型检查和转换。

五、泛型如何编译

        5.1 擦除机制

        通过命令:javap -c 查看字节码文件

       可以发现所有的T都是Object,在编译过程中,将所有T替换为Object这种机制,称为擦除机制,Java的泛型机制是在编译级别实现的。编译器生成的字节码在运行期间不包含泛型的类型信息。

        5.2 为什么不能实例化泛型类型的数组

class MyArray<T> {
        public T[] array = (T[])new Object[10];


        public T getValue(int pos) {
                return this.array[pos];
        }
        public void setValue(int pos,T val) {
                this.array[pos] = val;
        }
        public T[] getArray() {
                return array;
        }
}
public static void main(String[] args) {
        MyArray<Integer> myArray1 = new MyArray<>();
        Integer[] str = myArray1.getArray();//报错
}

        原因:T替换为Object后将Object[]分配给Integer[]引用。通俗讲就是:返回的Object数组里,可能存放的是任何数据类型,可能是String,可能是Person,运行时,直接转给Integer类型的数组,编译器认为不安全。

六、泛型的上界

        定义泛型类时,有时需要对传入的类型变量做一定约束,可以通过类型边界来约束。
        语法

class 泛型类名称<类型形参 extends 类型边界> {
        ...
}

        示例

public class MyArray<E extends Number> {
        ...
}    //表示只接受 Number 的子类型作为 E 的类型实参

MyArray<Integer> list1; // 编译通过, Integer 是 Number 的子类型
MyArray<String> list2; // 编译错误,String 不是 Number 的子类型

        没有指定类型边界 E,可以看作 E extends Object。

        复杂示例

public class MyArray<E extends Comparable<E>> {
         ...
}        //E必须是实现了Comparable接口的

七、泛型方法

        语法

访问限定符 <类型形参列表> 返回值类型 方法名称(形参列表) {

        ...

}

        示例1

public class Text{

        public static <E> void swap(E[] array, int i, int j) {
                E t = array[i];
                array[i] = array[j];
                array[j] = t;
        }

}

        使用示例1:可以类型推导

Integer[] a = { ... };
swap(a, 0, 9);
String[] b = { ... };
swap(b, 0, 9);

        使用示例2:不使用类型推导

Integer[] a = { ... };
Text.<Integer>swap(a, 0, 9);


String[] b={ ... };

Text.<String>swep(b,0,9);

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1003299.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

windows拷贝文件到WSL最快速的方法

windows拷贝文件到WSL最快速的方法 在WSL用 sudo ls /mnt/* 列出系统所有的挂载盘&#xff0c;可以看到windows系统的所有盘都列出来了 找到你需要拷贝的文件位置&#xff0c;然后使用 cp命令就可以了。例如我要拷贝h盘下面的gitlab备份文件去wsl&#xff0c;用下面的命令即可…

C语言经典100(49)--#if #ifdef和#ifndef的综合应用

目录 #if #endif #ifdef #ifnde 代码 运行结果 #if 通过下面这段小代码来分析一下&#xff1a; #define DEBUG 1 //首先定义一个宏 #if DEBUG printf("Value of i: %d\n", i); printf("Value of j: %d\n", j); #endif 在预处理过程中&a…

Spring(五):Spring Boot 的配置和日志

Spring Boot中的配置文件起到了非常重要的作用。在Spring Boot中&#xff0c;可以通过配置文件来配置应用程序的一系列属性&#xff0c;包括数据库连接、服务器端口号、日志级别等等。通过配置文件&#xff0c;可以方便地对应用程序进行配置管理和集中管理。另外&#xff0c;Sp…

文心一言 VS 讯飞星火 VS chatgpt (91)-- 算法导论8.3 4题

四、用go语言&#xff0c;说明如何在 O(n) 时间内&#xff0c;对 0 到 n 3 − 1 n^3-1 n3−1 区间内的n个整数进行排序。 文心一言&#xff1a; 在 O(n) 时间内对 0 到 n^3-1 区间内的 n 个整数进行排序&#xff0c;可以使用基数排序&#xff08;Radix Sort&#xff09;算法…

如何快速、高效、免费地将Excel文件转换为PDF格式?

如果你经常需要处理Excel文件&#xff0c;那么你一定遇到过这样的情况&#xff1a;你想把Excel文件分享给别人&#xff0c;但是对方没有安装Office软件&#xff0c;或者使用的版本和你不一致&#xff0c;导致打开时出现格式错乱、公式失效、图表显示不正常等问题。这时候&#…

USB Server助力苏美达,Ukey连接虚拟前置机

众所周知&#xff0c;银行的专用Ukey&#xff0c;和所有USB设备一样&#xff0c;有一个无法在虚拟机中被识别和调用的大问题。近日&#xff0c;苏美达集团想将银企直连的前置机程序迁移到虚拟机中时&#xff0c;就遇到了这个问题——大量Ukey因为无法在虚拟机中识别连接&#x…

C++——vector

作者&#xff1a;几冬雪来 时间&#xff1a;2023年9月12日 内容&#xff1a;C部分vector知识讲解 目录 前言&#xff1a; 1.vector&#xff1a; 1.vector的本质&#xff1a; 2.vector书写&#xff1a; vector创建空间&#xff1a; vector与reserve和vector和resize&a…

当promise遇上generator该如何应对?记一次工作中遇到的问题

问题背景 我们项目中有个保存功能&#xff0c;但是这个保存是一个异步函数&#xff0c;内部很多逻辑&#xff0c;比如说校验表单数据&#xff0c;获取子组件数据&#xff0c;数据处理&#xff0c;数据提交给后端获取中间值&#xff0c;最后保存。说明一下&#xff0c;我们的项…

项目经理摆脱「计划无用」的秘诀!

项目经理面临的最大挑战是项目计划执行不到位&#xff0c;导致项目进度严重滞后。这种情况下&#xff0c;尽管他们手中握有项目计划&#xff0c;但实际上却形同虚设&#xff0c;几乎无法发挥应有的作用。 许多项目经理喜欢陈述一些既定的事实&#xff0c;强调一些难以克服的困…

虚幻动画系统概述

本文主要整理一下高层次的概述&#xff0c;方便后续查阅 1.动画流程 DCC产出动画文件 -> UE动画导入 -> 动画蓝图驱动&#xff08;类似unity的动画状态机&#xff09; ->动画后处理蓝图驱动&#xff08;例如修型骨&#xff0c;骨骼矫正等后期处理&#xff09; 2.动…

淘宝直通车质量分怎么提高?

1、直通车关键词新质量分是什么 &#xff08;1&#xff09;创意质量&#xff1a;就是获取流量的能力&#xff0c;和关键词在店铺中基本数据的展现 &#xff08;2&#xff09;买家体验&#xff1a;最终的成交能力&#xff0c;在有相关性的前提下&#xff0c;可以提升创意分和体…

【C++】C++多线程库的使用

C线程库的使用 一、线程库&#xff08;thread&#xff09;1、线程的id类2、线程对象的构造3、thread提供的其他成员函数4、this_thread命名空间5、线程函数的参数问题 二、互斥量库&#xff08;mutex&#xff09;1、mutex的种类2、lock_guard和unique_locklock_guardunique_loc…

marisa-trie——一个基于高效Trie树实现的快速高效字符串压缩存储、查询匹配工具实践

在前文中&#xff0c;讲到了因为实际项目的需要&#xff0c;调研了一下当前比较好用字符串查询匹配算法&#xff0c;感兴趣的话可以直接看下&#xff1a; 《pyahocorasick——基于AC自动机的python高效字符串匹配实践》 本文的主要目的同前文相同&#xff0c;这里主要是介绍一…

Linux 6.6 中的 SELinux 删除了 NSA 的引用

导读Security Enhanced Linux (SELinux) 二十年来一直是主线内核的一部分&#xff0c;它提供了一个实现访问控制安全策略的模块&#xff0c;现在广泛用于增强生产 Linux 服务器和其他系统的安全性。长期接触 Linux 的人可能不知道 SELinux 源自美国国家安全局 (NSA)。但是现在 …

centos 下 Makefile 独立模块编译ko

1、安装编译内核环境包 编译需要用到kernel 源码&#xff0c;centos 下需先安装 kernel-devel 包&#xff0c;要下与自己kernel 对应版本 yum install kernel-devel 2、首先从内核或自己写的模块&#xff0c;发到编译环境中 注&#xff1a;就像我自己拷贝一个 bcache 驱动的目…

pinduoduo(商品优惠券)API接口

为了进行电商平台 的API开发&#xff0c;首先我们需要做下面几件事情。 1&#xff09;开发者注册一个账号 2&#xff09;然后为每个pinduoduo应用注册一个应用程序键&#xff08;App Key) 。 3&#xff09;下载pinduoduo API的SDK并掌握基本的API基础知识和调用 4&#xff…

在Ubuntu系统中安装Docker

&#x1f468; 作者简介&#xff1a;大家好&#xff0c;我是Taro&#xff0c;前端领域创作者 ✒️ 个人主页&#xff1a;唐璜Taro &#x1f680; 支持我&#xff1a;点赞&#x1f44d;&#x1f4dd; 评论 ⭐️收藏 文章目录 前言一、Ubuntu是什么&#xff1f;二、安装Docker1.…

2023-简单点-开启防火墙后,ping显示请求超时;windows共享盘挂在不上

情景描述 树莓派 挂载 windows共享盘 之前一直可以&#xff0c;突然有一天不行了 ping xxxx不通了 一查&#xff0c;或许是服务器被同事开了防火墙&#xff0c;默认关闭了ping的回显 操作&#xff1a; 开启ping回显cmd ping通了&#xff0c;但是挂载还是不行, 显示 dmesg命…

Mac电脑其他文件太占内存?如何进行删除

Mac老用户都知道在我们查看Mac内存时都会发现有一条“其他文件”占比非常高&#xff0c;它是Mac储存空间中的“其他”数据包含不可移除的移动资源&#xff0c;如&#xff0c;Siri 语音、字体、词典、钥匙串和 CloudKit 数据库、系统无法删除缓存的文件等。这些“其他文件”无用…

故障治理:如何进行故障复盘

故障复盘的重要性无需多说&#xff0c;每一次故障都是宝贵的学习机会&#xff0c;本人接手故障复盘工作已经半年有余&#xff0c;从一开始的手足无措&#xff0c;慢慢变得游刃有余。以下内容为本人从网上查阅学习多个专家经验&#xff0c;并结合工作经历总结而来&#xff0c;仅…