쿠릉쿠릉 쾅쾅 2022. 4. 4. 22:07
728x90

 

 

상품 엔티티 개발 (비즈니스 로직 추가)

package bookbook.shopshop.domain.item;


import bookbook.shopshop.domain.Category;
import bookbook.shopshop.exception.NotEnoughStockException;
import lombok.Getter;

import javax.persistence.*;
import java.util.ArrayList;
import java.util.List;

@Entity
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = "dtype")
@Getter
public abstract class Item {

    @Id @GeneratedValue
    @Column(name = "item_id")
    private Long id;

    private String name;
    private int price;
    private int stockQuantity;

    @ManyToMany(mappedBy = "items")
    private List<Category> categories = new ArrayList<>();

    // == 비즈니스 로직 == //

    /**
     * stock 증가
     */
    public void addStock(int quantity) {
        this.stockQuantity += quantity;
    }

    /**
     * stock 감소
     */
    public void removeStock(int quantity) {
        int resultStock = this.stockQuantity - quantity;

        if (resultStock <0) {
            throw new NotEnoughStockException("need more stock");
        }

        this.stockQuantity = resultStock;
    }

}

엔티티는 데이터를 가지고 있기 때문에 엔티티 안에 비즈니스 로직을 넣는것이 좋다.

 


 

상품 리포지토리 개발

package bookbook.shopshop.repository;

import bookbook.shopshop.domain.item.Item;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Repository;

import javax.persistence.EntityManager;
import java.util.List;

@Repository
@RequiredArgsConstructor
public class ItemRepository {

    private final EntityManager em;

    public void save(Item item) {
        if (item.getId()==null) {
            em.persist(item);
        } else  {
            em.merge(item);
        }
    }

    public Item findOne(Long id) {
        return em.find(Item.class, id);
    }

    public List<Item> findAll() {
        return em.createQuery("select i from Item i", Item.class)
                .getResultList();
    }

}

 


 

상품 서비스 개발

package bookbook.shopshop.service;

import bookbook.shopshop.domain.item.Item;
import bookbook.shopshop.repository.ItemRepository;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import java.util.List;

@Service
@Transactional(readOnly = true)
@RequiredArgsConstructor
public class ItemService {

    private final ItemRepository itemRepository;

    @Transactional
    public void saveItem(Item item) {
        itemRepository.save(item);
    }

    public List<Item> findItems() {
        return itemRepository.findAll();
    }

    public Item findOne(Long itemId) {
        return itemRepository.findOne(itemId);
    }
}

비즈니스 로직은 보통 여러 리포지토리를 호출하기 때문에 서비스 클래스에서 @Transactional 애노테이션을 적용시키는 것이 좋다.
만약에 비즈니스 로직에 문제가 발생할 경우에는 해당 비즈니스 로직과 관련된 부분을 모두 롤백해야한다. 그래서 일반적으로 비즈니스 로직의 시작점인 서비스 클래스에서 트랜잭션을 사용한다.

 

 

 

 


👀 참고 자료

https://www.inflearn.com/course/%EC%8A%A4%ED%94%84%EB%A7%81%EB%B6%80%ED%8A%B8-JPA-%ED%99%9C%EC%9A%A9-1/community

 

실전! 스프링 부트와 JPA 활용1 - 웹 애플리케이션 개발 질문 & 답변 - 인프런 | 강의

수강생이 남긴 질문과 지식공유자의 답변을 확인할 수 있어요. 질문 & 답변 | 인프런...

www.inflearn.com

 

728x90