찾다
Javajava지도 시간JAVA Reflection 사용의 장점과 단점

JAVA Reflection 사용의 장점과 단점

Dec 12, 2016 am 11:59 AM
자바 반사

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方法 

JAVA Reflection 사용의 장점과 단점

实体类如下:  

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으로 문의하세요.
Java는 주로 무엇입니까? 실제 개발에서 Java의 주요 용도 분석Java는 주로 무엇입니까? 실제 개발에서 Java의 주요 용도 분석May 16, 2025 pm 02:54 PM

Java는 주로 데스크탑 애플리케이션, 모바일 응용 프로그램, 엔터프라이즈 레벨 솔루션 및 빅 데이터 처리에 사용됩니다. 1. 엔터프라이즈 레벨 애플리케이션 : Javaee를 통한 은행 시스템과 같은 복잡한 응용 프로그램을 지원합니다. 2. 웹 개발 : Spring과 Hibernate를 사용하여 개발을 단순화하고 SpringBoot는 마이크로 서비스를 신속하게 구축합니다. 3. 모바일 애플리케이션 : 여전히 Android 개발을위한 주요 언어 중 하나입니다. 4. 빅 데이터 처리 : Hadoop 및 Spark Process Java를 기반으로 한 대규모 데이터. 5. 게임 개발 : Minecraft와 같은 중소형 게임 개발에 적합합니다.

Java를 중국어로 설정하는 방법 Java 개발 도구 중국 인터페이스 설정 튜토리얼Java를 중국어로 설정하는 방법 Java 개발 도구 중국 인터페이스 설정 튜토리얼May 16, 2025 pm 02:51 PM

Java 개발 도구를 중국 인터페이스로 설정하는 방법은 무엇입니까? 다음 단계를 통해 구현할 수 있습니다. Eclipse : Window-> Preferences-> General-> Lookance-> i18nsupport-> Language-> Chinese (단순화) 및 Eclipse를 다시 시작하십시오. Intellijidea : help-> findaction-> "SwitchLanguage"를 입력하십시오-> "Switchidelanguage & q를 선택하십시오

Java가 일하는 데 얼마나 걸립니까? Java 학습주기 및 고용 시간 추정Java가 일하는 데 얼마나 걸립니까? Java 학습주기 및 고용 시간 추정May 16, 2025 pm 02:48 PM

일반적으로 Java를 배우고 작업 수준에 도달하는 데 6 개월에서 12 개월이 걸리며 프로그래밍 재단이있는 사람들의 경우 3 ~ 6 개월로 단축 될 수 있습니다. 1) Foundation Zero가있는 학습자는 기본 사항과 일반적으로 사용 된 도서관을 6-12 개월 동안 마스터해야합니다. 2) 프로그래밍 재단이있는 사람들은 3-6 개월 이내에 마스터 할 수 있습니다. 3) 9-18 개월의 고용 후, 실제 프로젝트와 인턴쉽은 프로세스를 가속화 할 수 있습니다.

Java의 새로운 것은 무엇입니까? 새 연산자의 메모리 할당 프로세스Java의 새로운 것은 무엇입니까? 새 연산자의 메모리 할당 프로세스May 16, 2025 pm 02:45 PM

Java에서 새 연산자는 객체를 만드는 데 사용되며 프로세스에는 다음이 포함됩니다. 1) 힙 메모리에 공간 할당, 2) 객체 초기화, 3) 생성자 호출 및 4) 객체 참조를 반환합니다. 이러한 단계를 이해하면 메모리 사용을 최적화하고 응용 프로그램 성능을 향상시키는 데 도움이 될 수 있습니다.

배열을 정의하는 방법 Java 구문 형식 설명 배열 선언 설명배열을 정의하는 방법 Java 구문 형식 설명 배열 선언 설명May 16, 2025 pm 02:42 PM

Java에서 배열을 정의하기위한 구문은 다음과 같습니다. 1. 데이터 유형 [] 배열 이름 = 새 데이터 유형 [배열 길이]; 2. 데이터 유형 배열 이름 [] = 새 데이터 유형 [배열 길이]; 3. 데이터 유형 [] 배열 이름 = {요소 목록}; 배열은 객체이며 널이 될 수 있으며 첨자는 0에서 시작합니다.이를 사용하면 NullPointerException 및 ArrayIndExoutOfBoundSexception과 같은 잠재적 오류에주의를 기울여야합니다.

Java에서 새로운 키워드 사용 새로운 키워드의 개체 인스턴스 생성에 대한 자세한 설명Java에서 새로운 키워드 사용 새로운 키워드의 개체 인스턴스 생성에 대한 자세한 설명May 16, 2025 pm 02:39 PM

새로운 키워드는 Java에서 객체 인스턴스를 만드는 데 사용됩니다. 1) JVM에 메모리를 할당하고 생성자에게 전화하여 객체를 초기화하도록 지시합니다. 2) 컨텐츠가 동일하더라도 새로운 개체를 강제로 사용하도록합니다. 3) 생성자는 사용자 정의 초기화를 허용합니다. 4) 새로운 사용은 성능 문제와 메모리 누출로 이어질 수 있습니다. 5) 가능한 예외를 처리하기 위해 Try-Catch를 사용해야합니다. 6) 익명의 내부 클래스는 새로운 사용법입니다.

Java Chinese Garbled 솔루션 캐릭터 인코딩 변환을위한 몇 가지 기술Java Chinese Garbled 솔루션 캐릭터 인코딩 변환을위한 몇 가지 기술May 16, 2025 pm 02:36 PM

Java에서 중국어의 문제를 해결하려면 다음 단계를 사용할 수 있습니다. 1. UTF-8 또는 GBK와 같은 올바른 문자 인코딩을 설정하여 파일, 데이터베이스 및 네트워크 통신이 동일한 인코딩을 사용하도록하십시오. 2. Java의 캐릭터 인코딩 변환 클래스를 사용하여 필요한 인코딩 변환을 수행하십시오. 3. 디버깅 도구 및 로그를 통해 인코딩이 올바른지 확인하여 중국 디스플레이가 다른 환경에서 정상인지 확인하십시오.

Java의 두 가지 범주의 예외는 무엇입니까? 확인 된 예외와 확인되지 않은 예외의 차이Java의 두 가지 범주의 예외는 무엇입니까? 확인 된 예외와 확인되지 않은 예외의 차이May 16, 2025 pm 02:33 PM

Java의 예외는 점검 된 예외 및 확인되지 않은 예외로 나뉩니다. 점검 유형 예외는 명시 적으로 처리되어야합니다. 그렇지 않으면 컴파일러가 오류를보고합니다.이 오류는 종종 파일을 찾을 수없는 파일과 같은 오류를 복구하는 데 사용됩니다. 확인되지 않은 예외는 명시 적으로 처리 할 필요가 없으며 종종 NULL 포인터 예외와 같은 프로그래밍 오류에 사용됩니다.

See all articles

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

뜨거운 도구

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

VSCode Windows 64비트 다운로드

VSCode Windows 64비트 다운로드

Microsoft에서 출시한 강력한 무료 IDE 편집기

ZendStudio 13.5.1 맥

ZendStudio 13.5.1 맥

강력한 PHP 통합 개발 환경