1.실행단위 : cpu core에서 실행하는 하나의 단위로 프로세스와 스레드를 포괄하는개념

2.(부연설명이 없는)프로세스 - 하나의 스레드만 가지고있는 단일 스레드 프로세스

3.동시성 - 한순간에 여러가지 일이 아니라, 짧은 전환으로 여러가지 일을 동시에처리하는 것처럼 보이는것
(여러가지가 빠르게 전환되면서 동시에 일어난 것처럼보인다)

프로세스랑 쓰레드가 완전히 다른 무언가가 아니다.

 

목차 
1. process & thread
2.multi-process vs multi-thread
3.multi-core
4.요약


1 . 프로그램과 프로세스


피자레시피 : 내가 열심히 구현하고있는 코드파일(프로그램)
하지만 이 프로그램자체는 실행시키기전에는 그저 코드가 구현되어있는 파일일뿐이다.
이것을 사용하기 위해서는 종이 레시피가 피자가 되는것처럼 

실행이 되어서 사용 할 수있는 무언가가되어야하는데
그것이 바로 프로세스다.

*프로그램이 프로세스가 되면서 어떤일이 일어날까
일단 프로세스가 필요로 하는 재료들이 메모리에 올라가야 한다.

 

PCB 
code(실행명령을 포함하는 코드들) 

data(static변수 혹은 global변수)

heap(동적메모리 영역)

stack(지역변수,매개변수,반환값 등 일시적인데이터)


해당 프로세스에 대한 정보를 담고있는 PCB블럭이 프로세스 생성시 함께만들어진다.

 

두번째로 해당 프로세스에 대한 정보를 담고 있는 PCB블럭이 프로세스 생성시 함께 만들어 진다.

프로세스 상태준비, 대기상태의 큐를 구현하기 위해 필요한 포인터, 현재 프로세스의 상태를 담는 process state, 고유번호를 담는 PID, 다음 명령어를 가리키는 프로그램 카운터 등등이 있다.

 


* 코딩을 할때 예시를 들자면

  • 유투브에서 팝송 플레이리스트를 튼다
  • 코딩을 하기위해 인텔리제이를 킨다
  • 슬렉을킨다
  • 카톡, 크롬을킨다
    대부분의 사람들은 하나의 프로세스만 사용하기보다는 여러가지를 동시에 사용하고싶어한다.
    하지만 원래 한 프로세스가 실행되기 위해서 cpu를 점유하고있으면
    다른프로세스는 실행상태에 있을 수가없다.
    노래를 듣다가 코딩을 하기위해서 인텔리제이를 키면 노래가 꺼지게된다
    그래서 다수의 프로세스를 동시에 실행하기위해 여러개 프로세스를 시분할로, 즉 짧은텀을 반복하면서
    전환해서 실행을 시킨다.

컨텍스트 스위칭

동시에 실행하고 싶은 프로세스 두개가 있다고 가정하자.

먼저 프로세스 1이 실행상태에 있고 CPU에 적재되어있다.

프로세스 2는 준비상태에있다.

프로세스 2를 실행 하기 위해서는 프로세스 1이 먼저 준비상태로 내려가고 

프로세스 2가 cpu에 적재가 되고

다시 1로 전환하기 위해서는 2가 준비상태로 내려가고

1이 cpu에 올라와야한다. 이게 바로 컨테스트 스위칭이다.

 

 

*경량화된 프로세스버전인 스레드 등장

하나의 프로세스 안에 다수의 쓰레드가 있을때 공유되는 자원이 있기때문

스레드는 코드,데이터,힙영역을 공통된 자원으로 사용한다.

각 스레드는 스택부분만을 따로 가지고있는것

공유되는 자원이 있기 때문에 효율적이다.

컨텍스트 스위칭이 일어날때 캐싱 적중률이 올라간다

->모두 다 빼고 다시 다 넣을 필요가 없다.

 

 

*예를 들어 '회의실'을 예약을 하고 회의실을 사용한다고 가정하자.

개인 노트북을 공용 모니터 연결하고 스피커, 리모컨을 사용한다.

다음 예약한 팀이 들어올때 이전의 컨텍스트 스위칭은 또 사용할 자원인 TV와 스피커 리모콘을 모조리 챙겨서 나가는것

다음 팀은 그것을 다시 챙겨서 들어와야만 사용 할 수있다.

->스레드의 컨텍스트 스위칭은 공용으로 사용 할 것들은 두고 개인 노트북만 가지고 와서 연결을 하면된다.

간단하고 부담이 적다.

 

 

2 . 멀티프로세스 vs 멀티스레드 

두가지 모두 처리 방식의 일종

한 어플리케이션에 대한 처리방식

-> 한 어플리케이션에 대해서 두가지의 다른 처리방식

 

 

Multi-process Multi-thread
각 프로세스는 독립적
IPC를 사용한 통신
자원 소모적, 개별 메모리 차지
context switching 비용이 큼
동기화 작업이 필요하지 않음
ex. chrome
Thread끼리 긴밀하게 연결되어 있음
공유된 자원으로 통신비용 절감
context switching 비용이 적음
공유 자원 관리를 해야함
긴밀하게 연결되어있어 한 스레드게 문제가 생기면 전체프로세스에 영향이간다
ex. web server

 

 

3.  멀티코어 란?

하드웨어 측면에 가깝다.

 (멀티 프로세스랑 멀티스레드는 처리방식의 일종-소프트웨어 분야에 가까움)

  1. 동시성 : 여러 실행단위를 번갈아 실행하면서 동시에 일어난 것처럼 보이게한다. 하나의 코어에서 하나 이상의 프로세스(혹은 스레드)가 번갈아가면서 진행되지만 동시에 진행되는 것 처럼 보이는것 
  2. 병렬처리 :물리적으로 여러 코어를 사용해서 다수의 실행 단위를 한순간에 처리할 수있게 해준다. 둘 이상의 코어에서 동시에 하나 이상의 프로세스가 한꺼번에 진행되는것

 

* 리눅스 커널에서는 프로세스와 스레드를 동일하게 봅니다.

스레드는 사용자 스레드와 커널스레드로 나눈다.

 

 

4 . 요약

  1. 프로세스는 프로그램이 실행된 것이다.
  2. 스레드는 한 프로세스 내에서 나뉘어진 하나 이상의 실행 단위이다.
  3. 한 어플리케이션에 대한 작업을 동시에 하기 위해서는 2가지 처리방식(멀티 프로세스, 멀티스레드)이 있다.
  4. 동시에 실행이 되는 것처럼 보이기 위해서 실행단위는 시분할로 cpu를 점유하며 context switching을 한다.
  5. 멀티 프로세스는 독립적인 메모리를 가지고 있지만 멀티 스레드는 자원을 공유한다. 그것에 따른 각각의 장단점이 있다.
  6. 멀티코어는 하드웨어 측면에서 실행 단위를 병렬적으로 처리 할 수 있도록 여러 프로세서가 있는것이다

'* > What I did today' 카테고리의 다른 글

8/31  (0) 2021.08.31
8/29  (0) 2021.08.29
8/28  (0) 2021.08.29
JOIN  (0) 2021.08.21
dto와 entity를 구분해서 구현한 이유  (0) 2021.08.20
Web Server VS WAS  (0) 2021.08.16
스프링 핵심 원리 이해1 - 예제 만들기  (0) 2021.08.06
캡슐화/추상화/다형성  (0) 2021.08.03

String , StringBuffer,StringBuilder 차이점

1. String 
새로운 값을 할당할때 마다 새로 클래스에 대한 객체가 생성된다.
String에서 저장되는 문자열은 private final char[] 의 형태이기 때문에 String값은 변경 할 수없다.
String + String + String은 각각의 String주소값이 stack에 쌓이고 GC가 호출되기 전까지 생성된 String객체들은 Heap에 쌓이기 떄문에 메모리관리에 치명적이다.

 

인스턴스안의 저장된 문자열을 바꿀수없다

한번 생성되면 문자열을 바꿀 수 없다.
참조변수는 참조하는게 다임

str1이 str2를 참조해도 문제가안생긴다. 둘다 참조만가능하므로 ..

백개의 참조변수가생겨도 문제가 안생긴다.
새 인스턴스가 아니라 기존 값을 반환해준다.

새로운 인스턴스가 필요하다면 new로 새 객체생성가능

2. StringBuffer
동기화지원
각 메서드 별로 synchronized keyword가 존재한다.

3. StringBuilder
동기화 미지원
동기화를 지원하지 않아 속도가 빠르다.

'2021 Newlecture > JAVA' 카테고리의 다른 글

StringBuffer클래스  (0) 2021.09.19
클래스변수  (0) 2021.09.18
Wrapper class  (0) 2021.08.22
자바 컬렉션과 제네릭 강의  (0) 2021.04.25
자바 총 정리  (1) 2021.04.13
Inner class / Anonymous class  (0) 2021.04.12
인터페이스 구현  (0) 2021.04.12
(추가정리) 추상클래스 와 인터페이스  (0) 2021.04.11

dto 객체는 View layer와 데이터를 주고받을 때 사용된다.

entity객체는 db layer와 데이터를 주고받을 때 사용된다. 

