首页 Java java教程 使用JAVA反射的利与弊

使用JAVA反射的利与弊

Dec 12, 2016 am 11:59 AM
java反射

Java的核心技能有如下几项: 
(1)JVM的调优 
(2)类加载器 
(3)反射 
(4)动态编译 
(5)动态代理 
(6)注解 
(7)多线程 
(8)IO,NIO,Socket,Channel等网络编程 
除了JAVA的基础,面向对象的思想外,这些既是java里面核心技术,也是面试时候,面试官经常爱问的几个知识,了解,熟悉和掌握他们的重要性不言而喻,今天就先来谈谈反射。 

反射给java提供了,运行时获取一个类实例的可能,这一点非常灵活,你仅仅传一个类的全包名路径,就能通过反射,来获取对应的类实例,我们一般会用Class类,来调用这个被反射的Objcet类下的,构造方法,属性,或方法等,反射在一些开源框架里用的非常之多,Spring,Struts,Hibnerate,MyBatics都有它的影子,反射虽然很灵活,能够使得写的代码,变的大幅精简,所以在用的时候,一定要注意具体的应用场景,反射的优缺点如下: 

优点: 

(1)能够运行时动态获取类的实例,大大提高系统的灵活性和扩展性。 
(2)与Java动态编译相结合,可以实现无比强大的功能 

缺点: 
(1)使用反射的性能较低 
(2)使用反射相对来说不安全 
(3)破坏了类的封装性,可以通过反射获取这个类的私有方法和属性 

任何事物,都有两面性,反射的优点,也同是就是它的缺点,所以,没有好与坏,只有最合适的场景,一阴一阳,才是天道平衡的条件。 

下面来看个,使用java反射,来自动封装数据库对应的表的例子,初学java的人都会给每个实体类建立一个Dao对象,来专门操作这个对象对应的表,这样做没错,很好,是分层,分工明确的一个表现,但是如果有几十个实体类,那么这种重复增删改查的工作,就会大大增加,散仙初入门的时候也有如此的感受,虽然我们可以通过,抽象类和接口,使用适配器的设计模式来简化重复的代码,但是不可避免的就是类的臃肿了,下面看看如何使用反射来搞定这么多实体类的重复的增删改查的代码: 
使用前提: 
(1)每一个实体类都会对应一个数据库表 
(2)每个表的列,与对应的实体类的属性名是一样的 
(3)实体类要提供基本的get或set方法 

201611011003251.png

实体类如下:  

Java代码 

package com.qin.model;  
  
public class Dog {  
      
    private int id;  
    private String name;  
    private String type;  
    private String color;  
    private int weight;  
    public int getId() {  
        return id;  
    }  
    public void setId(int id) {  
        this.id = id;  
    }  
    public String getName() {  
        return name;  
    }  
    public void setName(String name) {  
        this.name = name;  
    }  
    public String getType() {  
        return type;  
    }  
    public void setType(String type) {  
        this.type = type;  
    }  
    public String getColor() {  
        return color;  
    }  
    public void setColor(String color) {  
        this.color = color;  
    }  
    public int getWeight() {  
        return weight;  
    }  
    public void setWeight(int weight) {  
        this.weight = weight;  
    }  
 public Dog() {  
    // TODO Auto-generated constructor stub  
}  
public Dog(int id, String name, String type, String color, int weight) {  
    super();  
    this.id = id;  
    this.name = name;  
    this.type = type;  
    this.color = color;  
    this.weight = weight;  
}  
@Override  
public String toString() {  
    return "Dog [id=" + id + ", name=" + name + ", type=" + type + ", color="  
            + color + ", weight=" + weight + "]";  
}  
  
  
   
      
}
登录后复制

Java代码

package com.qin.model;  
  
public class Person {  
      
    private int id;  
    private String name;  
    private int age;  
    private String address;  
    public int getId() {  
        return id;  
    }  
    public void setId(int id) {  
        this.id = id;  
    }  
    public String getName() {  
        return name;  
    }  
    public void setName(String name) {  
        this.name = name;  
    }  
    public int getAge() {  
        return age;  
    }  
    public void setAge(int age) {  
        this.age = age;  
    }  
    public String getAddress() {  
        return address;  
    }  
    public void setAddress(String address) {  
        this.address = address;  
    }  
      
    public Person() {  
        // TODO Auto-generated constructor stub  
    }  
    public Person(int id, String name, int age, String address) {  
        super();  
        this.id = id;  
        this.name = name;  
        this.age = age;  
        this.address = address;  
    }  
    @Override  
    public String toString() {  
        return "Person [id=" + id + ", name=" + name + ", age=" + age  
                + ", address=" + address + "]";  
    }  
      
      
  
}
登录后复制

