搜索
首页Javajava教程使用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
JVM性能与其他语言JVM性能与其他语言May 14, 2025 am 12:16 AM

JVM'SperformanceIsCompetitiveWithOtherRuntimes,operingabalanceOfspeed,安全性和生产性。1)JVMUSESJITCOMPILATIONFORDYNAMICOPTIMIZAIZATIONS.2)c提供NativePernativePerformanceButlanceButlactsjvm'ssafetyFeatures.3)

Java平台独立性:使用示例Java平台独立性:使用示例May 14, 2025 am 12:14 AM

JavaachievesPlatFormIndependencEthroughTheJavavIrtualMachine(JVM),允许CodeTorunonAnyPlatFormWithAjvm.1)codeisscompiledIntobytecode,notmachine-specificodificcode.2)bytecodeisisteredbytheybytheybytheybythejvm,enablingcross-platerssectectectectectross-eenablingcrossectectectectectection.2)

JVM架构:深入研究Java虚拟机JVM架构:深入研究Java虚拟机May 14, 2025 am 12:12 AM

TheJVMisanabstractcomputingmachinecrucialforrunningJavaprogramsduetoitsplatform-independentarchitecture.Itincludes:1)ClassLoaderforloadingclasses,2)RuntimeDataAreafordatastorage,3)ExecutionEnginewithInterpreter,JITCompiler,andGarbageCollectorforbytec

JVM:JVM与操作系统有关吗?JVM:JVM与操作系统有关吗?May 14, 2025 am 12:11 AM

JVMhasacloserelationshipwiththeOSasittranslatesJavabytecodeintomachine-specificinstructions,managesmemory,andhandlesgarbagecollection.ThisrelationshipallowsJavatorunonvariousOSenvironments,butitalsopresentschallengeslikedifferentJVMbehaviorsandOS-spe

Java:写一次,在任何地方跑步(WORA) - 深入了解平台独立性Java:写一次,在任何地方跑步(WORA) - 深入了解平台独立性May 14, 2025 am 12:05 AM

Java实现“一次编写,到处运行”通过编译成字节码并在Java虚拟机(JVM)上运行。1)编写Java代码并编译成字节码。2)字节码在任何安装了JVM的平台上运行。3)使用Java原生接口(JNI)处理平台特定功能。尽管存在挑战,如JVM一致性和平台特定库的使用,但WORA大大提高了开发效率和部署灵活性。

Java平台独立性:与不同的操作系统的兼容性Java平台独立性:与不同的操作系统的兼容性May 13, 2025 am 12:11 AM

JavaachievesPlatFormIndependencethroughTheJavavIrtualMachine(JVM),允许Codetorunondifferentoperatingsystemsswithoutmodification.thejvmcompilesjavacodeintoplatform-interploplatform-interpectentbybyteentbytybyteentbybytecode,whatittheninternterninterpretsandectectececutesoneonthepecificos,atrafficteyos,Afferctinginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginginging

什么功能使Java仍然强大什么功能使Java仍然强大May 13, 2025 am 12:05 AM

JavaispoperfulduetoitsplatFormitiondence,对象与偏见,RichstandardLibrary,PerformanceCapabilities和StrongsecurityFeatures.1)Platform-dimplighandependectionceallowsenceallowsenceallowsenceallowsencationSapplicationStornanyDevicesupportingJava.2)

顶级Java功能:开发人员的综合指南顶级Java功能:开发人员的综合指南May 13, 2025 am 12:04 AM

Java的顶级功能包括:1)面向对象编程,支持多态性,提升代码的灵活性和可维护性;2)异常处理机制,通过try-catch-finally块提高代码的鲁棒性;3)垃圾回收,简化内存管理;4)泛型,增强类型安全性;5)ambda表达式和函数式编程,使代码更简洁和表达性强;6)丰富的标准库,提供优化过的数据结构和算法。

See all articles

热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

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

热门文章

热工具

MinGW - 适用于 Windows 的极简 GNU

MinGW - 适用于 Windows 的极简 GNU

这个项目正在迁移到osdn.net/projects/mingw的过程中,你可以继续在那里关注我们。MinGW:GNU编译器集合(GCC)的本地Windows移植版本,可自由分发的导入库和用于构建本地Windows应用程序的头文件;包括对MSVC运行时的扩展,以支持C99功能。MinGW的所有软件都可以在64位Windows平台上运行。

安全考试浏览器

安全考试浏览器

Safe Exam Browser是一个安全的浏览器环境,用于安全地进行在线考试。该软件将任何计算机变成一个安全的工作站。它控制对任何实用工具的访问,并防止学生使用未经授权的资源。

DVWA

DVWA

Damn Vulnerable Web App (DVWA) 是一个PHP/MySQL的Web应用程序,非常容易受到攻击。它的主要目标是成为安全专业人员在合法环境中测试自己的技能和工具的辅助工具,帮助Web开发人员更好地理解保护Web应用程序的过程,并帮助教师/学生在课堂环境中教授/学习Web应用程序安全。DVWA的目标是通过简单直接的界面练习一些最常见的Web漏洞,难度各不相同。请注意,该软件中

Dreamweaver Mac版

Dreamweaver Mac版

视觉化网页开发工具

EditPlus 中文破解版

EditPlus 中文破解版

体积小,语法高亮,不支持代码提示功能