이렇게 구분한 이유는 아래 글의 내용을 반영하고자 했기 때문이다.

절대로 테이블과 매핑되는 
Entity 클래스를 Request/ Response 클래스로 사용해서는 안됩니다

Entity 클래스는 가장 Core한 클래스라고 보시면 되는데요. 
수많은 서비스 클래스나 비지니스 로직들이 Entity 클래스를 기준으로 동작합니다. 
Entity 클래스가 변경되면 여러 클래스에 영향
을 끼치게 되는 반면 Request와 Response용 DTO는 View를 위한 클래스라 정말 자주 변경이 필요합니다. 
View Layer와 DB Layer를 철저하게 역할 분리를 하는게 좋습니다.
실제로 Controller에서 결과값으로 여러 테이블을 조인해서 줘야할 경우가 빈번하기 때문에 Entity 클래스만으로 표현하기가 어려운 경우가 많습니다. 
꼭꼭 
Entity 클래스와 Controller에서 쓸 DTO는 분리
해서 사용하시길 바랍니다.

entity 클래스에는 setter를 최소한으로 사용한다.

명확한 의미를 가진 함수를 정의할 수 있다면 setter를 정의하지 않고 함수를 정의해 값을 세팅한다.

 

Entity 클래스를 생성하실때, 주의하실것은 
무분별한 setter 메소드 생성
입니다. 
자바빈 규약을 생각하시면서 getter/setter를 무작정 생성하시는 분들이 계시는데요. 
이렇게 되면 해당 클래스의 인스턴스 값들이 
언제 어디서 변해야하는지 코드상으로 명확히 구분할수가 없어, 차후 기능변경시 정말 복잡
해집니다. 
해당 필드의 값 변경이 필요하면 
명확히 그 목적과 의도를 나타낼 수 있는 메소드
를 추가하셔야만 합니다.

'* > What I did today' 카테고리의 다른 글

8/29  (0) 2021.08.29
8/28  (0) 2021.08.29
JOIN  (0) 2021.08.21
Process vs Thread  (0) 2021.08.20
Web Server VS WAS  (0) 2021.08.16
스프링 핵심 원리 이해1 - 예제 만들기  (0) 2021.08.06
캡슐화/추상화/다형성  (0) 2021.08.03
DNS  (0) 2021.05.02

Web Server VS WAS

 

Web Server란 

웹 브라우저(클라이언트)로 부터 HTTP요청을 받아 HTML 문서와 같은 정적 컨텐츠를 제공하는 프로그램

 

정적컨텐츠란 ? 

요청 인자 값에 상관없이 달라지지않는 컨텐츠 (html, css, image, ...)

어느 사용자 요청이든 항상 동일한 컨텐츠

 

Web Server의 기능

클라이언트로 부터 HTTP요청을 받을 수 있다.

1.정적 컨텐츠 요청시

정적 컨텐츠를 제공할 수있다

2. 동적 컨텐츠 요청시 

Web Application Server(WAS)로 전달하여 WAS가 처리한 결과를 클라이언트에게 전달

 

Web Application Server(WAS)란 

DB조회나 다양한 로직 처리를 요구하는 동적인 컨텐츠를 제공하기 위해 만들어진 프로그램

 

동적 컨텐츠 

요청 인자에 따라 바뀔 수 있는 컨텐츠

 

WAS의 기능

클라이언트로 부터 HTTP요청을 받을 수있다. (대부분의 WAS는 Web Server내장)

요청에 맞는 정적 컨텐츠를 제공할 수있다.

DB조회나 다양한 로직처리를 통해 동적 컨텐츠를 제공 할 수있다.

 

Q. WAS가 다 해 줄 수있는데 웹서버가 왜 필요할까 ? (웹서버를 같이 사용했을때의 장점)

책임분할을 통한 서버 부하 방지 

정적콘텐츠는 web server , 동적 콘텐츠는 WAS가 담당

  • 여러대의 WAS 로드밸런싱
  • 웹서버는 로드밸런싱 기능을 가지고있는데 앞단에 웹서버를 두고 뒷단에 여러 대의 WAS를 둬서 클라이언트 요청을 웹서버에서 열어보고 나누어서 여러 WAS에서 처리할 수 있도록 설정할 수있다.
  • Health Check : 서버에 주기적으로 HTTP 요청을 보내 서버의 상태를 확인 (ex 특정 url요청에 200 응답이 오는지)
  • 보안 : 리버스 프록시를 통해 실제 서버를 외부에 노출하지 않을 수 있다.

 

WAS만으로도 서비스는 가능하지만 서비스 확장성, 안정성을 고려한다면 앞단에 WEb server를 두는 것이 유리하다.

 

'* > What I did today' 카테고리의 다른 글

8/28  (0) 2021.08.29
JOIN  (0) 2021.08.21
Process vs Thread  (0) 2021.08.20
dto와 entity를 구분해서 구현한 이유  (0) 2021.08.20
스프링 핵심 원리 이해1 - 예제 만들기  (0) 2021.08.06
캡슐화/추상화/다형성  (0) 2021.08.03
DNS  (0) 2021.05.02
비전공자를 위한 이해할 수 있는 IT지식 (정리)  (0) 2021.04.13

프로젝트 생성

스프링 부트 스타터 사이트로 이동해서 스프링 프로젝트 생성

https://start.spring.io

프로젝트 선택

  • Project: Gradle Project
    • Spring Boot: 2.3.x
    • Language: Java
    • Packaging: Jar
    • Java: 11
  • Project Metadata
    • groupId: hello
    • artifactId: core
  • Dependencies: 선택하지 않는다.

Gradle 전체 설정

plugins {
	id 'org.springframework.boot' version '2.3.8.RELEASE'
	id 'io.spring.dependency-management' version '1.0.11.RELEASE'
	id 'java'
}

group = 'hello'
version = '0.0.1-SNAPSHOT'
sourceCompatibility = '11'

repositories {
	mavenCentral()
}

dependencies {
	implementation 'org.springframework.boot:spring-boot-starter'
	testImplementation('org.springframework.boot:spring-boot-starter-test') {
		exclude group: 'org.junit.vintage', module: 'junit-vintage-engine'
	}
}

test {
	useJUnitPlatform()
}

 

동작 확인

기본 메인 클래스 실행( CoreApplication.main() )

IntelliJ Gradle 대신에 자바 직접 실행

최근 IntelliJ 버전은 Gradle을 통해서 실행 하는 것이 기본 설정이다. 이렇게 하면 실행속도가 느리다.

다음과 같이 변경하면 자바로 바로 실행해서 실행속도가 더 빠르다.

Preferences Build, Execution, Deployment Build Tools Gradle
Build and run using: Gradle IntelliJ IDEA
Run tests using: Gradle IntelliJ IDEA

비즈니스 요구사항과 설계

  • 회원
    • 회원을 가입하고 조회할 수 있다.
    • 회원은 일반과 VIP 두 가지 등급이 있다.
    • 회원 데이터는 자체 DB를 구축할 수 있고, 외부 시스템과 연동할 수 있다. (미확정)
  • 주문과 할인 정책
    - 회원은 상품을 주문할 수 있다.
    - 회원 등급에 따라 할인 정책을 적용할 수 있다.
    - 할인 정책은 모든 VIP는 1000원을 할인해주는 고정 금액 할인을 적용해달라. (나중에 변경 될 수 있
    다.)
    - 할인 정책은 변경 가능성이 높다. 회사의 기본 할인 정책을 아직 정하지 못했고, 오픈 직전까지 고민을
    미루고 싶다. 최악의 경우 할인을 적용하지 않을 수 도 있다. (미확정)

요구사항을 보면 회원 데이터, 할인 정책 같은 부분은 지금 결정하기 어려운 부분이다. 그렇다고 이런 정책이 결정될 때 까지 개발을 무기한 기다릴 수 도 없다. 우리는 앞에서 배운 객체 지향 설계 방법이 있지 않은가! 인터페이스를 만들고 구현체를 언제든지 갈아끼울 수 있도록 설계하면 된다. 그럼 시작해보자.

참고: 프로젝트 환경설정을 편리하게 하려고 스프링 부트를 사용한 것이다. 지금은 스프링 없는 순수한 자바로만 개발을 진행한다는 점을 꼭 기억하자! 스프링 관련은 한참 뒤에 등장한다.

회원 도메인 설계

  • 회원 도메인 요구사항
    • 회원을 가입하고 조회할 수 있다.
    • 회원은 일반과 VIP 두 가지 등급이 있다.
    • 회원 데이터는 자체 DB를 구축할 수 있고, 외부 시스템과 연동할 수 있다. (미확정)

회원 도메인 협력 관계

회원 클래스 다이어그램

회원 객체 다이어그램


회원 서비스: MemberServiceImpl

회원 도메인 개발

회원 엔티티

회원 등급

package hello.core.member;

public enum Grade {
    BASIC,
    VIP
}

회원 엔티티

package hello.core.member;

public class Member {

    private Long id;
    private String name;
    private Grade grade;

    public Member(Long id, String name, Grade grade) {
        this.id = id;
        this.name = name;
        this.grade = grade;
    }