Java代码

package com.qin.db;  
  
import java.sql.Connection;  
import java.sql.DriverManager;  
import java.sql.PreparedStatement;  
import java.sql.ResultSet;  
/** 
 * 数据库连接的 
 * 测试类 
 * @author qindongliang 
 *  
 *  
 * **/  
public class ConnectionFactory {  
      
    public static Connection getCon()throws Exception{  
        Class.forName("com.mysql.jdbc.Driver");  
        //加上字符串编码指定,防止乱码  
        Connection connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/rate?characterEncoding=utf8", "root", "qin");  
        return connection;  
    }  
      
      
    public static void main(String[] args) throws Exception {  
          
        Class.forName("com.mysql.jdbc.Driver");  
        Connection connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/rate", "root", "qin");  
        System.out.println(connection);  
        connection.close();  
          
          
    }  
  
}
登录后复制

Java代码

package com.qin.commons;  
  
import java.lang.reflect.Field;  
import java.lang.reflect.Method;  
import java.sql.Connection;  
import java.sql.PreparedStatement;  
import java.sql.ResultSet;  
import java.util.ArrayList;  
import java.util.List;  
  
import com.qin.db.ConnectionFactory;  
import com.qin.model.Dog;  
import com.qin.model.Person;  
/*** 
 * 反射自动查询和封装的类 
 *@author qindongliang  
 *  
 * */  
public class CommonSupport {  
      
      
    /** 
     * @param obj需要保存的对象 
     * @param string 保存对象的sql语句 
     * */  
    public static String createSqlByObject(Object obj){  
          
         StringBuffer sb=new StringBuffer("insert into ");  
          
        //得到对象的类  
        Class c=obj.getClass();  
        //得到对象中的所有方法  
        Method[] ms=c.getMethods();  
          
        //得到对象中所有的属性,虽然在这个里面就能获取所有的字段名,但不建议这么用,破坏类的封装性  
        Field[]  fs=c.getDeclaredFields();  
        //得到对象类的名字  
        String cname=c.getName();  
        System.out.println("类名字: "+cname);  
        //表名字  
        String tableName=cname.split("\\.")[cname.split("\\.").length-1];  
        System.out.println("表名字: "+tableName);  
        //追加表名和(左边的符号  
        sb.append(tableName).append(" (");  
        //存放列名的集合  
        List<String> columns=new ArrayList<String>();  
        //存放值的集合  
        List values=new ArrayList();  
        //遍历方法  
        for(Method m:ms){  
             String methodName=m.getName();//获取每一个方法名  
             //只得到具有get方法的属性,getClass除外  
             if(methodName.startsWith("get")&&!methodName.startsWith("getClass")){  
                 //System.out.println("属性名:"+methodName);  
                 String fieldName = methodName.substring(3, methodName.length());  
//               System.out.println("字段名:"+fieldName);  
                 columns.add(fieldName);//将列名添加到列名的集合里  
                 try{  
                     Object value=m.invoke(obj, null);  
                     //System.out.println("执行方法返回的值:"+value);  
                     if(value instanceof String){  
//                       System.out.println("字符串类型字段值:"+value);  
                         values.add("&#39;"+value+"&#39;");//加上两个单引号,代表是字符串类型的  
                     }else{  
//                       System.out.println("数值类型字段值:"+value);  
                         values.add(value);//数值类型的则直接添加  
                     }  
                       
                 }catch(Exception e){  
                     e.printStackTrace();  
                 }  
                   
             }  
          
        }  
          
          
        for(int i=0;i<columns.size();i++){  
            String column=columns.get(i);  
            Object value=values.get(i);  
            System.out.println("列名:"+column+" 值:  "+value);  
        }  
          
        //拼接列名  
        for(int i=0;i<columns.size();i++){  
             if(i==columns.size()-1){  
                 sb.append(columns.get(i)).append(" ) ");  
             }else{  
                 sb.append(columns.get(i)).append(" , ");  
             }  
        }  
        System.out.println(" 拼接列名后的sql:"+sb.toString());  
        sb.append(" values ( ");  
        //拼接值  
        for(int i=0;i<values.size();i++){  
             if(i==values.size()-1){  
                 sb.append(values.get(i)).append(" ) ");  
             }else{  
                 sb.append(values.get(i)).append(" , ");  
             }  
        }  
      
        System.out.println(" 拼接值后的sql:"+sb.toString());  
    
        //返回组装的sql语句  
        return sb.toString();  
    }  
      
