飞道的博客

Java面向对象笔记 • 【第8章 内部类和泛型】

348人阅读  评论(0)

全部章节   >>>>


本章目录

8.1 内部类

8.1.1 内部类概述

8.1.2 内部类使用

8.1.3 实践练习

8.2 静态内部类

8.2.1 静态内部类的实现

8.2.2 实践练习

8.3 匿名内部类

8.3.1 匿名内部类的实现

8.3.2 实践练习

8.4 泛型

8.4.1 泛型概述

8.4.2 泛型应用

8.4.3 实践练习

总结:


8.1 内部类

8.1.1 内部类概述

  • 实现程序设计的需要,需要将一个类放在另一个类的内部定义,定义在其他类内部的类就称为内部类,包含内部类的类也被称为外部类,内部类的作用如下
  • 内部类提供了更好的封装,可以将内部类隐藏在外部类内,不允许同一个包中的其他类访问该类。
  • 内部类成员可以直接访问外部类的私有数据,因为内部类作为其外部类成员,同一个类的成员之间可以相互访问。但外部类不能访问内部类的实现细节,如内部类的成员变量。
  • 匿名内部类适用于创建那些仅需要一次使用的类。

8.1.2 内部类使用

语法:


  
  1. public class OutClass{
  2. ……
  3. //定义内部类
  4. public class innerClass{
  5. }
  6. }

内部类分为两种:静态内部类和非静态内部类,使用static修饰的内部类称为静态内部类,没有使用static修饰的内部类则称为非静态内部类

8.1.3 实践练习

 

8.2 静态内部类

8.2.1 静态内部类的实现

  • 使用static修饰的内部类,被称为静态内部类,静态部类属于外部类本身,而不属于外部类的某个对象。
  • 静态内部类可以包含静态成员,也可以包含非静态成员
  • 依据静态成员不能访问非静态成员的规则,静态内部类不能访问外部类的实例成员,只能访问外部类的类成员。即使是静态内部类的实例方法,也不能访问外部类的实例成员,只能访问外部类的静态成员。

示例: 演示静态内部类的定义


  
  1. public class OutClass {
  2. private int outPro= 5;
  3. private static int outPro2= 9;
  4. private static class InnerClass{
  5. private static int innerPro= 10; //静态内部类中可以定义静态成员
  6. private int innerPro2= 100; //静态内部类中的实例属性
  7. public void accessOutPro(){
  8. /*以下代码在编译时出现错误静态内部类无法直接访问外部类的实例成员*/
  9. //System.out.println(outPro);
  10. //可通过外部类实例访问外部类实例属性
  11. System.out.println( "外部类实例属性outPro="+ new OutClass().outPro);
  12. //访问外部类的静态成员,代码正确
  13. System.out.println( "外部类的静态属性outPro2="+outPro2);
  14. }
  15. }
  16. //未完接下一页
  17. }

  
  1. public class OutClass {
  2. //接上一页
  3. public void accessInnerPro(){
  4. //System.out.println(innerPro);
  5. //以上代码出现错误,应修改为如下形式,即通过类名访问静态内部类的类成员
  6. System.out.println( "内部类的静态属性innerPro="+InnerClass.innerPro);
  7. //System.out.println(innerPro2);
  8. //以上代码出现错误,应修改为如下格式
  9. InnerClass ic= new InnerClass();
  10. System.out.println( "内部类实例属性innerPro2="+ic.innerPro2);
  11. ic.accessOutPro(); //调用静态内部类的实例方法
  12. }
  13. public static void main(String[] args) {
  14. OutClass oc= new OutClass();
  15. oc.accessInnerPro();
  16. }
  17. }

8.2.2 实践练习

 

8.3 匿名内部类

8.3.1 匿名内部类的实现

匿名内部类适用于仅需要一次使用的类,匿名内部类的语法较为特别,创建匿名内部类时,会立即创建一个该类的实例,创建完毕之后该类的定义会立即消失

如果匿名内部类需要访问外部类的局部变量,则必须使用final修饰符修饰外部类的局部变量,否则系统将报错。

语法:


  
  1. new 父类构造器(参数列表)| 实现接口(){
  2. //匿名内部类的类体部分
  3. }

匿名内部类必须继承一个父类,或实现一个接口,但最多只能继承一个父类,或实现一个接口。

提醒:匿名内部类的两条规则

  • 匿名内部类不能是抽象类,因为系统在创建匿名内部类时,会立即创建匿名内部类的对象,因此不允许将匿名内部类定义成抽象类。
  • 匿名内部类不能定义构造器,因为匿名内部类没有类名,所以无法定义构造器,但匿名内部类可以定义实例化初始块,通过实例化初始块来完成构造器需要完成的事情

示例: 演示匿名内部类的定义


  
  1. //匿名内部类需要实现的接口
  2. public interface Product {
  3. double getPrice();
  4. String getName();
  5. }
  6. public class Anonymous {
  7. public void execute(Product product){
  8. System.out.println( "购买了一个:"+product.getName()+ " 花费:"+product.getPrice());
  9. }
  10. public static void main(String[] args) {
  11. Anonymous aym= new Anonymous();
  12. /*调用execute()方法时,需要创建一个Product类型参数,此处传入匿名实现类的实例*/
  13. aym.execute( new Product() {
  14. public double getPrice() {
  15. return 98.29;
  16. }
  17. public String getName() {
  18. return "费列罗巧克力";
  19. }
  20. });
  21. }
  22. }

