多态通过分离做什么和怎么做,从另一个角度将接口与实现分离开来;通过多态来消除类型之间的耦合关系,在Java中,多态也叫动态绑定,后期绑定或运行时绑定,那么什么是方法绑定?

方法调用绑定:

将一个方法与调用同一个方法的主体关联起来被称为绑定;若在程序执行前进行绑定(由编译器和连接程序实现),叫做前期绑定;还有一种叫后期绑定,就是在运行时根据对象的类型进行绑定,也叫动态绑定或运行时绑定,也就是说,编译器不知道对象的类型,但是方法调用机制能找到正确的方法体;在Java中除了static方法和final方法之外,所有的其他方法都是后期绑定;

因此,Java中所有的方法都是通过动态绑定来实现多态的,但如果直接访问某个域,则这个访问就会在编译其进行解析;一般我们都通过将子类向上转型为父类来实现多态,父类可以是抽象类,只要子类实现到父类的所有抽象方法,就可以将子类转型为抽象的父类;Java里的抽象类本身是不能被实例化,但可以将子类的引用向上转型为抽象的父类。

如:

abstract class Jack{
public abstract void doSomething();
}
class Product extends Jack{ @Override
public void doSomething() {
System.out.println("Product"); } }
public static void main(String[] args) {
//声明一个抽象类Jack的变量,并指向其子类的实例化对象,合法的,Java的多态性会保证在运行时可以得到其正确的类型;
Jack jack=new Product();
//Jack jack=new Jack();//非法,不能实例化抽象类对象
jack.doSomething();
}

Java反射

Java反射机制是指在运行状态时,可以知道任意一个类的的所有属性和方法,对任意一个对象都可以调用它的任意一个方法;通过反射,可以在运行时实例化对象

Java反射提供的功能包括:

  • 在运行时判断一个对象所属的类;
  • 运行时构造任意一个类的对象;
  • 运行时判断任意一个类的成员变量与方法;
  • 运行时调用任意一个对象的方法;
  • 生成动态代理;

我们想得到一个类的所有信息,第一步就是要得到类的Class对象,如果知道了一个对象或类的名字,就可以通过简单的:

Class<?> clz=对象.getClass();

Class<?> clz=类的名字.class

得到,但如果在编译期得不到具体类型,则可以通过Class.forName()来得到,但这个方法生成的结果在编译时是不可知的,所有的方法特征签名都是在运行时提取出来的。这是由Java的反射机制来提供足够的支持。在得到这个类的Class对象后,我们就可以反射来构造对象,进而得到这个类的所有的信息。

 public static Class<?> forName(String className)
throws ClassNotFoundException {
Class<?> caller = Reflection.getCallerClass();
return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
}

Java里的Class类与java.lang.reflect类库一起对反射进行支持,该类库包含了Field,Method,Constructor类(每个类都实现了Member接口),这些类型的对象是在JVM在运行时创建的,用以表示未知类里对应的成员。这样我们就可以使用Constructor创建新的对象,用get(),set()方法修改与Field对象关联的字段,用invoke()方法调用与Method对象关联的方法。(摘自《Java编程思想》第四版)

下面简要介绍利用反射实现的动态代理;

动态代理

步骤:

  • 新建委托类,实现动态代理要求委托类必须实现某个接口;
  • 新建中间类,用来连接代理类和委托类,这个中间必须实现InvocationHandler接口,这个接口只有一个invoke()方法;
  • 通过Proxy类新建代理类对象;

举例:

interface Operate{
void method1();
void method2();
void method3();
}
/**
* 委托类
* @author wood
*
*/
class Entrust implements Operate{ @Override
public void method1() {
// TODO Auto-generated method stub
System.out.println("*method1");
} @Override
public void method2() {
// TODO Auto-generated method stub
System.out.println("*method2");
} @Override
public void method3() {
// TODO Auto-generated method stub
System.out.println("*method3");
} }
/**
* 连接委托类与代理类的中间类;
* @author wood
*
*/
class DynamecProxyHandler implements InvocationHandler{
private Object proxied;//委托类对象
public DynamecProxyHandler(){ }
public DynamecProxyHandler(Object object){
this.proxied=object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] arg2)
throws Throwable {
// TODO Auto-generated method stub
Object object=method.invoke(proxied, arg2);
System.out.println(method.getName());
return object;
} }
//***********************
DynamecProxyHandler dymaProxy=new DynamecProxyHandler(new Entrust());
//通过Proxy类的静态函数生成代理对象;
Operate operate=(Operate)Proxy.newProxyInstance(Operate.class.getClassLoader(), new Class[]{Operate.class}, dymaProxy);
operate.method1();
operate.method2();
operate.method3();

我们通过Procy.newProcyInstance函数新建了一个代理对象,实际的代理类就是在这时候动态生成了,我们调用该代理对象的函数就会调用到中间类的invoke函数,而invoke函数实现调用委托类的对应函数;

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)

loader是类加载器;

interfaces是委托类的接口,生成代理需要实现这个接口;