    public Long getId() {
        return id;
    }

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

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Grade getGrade() {
        return grade;
    }

    public void setGrade(Grade grade) {
        this.grade = grade;
    }
}

회원 저장소

회원 저장소 인터페이스

package hello.core.member;

public interface MemberRepository {

    void save(Member member);

    Member findById(Long memberId);
}

메모리 회원 저장소 구현체

package hello.core.member;

import java.util.HashMap;
import java.util.Map;

public class MemoryMemberRepository implements MemberRepository{

    private static Map<Long, Member> store = new HashMap<>();

    @Override
    public void save(Member member) {
        store.put(member.getId(), member);
    }

    @Override
    public Member findById(Long memberId) {
        return store.get(memberId);
    }
}

데이터베이스가 아직 확정이 안되었다. 그래도 개발은 진행해야 하니 가장 단순한, 메모리 회원 저장소를 구
현해서 우선 개발을 진행하자.

참고: HashMap 은 동시성 이슈가 발생할 수 있다. 이런 경우 ConcurrentHashMap 을 사용하자.

회원 서비스

회원 서비스 인터페이스

package hello.core.member;

public interface MemberService {

    void join(Member member);

    Member findMember(Long memberId);
}

회원 서비스 구현체

package hello.core.member;

public class MemberServiceImpl implements  MemberService{

    private final MemberRepository memberRepository = new MemoryMemberRepository();

    @Override
    public void join(Member member) {
        memberRepository.save(member);
    }

    @Override
    public Member findMember(Long memberId) {
        return memberRepository.findById(memberId);
    }
}

회원 도메인 실행과 테스트

회원 도메인 - 회원 가입 main

package hello.core;

import hello.core.member.Grade;
import hello.core.member.Member;
import hello.core.member.MemberService;
import hello.core.member.MemberServiceImpl;

public class MemberApp {

    public static void main(String[] args) {
        MemberService memberService = new MemberServiceImpl();
        Member member = new Member(1L, "memberA", Grade.VIP);
        memberService.join(member);

        Member findMember = memberService.findMember(1L);
        System.out.println("new Member = " + member.getName());
        System.out.println("findMember = " + findMember);
    }
}

 

애플리케이션 로직으로 이렇게 테스트 하는 것은 좋은 방법이 아니다. JUnit 테스트를 사용하자.

회원 도메인 - 회원 가입 테스트

package hello.core.member;


import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.Test;

public class MemberServiceTest {

    MemberService memberService = new MemberServiceImpl();

    @Test
    void join() {
        //given
        Member member = new Member(1L, "memberA", Grade.VIP);

        //when
        memberService.join(member);
        Member findMember = memberService.findMember(1L);

        //then
        Assertions.assertThat(member).isEqualTo(findMember);
    }
}

 

회원 도메인 설계의 문제점

  • 이 코드의 설계상 문제점은 무엇일까요?
  • 다른 저장소로 변경할 때 OCP 원칙을 잘 준수할까요?
  • DIP를 잘 지키고 있을까요?
  • 의존관계가 인터페이스 뿐만 아니라 구현까지 모두 의존하는 문제점이 있음
    • 주문까지 만들고나서 문제점과 해결 방안을 설명

주문과 할인 도메인 설계

주문과 할인 정책

  • 회원은 상품을 주문할 수 있다.
  • 회원 등급에 따라 할인 정책을 적용할 수 있다.
  • 할인 정책은 모든 VIP는 1000원을 할인해주는 고정 금액 할인을 적용해달라. (나중에 변경 될 수 있다.)
  • 할인 정책은 변경 가능성이 높다. 회사의 기본 할인 정책을 아직 정하지 못했고, 오픈 직전까지 고민을 미루고 싶다. 최악의 경우 할인을 적용하지 않을 수 도 있다. (미확정)

주문 도메인 협력, 역할, 책임

  1. 주문 생성: 클라이언트는 주문 서비스에 주문 생성을 요청한다. (main 함수나 mvc에서 controller에 해당한다)
  2. 회원 조회: 할인을 위해서는 회원 등급이 필요하다. 그래서 주문 서비스는 회원 저장소에서 회원을 조회한다.
  3. 할인 적용: 주문 서비스는 회원 등급에 따른 할인 여부를 할인 정책에 위임한다.
  4. 주문 결과 반환: 주문 서비스는 할인 결과를 포함한 주문 결과를 반환한다.

참고: 실제로는 주문 데이터를 DB에 저장하겠지만, 예제가 너무 복잡해 질 수 있어서 생략하고, 단순히 주문 결과를 반환한다.

주문 도메인 전체

역할과 구현을 분리해서 자유롭게 구현 객체를 조립할 수 있게 설계했다. 덕분에 회원 저장소는 물론이고, 할인 정책도 유연하게 변경할 수 있다.

주문 도메인 클래스 다이어그램

주문 도메인 객체 다이어그램1

회원을 메모리에서 조회하고, 정액 할인 정책(고정 금액)을 지원해도 주문 서비스를 변경하지 않아도 된다. 역할들의 협력 관계를 그대로 재사용 할 수 있다.

주문 도메인 객체 다이어그램2

회원을 메모리가 아닌 실제 DB에서 조회하고, 정률 할인 정책(주문 금액에 따라 % 할인)을 지원해도 주문 서비스를 변경하지 않아도 된다.
협력 관계를 그대로 재사용 할 수 있다.

주문과 할인 도메인 개발

할인 정책 인터페이스

package hello.core.discount;

import hello.core.member.Member;

public interface DiscountPolicy {

    /*
    @return 할인 대상 금액액
    */
    int discount(Member member, int price);
}

정액 할인 정책 구현체

package hello.core.discount;

import hello.core.member.Grade;
import hello.core.member.Member;

public class FixDiscountPolicy implements DiscountPolicy {

    private int discountFixAmount = 1000; //1000원 할인

    @Override
    public int discount(Member member, int price) {
        if (member.getGrade() == Grade.VIP) {
            return discountFixAmount;
        } else {
            return 0;
        }
    }
}

VIP면 1000원 할인, 아니면 할인 없음

주문 엔티티

package hello.core.order;

public class Order {

    private Long membertId;
    private String itemName;
    private int itemPrice;
    private int discountPrice;

    public Order(Long membertId, String itemName, int itemPrice, int discountPrice) {
        this.membertId = membertId;
        this.itemName = itemName;
        this.itemPrice = itemPrice;
        this.discountPrice = discountPrice;
    }

    public int calculatePrice() {
        return itemPrice - discountPrice;
    }

    public Long getMembertId() {
        return membertId;
    }

    public void setMembertId(Long membertId) {
        this.membertId = membertId;
    }

    public String getItemName() {
        return itemName;
    }

    public void setItemName(String itemName) {
        this.itemName = itemName;
    }

    public int getItemPrice() {
        return itemPrice;
    }

    public void setItemPrice(int itemPrice) {
        this.itemPrice = itemPrice;
    }

    public int getDiscountPrice() {
        return discountPrice;
    }

    public void setDiscountPrice(int discountPrice) {
        this.discountPrice = discountPrice;
    }

    @Override
    public String toString() {
        return "Order{" +
            "membertId=" + membertId +
            ", itemName='" + itemName + '\'' +
            ", itemPrice=" + itemPrice +
            ", discountPrice=" + discountPrice +
            '}';
    }
}

주문 서비스 인터페이스

package hello.core.order;

public interface OrderService {
    Order createOrder(Long memberId, String itemName, int itemPrice);
}

주문 서비스 구현체

package hello.core.order;

import hello.core.discount.DiscountPolicy;
import hello.core.discount.FixDiscountPolicy;
import hello.core.member.Member;
import hello.core.member.MemberRepository;
import hello.core.member.MemoryMemberRepository;

public class OrderServiceImpl implements OrderService{

    private final MemberRepository memberRepository = new MemoryMemberRepository();
    private final DiscountPolicy discountPolicy = new FixDiscountPolicy();

    @Override
    public Order createOrder(Long memberId, String itemName, int itemPrice) {
        Member member = memberRepository.findById(memberId);
        int discountPrice = discountPolicy.discount(member, itemPrice);

        return new Order(memberId, itemName, itemPrice, discountPrice);
    }
}

주문 생성 요청이 오면, 회원 정보를 조회하고, 할인 정책을 적용한 다음 주문 객체를 생성해서 반환한다. 메모리 회원 리포지토리와, 고정 금액 할인 정책을 구현체로 생성한다.

주문과 할인 도메인 실행과 테스트

주문과 할인 정책 실행

package hello.core;

import hello.core.member.Grade;
import hello.core.member.Member;
import hello.core.member.MemberService;
import hello.core.member.MemberServiceImpl;
import hello.core.order.Order;
import hello.core.order.OrderService;
import hello.core.order.OrderServiceImpl;

public class OrderApp {

    public static void main(String[] args) {
        MemberService memberService = new MemberServiceImpl();
        OrderService orderService = new OrderServiceImpl();

        Long memberId = 1L;
        Member member = new Member(memberId, "memberA", Grade.VIP);
        memberService.join(member);

        Order order = orderService.createOrder(memberId, "itemA", 10000);

        System.out.println("order = " + order);
    }
}

