Полезные ссылки

Построитель стилей

Природа не понимает шуток. Она всегда честна, всегда серьёзна, всегда сурова. Она всегда права, а ошибки и промахи делают только люди. Человек, который относится к природе с пренебрижением, не способен её оценить, и только понимающему, чистому и правдивому, она покоряется и открывает свои тайны.
Иоган Вольфнг фон Гёте
Связываем Flex(Air) и Базу данных, используя RemoteObject, BlazeDS, Spring и Hibernate
28.06.2009 22:57

В прошлой статье я описал настройку связи Java сервера и Flex (Air) келиента, используя BlazeDS под управлением Spring. В этой статье мы добавим связь с БД (Postgres 8.3) используя набор библиотек Hibernate3.

В данной статье я не буду приводить все изменения в коде, но отмечу все основные моменты. В данном примере, мы на сервере добавим два метода:
- сохранить(добавить) объект.
- получить список объектов.
Шаг первый создадим доменный класс на стороне Java и на стороне Flex.
Java
package ru.gubber.elearning.domain;

import flex.messaging.annotations.IAnnotatedProxy;
import flex.messaging.annotations.FlexClass;

@FlexClass(classType = FlexClass.FlexClassType.RemoteObject)
public class Student implements IAnnotatedProxy {
private long id;
private String login;
private String firstName;
private String lastName;
private String patronomycName;
private String password;

public long getId() {
return id;
}

public void setId(long id) {
this.id = id;
}

....

}

Flex

package ru.gubber.domain {
//Указываем, что данный класс сереализуется и десериализуется из серверного класса ru.gubber.elearning.domain.Student.
[RemoteClass(alias="ru.gubber.elearning.domain.Student")]
//Измененние любого свойства объектта этого класса ведёт к генерации события (CHANGE)
[Bindable]
public class Student
{

private var id_:Number;
private var login_:String ;
private var firstName_:String ;
private var lastName_:String ;
private var patronomycName_:String ;
private var password_:String ;

public function Student()
{
id_ = 0;
}

public function get id():Number {
return id_;
}

public function set id(val:Number):void {
id_ = val;
}

...
}
}
В данном примере, я использую методы get и set как в серверной(Java) так и на клиентской(Flex) части. Во Flex части данного можно избежать, я просто так делаю в силу привычки (и идеалогических предпочтений).
Дальше нам надо связать наш teacherService через Spring и Hibernate с базой данных.
Изменения в файле applicationContext.xml

<!--Hibernate -->
<bean id="myDataSource" class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName" value="org.postgresql.Driver"/>
<property name="url" value="jdbc:postgresql://localhost/pgtest"/>
<property name="username" value="pgtest"/>
<property name="password" value="pgtest"/>
</bean>

<bean id="mySessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<property name="dataSource" ref="myDataSource"/>
<property name="mappingResources">
<list>
<value>
student.hbm.xml</value>
</list>
</property>
<property name="hibernateProperties">
<value>

hibernate.dialect=org.hibernate.dialect.PostgreSQLDialect
hibernate.show_sql=true
hibernate.connection.SelectMethod=cursor
hibernate.cglib.use_reflection_optimizer=false
hibernate.connection.pool_size=100
hibernate.jdbc.batch_size=0

</value>
</property>
</bean>

<bean id="hibernateTemplate" class="org.springframework.orm.hibernate3.HibernateTemplate">
<property name="sessionFactory" ref="mySessionFactory"/>

</bean>

<bean name="studentDAO" class="ru.gubber.elearning.dao.StudentDAO">
<property name="hibernateTemplate" ref="hibernateTemplate" />
</bean
>

<bean name="userProcessor" class="ru.gubber.elearning.business.UserProcessor">
<property name="studentDAO" ref="studentDAO" />
</bean
>
<!--Hibernate -->

<!-- FLEX -->

<!-- Implementation of ProductDAO using low-level JDBC -->
<bean id="teacherServiceBusiness" class="ru.gubber.elearning.flex.UserListService" >
<property name="userProcessor" ref="userProcessor" />
</bean
>

В данном случае я создал источник данных, связял с ним SessionFactory, на его основи созда hibernateTemplate, который уже использую в StudenDAO, для доступа к БД.
JAVA

package ru.gubber.elearning.dao;

