Skip to content

12. [TD]: تنفيذ طبقة [DAO] في TD باستخدام [Spring Data]

الكلمات المفتاحية: بنية متعددة الطبقات، Spring، حقن التبعية، JPA (Java Persistence APISpring Data.

سنتبع نفس النهج السابق لتنفيذ طبقة [DAO] في TD.

12.1. الدعم

  • في [1]، يحتوي المجلد [support / chap-12] على مشروع Eclipse الخاص بهذا الفصل؛

12.2. مشروع Eclipse

سيكون مشروع Eclipse على النحو التالي:

  
  • [elections.dao.config]: يحتوي على فئة تكوين مشروع Spring؛
  • [elections.dao.entities]: يحتوي على كيانات JPA بالإضافة إلى فئة الاستثناءات الخاصة بالمشروع؛
  • [elections.dao.repositories]: يحتوي على واجهات [CrudRepository] لجداول [CONF] و[LISTES
  • [elections.dao.service]: يحتوي على تنفيذ طبقة [DAO]. هذا هو ما نحتاج إلى كتابته؛
  • [elections.dao.console]: يحتوي على فئة اختبار [console
  • [pom.xml]: ملف تكوين مشروع Maven؛

يُنفذ هذا المشروع البنية التالية:

لا ترى طبقة [DAO] سوى الطبقة التي تنفذها [Spring Data].

12.3. تكوين Maven


المهمة: إنشاء ملف [pom.xml] الخاص بالمشروع.


12.4. كيانات طبقة [JPA]

  
  • [ElectionsConfig] هو نموذج الكائن المرتبط بصف في جدول [CONF
  • [VoterList] هو نموذج الكائن المرتبط بصف في جدول [LISTS
  • [AbstractEntity] هي الفئة الأم للفئتين السابقتين. وهي تفصل الحقول [id, version] المشتركة بين الفئتين؛
  • [ElectionsException] هي فئة الاستثناءات الخاصة بالمشروع؛

12.4.1. فئة [ElectionsException]

  

فئة [ElectionsException] هي الاستثناء غير المعالج الموصوف في القسم 4.3.

12.4.2. فئة [AbstractEntity]

  

فئة [AbstractEntity] هي كما يلي:


package elections.dao.entities;
 
import java.io.Serializable;
 
import javax.persistence.Column;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.MappedSuperclass;
import javax.persistence.Version;
 
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
 
@MappedSuperclass
public abstract class AbstractEntity implements Serializable{
    private static final long serialVersionUID = 1L;
 
    // properties
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name = "ID")
    protected Long id;
    @Version
    @Column(name = "VERSION")
    protected Long version;
 
    // manufacturers
    public AbstractEntity() {
 
    }
 
    public AbstractEntity(Long id, Long version) {
        this.id = id;
        this.version = version;
    }
 
    // redefine [equals] and [hashcode]
    @Override
    public int hashCode() {
        return (id != null ? id.hashCode() : 0);
    }
 
    @Override
    public boolean equals(Object entity) {
        if (!(entity instanceof AbstractEntity)) {
            return false;
        }
        String class1 = this.getClass().getName();
        String class2 = entity.getClass().getName();
        if (!class2.equals(class1)) {
            return false;
        }
        AbstractEntity other = (AbstractEntity) entity;
        return id != null && this.id == other.id;
    }
 
    // signature jSON
    public String toString() {
        ObjectMapper mapper = new ObjectMapper();
        try {
            return mapper.writeValueAsString(this);
        } catch (JsonProcessingException e) {
            e.printStackTrace();
            return null;
        }
    }
 
    // getters and setters
...
}

هذه هي الفئة الموصوفة في القسم 11.3.5.2، بدون مرشحات JSON. هنا، لا ترتبط الجداول [CONF] و[LISTES] بعلاقة مفتاح خارجي. ومع ذلك، فإن وجود مثل هذه العلاقة مع وضع [التحميل المتأخر] هو ما يستلزم مرشحات JSON.

12.4.3. فئة [ElectionsConfig]

  

فئة [ElectionsConfig] هي كيان JPA المرتبط بالجدول [CONF


المهمة: قم بتنفيذ فئة [ElectionsConfig].


12.4.4. فئة [VoterList]

  

فئة [VoterList] هي الكيان JPA المرتبط بالجدول [LISTS


المهمة: قم بتنفيذ فئة [VoterList].


12.5. طبقة [Spring Data]

  

المهمة: اكتب واجهتي [Spring Data] لإدارة الجدولين [CONF] و[LISTES


12.6. طبقة [DAO]

  

واجهة [IElectionsDao] لطبقة [DAO] هي كما يلي:


package dao.service;
 
import dao.entities.ElectionsConfig;
import dao.entities.ListeElectorale;
 
public interface IElectionsDao {
 
    // election configuration
    public ElectionsConfig getElectionsConfig();
 
    // candidate lists
    public ListeElectorale[] getListesElectorales();
 
    // updating candidate lists
    public void setListesElectorales(ListeElectorale[] listesElectorales);
}

المهمة: اكتب تنفيذ [ElectionsDaoJpa] لواجهة [IElectionsDao].


12.7. تكوين مشروع Spring

  

تقوم فئة [DaoConfig] بتكوين مشروع Spring.


المهمة: اكتب فئة [DaoConfig]


12.8. طبقة [console]

  

الفئة [Main] هي الفئة القابلة للتنفيذ التالية:


package dao.console;
 
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
 
import dao.config.AppConfig;
import dao.entities.ElectionsConfig;
import dao.entities.ListeElectorale;
import dao.service.IElectionsDao;
 
public class Main {
 
    // data source
    private static IElectionsDao dao;
 
    public static void main(String[] args) {
        // spring context retrieval
        AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(AppConfig.class);
        // data source recovery
        dao = ctx.getBean(IElectionsDao.class);
        // contents of both tables
        ElectionsConfig electionsConfig = dao.getElectionsConfig();
        ListeElectorale[] listes = dao.getListesElectorales();
        // display
        System.out.println(String.format("Nombre de sièges à pourvoir : %d", electionsConfig.getNbSiegesAPourvoir()));
        System.out.println(String.format("Seuil électoral : %5.2f", electionsConfig.getSeuilElectoral()));
        System.out.println("Listes candidates----------------");
        for (ListeElectorale liste : listes) {
            System.out.println(liste);
        }
        // closing Spring context
        ctx.close();
    }
 
}

فئة [Test01] هي اختبار JUnit:


package dao.tests;
 
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.SpringApplicationConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 
import dao.config.AppConfig;
import dao.entities.ElectionsConfig;
import dao.entities.ListeElectorale;
import dao.service.IElectionsDao;
 
@SpringApplicationConfiguration(classes = AppConfig.class)
@RunWith(SpringJUnit4ClassRunner.class)
public class Test01 {
 
    // layer [DAO]
    @Autowired
    private IElectionsDao electionsDao;
 
    @Before
    public void init() {
        // the table is cleaned [LISTES]
        // competing lists
        ListeElectorale[] listes = electionsDao.getListesElectorales();
        // set votes and seats to 0 and eliminate false
        int voix = 0;
        int sièges = 0;
        boolean elimine = false;
        for (ListeElectorale liste : listes) {
            liste.setVoix(voix);
            liste.setSieges(sièges);
            liste.setElimine(elimine);
        }
        // we make this data persistent using the [dao] layer
        electionsDao.setListesElectorales(listes);
    }
 
    @Test
    public void testElections01() {
        System.out.println("testElections01-------------------------------------");
        // election configuration recovery
        ElectionsConfig electionsConfig = electionsDao.getElectionsConfig();
        int nbSiegesAPourvoir = electionsConfig.getNbSiegesAPourvoir();
        double seuilElectoral = electionsConfig.getSeuilElectoral();
        Assert.assertEquals(6, nbSiegesAPourvoir);
        Assert.assertEquals(0.05, seuilElectoral, 1E-6);
 
        // competing lists
        ListeElectorale[] listes = electionsDao.getListesElectorales();
        // display read values
        System.out.println("Nombre de sièges à pourvoir : " + nbSiegesAPourvoir);
        System.out.println("Seuil électoral : " + seuilElectoral);
        System.out.println("Listes en compétition ---------------------");
        for (int i = 0; i < listes.length; i++) {
            System.out.println(listes[i]);
        }
 
        // votes and seats are allocated to lists
        int voix = 0;
        int sièges = 0;
        boolean elimine = false;
        for (ListeElectorale liste : listes) {
            liste.setVoix(voix);
            liste.setSieges(sièges);
            liste.setElimine(elimine);
            voix += 10;
            sièges += 1;
            elimine = !elimine;
        }
 
        // we make this data persistent using the [dao] layer
        electionsDao.setListesElectorales(listes);
 
        // data re-reading
        ListeElectorale[] listesElectorales2 = electionsDao.getListesElectorales();
        // check data read
        Assert.assertEquals(7, listesElectorales2.length);
        voix = 0;
        sièges = 0;
        elimine = false;
        for (ListeElectorale liste : listes) {
            Assert.assertEquals(voix, liste.getVoix());
            Assert.assertEquals(sièges, liste.getSieges());
            Assert.assertEquals(elimine, liste.isElimine());
            voix += 10;
            sièges += 1;
            elimine = !elimine;
        }
        System.out.println("Listes en compétition ---------------------");
        for (int i = 0; i < listes.length; i++) {
            System.out.println(listes[i]);
        }
    }
}

المهمة: قم بتشغيل اختبارات [console] و [JUnit] على طبقة [DAO] الخاصة بك.


12.9. إنشاء أرشيف Maven للمشروع

باتباع المثال الوارد في القسم 11.3.12، قم بإنشاء أرشيف Maven للمشروع.