결과

order = Order{memberId=1, itemName='itemA', itemPrice=10000,
discountPrice=1000}

할인 금액이 잘 출력되는 것을 확인할 수 있다.
애플리케이션 로직으로 이렇게 테스트 하는 것은 좋은 방법이 아니다. JUnit 테스트를 사용하자.

주문과 할인 정책 테스트

package hello.core.order;

import hello.core.member.Grade;
import hello.core.member.Member;
import hello.core.member.MemberService;
import hello.core.member.MemberServiceImpl;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.Test;

public class OrderServiceTest {

    MemberService memberService = new MemberServiceImpl();
    OrderService orderService = new OrderServiceImpl();

    @Test
    void createOrder() {
        Long memberId = 1L;
        Member member = new Member(memberId, "memberA", Grade.VIP);
        memberService.join(member);

        Order order = orderService.createOrder(memberId, "iteamA", 10000);
        Assertions.assertThat(order.getDiscountPrice()).isEqualTo(1000);
    }
}

 

'* > What I did today' 카테고리의 다른 글

JOIN  (0) 2021.08.21
Process vs Thread  (0) 2021.08.20
dto와 entity를 구분해서 구현한 이유  (0) 2021.08.20
Web Server VS WAS  (0) 2021.08.16
캡슐화/추상화/다형성  (0) 2021.08.03
DNS  (0) 2021.05.02
비전공자를 위한 이해할 수 있는 IT지식 (정리)  (0) 2021.04.13
네트워크 개요  (0) 2021.04.11

The CodeDeploy agent did not find an AppSpec file within the unpacked revision directory at revision-relative path "appspec.yml". The revision was unpacked to directory "/opt/codedeploy-agent/deployment-root/7a4336d8-06e1-4025-abe0-192e0bffa4e1/d-L0T638J4B/deployment-archive", and the AppSpec file was expected but not found at path "/opt/codedeploy-agent/deployment-root/7a4336d8-06e1-4025-abe0-192e0bffa4e1/d-L0T638J4B/deployment-archive/appspec.yml". Consult the AWS CodeDeploy Appspec documentation for more information at http://docs.aws.amazon.com/codedeploy/latest/userguide/reference-appspec-file.html

 

디렉토리문제

프로젝트폴더 안에 pom이없고  프로젝트폴더 - 프로젝트폴더2 - pom.xml으로 

경로맵핑때문에 오류가났었다.

 

디렉토리구조를 프로젝트폴더 - pom.xml로 바꾸니까 해결 !

The overall deployment failed because too many individual instances failed deployment, too few healthy instances are available for deployment, or some instances in your deployment group are experiencing problems.

 

 

캡슐화

비슷한 역할을 하는 속성과 메소드들을 하나의 클래스로 모은것을 캡슐화 라고 한다. 캡슐화에 속한 개념으로 정보 은닉이라는것이 있는데, 캡슐 내부의 로직이나 변수들을 감추고 외부에는 기능(api)만을 제공하는것을 의미한다.

 

상속

상속이란 클래스를 재사용 하는것이다. 상위 클래스를 하위 클래스에서 상속 받게 되면 상위 클래스의 멤버변수나 메소드를 그대로 물려 받을 수 있다. 상속이 있기 때문에 코드를 재활용할 수 있고 그렇기 때문에 생산성이 높고 유지보수 하기가 좋다. 

 

추상화

추상화라는것은, 어떤 실체로부터 공통적인 부분이나 관심 있는 특성들만 한곳에 모은것을 의미한다. 예를들어서, 지구를 본따 만든 지구본을 예로 들 수 있다. 지구본은 실제 지구로 부터 관심 있는 특성들(대륙의 위치, 위도,경도)만 뽑아서 만든것이다. 지구를 추상화해서 지구본을 만들었다.

객체지향에서의 추상화는 어떤 하위클래스들에 존재하는 공통적인 메소드를 인터페이스로 정의하는것을 예로 들 수 있다.

 

다형성

다형성은, 같은 모양의 함수가 상황에 따라 다르게 동작 하는것을 의미한다. 오버로딩과 오버라이딩이 있는데, 오버로딩이란것은 함수의 이름은 같으나 함수의 매개변수 숫자, 타입등을 달리해서 다르게 사용하는것을 의미하고, 오버라이딩은 상위 클래스의 메소드를 하위 클래스에서 똑같은 이름으로 재정의 하는것을 의미한다.(덮어씌우기) 이렇게 되면, c++의 경우에는 상위 클래스 타입 변수에 하위 클래스를 담은 상태에서 메소드를 호출하면 상위 클래스의 메소드가 호출되고, 하위 클래스 타입 변수에 하위 클래스를 담으면 하위 클래스의 메소드가 호출된다. 즉, 메소드의 이름은 똑같은데, 상황(상위 클래스의 참조 변수냐 하위 클래스의 참조 변수냐)에 따라 호출 되는 메소드가 다른것이다.

'* > What I did today' 카테고리의 다른 글

JOIN  (0) 2021.08.21
Process vs Thread  (0) 2021.08.20
dto와 entity를 구분해서 구현한 이유  (0) 2021.08.20
Web Server VS WAS  (0) 2021.08.16
스프링 핵심 원리 이해1 - 예제 만들기  (0) 2021.08.06
DNS  (0) 2021.05.02
비전공자를 위한 이해할 수 있는 IT지식 (정리)  (0) 2021.04.13
네트워크 개요  (0) 2021.04.11

웹호스팅 링크 정리

  1. AWS계정 생성 및 인스턴스 생성 https://jiwontip.tistory.com/43?category=367314
  2. EC2 인스턴스에 java 설치 https://yhmane.tistory.com/23?category=773048
  3. openjdk 환경변수 https://eodevelop.tistory.com/42
  4. jar파일 생성 https://jhleeeme.github.io/create-jar-with-maven-in-vscode/
  5. maven 환경변수 설정 https://devpad.tistory.com/19
  6. ubuntu 파일질라 연결 https://onlyfor-me-blog.tistory.com/121
  7. ubuntu에서 jar파일 백그라운드로 실행하기 https://kyome.tistory.com/44
  8. S3버킷 https://velog.io/@reedfoxy/AWS-S3-SDK-2.0-for-Java
  9. S3 업로드 https://leveloper.tistory.com/46
  10. https://victorydntmd.tistory.com/334

 

 

 

분명 간단하다고 들었는데 끝이 없다.. 

로컬서버에서 연동되던 것들이 웹서버에서는 오류가 발생했다.

viewResolver가 prefix와 suffix를 결합해주는데 이때 중간에 절대경로로 넣은 '/'가 두번들어가게되서 오류나는거라고 천재 고라니가 말해줬다..

배포 완료 된 줄 알고 자려고 누웠을때 폰으로 웹서버에 접속했더니 푸터연결이 끊겨서 접속이 안되는것..백그라운설정 후 실시간 접속이 가능해졌다.

이제는 이미지 저장소 연결 문제가 남았다. 사용자가 업로드한 이미지를 저장할 서버를 따로 만들어야한다. 보통 EC2와 S3를 같이 쓴다고해서 버킷만들고 서비스,컨트롤러 만들었는데 테스트페이지에서 에러가 난다.

디펜던시오류인가..프라퍼티즈설정 오류인가... 자바코드 오류인가..............ㅠㅠㅠㅠ

멀티파일을 파일로 변환하지못하고있고,, 아마존클라이언트도 인식못하고 있는것같다. 

SDK도 다시찾아봐야겠다.

 

https://about.weverse.io/ko.html

https://www.youtube.com/watch?v=6P6HoumuTOw&list=PLuHgQVnccGMCFHj64mNZxlbeNWOYUpua4

 

원래있던거 삭제하고 새로운거 넣기

root@ip-172-31-5-40:/var/lib/tomcat9/webapps# ls -l


root@ip-172-31-5-40:/var/lib/tomcat9/webapps# rm bootPrj-1.0.jar (원래있던자르파일없애기)
root@ip-172-31-5-40:/var/lib/tomcat9/webapps# ls -l


root@ip-172-31-5-40:/home/ubuntu# mv bootPrj-1.0.jar /var/lib/tomcat9/webapps

 


1 .자르파일 만들기

2 . home/ubuntu에 옮겨주기(파일질라에서 더블클릭)

3. 홈 우분투에있는거를 웹앱스로 옮기기 (이때 기존것 삭제하고 옮기기)

webapps 

 

java -jar 파일명.jar 자르파일이있는데서 자르실행

 

 

 

Pull Request

Pull Request 란 ?

  • 코드리뷰를 위해 / Push 권한이 없는 오픈 프로젝트에 기여할때
  • 레파지토리의 한 브랜치에 푸시한 변화를 다른사람에게 알려준다.
  • pull request가 open되면 기본 브랜치에 머지하기전에 콜라보레이터와 코드변화를 리뷰하고 논의 할 수있다.

Pull Request 보내는 방법

1. 기여하려는 저장소 fork 

기여하려는 프로젝트의 상단에서 fork 클릭 !

owner 의 레파지토리를 fork