    /** 
     * 将对象保存在数据库中 
     * @param obj 保存的对象 
     * **/  
    public static void addOne(Object obj){  
        try {  
            Connection con=ConnectionFactory.getCon();  
            String sql=createSqlByObject(obj);  
            PreparedStatement ps=con.prepareStatement(sql);  
            int result=ps.executeUpdate();  
            if(result==1){  
                System.out.println("保存成功!");  
            }else{  
                System.out.println("保存失败!");  
            }  
            ps.close();  
            con.close();  
        } catch (Exception e) {  
            // TODO Auto-generated catch block  
            e.printStackTrace();  
        }  
          
    }  
      
    /** 
     * 根据类名字和一个查询条件 
     * 自动封装一个Bean对象 
     * @param columnName 列名 
     * @param value 列值 
     * @return {@link Object} 
     *  
     * */  
    public static Object getOneObject(String className,String columnName,String value){  
          
        String tableName=className.split("\\.")[className.split("\\.").length-1];  
        System.out.println("表名字: "+tableName);  
          
        //根据类名来创建对象  
        Class c=null;  
        try{  
            c=Class.forName(className);//反射生成一个类实例  
        }catch(Exception e){  
            e.printStackTrace();  
        }  
        //拼接sql语句  
        StringBuffer sb=new StringBuffer();  
        sb.append("select * from ")  
        .append(tableName)  
        .append(" where ")  
        .append(columnName).append(" = ").append("&#39;").append(value).append("&#39;");  
          
        String querySql=sb.toString();  
        System.out.println("查询的sql语句为:"+querySql);  
          
        Object obj=null;  
        try{  
        Connection con=ConnectionFactory.getCon();//得到一个数据库连接  
        PreparedStatement ps=con.prepareStatement(querySql);//预编译语句  
        ResultSet rs=ps.executeQuery();//执行查询  
        //得到对象的所有的方法  
        Method ms[]=c.getMethods();  
          
        if(rs.next()){  
            //生成一个实例  
            obj=c.newInstance();  
              
            for(Method m:ms){  
                String mName=m.getName();  
                if(mName.startsWith("set")){  
                    //根据方法名字自动提取表中对应的列名  
                      String cname = mName.substring(3, mName.length());  
                      //打印set的方法名  
                     // System.out.println(cname);  
                    //得到方法的参数类型  
                      Class[] params=m.getParameterTypes();  
//                    for(Class cp : params){  
//                        System.out.println(cp.toString());  
//                    }  
                      //如果参数是String类型,则从结果集中,按照列名取到的值,进行set  
                      //从params[0]的第一个值,能得到该数的参数类型  
                      if(params[0]==String.class){//  
                          m.invoke(obj, rs.getString(cname));  
                      //如果判断出来是int形,则使用int  
                      }else if(params[0]==int.class){  
                          m.invoke(obj, rs.getInt(cname));  
                      }  
                }  
            }  
              
              
              
        }else{  
            System.out.println("请注意:"+columnName+"="+value+"的条件,没有查询到数据!!");  
        }  
        rs.close();  
        ps.close();  
        con.close();  
        }catch(Exception e){  
            e.printStackTrace();  
        }  
          
          
          
        return obj;  
    }  
      
      
      
      
    public static void main(String[] args) throws Exception{  
        //====================添加======================  
        Dog d=new Dog(21, "小不点", "藏獒", "灰色", 25);  
        Person p=new Person(6, "大象hadoop", 10, "家住Apache基金组织");  
         //createSqlByObject(d);  
        //addOne(d);给dog表添加一条数据  
        //addOne(p);//给person表添加一条数据  
          
        //=======================查询=======================  
        //强制转换为原始类  
//    Dog d1=(Dog)getOneObject("com.qin.model.Dog", "id", "1");  
//    System.out.println(d1);  
          
        Person d1=(Person)getOneObject("com.qin.model.Person", "id", "1");  
        //Person d1=(Person)getOneObject("com.qin.model.Person", "name", "王婷");  
        System.out.println(d1);  
        
        
    }  
       
      
  
}
登录后复制

代码量是非常的少的,而且具有通用型,如果再有10个这个实体类,我们代码根本不用任何改动,只需要传入不同的实体类名字即可,当然这一点和Hibernate的自动化ORM非常接近了,在Hibnerate里,可以自动通过表生成类,也可以通过类生成数据库的表,原理其实就是利用了反射的特性,帮我们做了大量的重复工作,当然Hibnerate提供了更多的特性,也这只是一个简单的例子,具体的应用场景中,我们也需要因地制宜,否则,则为适得其反! 

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

