Javaにおけるサービスレイヤーパターン:堅牢なサービスレイヤーによるアプリケーションアーキテクチャの強化
別名
- アプリケーションファサード
サービスレイヤーのデザインパターンの意図
サービスレイヤーパターンは、ビジネスプロセスをユーザーインターフェースの懸念事項から分離する堅牢なアプリケーションアーキテクチャの構築に焦点を当てているJava開発者にとって非常に重要です。
このパターンは、関心の分離を促進し、プレゼンテーションレイヤーに明確に定義されたAPIを提供するために、ビジネスロジックを明確なレイヤーにカプセル化します。
実際の例を用いたサービスレイヤーパターンの詳細な説明
実際の例
レストランの複雑なシステムを想像してみてください。注文は、効率的な運用とフロントとバックの円滑なコミュニケーションを確保するために、集中型の「サービスレイヤー」を通じて管理されます。各セクションは食事の一部を専門としていますが、ウェイターは厨房スタッフと直接やり取りしません。代わりに、すべての注文はワークフローを調整するヘッドシェフを通じて行われます。ヘッドシェフはサービスレイヤーのように機能し、ビジネスロジック(注文調整)を処理し、ウェイター(プレゼンテーションレイヤー)がキッチン(データアクセスレイヤー)とやり取りするための統一されたインターフェースを提供します。
平易な言葉で
ビジネスロジックを明確なレイヤーにカプセル化して、関心の分離を促進し、プレゼンテーションレイヤーに明確なAPIを提供するパターン。
Wikipediaによると
サービスレイヤーは、サービス指向の設計パラダイム内で適用されるアーキテクチャパターンであり、サービスインベントリ内のサービスを論理レイヤーのセットに整理することを目的としています。特定のレイヤーに分類されるサービスは、機能を共有します。これにより、同じレイヤーに属するサービスがより小さな一連のアクティビティに対応するため、サービスインベントリの管理に関連する概念的なオーバーヘッドを削減するのに役立ちます。
Javaでのサービスレイヤーパターンのプログラム例
当社のJava実装では、サービスレイヤーパターンを使用して、データアクセスオブジェクト(DAO)とビジネスロジック間のやり取りを効率化し、関心の明確な分離を確保します。
このアプリケーションの例では、クライアントApp
と、ウィザード、魔法の書、呪文間のやり取りを可能にするサービスMagicService
との間のやり取りを示しています。サービスは3層アーキテクチャで実装されています。
(エンティティ、DAO、サービス)。
この説明では、システムの1つの垂直スライスを見ています。エンティティレイヤーから始めて、Wizard
クラスを見てみましょう。ここに示されていない他のエンティティは、Spellbook
とSpell
です。
@Entity
@Table(name = "WIZARD")
@Getter
@Setter
public class Wizard extends BaseEntity {
@Id
@GeneratedValue
@Column(name = "WIZARD_ID")
private Long id;
private String name;
@ManyToMany(cascade = CascadeType.ALL)
private Set<Spellbook> spellbooks;
public Wizard() {
spellbooks = new HashSet<>();
}
public Wizard(String name) {
this();
this.name = name;
}
public void addSpellbook(Spellbook spellbook) {
spellbook.getWizards().add(this);
spellbooks.add(spellbook);
}
@Override
public String toString() {
return name;
}
}
エンティティレイヤーの上にDAOがあります。Wizard
の場合、DAOレイヤーは次のようになります。
public interface WizardDao extends Dao<Wizard> {
Wizard findByName(String name);
}
public class WizardDaoImpl extends DaoBaseImpl<Wizard> implements WizardDao {
@Override
public Wizard findByName(String name) {
Transaction tx = null;
Wizard result;
try (var session = getSessionFactory().openSession()) {
tx = session.beginTransaction();
var criteria = session.createCriteria(persistentClass);
criteria.add(Restrictions.eq("name", name));
result = (Wizard) criteria.uniqueResult();
tx.commit();
} catch (Exception e) {
if (tx != null) {
tx.rollback();
}
throw e;
}
return result;
}
}
次に、サービスレイヤーを見てみましょう。この場合、単一のMagicService
で構成されています。
public interface MagicService {
List<Wizard> findAllWizards();
List<Spellbook> findAllSpellbooks();
List<Spell> findAllSpells();
List<Wizard> findWizardsWithSpellbook(String name);
List<Wizard> findWizardsWithSpell(String name);
}
public class MagicServiceImpl implements MagicService {
private final WizardDao wizardDao;
private final SpellbookDao spellbookDao;
private final SpellDao spellDao;
public MagicServiceImpl(WizardDao wizardDao, SpellbookDao spellbookDao, SpellDao spellDao) {
this.wizardDao = wizardDao;
this.spellbookDao = spellbookDao;
this.spellDao = spellDao;
}
@Override
public List<Wizard> findAllWizards() {
return wizardDao.findAll();
}
@Override
public List<Spellbook> findAllSpellbooks() {
return spellbookDao.findAll();
}
@Override
public List<Spell> findAllSpells() {
return spellDao.findAll();
}
@Override
public List<Wizard> findWizardsWithSpellbook(String name) {
var spellbook = spellbookDao.findByName(name);
return new ArrayList<>(spellbook.getWizards());
}
@Override
public List<Wizard> findWizardsWithSpell(String name) {
var spell = spellDao.findByName(name);
var spellbook = spell.getSpellbook();
return new ArrayList<>(spellbook.getWizards());
}
}
そして最後に、クライアントApp
がサービスレイヤーのMagicService
とどのようにやり取りするかを示すことができます。
@Slf4j
public class App {
public static final String BOOK_OF_IDORES = "Book of Idores";
public static void main(String[] args) {
// populate the in-memory database
initData();
// query the data using the service
queryData();
}
public static void initData() {
// spells
var spell1 = new Spell("Ice dart");
var spell2 = new Spell("Invisibility");
var spell3 = new Spell("Stun bolt");
var spell4 = new Spell("Confusion");
var spell5 = new Spell("Darkness");
var spell6 = new Spell("Fireball");
var spell7 = new Spell("Enchant weapon");
var spell8 = new Spell("Rock armour");
var spell9 = new Spell("Light");
var spell10 = new Spell("Bee swarm");
var spell11 = new Spell("Haste");
var spell12 = new Spell("Levitation");
var spell13 = new Spell("Magic lock");
var spell14 = new Spell("Summon hell bat");
var spell15 = new Spell("Water walking");
var spell16 = new Spell("Magic storm");
var spell17 = new Spell("Entangle");
var spellDao = new SpellDaoImpl();
spellDao.persist(spell1);
spellDao.persist(spell2);
spellDao.persist(spell3);
spellDao.persist(spell4);
spellDao.persist(spell5);
spellDao.persist(spell6);
spellDao.persist(spell7);
spellDao.persist(spell8);
spellDao.persist(spell9);
spellDao.persist(spell10);
spellDao.persist(spell11);
spellDao.persist(spell12);
spellDao.persist(spell13);
spellDao.persist(spell14);
spellDao.persist(spell15);
spellDao.persist(spell16);
spellDao.persist(spell17);
// spellbooks
var spellbookDao = new SpellbookDaoImpl();
var spellbook1 = new Spellbook("Book of Orgymon");
spellbookDao.persist(spellbook1);
spellbook1.addSpell(spell1);
spellbook1.addSpell(spell2);
spellbook1.addSpell(spell3);
spellbook1.addSpell(spell4);
spellbookDao.merge(spellbook1);
var spellbook2 = new Spellbook("Book of Aras");
spellbookDao.persist(spellbook2);
spellbook2.addSpell(spell5);
spellbook2.addSpell(spell6);
spellbookDao.merge(spellbook2);
var spellbook3 = new Spellbook("Book of Kritior");
spellbookDao.persist(spellbook3);
spellbook3.addSpell(spell7);
spellbook3.addSpell(spell8);
spellbook3.addSpell(spell9);
spellbookDao.merge(spellbook3);
var spellbook4 = new Spellbook("Book of Tamaex");
spellbookDao.persist(spellbook4);
spellbook4.addSpell(spell10);
spellbook4.addSpell(spell11);
spellbook4.addSpell(spell12);
spellbookDao.merge(spellbook4);
var spellbook5 = new Spellbook(BOOK_OF_IDORES);
spellbookDao.persist(spellbook5);
spellbook5.addSpell(spell13);
spellbookDao.merge(spellbook5);
var spellbook6 = new Spellbook("Book of Opaen");
spellbookDao.persist(spellbook6);
spellbook6.addSpell(spell14);
spellbook6.addSpell(spell15);
spellbookDao.merge(spellbook6);
var spellbook7 = new Spellbook("Book of Kihione");
spellbookDao.persist(spellbook7);
spellbook7.addSpell(spell16);
spellbook7.addSpell(spell17);
spellbookDao.merge(spellbook7);
// wizards
var wizardDao = new WizardDaoImpl();
var wizard1 = new Wizard("Aderlard Boud");
wizardDao.persist(wizard1);
wizard1.addSpellbook(spellbookDao.findByName("Book of Orgymon"));
wizard1.addSpellbook(spellbookDao.findByName("Book of Aras"));
wizardDao.merge(wizard1);
var wizard2 = new Wizard("Anaxis Bajraktari");
wizardDao.persist(wizard2);
wizard2.addSpellbook(spellbookDao.findByName("Book of Kritior"));
wizard2.addSpellbook(spellbookDao.findByName("Book of Tamaex"));
wizardDao.merge(wizard2);
var wizard3 = new Wizard("Xuban Munoa");
wizardDao.persist(wizard3);
wizard3.addSpellbook(spellbookDao.findByName(BOOK_OF_IDORES));
wizard3.addSpellbook(spellbookDao.findByName("Book of Opaen"));
wizardDao.merge(wizard3);
var wizard4 = new Wizard("Blasius Dehooge");
wizardDao.persist(wizard4);
wizard4.addSpellbook(spellbookDao.findByName("Book of Kihione"));
wizardDao.merge(wizard4);
}
public static void queryData() {
var wizardDao = new WizardDaoImpl();
var spellbookDao = new SpellbookDaoImpl();
var spellDao = new SpellDaoImpl();
var service = new MagicServiceImpl(wizardDao, spellbookDao, spellDao);
LOGGER.info("Enumerating all wizards");
service.findAllWizards().stream().map(Wizard::getName).forEach(LOGGER::info);
LOGGER.info("Enumerating all spellbooks");
service.findAllSpellbooks().stream().map(Spellbook::getName).forEach(LOGGER::info);
LOGGER.info("Enumerating all spells");
service.findAllSpells().stream().map(Spell::getName).forEach(LOGGER::info);
LOGGER.info("Find wizards with spellbook 'Book of Idores'");
var wizardsWithSpellbook = service.findWizardsWithSpellbook(BOOK_OF_IDORES);
wizardsWithSpellbook.forEach(w -> LOGGER.info("{} has 'Book of Idores'", w.getName()));
LOGGER.info("Find wizards with spell 'Fireball'");
var wizardsWithSpell = service.findWizardsWithSpell("Fireball");
wizardsWithSpell.forEach(w -> LOGGER.info("{} has 'Fireball'", w.getName()));
}
}
プログラムの出力
INFO [2024-05-27 09:16:40,668] com.iluwatar.servicelayer.app.App: Enumerating all wizards
INFO [2024-05-27 09:16:40,671] com.iluwatar.servicelayer.app.App: Aderlard Boud
INFO [2024-05-27 09:16:40,671] com.iluwatar.servicelayer.app.App: Anaxis Bajraktari
INFO [2024-05-27 09:16:40,671] com.iluwatar.servicelayer.app.App: Xuban Munoa
INFO [2024-05-27 09:16:40,671] com.iluwatar.servicelayer.app.App: Blasius Dehooge
INFO [2024-05-27 09:16:40,671] com.iluwatar.servicelayer.app.App: Enumerating all spellbooks
INFO [2024-05-27 09:16:40,675] com.iluwatar.servicelayer.app.App: Book of Orgymon
INFO [2024-05-27 09:16:40,675] com.iluwatar.servicelayer.app.App: Book of Aras
INFO [2024-05-27 09:16:40,675] com.iluwatar.servicelayer.app.App: Book of Kritior
INFO [2024-05-27 09:16:40,675] com.iluwatar.servicelayer.app.App: Book of Tamaex
INFO [2024-05-27 09:16:40,675] com.iluwatar.servicelayer.app.App: Book of Idores
INFO [2024-05-27 09:16:40,675] com.iluwatar.servicelayer.app.App: Book of Opaen
INFO [2024-05-27 09:16:40,675] com.iluwatar.servicelayer.app.App: Book of Kihione
INFO [2024-05-27 09:16:40,675] com.iluwatar.servicelayer.app.App: Enumerating all spells
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Ice dart
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Invisibility
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Stun bolt
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Confusion
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Darkness
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Fireball
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Enchant weapon
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Rock armour
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Light
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Bee swarm
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Haste
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Levitation
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Magic lock
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Summon hell bat
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Water walking
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Magic storm
INFO [2024-05-27 09:16:40,679] com.iluwatar.servicelayer.app.App: Entangle
INFO [2024-05-27 09:16:40,680] com.iluwatar.servicelayer.app.App: Find wizards with spellbook 'Book of Idores'
INFO [2024-05-27 09:16:40,680] com.iluwatar.servicelayer.app.App: Xuban Munoa has 'Book of Idores'
INFO [2024-05-27 09:16:40,681] com.iluwatar.servicelayer.app.App: Find wizards with spell 'Fireball'
INFO [2024-05-27 09:16:40,683] com.iluwatar.servicelayer.app.App: Aderlard Boud has 'Fireball'
実際の例を用いたサービスレイヤーパターンの詳細な説明