2. fork한 저장소의 git주소를 clone 

- fork한 자신의 레파지토리 주소를 clone한다.

- clone한 프로젝트가 열린다.

 

3. 원격저장소 설정

Git - manage remotes - '+'버튼

- origin에는 본인의 레파지토리가 들어가있고

- 원격저장소에 owner의 레파지토리를 추가한다.

 

원격저장소 추가

4. Branch 생성 

- Git - branches - new branches 

develop라는 브랜치를 추가한다.

- 브랜치 추가는 필수는 아니라고함..어떻게 관리하느냐에따라 달라지는듯..

 

5. 코드 수정

 

6. commit and push

- commit : git → commit or ctrl + k

- 커밋메세지 적고 커밋

- push : git →  push or ctrl + shft + k 

- 이때 본인의 레파지토리 (origin)에다가 push를 한다.

- owner에 push는 불가능한 상황임..

푸시 완료 

7. github사이트에서 fork한 나의 저장소에 들어가기 

- 방금 push한 알림이 떠있다.

- 또는 pull requests에서도 확인가능 

8. compare & pull request 클릭

- 내가 수정한 코드가 확인이 된다.

- 기여한것들을 작성 후 create pull request를 눌러 Owner의 승인을 기다린다. 

- onwer가 승인하면 그때 merge가 된다. 

 

참조

https://docs.github.com/en/github/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-pull-requests

 

https://www.gitkraken.com/learn/git/tutorials/what-is-a-pull-request-in-git

 

'2021 Newlecture > Git&GitHub' 카테고리의 다른 글

GIT Commit and push  (0) 2021.05.19
  • 학생인증이 되면 intellij Ultimate버전을 1년동안 무료로 사용이 가능
  • 매년 갱신해야함
  • 졸업 후더라도 학교메일계정이 살아있으면 인증가능
  • 학생인증에는 깃헙 및 다른 방법도 있다고함..

1. Jetbrains 접속 → Apply now 

https://www.jetbrains.com/community/education/#students

 

2. 정보입력 - 학교이메일 계정입력

입력을 완료하면 인증을 위한 메일을 발송했다고 메세지가 뜬다.

3. 학교 이메일 확인 → follow this link 클릭

 

4. 최종인증 → I accept 클릭

 

5. 회원가입 → Sign up →  메일주소 기입

 

6. 가입 이메일 확인   Confirm your account 클릭  정보기입

7. 로그인하면 학생 라이센스 확인 가능

유효 기간 및 라이센스번호가 확인된다.

 


IntelliJ IDEA 다운로드 

https://www.jetbrains.com/idea/download/#section=windows

리액트복습

  • 리액트 컴포넌트껴넣는 과정 익숙해지기
  • 스타일이 깨지지않게 주의할것
  • ReactDom이 필요하면 import해야함..
  • 컴포넌트 두가지 : 함수형(가볍게 페이지분리해서 사용) / 클래형 (state를 이용한 데이터 바인딩필요)

원하는 프로젝트폴더에서 react app 설치

 

리액트앱설치한 폴더(package json)이 들어있는 폴더에서 npm스타트

index.js가 App을 import하고있다.

App.js에서 리턴값에 원하는걸 넣어서 화면에 출력한다.


html폴더에서 필요한것만 쓰자

html폴더의 index에서 바디부분만 가져오자.

function App()의 리턴에 넣어준다.

이때 에러가 발생

반드시 전체를 감싸는 태그가 있어야한다.

<div>중첩이 발생

 

Header는 주로 데이터를 바인딩하지않는다

메인쪽이라 함수형으로하자

 

()=>3; // 함수를 정의하는데 3을 리턴하는 함수, 무명

header.js만들고, header부분 리턴하는 함수만들기

class를 className으로 변경


라우팅하기위한 라이브러리 설치

url요청에따라서 스위칭

a 태그대신 Link태그로

/이게 우선순위가 제일위로..처리됨.

"/"를 밑으로둘수도있지만 exact를 넣어주면 딱 정확한 url이된다(패턴x)

'2021 Newlecture > Node & React' 카테고리의 다른 글

React : bind() / Mounting / fetch  (0) 2021.06.23

bind()

값이 변경하면 그것에대한 변수를 자동으로 변경해주지않는다. onChange를 써줘야한다.

함수를 다른함수로 위임할때, 객체없이 호출하는 일이 발생했다.

this를 사용할수없는 인스턴스함수가되었다.

위임했던 메소드는 this를 쓸수없다

해결방법은 ?

 

자료형식으로부터 유연함

자료형식에 구애를 받지않는,

 

함수형태를 담는 형식ㅇ ㅣ정의되어있어야한다 

스크립트언어들은 형식명을 거론하지 x (ex var)

다 담을  수 있다는얘기

 

exam을 넘길수있는대안을 써야한다.

f1한테 위임할때, 바인드함수를 사용할수있다. bind()

bind(exam)을 쓰면 객체가 안에 바인드되서 전달된다.

f1() : 객체없이 함수호출하고있음.

인스턴스메소드를 위임할때 bind()를 쓸것

 

state는 여러개로 들어온다.

나중에뒤에있ㄴ는 state로 덮어쓰여진다.

 

이벤트가 발생하면 정보를 주는것 

이벤트가 발생한 객체를 알려줌

저장될 구조를 미리 정의 (state가있을것이다.)

 

1.객체를 얻어서 값을 얻는 방법

2. 리액트 방식으로 객체를 얻어서 값을 얻는 방법

input의 value값을 얻어온다 value={this.state.x}

3. ..


백엔드로만 할래, 프론트로만 할래

return값을 실제연결할 html이름으로 변경


1.데이터요청도구가 달라졌다

promise을 이용한 비동기처리 (fetch API)

 

mount : 화면에  뜨다

constructor : 생성자가 제일먼저 호출됨(멤버변수만 초기화) : 화면과 상관없이 변수를 마련해서 변수에 초기값설정

화면을 초기화하려면 ? 

render : 화면에 출력할것

componentDidMount : 화면에 로드되자마자 초기화할떄

 

순서 : render 를 하면서 <List>객체가 생성되고 , contructor생성자가 호출되고 render() {}가되고 componentDidMount가 된다

1.constructor, 2.render 3. mount

렌더는 언제호출될까

인풋이 바뀔때마다 랜더가 호출된다. 화면상태가 바뀔때마다 호출


서버쪽에있는 데이터를 가져오자

페치하는것을 어디서 가져오는게 좋을까

1 생성자 2 랜더 3 컴포넌트디드마운트

 

랜더는 문제가 있음 : 데이터 가져오는 작업을 계속 반복한다

 사용자가 인풋에 값을넣을때마다 랜더가 계속 호출된다(콘솔에 랜더)

사용자의 화면상태가 바뀔때마다 랜더호출 , 무한반복이된다.

 

화면이 바뀌었다가 하고싶은 게있어 ? ->componendidupdate

 

순서 : 생성자 - 랜더 - 딛어마운트 - 랜더(다시) - 후처리(딛업데이트)

 

바겼다가 후처리 : didUpdate

랜더-업데이트

데이터를 제공해주는 녀석만들기

api는 하나의 약속이다.

한번 api/notice/3으로 서비스했으면 못바꾼다

경로가 데이터형태.

계층화된 데이터구조

질의x

 

삭제할떄는 요청자체를 delete

list와 count 모두 반환할 수있는것

1.dto

2Map (키:문자열) 

 

@RestController 데이터 자체를 리턴

json형태나옴

fetch절대경로

 

 

목록으로된것을 바인딩

list배열[]이 갖고있는것 {"id":92032943},{"id":92032942}
배열의 기능이용

map : 배열의 항목을 가지고 변환해서 새로운 배열을 만들때 사용

다른모양

제이슨과 list의 구조가같아서 바로넣는데, 다를경우 각각 만들어줘야한다

'2021 Newlecture > Node & React' 카테고리의 다른 글

리액트복습  (0) 2021.06.28

ECMA Script 6 / 2015 강의 08 -object destructuring


ECMA Script 6 / 2015 강의 09 - Object Destructuring #2


ECMA Script 6 / 2015 강의 10 - Object Destructuring #3 중첩과 적응(뽀개기 중첩과 적용)

객체가 가지고 있는 속성의 이름을 전체네임으로 바꾸려면 ?

객체가 중첩된 경우


ECMA Script 6 / 2015 강의 11 - Array Destructuring #1

let을 통해 변수 선언

나중에 temp배열에 담긴 값으로 바꾸려면 ? (다른배열을 이용하고싶을때)

배열뽀개기를 이용한 순서 재배열


ECMA Script 6 / 2015 강의 12 - Array Destructuring #2

기본값 설정하기

배열이 가지고있지않은  속성에  기본값 설정하기

변수선언과 동시에 값대입 kor4=40
,콤마를 써서 두번째배열을 가리키게된다


ECMA Script 6 / 2015 강의 13 - Set 콜렉션

객체지향언어에서 컬렉션이 내장되어있다.

언어가 컬렉션을 지원 (자바,씨샵 등등)

컬렉션 : set, list , map

컬렉션들은 데이터를 수집한다(데이터를 담을 수 있다). 가변길이의 데이터공간