实际上Java的动态代理就是两层的静态代理:开发者提供一个委托类A,程序动态生成了一个代理类B,开发者还需要提供一个实现了InvocationHandler的接口C,用类C来连接委托类A和委托类B,类C是代理类B的委托类,是类A的代理类;用户直接调用代理类B,B将调用转发给委托类C,C再将调用转发给委托类A;

参考:

《Java编程思想》第四版

公共技术点之 Java 动态代理

Java多态与反射的更多相关文章

  1. Java中的反射机制和动态代理

    一.反射概述 反射机制指的是Java在运行时候有一种自观的能力,能够了解自身的情况为下一步做准备,其想表达的意思就是:在运行状态中,对于任意一个类,都能够获取到这个类的所有属性和方法:对于任意一个对象 ...

  2. 在java中有关于反射的皮毛----自己的简略认知

    白首为功名.旧山松竹老,阻归程.欲将心事付瑶琴.知音少,弦断有谁听? 反射(reflection): 当我们在看到这个名词首先会想到的是,我们在上高中时学的物理,那么在java开发中,反射这个名词是怎 ...

  3. java中的反射(二)

    java中的反射(一):https://www.cnblogs.com/KeleLLXin/p/14060555.html 目录 一.反射 1.class类 2.访问字段 3.调用方法 4.调用构造方 ...

  4. Java中的反射和注解

    前言 在Java中,反射机制和注解机制一直是一个很重要的概念,那么他们其中的原理是怎么样呢,我们不仅仅需要会使用,更要知其然而之所以然. 目录 反射机制 反射如何使用 注解定义 注解机制原理 注解如何 ...

  5. java中的反射机制在Android开发中的用处

    JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反 ...

  6. Java里面,反射父类里面数字类型字段,怎么set值

    Java里面,反射父类里面数字类型字段,怎么set值,我的做法是这样: /** * TODO 直接设置对象属性值, 忽略private/protected 修饰符, 也不经过setter * @aut ...

  7. Java 多态——与C++的比较

    学习了Java和C++之后,由于长期不使用C++,而java的基础知识掌握不牢,现在已经搞不清java多态了.现在先来谈谈java多态,稍后有时间再更新C++的多态,并进行比较~ 一. Java的多态 ...

  8. C++和java多态的区别

    C++和java多态的区别 分类: Java2015-06-04 21:38 2人阅读 评论(0) 收藏 举报  转载自:http://www.cnblogs.com/plmnko/archive ...

  9. java 27 - 2 反射之 反射的概述以及获取Class文件对象的方式

    反射: JAVA语言的反射机制: JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法: 对于任意一个对象,都能够调用它的任意一个方法和属性: 这种动态获取的信息以及动态调 ...

随机推荐

  1. 第7章 权限管理(2)_文件特殊权限(SUID、SGID、SBIT)

    2. 文件特殊权限(主要用来临时提升命令执行者或其组身份) 2.1 SetUID (1)SetUID的功能 ①只有可以执行的二进制程序才能设定SUID权限.用来临时提升执行程序(或某条命令)的用户身份 ...

  2. jmeter(六)元件的作用域与执行顺序

    jmeter是一个开源的性能测试工具,它可以通过鼠标拖拽来随意改变元件之间的顺序以及元件的父子关系,那么随着它们的顺序和所在的域不同,它们在执行的时候,也会有很多不同. jmeter的test pla ...

  3. http协议进阶(一)http概述

    参考书籍——<HTTP权威指南> 1.web客户端和服务器 http客户端发出请求,其中包含请求内容,发给服务器,服务器再返回内容中回送请求的数据,http客户端和服务器构成了万维网的基本 ...

  4. Python-13-堡垒机开发

    今天主要是熟悉堡垒机开发流程.

  5. Python list

    序列是Python中最基本的数据结构.序列中的每个元素都分配一个数字 - 它的位置,或索引,第一个索引是0,第二个索引是1,依此类推. Python有6个序列的内置类型,但最常见的是列表和元组. 序列 ...

  6. .NET跨平台之旅:将示例站点从ASP.NET 5 Beta5升级至Beta7

    9月2日,微软发布了ASP.NET 5 Beta7(详见Announcing Availability of ASP.NET 5 Beta7).其中最大的亮点是dnx已经可以完全基于CoreCLR运行 ...

  7. 非对称加密算法RSA

    RSA公钥加密算法是1977年由罗纳德·李维斯特(Ron Rivest).阿迪·萨莫尔(Adi Shamir)和伦纳德·阿德曼(Leonard Adleman)一起提出的.1987年首次公布,当时他们 ...

  8. [转]Eclipse SVN冲突解决

    原文地址:http://www.devnote.cn/article/82.html 基本原则是:每次提交前需要先和线上的对比,先把冲突解决掉,然后把线上的更新到本地,最后把本地的提交上去. 右键项目 ...

  9. TortoiseGit的使用教程(转)

    1.windows安装TortoiseGit详细使用教程[基础篇] http://www.mamicode.com/info-detail-423481.html 2.Windows下Git与Tort ...

  10. [转]office 2016 4合1/3合1 专业版 增强版 精简绿色安装版

    Office 2016是微软的一个庞大的办公软件集合,其中包括了Word.Excel.PowerPoint.OneNote.Outlook.Skype.Project.Visio以及Publisher ...