<🎜>:泡泡胶模拟器无穷大 - 如何获取和使用皇家钥匙
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆树的耳语 - 如何解锁抓钩
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系统,解释
3 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

热门话题

Java教程
1668
14
CakePHP 教程
1427
52
Laravel 教程
1329
25
PHP教程
1273
29
C# 教程
1256
24
使用Java反射进行逆向工程:揭秘软件的内部运作方式 使用Java反射进行逆向工程:揭秘软件的内部运作方式 Feb 19, 2024 pm 11:20 PM

Java反射是一个强大的工具,它可以让你访问类的私有字段和方法,从而揭秘软件的内部运作方式。这在逆向工程、软件分析和调试等领域非常有用。要使用Java反射,首先需要导入java.lang.reflect包。然后,你可以使用Class.forName()方法来获取一个类的Class对象。一旦有了Class对象,你就可以使用各种方法来访问类的字段和方法。例如,你可以使用getDeclaredFields()方法来获取类的所有字段,包括私有字段。你也可以使用getDeclaredMethods()方法

java反射如何获取属性的值 java反射如何获取属性的值 Jan 03, 2024 pm 03:05 PM

获取方法:1、创建一个示例对象;2、通过field.get(person)获取了字段的值,其中person是示例对象,而field是Field对象,表示一个字段;3、通过setAccessible(true)设置字段为可访问状态,即使是私有字段也可以获取其值;4、遍历字段数组,可以获取每个字段的名称和对应的值,并打印出来即可。

如何使用java反射获取对象属性和值 如何使用java反射获取对象属性和值 Jan 03, 2024 pm 02:43 PM

获取方法:1、创建一个Person类,通过反射获取了该类的Class对象;2、使用getDeclaredFields方法获取了该类的所有字段;3、通过遍历字段数组,设置字段为可访问状态,然后使用get方法获取字段的值,并打印字段名和值即可。

java的反射机制原理是什么 java的反射机制原理是什么 Jun 21, 2023 am 10:53 AM

Java反射机制原理是,当一个字节码文件加载到内存的时候,jvm会对该字节码进行解剖,创建一个对象的Class对象,jvm把字节码文件信息都存储到Class对象中,只要获取到Class对象,就能使用该对象设置对象的属性或方法等。反射机制是,在运行状态中对任意一个类,都知道这个类的所有属性和方法,对于任意一个对象,能够调用其任意属性和方法,动态获取信息以及动态调用对象方法的功能。

深入理解Java反射机制的原理与应用 深入理解Java反射机制的原理与应用 Dec 23, 2023 am 09:09 AM

深入理解Java反射机制的原理与应用一、反射机制的概念与原理反射机制是指在程序运行时动态地获取类的信息、访问和操作类的成员(属性、方法、构造方法等)的能力。通过反射机制,我们可以在程序运行时动态地创建对象、调用方法和访问属性,而不需要在编译时知道类的具体信息。反射机制的核心是java.lang.reflect包中的类和接口。其中,Class类代表一个类的字节

如何使用Java反射机制创建对象? 如何使用Java反射机制创建对象? Apr 15, 2024 pm 04:18 PM

通过Java反射机制创建对象步骤如下:加载目标类:使用Class.forName()方法。获取构造函数:使用getDeclaredConstructor()方法。创建对象:使用newInstance()方法传递参数。

Java中的NoSuchFieldException异常在什么场景下出现? Java中的NoSuchFieldException异常在什么场景下出现? Jun 25, 2023 am 11:51 AM

Java中的NoSuchFieldException异常指的是在反射过程中试图访问不存在的字段(Field)时抛出的异常。在Java中,反射可以让我们通过代码来操纵程序中的类、方法、变量等,使得程序具有更高的灵活性和扩展性。但是,在使用反射时,如果访问的字段不存在,则会抛出NoSuchFieldException异常。NoSuchFieldException

java反射有哪些调用方法 java反射有哪些调用方法 Dec 22, 2023 pm 05:09 PM

java反射调用方法有:1、Class类;2、Constructor类;3、Method类;4、Field类;5、ClassLoader类。详细介绍:1、Class类,用于获取类的信息,包括类的名称、成员变量和方法等,可以通过Class类的"newInstance()"方法创建类的实例;2、Constructor类,用于获取构造函数的参数类型、修饰符和返回类型等信息等等。

See all articles