A:获取所有成员变量
public Field[] getFields() 获取所有的成员变量包含从父类继承过来的
public Field[] getDeclaredFields() 获取所有的成员变量 包含私有的 也包含从父类继承过来的成员变量
B:获取单个成员变量
public Field getField(String name)
public Field getDeclaredField(String name)
C:案例演示: 通过反射获取成员变量并使用
public class MyTest {
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, InstantiationException {
Class teacherClass = Teacher.class;
Field field = teacherClass.getField("name");
Object obj = teacherClass.newInstance();
field.set(obj,"一一");
Object o = field.get(obj);
System.out.println(o);
Field sexFiled = teacherClass.getDeclaredField("sex");
sexFiled.setAccessible(true);
sexFiled.set(obj,"女");
Object o1 = sexFiled.get(obj);
System.out.println(o1);
}
}
反射(通过反射获取无参无返回值成员方法并使用)
A:获取所有成员方法
public Method[] getMethods() //获取所有的公共的成员方法不包含私有的 包含从父类继承过来的过来的公共方法
public Method[] getDeclaredMethods()//获取自己的所有成员方法 包含私有的
B:获取单个成员方法
//参数1: 方法名称 参数2:方法行参的class 对象
public Method getMethod(String name,Class<?>... parameterTypes) //获取单个的方法 不包含私有的
public Method getDeclaredMethod(String name,Class<?>... parameterTypes) 获取单个方法包括私有的
C:案例演示: 通过反射获取无参无返回值成员方法并使用
public class MyTest {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException {
Class studentClass = Class.forName("Demo6.Student");
Method showMethod = studentClass.getMethod("show");
Object obj = studentClass.newInstance();
showMethod.invoke(obj);
}
}
反射(通过反射获取带参带返回值成员方法并使用)
A:案例演示: 通过反射获取带参带返回值成员方法并使用
public class MyTest4 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException {
Class studentClass = Class.forName("Demo6.Student");
Method show3 = studentClass.getMethod("show3", String.class, int.class, double.class);
Integer invoke = (Integer) show3.invoke(obj, "王五", 25, 3.5);
System.out.println(invoke);
Method test = studentClass.getDeclaredMethod("test", String.class, int.class, double.class);
test.setAccessible(true);
Integer result =(Integer) test.invoke(obj, "二二", 18, 5.5);
System.out.println(result);
}
}
反射(通过反射运行配置文件内容)
A:案例演示: 通过反射运行配置文件内容
public class MyTest {
public static void main(String[] args) throws IOException, ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
Properties properties = new Properties();
properties.load(new FileReader("src/peizhi.txt"));
Class dogClass = Class.forName(properties.getProperty("className"));
Object obj = dogClass.getDeclaredConstructor().newInstance();
Method methodName = dogClass.getMethod(properties.getProperty("methodName"));
methodName.invoke(obj);
}
}
public class Dog {
public void eat(){
System.out.println("狗吃骨头");
}
}
public class Cat {
public void eat(){
System.out.println("猫吃鱼");
}
}
peizhi.txt
className=Demo7.Cat
methodName=eat
反射(通过反射越过泛型检查)
A:案例演示: 我给你ArrayList<Integer>的一个对象,我想在这个集合中添加一个字符串数据,如何实现呢?
public class MyTest {
public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
ArrayList<Integer> list = new ArrayList<>();
list.add(200);
list.add(200);
list.add(200);
//list.add("abc");
Class aClass = list.getClass();
Method add = aClass.getMethod("add", Object.class);
add.invoke(list,"abc");
System.out.println(list);
}
}
反射(通过反射写一个通用的设置某个对象的某个属性为指定的值)
A:案例演示
public void setProperty(Object obj, String propertyName, Object value){},
此方法可将obj对象中名为propertyName的属性的值设置为value。
public class MyTest {
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
Student student = new Student();
MyUtils.setProperty(student,"name","张三");
MyUtils.setProperty(student,"age",18);
System.out.println(student.getName());
System.out.println(student.getAge());
}
}
public class Student {
private String name;
private int age;
public String getName() {
return name;
}
public int getAge() {
return age;
}
}
public class MyUtils {
public static void setProperty(Object obj, String propertyName, Object value) throws NoSuchFieldException, IllegalAccessException {
Class aClass = obj.getClass();
Field declaredField = aClass.getDeclaredField(propertyName);
declaredField.setAccessible(true);
declaredField.set(obj,value);
}
}
反射(通过用户的增删改查引出中介)
A:案例演示: 用户的增删改查
public class MyTest {
public static void main(String[] args) {
UserDaoImpl userDao = new UserDaoImpl();
userDao.insert();
userDao.delete();
userDao.update();
userDao.query();
}
}
public interface UserDao {
void insert();
void delete();
void update();
void query();
}
public class UserDaoImpl implements UserDao {
@Override
public void insert() {
System.out.println("全权限校验");
System.out.println("添加一个用户");
System.out.println("记录日志");
}
@Override
public void delete() {
System.out.println("删除用户");
}
@Override
public void update() {
System.out.println("修改用户");
}
@Override
public void query() {
System.out.println("查询用户");
}
}
反射(动态代理的概述和实现)
A:动态代理概述
代理:本来应该自己做的事情,却请了别人来做,被请的人就是代理对象。
举例:春季回家买票让人代买
动态代理:在程序运行过程中产生的这个对象
而程序运行过程中产生对象其实就是我们刚才反射讲解的内容,所以,动态代理其实就是通过反射来生成一个代理
在Java中java.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口,
通过使用这个类和接口就可以生成动态代理对象。JDK提供的代理只能针对接口做代理。
我们有更强大的代理cglib,Proxy类中的方法创建动态代理类对象
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
最终会调用InvocationHandler的方法
InvocationHandler Object invoke(Object proxy,Method method,Object[] args)
B:案例演示: 动态代理的实现
public class MyTest {
public static void main(String[] args) {
UserDao userDao = new UserDaoImpl();
UserDao proxy = ProxyUtils.getProxy(userDao);
proxy.insert();
System.out.println("----------------");
proxy.delete();
System.out.println("----------------");
proxy.update();
System.out.println("----------------");
proxy.query();
}
}
public interface UserDao {
void insert();
void delete();
void update();
void query();
}
public class ProxyUtils {
public static UserDao getProxy(UserDao userDao){
UserDao obj = (UserDao) Proxy.newProxyInstance(userDao.getClass().getClassLoader(), userDao.getClass().getInterfaces(), new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//全部
System.out.println("权限校验");
Object result = method.invoke(userDao);
System.out.println("记录日志");
//单独
Object result=null;
if (method.getName().equals("delete")){
System.out.println("权限校验");
result = method.invoke(userDao);
System.out.println("记录日志");
}
return result;
}
});
return obj;
}
}
public class UserDaoImpl implements UserDao {
@Override
public void insert() {
System.out.println("添加一个用户");
}
@Override
public void delete() {
System.out.println("删除用户");
}
@Override
public void update() {
System.out.println("修改用户");
}
@Override
public void query() {
System.out.println("查询用户");
}
}
我们可以通过Proxy类中的静态方法获取一个代理对象:
-
- public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
A: JDK1.5的新特性: 自动拆装箱 , 泛型 , 增强for , 可变参数 , 枚举
B:枚举概述: 就是一个类只能存在几个固定的对象,那么这个就是枚举.我们就可以使用这些对象可以表示一些固定的值.
举例:一周只有7天,一年只有12个月等。
C:案例演示: 自己实现枚举类
public class MyTest {
public static void main(String[] args) {
Direction front = Direction.front;
Direction behind = Direction.behind;
Direction left = Direction.left;
Direction right = Direction.right;
System.out.println(front.name);
System.out.println(behind.name);
System.out.println(left.name);
System.out.println(right.name);
front.show("前");
behind.show("后");
left.show("左");
right.show("右");
}
}
public abstract class Direction {
public static final Direction front= new Direction("前") {
@Override
public void show(String name) {
System.out.println(name);
}
};
public static final Direction behind= new Direction("后") {
@Override
public void show(String name) {
System.out.println(name);
}
};
public static final Direction left= new Direction("左") {
@Override
public void show(String name) {
System.out.println(name);
}
};
public static final Direction right= new Direction("右") {
@Override
public void show(String name) {
System.out.println(name);
}
};
String name;
public Direction(String name) {
this.name = name;
}
public abstract void show(String name);
}
JDK5新特性(通过enum实现枚举类)
A:案例演示: 通过enum实现枚举类
public class MyTest {
public static void main(String[] args) {
Direction front = Direction.FRONT;
String name = front.name;
System.out.println(name);
Direction behind = Direction.BEHIND;
String name1 = behind.name;
System.out.println(name1);
Direction left = Direction.LEFT;
String name2 = left.name;
System.out.println(name2);
Direction right = Direction.RIGHT;
String name3 = right.name;
System.out.println(name3);
}
}
public enum Direction {
FRONT("前"),BEHIND("后"),LEFT("左"),RIGHT("右"),
;
String name;
Direction(String name) {
this.name=name;
}
}
JDK5新特性(枚举的注意事项)
A:案例演示
定义枚举类要用关键字enum
所有枚举类都是Enum的子类,但是不要显式的写出来
枚举类的第一行上必须是枚举项,最后一个枚举项后的分号是可以省略的,但是如果枚举类有其他的东西,这个分号就不能省略。建议不要省略
枚举类可以有构造器,但必须是private的,它默认的也是private的。枚举项的用法比较特殊:枚举(“”);
枚举类也可以有抽象方法,但是枚举项必须重写该方法
枚举在switch语句中的使用
public class MyTest {
public static void main(String[] args) {
Direction front = Direction.FRONT;
switch (front){
case FRONT:
System.out.println("前");
break;
case BEHIND:
System.out.println("后");
break;
case LEFT:
System.out.println("左");
break;
case RIGHT:
System.out.println("右");
break;
}
}
}
public enum Direction {
FRONT,BEHIND,LEFT,RIGHT,
;
}
JDK5新特性(枚举类的常见方法)
A:枚举类的常见方法
int ordinal() 返回枚举项的序号
int compareTo(E o) 比较两个枚举项的 返回的是两个枚举项序号的 差值
String name() 获取枚举项的名称
String toString()获取枚举项的名称
<T> T valueOf(Class<T> type,String name) 用来获取指定的枚举项 参数1:枚举类对应的字节码对象 参数2 枚举项的名称
values() 获取所有的枚举项
此方法虽然在JDK文档中查找不到,但每个枚举类都具有该方法,它遍历枚举类的所有枚举值非常方便
B:案例演示: 枚举类的常见方法
public class MyTest {
public static void main(String[] args) {
System.out.println(Direction.FRONT.ordinal());
Direction left = Direction.LEFT;
int index = left.ordinal();
System.out.println(index);
String s = left.toString();
String name = left.name();
System.out.println(s);
System.out.println(name);
Direction[] values = Direction.values();
for (Direction value : values) {
System.out.println(value.name());
}
}
}
public enum Direction {
FRONT,BEHIND,LEFT,RIGHT,
;
}
public static void main(String[] args) {
// 测试
Direction front = Direction.FRONT ;
Direction behind = Direction.BEHIND;
Direction left = Direction.LEFT ;
Direction right = Direction.RIGHT ;
System.out.println(front.ordinal());
System.out.println(behind.ordinal());
System.out.println(left.ordinal());
System.out.println(right.ordinal());
System.out.println("----------------------------------");
System.out.println(front.compareTo(right));
System.out.println("----------------------------------");
System.out.println(front.name());
System.out.println("----------------------------------");
System.out.println(front.toString());
System.out.println(front);
System.out.println("----------------------------------");
// <T> T valueOf(Class<T> type,String name): 用来获取指定的枚举项
// type: 表示的是对应的枚举的字节码文件对象
// name: 就是枚举项的名称
Direction direction = Direction.valueOf(Direction.class, "RIGHT") ;
System.out.println(direction);
System.out.println("----------------------------------");
Direction[] directions = Direction.values() ;
for(Direction d : directions){
System.out.println(d);
}
}
JDK7新特性(JDK7的六个新特性回顾和讲解)
A:二进制字面量
JDK7开始,终于可以用二进制来表示整数(byte,short,int和long)。
使用二进制字面量的好处是,可以使代码更容易被理解。语法非常简单,只要在二进制数值前面加 0b或者0B
int x = 0b110110
B:数字字面量可以出现下划线
为了增强对数值的阅读性,如我们经常把数据用逗号分隔一样。JDK7提供了_对数据分隔。
举例:
int x = 100_1000;
注意事项:
不能出现在进制标识和数值之间
不能出现在数值开头和结尾
不能出现在小数点旁边
C:switch 语句可以用字符串
D:泛型简化
public class MyTest {
public static void main(String[] args) {
System.out.println(0b100);
System.out.println(0100);
System.out.println(100);
System.out.println(0x100);
int num=1_000_000_000;
ArrayList<String> strings = new ArrayList<>();
}
}