*데이터를 식별하는 키값이 다르다.

map : 데이터를 수집하면서 데이터식별키를 같이 심는다.

        "내가 맡겼던 타이틀줘"

list : 데이터를 담고 , 식별자를 담지않는다. (데이터를 담는 순서가 식별자)

     "내가 맡긴 0번째 줘"

set : 식별자가없다. 데이터가 식별자 (so, 같은값 저장불가능, 중복된 값 허용 x)

     "컬렉션에 5가 담겨있니" -> 값들을 특정 지어서 꺼낼 수 없으니 나열할 수 있는 기능이 필요 ->이터레이터등장

 

*배열의 항목에 중복값이있을때 중복제거가능 ->set이 중복제거해줌

 

*for in : 키 값 얻어줌

*for of : 값을 하나씩 꺼내줌

https://haileykim2014.tistory.com/211

forEach , entries로 키와 값을 출력할수있다.


ECMA Script 6 / 2015 강의 14 - Map 콜렉션

map을 사용할때 exam.set할수도있고 배열형태로 중첩해서 사용가능, 반환해서 쓸때도 배열형태모양으로 가능


ECMA Script 6 / 2015 강의 15 - Rest Parameters

...values 나머지것들에 대해서 컬렉팅(나머지것들에 대해서 인덱스 0부터)

별칭을 지정하지않은것은 arguments로도 가능

 


ECMA Script 6 / 2015 강의 16 - Spread Operator

갖고있는값을 흩뿌려줘

각각 나누어서 전달

단순히 함수에 전달하는 목적x + 값을 나열하기 위해 사용


ECMA Script 6 / 2015 강의 17 - Default Value의 이전방식과 현재방식


ECMA Script 6 / 2015 강의 18 - Arrow Functions와 그 특징

print()안의 this : window객체라서 console.log(kor)하면 10이 출력, 전역변수

Arrow function은 애초부터 함수의 기능만 수행하도록 만들어짐

오로지 리턴만있다면 괄호제외가능

매개변수하나일경우

var fun = x =>x


ECMA Script 6 / 2015 강의 19 - class를 이용한 객체 정의

은닉화 문제발생

console.log(exam.kor) // 10 출력, 속성을 은닉화해야한다.


ECMA Script 6 / 2015 강의 20 - class 멤버 은닉하기

함수도 은닉화가능


ECMA Script 6 / 2015 강의 21 - static 멤버 정의하기

static : 객체마다 동일한 속성이 존재할 경우 사용, 객체와 상관없이 전역변수화

클래스명.스테틱변수명 (울타리내에있다)

객체생성과 상관없이 사용가능

객체를 만들지않고도 사용할 수있는 static메소드

메서드는 객체화되지않는다. 메서드는 데이터가 아니다. 함수는 절차니까 흐름에왔다갔다.

호출될떄마다 만들어질필요 x

false (같은기능의 함수인데, 다르다고나온다)

프로토타입

객체가 다르면 데이터가 달라지지만 함수는 같은함수쓰도록하자.

프로토타입이란 ?

 

ex 배경색바꾸기, 기본도형설정 

new할때마다 aa라는 속성을 가져야할때 

Exam.prototype.aa = 30;

객체가 생성되지않아도 프로그램이 실행이 되는순간 만들어짐.전역변수,

하나만 가지고 공유

private static : 클래스안에서만 사용

false : Exam을 만들때마다 객체생성이되므로 각각 다르다.
true 같은애들

기본도형에 넣으면 하나로 공유.

인스턴스 호출전에 항상 먼저 준비됨.

단일하게 처리

함수는 프로토타입으로 ! 

 


객체지향

-행위 또는 처리하는 함수

-데이터객체를 만들어서 초기화하는 목적의 함수 :  생성자 함수

위 : true / 아래 : false

=== : 참조비교

print()안의 this : 전역객체, window 

Exam()안의 this : new를 통해 호출한 함수. exam을 가리킴

 

this생략불가능. 무조건 써줘야한다.

1.지역변수

2.매개변수

3.인스턴스변수


클래스 특징

new를 통해서만 호출

클래스내 데이터만열거(메소드제외)

strict 모드 : 변수 중복선언 불가능, 변수선언안하고 쓰는것도 오류발생 ->let ,const (es6)

 

캡슐화 : 속성+ 기능

컨트롤러 :

서비스(업무로직)

다오(데이터)

데이터베이스

뷰(JSP)

 

DI작업

트랜잭션 

 

JDBC에다가 추가적으로 마이바티스라는 매퍼작업

백엔드기술로서 이런작업을 처리할때 추가적인라이브러리 : 스프링, mvc라이브러리,

트렌트가 바뀐다.

JSP대신 -> 타임리프

스프링을 이용하다보니 조금더 새로운방식, 새로운표기법

타임리프가 JSP대신해서 많이쓰인다.

 

서버사이드자바 템플릿엔진

웹뿐만 아니라 다른문서를 만드는곳에서도 쓰일 수 있다. 

주된목표는 지금까지 사용ㅎ왔단 템플릿에 영향을 주지않으면서 서버코드를 꽂아넣을수있다.

jsp는 퍼블리셔서 만들어놓은 페이지를 바꿔가면서 데이터를꽂는다. (타이틀을없애도 el태그)

반복할떄도 기존것삭제후 ->코드블럭

서로 협업할떄 영향을준다.

퍼블리셔서 완벽하게 작성후전달해야한다.

 

기존것을 그대로 나둔다. 꽂을텍스트에대해서 추가한다

타임리프의 장점은 6가지 종류의 템플릿을 만드는데 가능하다 (다른 템플릿엔진도 다 가능...ㅎ)

 

타임리프설정하기

 

1. dependency 중 secutiry, security-test주석처리
2. spring->starters에서 타임리프라이브러리추가
3. websecutiryconfig 어노테이션 지우기

 

css,html은 static

서버단은 웹앱

 

templates 에 index.html추가

 

1.타일즈 리졸버 (우선순위높음) src를 먼저찾음 -> 인덱스를 찾는다. "index"

2.리소스 리졸버