示例:将匿名内部类更改为内部类


  
  1. public class Consumer {
  2. public void execute(Product product){
  3. System.out.println( "购买了一个:"+product.getName()+ " 花费:"+product.getPrice());
  4. }
  5. public class MyProduct implements Product{ //内部类
  6. public double getPrice() {
  7. return 98.29;
  8. }
  9. public String getName() {
  10. return "费列罗巧克力";
  11. }
  12. }
  13. public static void main(String[] args) {
  14. Consumer consumer = new Consumer(); //实例化一个外部类对象
  15. //通过外部类对象创建内部类对象
  16. Product product = consumer.new MyProduct();
  17. consumer.execute(product);
  18. }
  19. }

8.3.2 实践练习

 

8.4 泛型

8.4.1 泛型概述

没有使用泛型时取出集合元素


  
  1. List listUser = new ArrayList();
  2. listUser.add( new User(“张三”, 20)) ;
  3. User user = (User)listUser.get( 0) ; //使用了强制类型转换。

使用泛型时取出集合元素


  
  1. List<User> listUser = new ArrayList<User>();
  2. listUser.add( new User(“张三”, 20)) ;
  3. User user = listUser.get( 0) ; //把类型转换交给了编译器

型的好处是在编译的时候检查类型安全,并且所有的强制转换都是自动和隐式的,提高代码的重用率。

泛型的原理就是参数化类型

  • 泛型把类型看做参数,也就是说把所要操作的数据类型看做参数,就像方法的形式参数是运行时传递的值的占位符一样
  • 型中的类型变量扮演的角色就如同一个参数,它提供给编译器用来类型检查的信息

型的分类

  • 型类
  • 型方法

8.4.2 泛型应用

1、泛型类:就是具有一个或多个类型变量的

示例: 定义泛型类并实例化泛型类


  
  1. public classPair<T>{
  2. private T first
  3. public Pair (T first){ this.first=first;}
  4. public void setFirst(T newValue){first=newValue;}
  5. public T getFirst(){ return first;}
  6. }
  7. public class PairTest{
  8. public static void main(String[] args){
  9. //用具体的类型替换类型变量就可以实例化泛型类型
  10. Pair<String> pair= new Pair<String>( "Hello");
  11. System.out.println( "first="+pair.getFirst());
  12. }
  13. }

2、泛型方法:

  • 带有参数类型的方法即泛型方法,泛型方法使得该方法能够独立于类而产生变化。
  • 泛型方法所在的类可以是泛型类,也可以不是泛型类。

示例:声明一个f(T x)泛型方法,用于返回调用该方法时,所传入的参数类型的类名


  
  1. class GenericMethod{
  2. //T是泛型类型参数
  3. public <T> void f(T x){
  4. System.out.println(x.getClass().getName());
  5. }
  6. }
  7. public class GenericMethodTest{
  8. public static void main(String[] args){
  9. GenericMethod gm= new GenericMethod();
  10. gm.f( " ");
  11. gm.f( 1);
  12. gm.f( 1.0f);
  13. gm.f( 'c');
  14. gm.f(gm);
  15. }
  16. }

3、泛型定义泛型定义包括泛型类定义和泛型对象定义,其中泛型对象定义的应用最为普遍。

语法:

类名称<具体类>  对象名称 = new  类名称<具体类>();

例如:


  
  1. List<String> citys = new ArrayList<String>(); //城市列表
  2. List<Address> empAdds = new ArrayList<Address>(); //员工住址列表
  3. Map<String, Emp> empnoToEmp = new HashMap<String, Emp>(); //员工编号到员工映射

示例: 型集合List<T>Map<K,V>应用


  
  1. List<String> strList= new ArrayList<String>(); //声明泛型集合,该集合元素为String类型
  2. //strList集合中只能添加字符串元素,否则无法通过编译
  3. strList.add( "apple");
  4. strList.add( "peer");
  5. strList.add( "banana");
  6. //strList.add(new Date());此处如果不将其注释,在编译时系统报错
  7. //集合使用泛型后,从集合中获取数据无须强制类型转换
  8. String str=strList.get(strList.size()- 1);
  9. //创建Map泛型集合,map的Key为String类型,value为Date类型
  10. Map<String,Date> map= new HashMap<String,Date>();
  11. //向集合中添加数据
  12. map.put( "Java", new Date( 60000));
  13. map.put( "C#", new Date( 300000000));
  14. map.put( "MySQL", new Date( 1000000));
  15. //调用keySet()方法返回装有所有Key值的Set集合
  16. Set<String> keySet=map.keySet();
  17. //通过keySet获取Iterator迭代器
  18. Iterator<String> iter=keySet.iterator();
  19. while(iter.hasNext()){ //遍历迭代器
  20. String key=iter.next();
  21. System.out.println( "Map["+key+ "]="+map.get(key));
  22. }

8.4.3 实践练习

总结:

  • 内部类提供了更好的封装,可以将内部类隐藏在外部类内。内部类成员可以直接访问外部类的私有数据,但外部类不能访问内部类的成员变量
  • 使用static修饰的内部类,被称为静态内部类,静态内部类属于外部类本身,而不属于外部类的某个对象
  • 匿名内部类适用于创建那些仅需要一次使用的类,匿名内部类不能是抽象类,不能定义构造器,但是可以通过实例化初始块来完成构造器需要完成的事情
  • 泛型的原理就是参数化类型,泛型的好处是在编译的时候检查类型安全,并且所有的强制转换都是自动和隐式的,提高代码的重用率。

转载:https://blog.csdn.net/weixin_44893902/article/details/106699092
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场