Javaでサービスレイヤーパターンを使用するタイミング
- ビジネスロジックをプレゼンテーションロジックから分離する必要がある場合に使用します。
- 複雑なビジネスルールとワークフローを持つアプリケーションに最適です。
- プレゼンテーションレイヤーに明確なAPIを必要とするプロジェクトに適しています。
Javaにおけるサービスレイヤーパターンの実際のアプリケーション
- Enterprise JavaBeans(EJB)を使用してサービスレイヤーを実装するJava EEアプリケーション。
- サービスレイヤークラスを示すために@Serviceアノテーションを使用するSpring Frameworkアプリケーション。
- ビジネスロジックをコントローラーロジックから分離する必要があるWebアプリケーション。
サービスレイヤーパターンのメリットとトレードオフ
メリット
Javaでのサービスレイヤーの実装
- ビジネスロジックを1箇所にカプセル化することにより、コードの再利用を促進します。
- ビジネスロジックを分離することにより、テスト可能性を向上させます。
- エンタープライズアプリケーションの保守性と柔軟性を向上させます。
トレードオフ
- アプリケーションに別のレイヤーを追加することにより、複雑さを増す可能性があります。
- 抽象化の追加レイヤーにより、パフォーマンスのオーバーヘッドが発生する可能性があります。
関連するJavaデザインパターン
- ファサード:統一されたインターフェースを提供することにより、複雑なサブシステムとのやり取りを簡素化します。
- DAO(データアクセスオブジェクト):データ永続性を処理するために、サービスレイヤーと一緒に使用されることがよくあります。
- MVC(モデルビューコントローラー):サービスレイヤーは、モデルコンポーネントのビジネスロジックをカプセル化するために使用できます。