`
shlei
  • 浏览: 282886 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

flex的hibernate(延迟加载)解决方案

    博客分类:
  • JAVA
阅读更多
   最近在搞flex+java,用了spring和hibernate,通过blazeds通信。但是我发现blazeds无法实现hibernate的延迟加载功能,更可恨的是当进行通信的时候,对象中的所有关联关系都被强制加载了,这与blazeds的原理有关(blazeds将对象所有的属性进行了遍历,造成了属性的加载)。网上有些flex的延迟加载解决方案,比如dbhibernate,但是并不好用。

    我现在的解决方法是:去掉了绝大部分的关联关系,关掉了延迟加载功能(没有了关联关系,延迟加载也没有意义了),这样就靠自己维护关系了。只是比以前多出了一点代码量。
   
    你也可以直接不用hibernate。
    或者自己对hibernate取到的对象进行一次封装,再传给flex,这样就不会被强制加载了
    针对这个问题,有一些开源的解决方案:dpHibernate,Glead等,但笔者感觉这些框架侵入性太强,不宜采用。
    在不引入第三方框架的情况下,可以采取两个办法:1.增加一层Vo类,将类属性和集合属性的代理进行转化,然后与as映射,传给flex端 ,这样序列化是ok的。
2.采用Spring 的aop解决方案,对可能返回无法正常序列化的pojo方法进行拦截。步骤如下:
1.开启@AspectJ
<aop:aspectj-autoproxy/>

2.编写针对具有代理属性类的转化工具类:假如Student类具有多对一的类属性Grade:
public class Student {
private Long id;
private Grade grade;
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public Grade getGrade() {
return grade;
}
public void setGrade(Grade grade) {
this.grade = grade;
}
}

用代码生成器生成StudentConvert.java :
public class StudentConverter {

    public static Student convert2Student(Object sourceObj) {
    Student source=(Student) sourceObj;
        Student target = new Student();
        target.setId(source.getId());
      //Hibernate.initialize(source.getGrade());  可以选择将类或集合代理初始化
       //target.setGrade(source.getGrade());
        return target;
    }

    public static List  convert2StudentList(List oldlist) {
        List newlist= new ArrayList();
        for(Object sourceObj : oldlist) {
        newlist.add(convert2Student(sourceObj));
        }
        return newlist;
    }
}
3.编写aop的advisor进行环绕通知,假设StudentDao 有两个方法
Student findById(Long id)
List  findAll();
由代码生成器生成对应的advisor如下 HibernateAdvisor.java :

@Aspect
public class HibernateAdvisor {

//insert aop code  for hibernate
  
///////////////////////////
// Student readme
///////////////////////////
/**
* 必须为final String类型的,注解里要使用的变量只能是静态常量类型的
*/
// 切入点定义
public static final String StudentList = "execution(* com.lai.flex.service.StudentDao.findAll(..))";
public static final String Student = "execution(* com.lai.flex.service.StudentDao.findBy*(..))";

@Around(StudentList)
public List convertStudentList(ProceedingJoinPoint joinPoint) {

//System.out.println("Convert之前");
Object[] args = joinPoint.getArgs();
Object obj = null;
List newlist= null;
try {
obj = joinPoint.proceed(args);
List old= (List) obj;
newlist=(StudentConverter.convert2StudentList(oldlist));
} catch (Throwable e) {
e.printStackTrace();
}
//System.out.println("Convert之后"); // 方法执行后的代理处理
return newlist;
}

@Around(Student)
public Student convertStudent(ProceedingJoinPoint joinPoint) {

//System.out.println("Convert之前");
Object[] args = joinPoint.getArgs();
Object obj = null;

Student newsStudent = null;
try {
obj = joinPoint.proceed(args);
newsStudent = StudentConverter.convert2Student(obj);

} catch (Throwable e) {
e.printStackTrace();
}
//System.out.println("Convert之后"); // 方法执行后的代理处理
return newsStudent;
}

// generator-insert-location
 


}

最后,由StudentDao方法返回的代理类和集合属性在序列化到flex端之前,都被拦截,进行转化后,可以正常序列化了。

转自http://67566894.iteye.com/blog/686601
http://archive.cnblogs.com/a/1843018/
分享到:
评论
2 楼 shlei 2010-12-14  
359468787 写道
版主的意思我懂,不过这只是一个student类,要是类多了那就得写一大堆类似convert2StudentList这样的方法,太麻烦了,有没有更好的方法?

我QQ:359468787有时间交流一下!


到目前我还没找到比较好的方法,遇到这种问题,我一般会在后台手动把数据封装好,然后再回传给前台,对于数据前台只显示和传递,不处理。此博文是我在网上找到的相对较省力的解决方案,但还没有亲自试过。
1 楼 359468787 2010-12-14  
版主的意思我懂,不过这只是一个student类,要是类多了那就得写一大堆类似convert2StudentList这样的方法,太麻烦了,有没有更好的方法?

我QQ:359468787有时间交流一下!

相关推荐

Global site tag (gtag.js) - Google Analytics