3. 타임리프 리졸버 (aa- >템플릿폴더에서 aa.html을찾는다.

타임리프라이브러리를 설치하면 타임리프를 먼저찾는다.

 

컨트롤러에서 타일즈를뺴고 타임리프만 가져가는게 좋다.

타임리프쓸때 타일즈 걷어내기

혼합해서쓰지말것

타일즈 컨피규레이션 주석처리하기

 


퍼블리셔서 나중에 페이지를 변경해도 서로간에 데이터를 침범하지않는다.

 


text

utext: unescaped text

 

출력되는게 아니라 볼드체로 적용하려면 ?


jsp에서 el태그와 비슷하게생겼다

하지만 분명히 다르다.

암기하기

message : #{} 

메세지 : 외부파일

aa.properties 등 설정.properties로 확장자만 프라퍼티즈

프라퍼티즈컬렉션이 자바에있음

프라퍼티즈 클래스

파일로 저장할 수있다. xml 또는 프라퍼티즈확장자로 

맵컬렉션인데 추가적으로 실제파일과 연관되서 읽을수있다.

 

저장,읽기 가능

외부파일로 저장해보자 원하는 설정을

 

#{}프라퍼티즈확장자인 외부파일에서 불러옴 ->메세지(외부파일) 꺼내오는것

 

컨트롤러에서 모델에서 뽑아올떄는 ${}

#{} 메세지를 불러온다.

${}은 모델에서 가져옴


컨텍스트라는 저장소에 데이터를 담았다가 꺼내쓸수있다.

setVariable로 담았다가 getVariavle로 꺼내쓸수 있다.

OGNL

JSP에서도 ognl을 쓰고있었다.

객체라면 그안에있는것들을 점으로 호출했었다.

 



fragment 이름이 카피인것 집어넣는다

 

 

~ 절대경로

한글깨짐

.

https://www.thymeleaf.org/doc/tutorials/3.0/usingthymeleaf.html#expression-basic-objects

div외에도 가능

insert 

리플레이스 :덮어쓰기, 프래그먼트시정시 대신해서들어가는 것,

인클루드:안쪽으로

 

물결생략가능


비포

id 또는 fragment로 가져온다.

에프터

 


레이아웃처리방법 두가지

 

레이아웃에서 메인을뺴서 리스트에 메인넣기

list.html의 div안쪽이 다 사라지고 layout의 th:fragment="body"로 덮어진다.

왼쪽에있는애들이 모두 오른쪽으로 대체됨.

 


문제점 : 레이아웃에서 헤더,푸터를 가져왔는데, list.html의 메인영역위에 덮어씌워져서 list의 메인이 사라졌다.

레이아웃에세 내 main을 꽂아넣어주고 가져오려면 ?

 

레이아웃을 가져오는데,

헤더,푸터를 가져왔다. 

메인은 어떻게가져온담 ?

 

list.html에서 헤더,푸터를 레이아웃에서가져온뒤

바디에게 내꺼써서줘라고 할수있다

 

레이아웃의 내 바디의정볼르 꽂아넣을수있다

 

list의 자기영역에서 main클래스로 선택된애를 레이아웃에게준다.(.main)

 

레이아웃의 body(x) x를 자기 몸에 가져다붙인다. 자기영역에 가져다붙인다.

1.list.html의 body(~{this::.main}) 이 레이아웃의 바디부분을 가져오고 , 덮어씌운다

2. list의 메인부분이 레이아웃의 메인때문에 안보이게되므로 인자를 넘겨준다 

.main은 내껄쓰겠다

3. layout.html의 th:fragment ="body(x)" x는 list가 지정한부분을 넣겠다

4. th:replace="${x}" main부분을 x에 넣어줘

 

남의것을가져오면서 내것을 거기에 꽂을수있어

외부라이브러리를 가져올 필요가없다.


타임리프말고, 타일즈같은 라이브러리를 쓰는 방법

 

스프링 MVC

-컨트롤러, 
@Controller
@RestController

-입력, 
@RequestParam, @PathVariable, @Cookie
-출력
@ResponseBody

스프링 DI:Ioc Container/역제어/
@Component->@Service,@Repository,@Controller,@Configuration,@Bean
@Autowired

마이바티스(SQL 매퍼를 이용해서 JDBC를 대신 작성해주는 프레임워크)
@Mapper->@Select,@Insert,@...
XML ->resultType,resultMap,Dynamic Query <if><foreach>...

JUnit을 이용해서 테스트
Log 를 통해서 쿼리를 보는 방법

MVC구현과 3Tier(티어)[UI<->Service<->DAO]에서 필요로하는 기본적인 솔루션 2가지
-계층간의 사용방식
-UI에서 중첩되는 데이터 사용방식

스프링 트랜잭션
@Transactional


 

데이터베이스를 사용하기위해 다오

orm 나중에설명

셀렉트할떄 (값을가져올떄), 업데이트할때 고민되는 부분이있다

 

select * from notice where id = 3 (or id=5 or 9)

가끔은 이 쿼리문을 3이 아니라 3또는 5 또는 9 여러개를 담는 경우가 있다.

select * from notice where id in(3,5,9...)

1. 10번의 쿼리를 실행하고 10번위 데이터를불러와야한다.

또는 다오에게 아이디를 통째로주는것

 

다오 :sql대신, 필터링, 집계,정렬,배열선택

마이바티스는 해당시퀄이가능하도록 기능제공

 

아이디즈에서 하나꺼내서 item에 담는다.

(시작  )끝남

id출력

구분자seperator =","로 반복

 

colletion에는 배열이나 리스트만 온다

아이디를 배열에 전달
콘트롤러 / 마이바티스다오
매퍼 / 콘솔

foreach : 전달하는 ids를받아서 id하나를 꺼낸다. 쉼표를이용해 아이디를 반복하게해준다.

1
2
3
4
5
6
7
8
9
    <select id="getListIn" resultType="com.newlecture.web.entity.Notice">
        SELECT *
        FROM Notice
        WHERE ID in
        <foreach item="id" index="index" collection="ids"
         open="(" separator="," close=")">
            #{id}
        </foreach>
    </select>
cs

 

단위테스트 

junit 테스트툴

juit은 스프링이 아니다.

마이바티스는 스프링이랑 상관없다..

 

스프링은 BootPrjApplication클래스 실행 -> SpringApplication.run() 실행 -> 스프링깨어남 ->마이바티스 깨어남

스프링이 일어나야 컨테이너에 객체가 찬다.

스프링이 일어나야 마이바티스를 꺠운다.

 

테스트하고저ㅏ하는 클래스 선택- 마우스오른쪽 - 테스트유닛

스프링을 깨우지않은상태

test가 메인함수가 아니니까

실행자가 junit () 

@Test를 통해 실행해준다.

 

1.메인함수로 반환값확인

2.트레이스작업

 

런에즈- 제이유닛으로 실행

코드 : 이클립스가도와줌,

 

메크로함수테스트해보기


스프링이일어나야 객체들이찬다

 

마이바티스에 제이유닛플러그인을 ..

라이브러리추가

pom.xml에 추가

마이바티스가 이미 개발목적

테스트할떄사용하면 ,,,

http://mybatis.org/spring-boot-starter/mybatis-spring-boot-test-autoconfigure/index.html

 

 


사용자가 액션하면 ..

사용자행위가 그대로묻어난다

likeToggle

 

부품의부품화...잘못!

서비스가 서비스를 부품으로 xx -> 인터페이스라서 ?

/ 아마 오른쪽으로가게될것..(병목발생) ->고립화필요

남이만든 서비스를 쓰면 민폐를 끼칠수있다...협업에따라다름


대댓글기능

포이치안에 포이치

 


와스같은컨테이너가있다

엔터프라이즈자바빈

 

서비스에서 직접jdbc가편할수도있다..

AOP

어노테이션이 aop로 구현되어있다.

 

관점지향 프로그래밍

 

업무가 말했던것만 코드로 만들지않는다.

내가 만드는 내용중에는 업무로 분석된것이 아니라,

ex 성능에 문제가있다면 , 실행시간이 느린 함수체크를위해 로그기록, 시간을 재기위한 코드를 포함시키게된다.

관리자가 할일, 사용자가 할일 (조건처리필요)

 

프로그래밍은 업무외에 다른사람입장(새로운관점)이 필요하다.

개발자,운영자 목적

 

코드마다 똑같은내용발생.

노가다작업발생

->방법론나옴 AOP

 

업무자 + 다른관점 : 위아래에 꽂는다. 뺄수도있다

 

AOP관련용어

업무자/클라이언트/사용자 : 주업무, core concern , primary concern

부업무 :  cross cutting 방향성

 

크로스컷팅을 쉽게하기위한 방법론

 

객체지향안에 사용되는 함수안에서 쉽게 꽂고뺄수있는 방법론

 

한번만큼 코드를 직접하지않고 모든함수에 꽂아 줄 수 있니 ?

빵또아->아이스크림고 ㅏ빵 분리-> 위에있는 빵 실행 -> 안에실행 -> 밑에있는빵실행

 

Proxy준비 

 

메소드가 빵을 통해 core concern을 호출,사용

 

과거방식

AOP구현방식

 

오른쪽업무에는 proxy부분을 제외(분리)

proxy / 업무

 

가짜인스턴스생성.

거쳐서호출하기위한 핸들러

 

가짜의토탈(가짜의토탈 호출) 로드할때 이그젬인터페이스를통해 호출

 

aop는 방법론 : 업무로직+로그,권한,다양한 역할자관점을 코드에 넣다빼는방법

프락시로 객체를 호출

프렘워크마다 사용방법이다르므로 개념을알고있으면 사용시 ㄷ움말참저

 

트랜잭션도 이런 프락시를 제공하면서 사용


트랜잭션 특징

 

  • 원자성 (Atomicity)
  • 일관성 (Consistency) :데이터결함이 발생하지않게
  • 고립성 (Isolation)
  • 지속성 (Durability) : 커밋한 내용은 계속 저장되어야한다

억지로 두번의 업데이트를실행했었다

다완전히 ㅅ ㅣㄹ행되길원했었다

결과는 0이되어야했다

중간에오류가났는데 100이다.

 

 

 

 

@Transactional

오류가나면 위에 성공한것이 복구된다

원자성지켜짐

오류는 나지만, 데이터 hit가 처음값1으로 나온다

100이 적용된 후 0으로 덮어씌워져야하는데, 아이디오류로 아예 롤백되어 100, 0모두 적용이안되고

조회하면 원래값인 1이 조회된다.


트랜잭션 : 애ㅔ러나면 원래상태로복구됨

하위부분 호출에대한 트랜잭션 

  • PK − Primary Key
  • NN − NOT NULL
  • BIN − Binary
  • UN − Unsigned
  • UQ − Unique
  • ZF − Zero Filled
  • G − Generate Column
  • AI − Auto Increment

ioc컨테이너에 담긴 객체는 autowired를 통해 불러온다.

@Repository ioc에 담아줘

@Autowired ioc에서 관련 객체 꺼내줘

 

맵핑방식을 xml로 바꾸자~

@Mapper대신 xml구현체를 직접 만들어보자

 

1. https://mybatis.org/mybatis-3/configuration.html#mappers

  NoticeDaoMapper.xml에 매퍼태그를 넣는다.

2. NoticeDao에는 인터페이스이므로 구현할 함수목록만 남긴다.

3. MybatisNoticeDao 생성 (클래스생성시 add눌러서 NoticeDao 인터페이스선택)

현재쓰고있는 마이바티스버전 / resultType = 반환할때의 그릇형식, 반환하게될 객체의 그릇(select의 컬럼)

마이바티스를 스프링에서 사용할때 필요한설정을 갖고있는 라이브러리가 추가되어있다.(자동) 3.5.

 

'%${query}%' 를 쓴이유 ?

#을 쓰면 't'가 들어간다. (값형태)

$는 날것그대로


마이바티스는 동일이름 메소드 중복불가


 

주황:스프링

@controller, @servicem @repository를  ioc에 담는다

@mapper 객체가 바로 ioc에 담긴다

 

그린:마이바티스

마이바티스는 mapper Container를 가지고있다. Mapper객체가 담겨진다.

두 종류로 설정할 수있다.

1. 어노테이션 @Mapper (매퍼객체를 바로씀), 마이바티스에 특화된 인터페이스(제약이생김) , ioc에서 바로 사용가능

2. xml ??Mapper.xml (매퍼컨테이너에 담기만) , 간접적으로쓴다, 자기컨테이너에담는다.Dao는 내가 직접구현할게

두개다 매퍼객체가 매처컨테이너에 담긴다

ioc컨테이너가 매퍼객체를 사용 할 수있게 넣어준다

다오를 구현할때 쓰기위해 객체를 꺼낸다. (MybatisNoticeDao)

 

매퍼개체를 꺼내오는 도구 : sqlSession클래스사용 ( private SqlSession sqlSession)

getMapper()매퍼객체를 가져와

 

마이바티스가 Mapper.xml을 읽으면 Mapper Container 에 Mapper객체를 넣는다.

우리는 그 컨테이너에 있는것을 꺼내서(getMapper) 실제 마이바티스를 구현할때 쓴다. (class MybatisNoticeDao)

무슨 장점 ?

구현체에 대한 내용이 노출되지않는다.

인터페이스는 유지.

쉽게 바꿀수있다. 

독립적인 인터페이스

다오를 직접구현하는게 불편해보여도 정상적인 방법

인터페이스에 @select등 붙이지 않는게 바람직

Jdbc 코드를 2줄코드로 완성

NoticeDao mapper = sqlSession.getMapper(??;

return mapper.getList();

 

모든Mapper의 위치 지정(내가 만든 메퍼정보를 마이바티스에게 알린다) (xml위치지정)

 


getList 인자값이 각각 다른경우 (인자가없거나 하나일떄)

기본값을 넣어준다.

다른데로 sendredirection

마이바티스는 xml의 한줄맵핑으로 ~~

 


디테일페이지

id가 널인 가능성이없으므로 integer말고 int로하기


삭제페이지

update,insert,delete는 반환타입을 주로 int쓴다.

1. 컨트롤러
1
2
3
4
5
6
7
8
9
10
11
12
13
 
@Controller("adminNoticeController")
@RequestMapping("/admin/notice/")//공통분모
public class NoticeController {
 
 @RequestMapping("del")
    public String del(int id) {
 
     service.delete(id);
 
    return"redirect:list";
   }
}
cs

2. 다오매퍼

1
2
3
4
NoticeDaoMapper
<delete id="delete" parameterType="int">
  delete from Notice where id=#{id}
</delete>
cs

3. 서비스인터페이스

1
2
3
public interface NoticeService {
int delete(int id);
}
cs

4. 서비스인터페이스 구현

1
2
3
4
5
6
7
@Service//component
   public class NoticeServiceimp implements NoticeService {
   @Override
   public int delete(int id) {
   return dao.delete(id);
  }
}
cs

5.다오인터페이스

1
2
3
4
public interface NoticeDao {
 
     int delete(int id);
}
cs

6. 다오인터페이스 구현

1
2
3
4
5
6
7
8
9
@Repository//ioc에 MybatisNoticeDao를 담아줘
public class MybatisNoticeDao implements NoticeDao {
 
  @Override
  public int delete(int id) {
 
  return mapper.delete(id);
  }
}
cs

 


 

resultType SELECT문 실행 결과를 담을 객체
parameterType 이 속성에 지정한 객체의 프로퍼티값이 SQL문의 입력 파라미터에 지정된다.(insert,update,delete)

다오 -> 서비스->컨트롤러 순으로 만들어보자

업데이트

1. 다오 인터페이스

1
2
3
4
5
public interface NoticeDao {
    int update(Notice notice);
 
}
 
cs

2. 다오매퍼

1
2
3
4
5
6
7
8
9
10
11
12
    <update id="update" parameterType="com.newlecture.web.entity.Notice">
        update Notice
        SET
        title = #{title},
        writerId = #{writerId},
        content = #{content},
        hit = #{hit: 0},
        files = #{files},
        pub = #{pub}
        where id=#{id}
 
    </update>
cs

3. 마이바티스다오 (다오인터페이스 구현)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
@Repository//ioc에 MybatisNoticeDao를 담아줘
public class MybatisNoticeDao implements NoticeDao {
    
    //@Autowired//ioc에있는 연관객체를 가져와//mybatis-spring라이브러리:스프링에담아주는역할
    private SqlSession sqlSession;    
    private NoticeDao mapper;
    
    @Autowired
    public MybatisNoticeDao(SqlSession sqlSession) {
        this.sqlSession = sqlSession;
        mapper = sqlSession.getMapper(NoticeDao.class);
    }
 
    @Override
    public int update(Notice notice) {
 
        return mapper.update(notice);
    }
 
}
 
cs

 

ㅇ인서트,업데이트는 인자쓸수잇다

셀렉트는 파라미터타입지정할 수없다.

약속되어있는 param1, param2를 쓴다

 

프로젝트진행 : 디비사용되어야한다 ->데이터베이스구축되어야한다..

https://elfinlas.github.io/2018/02/18/spring-parameter/

https://logical-code.tistory.com/25


11111

컨트롤 + 마우스휠 위로 : 확대

컨트롤 + 엔터

먼저 newlect선택하고 실행해줘야한다


Mybatis 라이브러리 추가

쿼리만들고 구하려는것의 반환타입만 적으면 mybatis가 도와준다


스프링의 역할은 ?

디펜던시인젝션역할

두개의 객체를 결합해준다.

 

 

B가 C를 이용, A가 C를 이용

제품과 부품

제품이 될 수도 부품이될 수도있다.

디펜던시 : 내가 만드는게 디펜던시가될수있다.

 

DI의 장점 : 기능에 대한 인터페이스가 올바르면 , 구현된 다른제품을 꽂아쓸 수있다. 

                협업용이, 다른이유로 버전업을 시켜야하거나 , 구현성능이 좋아진것으로 바꿔낄때 유용

              재사용, 부품교환이 자유롭다

 

1. A는 부품으로쓰임 / 세번째 Frame클래스가 DI 3. 스프링이 코드재배포(재작성) 개선을 위해 설정바꿔준다. 결합관계를 표시만해주면된다

 

"내가 frame하나 객체화했는데 그 이름을 f로 해줘"

"A라는 클래스도 객체화했는데 그 이름을 a로해줘"

"그 a를 Frame의 set함수에다가 넣어줘" (인젝션)"

디펜던시 설정, 디펜던시결합해줌

설정만 바꾸면 알아서 자바코드가 변경되는 식

 

내가 만든건 @Component

남이 만든 클래스는 @Component , @bean

 

기존컨테이너는 각각 써야했는데

ioc컨테이너는 결합 및 각각 쓸 수있다

위 : 일체형 A -> B- >C-> D

아래 : 결합형 D부터만들어짐 , 역순으로 객체를 생성, 부품이 결합

 


반환타입에 맞게끔 NoticeDao를 구현해서 장바구니에 담는다 

 

플랫폼을 공부목적으로..?xxx


Mybatis 역할 :

NoticeDao(@mapper)의 List<Notice>를 구현한다 . Ioc컨테이너에 담아준다. 우리는 꺼내쓰는 작업만 하면된다

 

XML 과 어노테이션

 

새로운버전때문에 설정을 바꾸면 , 

스프링에서는 

어노테이션 : 명찰역할 (훨씬 편리)

xml : 변경발생시 매번설정바꿔야함 (매번 장부작성과 비슷)

 

Responsebody : 뷰페이지를 찾는게 아니라 list()리턴값이 뷰다 ->바로 출력된다.

1.생성자 인젝션 : 가져오자마자 다른 초기화 로직이 동작하게 할 수있다. 세팅될때 더불어서 하게할수있다.

  셋팅이 나중에 바뀌면안된다면 생성자가 적합

2.세터 인젝션 : 언제든지 또 호출될 수있다.  

콘트롤러/서비스에게반환/구현체

IoC에 실린 객체들
noticeDao

noticeServiceimp

noticeController

마이바티스가 sql문을 맵핑해준다.

우리가 직접구현한다면 Jdbc 를 구현했을것..

마이바티스가 이것을 자동으로 구현해준다.

 

IoC container는 스프링에서 자유롭게 객체를 넣거나 꺼내쓸수있는 
스프링 공용 객체보관 container다

@Service

Spring에서 @Component로 다 쓰지 않고 @Repository, @Service, @Controller등을 사용하는 이유는 

또한 가독성에서도 해당 애노테이션을 갖는 클래스가 무엇을 하는지 단 번에 알 수 있다.

 

@Autowired

생성자에 @Autowired 가 있으면 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어준다.

이렇게 객체 의존관계를 외부에서 넣어주는 것을 DI (Dependency Injection), 의존성 주입이라 한다.

 

@Component 

컴포넌트 스캔 원리 @Component 애노테이션이 있으면 스프링 빈으로 자동 등록된다.

@Controller 컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문이다.

@Component 를 포함하는 다음 애노테이션도 스프링 빈으로 자동 등록된다.

 @Controller @Service @Repository 회원 서비스 스프링 빈 등

 

+ Recent posts