import org.springframework.orm.hibernate3.HibernateTemplate;
import org.springframework.orm.hibernate3.HibernateCallback;
import org.hibernate.Session;
import org.hibernate.HibernateException;
import ru.gubber.elearning.domain.Student;

import java.sql.SQLException;
import java.util.List;

public class StudentDAO {

private HibernateTemplate hibernateTemplate;

public void setHibernateTemplate(HibernateTemplate hibernateTemplate) {
this.hibernateTemplate = hibernateTemplate;
}

public void saveStudent(final Student student) {
HibernateCallback callback = new HibernateCallback() {
@Override
public Object doInHibernate(Session session) throws HibernateException, SQLException {
session.saveOrUpdate(student);
return null;
}
};

hibernateTemplate.execute(callback);
}

public List getStudentList() {
HibernateCallback callback = new HibernateCallback() {
@Override
public Object doInHibernate(Session session) throws HibernateException, SQLException {
List list = session.createCriteria(Student.class).list();
return list;
}
};

return (List) hibernateTemplate.execute(callback);
}
}

Классы UserProcessor и UserListService, не делают ни каких действий, просто прокидывают вызовы в StudentDAO. Опять таки это сделано по идеалогическим причинам. Класс UserProcessor отвечает за обработку бизнес логики, когда она будет. Класс же UserListService получает запрос от клиентской части, передаёт запрос в уровень бизнес логики, преобразует результат к виду предназначенному для клиента и отдаёт преобразованный результат клиенту.
Flex

<?xml version="1.0" encoding="utf-8"?>
<mx:WindowedApplication xmlns:mx="http://www.adobe.com/2006/mxml"
layout="absolute" width="500" height="400"
creationComplete="init()">
<mx:Script>
<![CDATA[
....

import ru.gubber.domain.Student;

[Bindable]
private var studentList:ArrayCollection = new ArrayCollection();
[Bindable]
private var student:Student = new Student();

// вызов серверной функции
private function handleGetStudentListError(event:FaultEvent):void {
Alert.show("Не возможно получить список студентов.\n" + event.toString())
}

private function handleGetStudentListSuccess(event:ResultEvent):void {
studentList = event.result as ArrayCollection;
}

private function handleSaveStudentError(event:FaultEvent):void {
Alert.show("Не возможно сохранить информацию о студенте.\n" + event.toString())
}

private function handleSaveStudentSuccess(event:ResultEvent):void {
testDAO.getStudentList();
Alert.show("Информация о студенте успешно сохранена.")
}

.......

//Формирование информации о пользователе и отправление её на сервер для сохранения
private
function saveUser(ev:Event):void{
student.login = tiLogin.text;
student.password = tiPassword.text;
student.firstName = tiFName.text;
student.lastName = tiLName.text;
student.patronomycName = tiPatronomyc.text;
testDAO.saveStudent(student);
}

]]>
</mx:Script>
<!-- Дальне идёт создание интерфейса, который моджно увидеть в исходниках-->

</mx:WindowedApplication>

Результат, после запуска сервера и клиентского приложения должен выглядеть следующим образом

air-blazeds-0002

.Небольшие хотрости, которые были использованы при написании этой статьи. Что бы не писать два метода на сохранение и обновление информации о студенте, использовалось свойство Hibernate, когда указывается при каком значении id, счтитать, что запись новая. В нашем примере, значение id =0, говорит Hibernate, что надо сделать операцию Insert, а не Update. Именно по этому во Flex части приложения в конструкторе класса Student в поле id_, зписывается значение 0. В данном примере данные прочитанные из БД сериализуются для передачи на клиентскую часть, это не самая хорошая практика, лучше разделить доменные классы сервера и классы, которые передаются на клиента.
Обновление исходников можно сказать по ссылке (открывается в новом окне, весит 3,82 МБ):
Комментарии
Добавить новый Поиск
Большой Синий Кит   |2009-10-01 08:23:56
Отличная статья! Спасибо большое!
Оставить комментарий
Имя:
Email:
 
Веб-сайт:
Тема:
UBB-Код:
[b] [i] [u] [url] [quote] [code] [img] 
 
 
:angry::0:confused::cheer:B):evil::silly::dry::lol::kiss::D:pinch:
:(:shock::X:side::):P:unsure::woohoo::huh::whistle:;):s
:!::?::idea::arrow:
 

!joomlacomment 4.0 Copyright (C) 2009 Compojoom.com . All rights reserved."