๐Ÿ“š
TIL
  • README
  • Git
    • Basic
    • Remote Repository
    • Log & Diff
    • Rebase&Cherri-Pick
    • git-flow
  • DevOps
    • Monolithic vs MSA
    • Jenkins ์‹œ์ž‘ํ•˜๊ธฐ
    • Airflow ์‹œ์ž‘ํ•˜๊ธฐ
    • Airflow ์‹œ์ž‘ํ•˜๊ธฐ
    • Build Tools
      • maven
  • ๊ฐœ๋ฐœ ๋ฐฉ๋ฒ•๋ก 
    • TDD
  • Spring
    • IoC
    • Is Spring Bean Thread-Safe?
    • Spring Singleton
    • Component Scan
    • Spring Annotation
    • ์˜์กด ๊ด€๊ณ„ ์ฃผ์ž…(DI)
    • Lombok ํ™œ์šฉํ•˜๊ธฐ
    • Bean ์ƒ๋ช…์ฃผ๊ธฐ์™€ ์ฝœ๋ฐฑ
    • Bean Scope
    • AOP(1) - AOP๋ž€
    • AOP(2) - Aop Proxy
    • AOP(3) - Dynamic Proxy
    • AOP(4) - AspectJ
    • POJO
    • Spring ์„œ๋น„์Šค ๊ตฌ์กฐ
    • Transaction
    • JPA๋ž€?
    • JPA Entity
    • Spring Data JPA
    • Spring Data Specification
    • Model Mapping
    • Cache
    • restTemplate
    • YAML ํŒŒ์ผ ์„ค์ •
    • Spring Boot
      • H2 DB ์„ค์ •
      • ๋‹ค์ค‘ ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค ์„ค์ •
      • Mybatis ์—ฐ๋™ํ•˜๊ธฐ
    • Spring Batch
      • Batch ์‹œ์ž‘ํ•ด๋ณด๊ธฐ
      • Batch Job Flow
      • Job
      • Step
      • Batch Scope & Job Parameter
      • JobRepository์™€ ๋ฉ”ํƒ€ํ…Œ์ด๋ธ”
      • Chunk ์ง€ํ–ฅ ํ”„๋กœ๊ทธ๋ž˜๋ฐ
      • ItemReader
      • ItemProcessor
      • ItemWriter
      • Batch Schedular
      • Job๋ณ„ Bean๋“ฑ๋กํ•˜๊ธฐ
      • Batch ๊ตฌํ˜„์‹œ ๋ฐœ์ƒํ•œ ์˜ค๋ฅ˜ ์ •๋ฆฌ
      • Spring Batch Scaling
        • Multithread Job๊ตฌํ˜„์‹œ ์ด์Šˆ์‚ฌํ•ญ
    • Spring test
      • Junit5
        • ํ…Œ์ŠคํŠธ ์ด๋ฆ„ ํ‘œ๊ธฐ
        • ํ…Œ์ŠคํŠธ ๊ทธ๋ฃน ์‚ฌ์ด์˜ ๊ด€๊ณ„
        • ํƒœ๊ทธ์™€ ํ•„ํ„ฐ๋ง
        • ๋™์  ํ…Œ์ŠคํŠธ
        • ํ…Œ์ŠคํŠธ LifeCycle
        • ํ…Œ์ŠคํŠธ ๋ฉ”์„œ๋“œ
        • ํ…Œ์ŠคํŠธ ์ˆœ์„œ
        • AssertJ
        • ํ…Œ์ŠคํŠธ ๋ณ‘๋ ฌ ์‹คํ–‰
        • AssertJ
        • Mock
      • Spring Boot Test DB ๋ถ„๋ฆฌ
      • Spring Batch Test
  • Web Application
    • Web Server & WAS
    • ๊ด€๋ จ ๊ฐœ๋… - HTTP API, HTML, CSR, SSR
    • Servlet
    • JSP
    • Cookie And Session
    • ์˜ˆ์™ธํŽ˜์ด์ง€
    • Java Bean
    • JDBC
    • Connection Pool
    • ํŒŒ์ผ ์—…๋กœ๋“œ
    • Expression Language
    • JSTL
    • FrontControllerํŒจํ„ด Command ํŒจํ„ด
    • Forwarding
    • MVC
    • ํšŒ์›๊ฐ€์ž…์˜ˆ์ œ
    • ์ฐธ๊ณ 
      • ๊ฐœ๋ฐœํ™˜๊ฒฝ์„ค์ •
  • Java+
    • SOAP/WSDL vs REST
    • WSDL์„ JAVA๋กœ ๋ณ€ํ™˜ํ•˜๊ธฐ
    • SOAP ํ†ต์‹  OPEN API๋กœ ๊ฐœ๋ฐœํ•ด๋ณด๊ธฐ
  • Java
    • Basic
      • ๋ณ€์ˆ˜์™€ ํƒ€์ž…
      • ์—ฐ์‚ฐ์ž
      • ์กฐ๊ฑด๋ฌธ๊ณผ ๋ฐ˜๋ณต๋ฌธ
      • ์ฐธ์กฐ ํƒ€์ž…
      • ํด๋ž˜์Šค
      • ์ƒ์†(Inheritance)
      • ์ธํ„ฐํŽ˜์ด์Šค(Interface)
      • ์ค‘์ฒฉ ํด๋ž˜์Šค์™€ ์ค‘์ฒฉ ์ธํ„ฐํŽ˜์ด์Šค
      • ์˜ˆ์™ธ ์ฒ˜๋ฆฌ
      • API - Object, System, Class, Math, Wrapper
      • API - String, StringBuffer, StringBuilder
      • Thread
      • Generic
      • Lambda
      • Collection - List, Set
      • Collection - Map
      • Collection - Tree
      • Collection - Stack, Queue
      • Stream
      • Reflection
      • ์ •๊ทœํ‘œํ˜„์‹
      • GUI
      • UML
      • Serializable
    • Advanced
      • OutOfMemoryError
      • AutoValue
      • meta-annotation
        • @Retention
        • @Target
        • @Repeatable
    • Effective Java 3/E
      • ITEM 1: Static Factory Method(์ •์  ๋ฉ”์†Œ๋“œ)
      • ITEM 2: Builder Pattern
      • ITEM 3: Singleton
      • ITEM 4: Private Constructor
      • ITEM 5: Dependency Injection
      • ITEM 6: Avoid Unnecessary Object
      • ITEM 7: Eliminate Object Reference
      • ITEM 8: Avoid finalizer and cleaner
      • ITEM 9: try-with-resources
      • ITEM 10: The gerneral contract when overriding equlas
      • ITEM 11: Overriding hashCode
      • ITEM 12: overriding toString
      • ITEM 13: overriding clone judiciously
      • ITEM 14: Consider implementing comparable
      • ITEM 15: ํด๋ž˜์Šค์™€ ๋ฉค๋ฒ„์˜ ์ ‘๊ทผ์„ ์ตœ์†Œํ™”ํ•ด๋ผ
      • ITEM 16: Use Accessor methods
      • ITEM 17: ๋ณ€๊ฒฝ ๊ฐ€๋Šฅ์„ฑ์„ ์ตœ์†Œํ™”ํ•ด๋ผ(๋ถˆ๋ณ€ ํด๋ž˜์Šค)
      • ITEM 18: ์ƒ์†๋ณด๋‹จ ์ปดํฌ์ง€์…˜์„ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 19: ์ƒ์†์„ ๊ณ ๋ คํ•ด ์„ค๊ณ„ํ•˜๊ณ  ๋ฌธ์„œํ™”ํ•ด๋ผ
      • ITEM 20: ์ถ”์ƒ ํด๋ž˜์Šค๋ณด๋‹ค ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ์šฐ์„ ํ•˜๋ผ
      • ITEM 21: ์ธํ„ฐํŽ˜์ด์Šค๋Š” ๊ตฌํ˜„ํ•˜๋Š” ์ชฝ์„ ์ƒ๊ฐํ•ด ์„ค๊ณ„ํ•ด๋ผ.
      • ITEM 22: ์ธํ„ฐํŽ˜์ด์Šค๋Š” ํƒ€์ž…์„ ์ •์˜ํ•˜๋Š” ์šฉ๋„๋กœ๋งŒ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 23: ํƒœ๊ทธ ๋‹ฌ๋ฆฐ ํด๋ž˜์Šค๋ณด๋‹ค ํด๋ž˜์Šค ๊ณ„์ธต๊ตฌ์กฐ๋ฅผ ํ™œ์šฉํ•ด๋ผ
      • ITEM 24: ๋ฉค๋ฒ„ ํด๋ž˜์Šค๋Š” ๋˜๋„๋ก static์œผ๋กœ ๊ตฌํ˜„ํ•ด๋ผ
      • ITEM 25: ํ†ฑ๋ ˆ๋ฒจ ํด๋ž˜์Šค๋Š” ํ•œ ํŒŒ์ผ์— ํ•˜๋‚˜๋งŒ ์ƒ์„ฑํ•ด๋ผ.
      • ITEM 26: Raw type์€ ์‚ฌ์šฉํ•˜์ง€ ๋งˆ๋ผ
      • ITEM 27: ๋น„๊ฒ€์‚ฌ ๊ฒฝ๊ณ ๋ฅผ ์ œ๊ฑฐํ•ด๋ผ
      • ITEM 28: ๋ฐฐ์—ด๋ณด๋‹ค๋Š” ๋ฆฌ์ŠคํŠธ๋ฅผ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 29: ์ด์™•์ด๋ฉด ์ œ๋„ค๋ฆญ ํƒ€์ž…์œผ๋กœ ๋งŒ๋“ค์–ด๋ผ
      • ITEM 30: ์ด์™•์ด๋ฉด ์ œ๋„ค๋ฆญ ๋ฉ”์„œ๋“œ๋กœ ๋งŒ๋“ค์–ด๋ผ
      • ITEM 31 : ํ•œ์ •์  ์™€์ผ๋“œ์นด๋“œ๋ฅผ ์‚ฌ์šฉํ•ด API ์œ ์—ฐ์„ฑ์„ ๋†’์—ฌ๋ผ
      • ITEM 32: ์ œ๋„ค๋ฆญ๊ณผ ๊ฐ€๋ณ€์ธ์ˆ˜๋ฅผ ํ•จ๊ป˜ ์“ธ ๋•Œ๋Š” ์‹ ์ค‘ํ•ด๋ผ
      • ITEM 33: ํƒ€์ž… ์•ˆ์ „ ์ด์ข… ์ปจํ…Œ์ด๋„ˆ๋ฅผ ๊ณ ๋ คํ•ด๋ผ
      • ITEM 34: int ์ƒ์ˆ˜ ๋Œ€์‹  ์—ด๊ฑฐ ํƒ€์ž…์„ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 35: ordinal ๋ฉ”์„œ๋“œ ๋Œ€์‹  ์ธ์Šคํ„ด์Šค ํ•„๋“œ๋ฅผ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 36: ๋น„ํŠธ ํ•„๋“œ ๋Œ€์‹  EnumSet์„ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 37: ordinal ์ธ๋ฑ์‹ฑ ๋Œ€์‹  EnumMap์„ ์‚ฌ์šฉํ•ด๋ผ
      • TEM 38 : ํ™•์žฅํ•  ์ˆ˜ ์žˆ๋Š” ์—ด๊ฑฐํƒ€์ž…์ด ํ•„์š”ํ•˜๋ฉด ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 39: ๋ช…๋ช… ํŒจํ„ด๋ณด๋‹ค ์• ๋„ˆํ…Œ์ด์…˜์„ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 40: @Override ์–ด๋…ธํ…Œ์ด์…˜์„ ์ผ๊ด€๋˜๊ฒŒ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 41: ์ •์˜ํ•˜๋ ค๋Š” ๊ฒƒ์ด ํƒ€์ž…์ด๋ผ๋ฉด ๋งˆ์ปค ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 42: ์ต๋ช… ํด๋ž˜์Šค๋ณด๋‹ค๋Š” ๋žŒ๋‹ค๋ฅผ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 43: ๋žŒ๋‹ค๋ณด๋‹ค๋Š” ๋ฉ”์„œ๋“œ ์ฐธ์กฐ๋ฅผ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 44: ํ‘œ์ค€ ํ•จ์ˆ˜ํ˜• ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 45: ์ŠคํŠธ๋ฆผ์€ ์ฃผ์˜ํ•ด์„œ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 46: ์ŠคํŠธ๋ฆผ์—์„œ ๋ถ€์ž‘์šฉ ์—†๋Š” ํ•จ์ˆ˜๋ฅผ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 47: ๋ฐ˜ํ™˜ ํƒ€์ž…์œผ๋กœ๋Š” ์ŠคํŠธ๋ฆผ๋ณด๋‹ค ์ปฌ๋ ‰์…˜์ด ๋‚ซ๋‹ค.
      • ITEM 48: ์ŠคํŠธ๋ฆผ ๋ณ‘๋ ฌํ™”๋Š” ์ฃผ์˜ํ•ด์„œ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 49: ๋งค๊ฐœ๋ณ€์ˆ˜๊ฐ€ ์œ ํšจํ•œ์ง€ ๊ฒ€์‚ฌํ•ด๋ผ
      • ITEM 50: ์ ์‹œ์— ๋ฐฉ์–ด์  ๋ณต์‚ฌ๋ณธ์„ ๋งŒ๋“ค์–ด๋ผ
      • ITEM 51: ๋ฉ”์„œ๋“œ ์‹œ๊ทธ๋‹ˆ์ฒ˜๋ฅผ ์‹ ์ค‘ํžˆ ์„ค๊ณ„ํ•ด๋ผ
      • ITEM 52: ๋‹ค์ค‘์ •์˜๋Š” ์‹ ์ค‘ํžˆ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 53: ๊ฐ€๋ณ€์ธ์ˆ˜๋Š” ์‹ ์ค‘ํžˆ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 54: null์ด ์•„๋‹Œ, ๋นˆ ์ปฌ๋ ‰์…˜์ด๋‚˜ ๋ฐฐ์—ด์„ ๋ฐ˜ํ™˜ํ•ด๋ผ
      • ITEM 55: Optional ๋ฐ˜ํ™˜์€ ์‹ ์ค‘ํ•˜๊ฒŒ ํ•ด๋ผ
      • ITEM 56: ๊ณต๊ฐœ๋œ API ์š”์†Œ์—๋Š” ํ•ญ์ƒ ์ฃผ์„์„ ์ž‘์„ฑํ•ด๋ผ
      • ITEM 57: ์ง€์—ญ๋ณ€์ˆ˜์˜ ๋ฒ”์œ„๋ฅผ ์ตœ์†Œํ™”ํ•ด๋ผ
      • ITEM 58: ์ „ํ†ต์ ์ธ for ๋ฌธ๋ณด๋‹ค๋Š” for-each๋ฌธ์„ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 59: ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ๋ฅผ ์ตํžˆ๊ณ  ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 60: ์ •ํ™•ํ•œ ๋‹ต์ด ํ•„์š”ํ•˜๋‹ค๋ฉด float์™€ double์€ ํ”ผํ•ด๋ผ
      • ITEM 61: ๋ฐ•์‹ฑ๋œ ๊ธฐ๋ณธ ํƒ€์ž…๋ณด๋‹ค๋Š” ๊ธฐ๋ณธ ํƒ€์ž…์„ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 62: ๋‹ค๋ฅธ ํƒ€์ž…์ด ์ ์ ˆํ•˜๋‹ค๋ฉด ๋ฌธ์ž์—ด ์‚ฌ์šฉ์„ ํ”ผํ•ด๋ผ
      • ITEM 63: ๋ฌธ์ž์—ด ์—ฐ๊ฒฐ์€ ๋А๋ฆฌ๋‹ˆ ์ฃผ์˜ํ•ด๋ผ
      • ITEM 64: ๊ฐ์ฒด๋Š” ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ์‚ฌ์šฉํ•ด ์ฐธ์กฐํ•ด๋ผ
      • ITEM 65: ๋ฆฌํ”Œ๋ ‰์…˜๋ณด๋‹ค๋Š” ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 66: ๋„ค์ดํ‹ฐ๋ธŒ ๋ฉ”์„œ๋“œ๋Š” ์‹ ์ค‘ํžˆ ์‚ฌ์šฉํ•ด๋ผ
      • ITEM 67: ์ตœ์ ํ™”๋Š” ์‹ ์ค‘ํžˆ ํ•ด๋ผ
      • ITEM 68: ์ผ๋ฐ˜์ ์œผ๋กœ ํ†ต์šฉ๋˜๋Š” ๋ช…๋ช… ๊ทœ์น™์„ ๋”ฐ๋ผ๋ผ
    • ๊ฐ์ฒด์ง€ํ–ฅ ์„ค๊ณ„ ์›์น™(SOLID)
    • ๋””์ž์ธํŒจํ„ด
      • Strategy Pattern
      • Template Method Pattern
      • Factory Method Pattern
      • Singleton
      • Delegation
      • Proxy
      • Adapter Pattern
    • ์‹ค์Šต
      • ์ธํ„ฐํŽ˜์ด์Šค ์‹ค์Šต - Vehicle
      • ์ธํ„ฐํŽ˜์ด์Šค ์‹ค์Šต - Remote
      • GUI ์‹ค์Šต - Calculator
      • GUI ์‹ค์Šต - button
      • GUI ์‹ค์Šต - lotto
      • Thread ์‹ค์Šต - ์ขŒ์„์˜ˆ์•ฝ, ๋ฉ”์„ธ์ง€๋ณด๋‚ด๊ธฐ
    • Jar vs War
  • ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค
    • KEY
    • Index
    • Transaction
    • Trigger
    • Procedure / Function
    • Package
    • ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค ๋ฐฐ์›€ํ„ฐ
      • ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค ์‹œ์Šคํ…œ
      • ๊ด€๊ณ„๋ฐ์ดํ„ฐ ๋ชจ๋ธ
      • ๊ด€๊ณ„๋Œ€์ˆ˜์™€ SQL
    • MySQL
      • Database๋ž€
      • MySQL ์‹œ์ž‘ํ•˜๊ธฐ
      • MySQL Database
      • MySQL Table
      • CRUD
      • ๊ด€๊ณ„ํ˜• ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค
      • Server์™€ Client
    • PostgreSQL
    • NoSQL
      • Install Cassandra on mac
      • Cassandra๋ž€?
      • NiFi๋ž€
  • Algorithm
    • String
    • Recursion
    • Dynamic Programming
    • Array, Struct, Pointer
    • Math
    • Sort
    • List
    • Stack
    • Queue
    • Graph
    • Tree
    • Maze
    • AVL
    • ์ด์ง„ํƒ์ƒ‰ํŠธ๋ฆฌ(Binary Search Tree)
    • DFS์™€ BFS
    • ๋‹ค์ต์ŠคํŠธ๋ผ ์•Œ๊ณ ๋ฆฌ์ฆ˜(Dijkstra's Algorithm)
    • Red-Black ํŠธ๋ฆฌ
    • A* ์•Œ๊ณ ๋ฆฌ์ฆ˜
    • Heap
    • Huffman Coding
    • Priority Queue
    • Bellman-Ford ์•Œ๊ณ ๋ฆฌ์ฆ˜
    • C++
      • Class
      • STL
        • STL pair
        • STL Container - Associate Container
        • STL Container - Sequence Container
        • STL Container - Container Adapter
  • JavaScript
    • JABASCRIPT BASIC
    • Shallow Copy vs Deep Copy
    • OBJECT MODEL
    • NODE
    • ๋™๊ธฐ ์ฒ˜๋ฆฌ vs ๋น„๋™๊ธฐ ์ฒ˜๋ฆฌ
    • AJAX
    • CALLBACK
    • PROMISE
    • DEFERRER
    • UNDERSCORE
    • WEBPACK
    • SCOPE
    • EXECUTION CONTEXT
    • Image Object
    • BFCache๋ž€?
    • history.scrollRestoration
    • Intersection Observer
    • JWT - JSON Web Token
    • HTML vs JSON
  • Vue.js
    • ํ™˜๊ฒฝ์„ค์ •
    • Vue.js๋ž€?
    • Vue Instance
    • Vue Component
    • Vue Router
    • HTTP ํ†ต์‹ 
    • Template
    • Single File Component
    • Vue Animation
    • Vuex
    • Djnago์™€ ์—ฐ๋™ํ•˜๊ธฐ
  • Backbone.js
    • Model
    • Collection
    • Sync
    • view
  • Node.js
    • Doit! - ๋…ธ๋“œ๋กœ ๋งŒ๋“ค ์ˆ˜ ์žˆ๋Š” ๋Œ€ํ‘œ์ ์ธ ์„œ๋ฒ„์™€ ์šฉ๋„
    • Doit! - ๋…ธ๋“œ์— ๋Œ€ํ•ด ์•Œ์•„๋ณด๊ณ  ๊ฐœ๋ฐœ ๋„๊ตฌ ์„ค์น˜ํ•˜๊ธฐ
    • Doit! - ๋…ธ๋“œ ๊ฐ„๋‹จํ•˜๊ฒŒ ์‚ดํŽด๋ณด๊ธฐ
    • Doit! - ๋…ธ๋“œ์˜ ์ž๋ฐ”์Šคํฌ๋ฆฝํŠธ์™€ ์นœํ•ด์ง€๊ธฐ
    • Doit! - ๋…ธ๋“œ์˜ ๊ธฐ๋ณธ ๊ธฐ๋Šฅ ์•Œ์•„๋ณด๊ธฐ
    • Doit! - ์›น ์„œ๋ฒ„ ๋งŒ๋“ค๊ธฐ
    • Doit! - ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค ์‚ฌ์šฉํ•˜๊ธฐ
    • Doit! - ์ต์Šคํ”„๋ ˆ์Šค ํ”„๋กœ์ ํŠธ๋ฅผ ๋ชจ๋“ˆํ™”ํ•˜๊ธฐ
    • Doit! - ๋ทฐ ํ…œํ”Œ๋ฆฟ ์ ์šฉํ•˜๊ธฐ
    • Doit! - ํŒจ์ŠคํฌํŠธ๋กœ ์‚ฌ์šฉ์ž ์ธ์ฆํ•˜๊ธฐ
    • Doit! - ์ฑ„ํŒ…์„œ๋ฒ„ ๋งŒ๋“ค๊ธฐ
    • Doit! - JSON-RPC ์„œ๋ฒ„ ๋งŒ๋“ค๊ธฐ
  • Python
    • Warning-Could not import the lzma module
    • Pandas
      • Pandas ์ž๋ฃŒ๊ตฌ์กฐ
      • Pandas ๋ฐ์ดํ„ฐ ์ž…์ถœ๋ ฅ
      • DataFrame Data ์‚ดํŽด๋ณด๊ธฐ
      • ์‹œ๊ฐํ™” ๋„๊ตฌ - Matplotlib
  • ML
    • ์ถ”์ฒœ ์‹œ์Šคํ…œ
      • Collaborative Filtering
      • Matrix Factorization
  • Django
    • Basic
      • ํ™˜๊ฒฝ์„ค์ •
      • About Django
      • Start Django Project
      • Secret Key ๊ด€๋ฆฌํ•˜๊ธฐ
      • Settings ๋ถ„๋ฆฌํ•˜๊ธฐ
      • Django App
      • Django View & URL (1)
      • Django Model
        • MySQL ์—ฐ๋™
      • Django Admin
      • Django View & URL (2)
      • Django Template
      • Django Template & View & URL
      • Django Static
      • Django form
    • Advanced
      • Django Generic View
      • Django Automated Testing
      • Django Extenstion Template
      • Django Model Package
      • Django OpenSSL setting
    • REST framework
      • Rest API
      • Serializers
      • ViewSet
    • Error
      • ํ™˜๊ฒฝ์„ค์ • zlib ์˜ค๋ฅ˜๋ฐœ์ƒ
      • ModuleNotFoundError
    • ํŒจํ‚ค์ง€
      • django-debug-toolbar
    • Vue.js ์—ฐ๋™ํ•˜๊ธฐ
  • Ruby
    • variable & input/output
    • ์กฐ๊ฑด๋ฌธ
    • ๋ฐ˜๋ณต๋ฌธ
    • Array & Hash
    • Method
    • Proc&Lamda
    • Class
  • Ruby on Rails
    • Scaffolding
    • Controller
    • Model
    • Model-M:N relation
    • Model Validation
    • ๋ฉ‹์‚ฌ 10์ฃผ์ฐจ ์ˆ˜์—…(Tip)
  • HTML/CSS
    • Udacity - Intro to HTML/CSS
    • Udacity - Responsive Web Design
    • Udacity - Responsive Images
    • HTML Basic
    • CSS Basic
    • HTML5 Sementic Tag
    • HTML ํ…์ŠคํŠธ ๊ด€๋ จ ํƒœ๊ทธ๋“ค
    • HTML5 ๋ฉ€ํ‹ฐ๋ฏธ๋””์–ด
    • HTML ํผ ๊ด€๋ จ ํƒœ๊ทธ๋“ค
    • ํ…์ŠคํŠธ ๊ด€๋ จ ์Šคํƒ€์ผ
    • ์ƒ‰์ƒ๊ณผ ๋ฐฐ๊ฒฝ์„ ์œ„ํ•œ ์Šคํƒ€์ผ
    • ๋ ˆ์ด์•„์›ƒ์„ ์œ„ํ•œ ์Šคํƒ€์ผ
    • CSS ํฌ์ง€์…”๋‹
    • ๋‹ค์žฌ๋‹ค๋Šฅํ•œ CSS3 ์„ ํƒ์ž
    • CSS์™€ ์• ๋‹ˆ๋ฉ”์ด์…˜
    • ๋ฐ˜์‘ํ˜• ์›น์ด๋ž€?
  • OS(์šด์˜์ฒด์ œ)
    • Linux
      • Daemon
      • Cron
      • ํ”„๋กœ์„ธ์Šค ๊ด€๋ จ ๋ช…๋ น์–ด
      • ํ…์ŠคํŠธ ํŒŒ์ผ ๋ช…๋ น์–ด
  • Network
    • ๋„คํŠธ์›Œํฌ ๊ธฐ๋ณธ ๊ฐœ๋…
    • ๋„คํŠธ์›Œํฌ ๊ธฐ๋ณธ ๊ทœ์น™
    • ๋ฌผ๋ฆฌ ๊ณ„์ธต
    • ๋ฐ์ดํ„ฐ ๋งํฌ ๊ณ„์ธต
    • ๋„คํŠธ์›Œํฌ ๊ณ„์ธต
    • ์ „์†ก ๊ณ„์ธต
    • ์‘์šฉ ๊ณ„์ธต
    • ๋„คํŠธ์›Œํฌ ์ „์ฒด ํ๋ฆ„
    • ๋ฌด์„  ๋žœ
  • IT ๊ธฐํƒ€์ง€์‹
    • NAS๋ž€
Powered by GitBook
On this page
  • JobLauncher
  • JobExplorer
  • Job
  • Job Runner
  • JobBuilderFactory
  • JobBuilder
  • JobInstance
  • JobExecution
  • JobParameters
  • ExecutionContext

Was this helpful?

  1. Spring
  2. Spring Batch

Job

PreviousBatch Job FlowNextStep

Last updated 3 years ago

Was this helpful?

JobLauncher

Job, JobParameters์™€ ํ•จ๊ป˜ ๋ฐฐ์น˜๋ฅผ ์‹คํ–‰ํ•˜๋Š” ์ธํ„ฐํŽ˜์ด์Šค๋กœ, ๋ฉ”์„œ๋“œ๋Š” run ํ•œ๊ฐœ์ด๋‹ค.

  • Job์„ ์‹คํ–‰ํ•˜๋Š” ์—ญํ• 

  • Job.execute() ํ˜ธ์ถœ

  • Job์žฌ์‹คํ–‰ ๊ฐ€๋Šฅ ์—ฌ๋ถ€ ๊ฒ€์ฆ

  • Job ์‹คํ–‰ ๋ฐฉ๋ฒ•(ํ˜„์žฌ ์Šค๋ ˆ๋“œ์—์„œ ์ˆ˜ํ–‰ํ• ์ง€, ์Šค๋ ˆ๋“œ ํ’€์„ ํ†ตํ•ด ์‹คํ–‰ํ• ์ง€ ๋“ฑ)

  • ํŒŒ๋ผ๋ฏธํ„ฐ ์œ ํšจ์„ฑ ๊ฒ€์ฆ

  • ์Šคํ”„๋ง๋ถ€ํŠธ๋ฅผ ์‚ฌ์šฉํ•˜๋ฉด ์Šคํ”„๋ง ๋ถ€ํŠธ๊ฐ€ ์ฆ‰์‹œ Job๋ฅผ ์‹œ์ž‘ํ•˜๋Š” ๊ธฐ๋Šฅ์„ ์ œ๊ณตํ•ด, ์ผ๋ฐ˜์ ์œผ๋กœ๋Š” ์ง์ ‘ ๋‹ค๋ฃฐ ํ•„์š” ์—†์Œ

public interface JobLauncher {
  	// Job๊ณผ JobParameters๋ฅผ ๋งค๊ฐœ๋ณ€์ˆ˜๋กœ ๋ฐ›์•„ JobExecution ๋ฐ˜ํ™˜
    // ๋งค๊ฐœ๋ณ€์ˆ˜๊ฐ€ ๋™์ผํ•˜๋ฉฐ, ์ด์ „ JobExecution๊ฐ€ ์ค‘๋‹จ๋œ ์ ์ด ์žˆ์œผ๋ฉด ๋™์ผํ•œ JobExecution ๋ฐ˜ํ™˜
    JobExecution run(Job var1, JobParameters var2) throws JobExecutionAlreadyRunningException, JobRestartException, JobInstanceAlreadyCompleteException, JobParametersInvalidException;
}

JobExplorer

  • org.springframework.batch.core.explore.JobExplorer

  • JobRepository ์— ์žˆ๋Š” ์ด๋ ฅ ๋ฐ์ดํ„ฐ๋‚˜ ์ตœ์‹  ๋ฐ์ดํ„ฐ์— ์ ‘๊ทผํ•˜๋Š” ์‹œ์ž‘์ ์ด๋‹ค.

  • JobExplorer๋Š” ์žก ์ •๋ณด๋ฅผ ์–ป๊ธฐ์œ„ํ•ด ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค์— ์ง์ ‘ ์ ‘๊ทผํ•œ๋‹ค.

๋ฉ”์„œ๋“œ
์„ค๋ช…

Set<JobExecution> findRunningJobExecutions(@Nullable String jobName)

์ข…๋ฃŒ ์‹œ๊ฐ„์ด ์กด์žฌํ•˜์ง€ ์•Š๋Š” ๋ชจ๋“  JobExecution ๋ฐ˜ํ™˜

List<JobInstance> findJobInstancesByJobName(String jobName, int start, int count)

์ „๋‹ฌ๋ฐ›์€ ์ด๋ฆ„์„ ๊ฐ€์ง„ JobInstance ๋ชฉ๋ก ๋ฐ˜ํ™˜(ํŽ˜์ด์ง• ์ฒ˜๋ฆฌ)

JobExecution getJobExecution(@Nullable Long executionId)

์ „๋‹ฌ๋ฐ›์€ ID๋ฅผ ๊ฐ€์ง„ JobExecution ๋ฐ˜ํ™˜ ์กด์žฌํ•˜์ง€ ์•Š๋Š”๋‹ค๋ฉด null ๋ฐ˜ํ™˜

List<JobExecution> getJobExecutions(JobInstance jobInstance)

์ „๋‹ฌ๋ฐ›์€ JobInstance ์™€ ์—ฐ๊ด€๋œ ๋ชจ๋“  JobExecution ๋ชฉ๋ก ๋ฐ˜ํ™˜

JobInstance getJobInstance(@Nullable Long instanceId)

์ „๋‹ฌ๋ฐ›์€ ID๋ฅผ ๊ฐ€์ง„ JobInstance ๋ฐ˜ํ™˜ ์กด์žฌํ•˜์ง€ ์•Š์œผ๋ฉด null ๋ฐ˜ํ™˜

List<JobInstance> getJobInstances(String jobName, int start, int count)

์ „๋‹ฌ๋ฐ›์€ ์ด๋ฆ„์„ ๊ฐ€์ง„ JobInstance ๋ชฉ๋ก ๋ฐ˜ํ™˜(ํŽ˜์ด์ง• ์ฒ˜๋ฆฌ)

int getJobInstanceCount(@Nullable String jobName)

์ „๋‹ฌ๋ฐ›์€ ์žก ์ด๋ฆ„์œผ๋กœ ์ƒ์„ฑ๋œ JobInstance ์ˆ˜

List<String> getJobNames()

JobRepository์— ์ €์žฅ๋ผ ์žˆ๋Š” ๊ณ ์œ ํ•œ ๋ชจ๋“  ์žก ์ด๋ฆ„ ์•ŒํŒŒ๋ฒณ ์ˆœ ๋ฆฌ์ŠคํŠธ

StepExecution getStepExecution(@Nullable Long jobExecutionId, @Nullable Long stepExecutionId)

์ „๋‹ฌ๋ฐ›์€ jobExecutionId์™€ stepExecutionId๋ฅผ ๊ฐ€์ง„ StepExecution ๋ฐ˜ํ™˜

@AllArgsConstructor
public class ExploringTasklet implements Tasklet {

    private JobExplorer jobExplorer;

    @Override
    public RepeatStatus execute(StepContribution contribution, ChunkContext chunkContext) throws Exception {

        // ํ˜„์žฌ Job์ด๋ฆ„ ์กฐํšŒ
        String jobName = chunkContext.getStepContext().getJobName();

        // ์ด๋•Œ๊ฐ€์ง€ ์‹คํ–‰๋œ ๋ชจ๋“  JobInstance ์กฐํšŒ
        List<JobInstance> instaces = jobExplorer.getJobInstances(jobName, 0, Integer.MAX_VALUE);

        System.out.println(String.format("%d job instances for the job %s", instaces.size(), jobName));

        System.out.println("===========================");

        for (JobInstance instance : instaces) {

            // JobInstance์™€ ๊ด€๋ จ๋œ JobExecution
            List<JobExecution> jobExecutions = this.jobExplorer.getJobExecutions(instance);
            System.out.println(String.format("Instance %d had %d executions", instance.getInstanceId(), jobExecutions.size()));

            for (JobExecution jobExecution : jobExecutions) {
                System.out.println(String.format("Execution %d resulted in ExitStatus %s", jobExecution.getId(), jobExecution.getExitStatus()));
            }
        }

        return RepeatStatus.FINISHED;

    }
}
@EnableBatchProcessing
@Configuration
public class DemoTaskletConfiguration {
    @Autowired
    private JobBuilderFactory jobBuilderFactory;

    @Autowired
    private StepBuilderFactory stepBuilderFactory;

    @Autowired
    private JobExplorer jobExplorer;

    @Bean
    public Tasklet explorerTasklet() {
        return new ExploringTasklet(this.jobExplorer);
    }

    @Bean
    public Step explorerStep() {
        return this.stepBuilderFactory.get("explorerStep")
                .tasklet(explorerTasklet())
                .build();
    }

    @Bean
    public Job explorerJob() {
        return this.jobBuilderFactory.get("explorerJob")
                .start(explorerStep())
                .build();
    }
}

Job

  • ๋ฐฐ์น˜ ์ฒ˜๋ฆฌ ๊ณผ์ •์„ ํ•˜๋‚˜์˜ ๋‹จ์œ„๋กœ ํ‘œํ˜„ํ•œ ๊ฐ์ฒด๋กœ ์ „์ฒด ๋ฐฐ์น˜ ์ฒ˜๋ฆฌ์— ์žˆ์–ด ์ตœ์ƒ๋‹จ ๊ณ„์ธต

  • Job์€ ์ค‘๋‹จ์ด๋‚˜ ์ƒํ˜ธ์ž‘์šฉ ์—†์ด ์ฒ˜์Œ๋ถ€ํ„ฐ ๋๊นŒ์ง€ ์‹คํ–‰๋˜๋Š” ์ฒ˜๋ฆฌ

  • Spring Batch์—์„œ Job ๊ฐ์ฒด๋Š” ์—ฌ๋Ÿฌ step ์ธ์Šคํ„ด์Šค๋ฅผ ํฌํ•จํ•˜๋Š” ์ปจํ…Œ์ด๋„ˆ

Job Runner

Job์˜ ์‹คํ–‰์€ job runner์—์„œ ์‹œ์ž‘๋œ๋‹ค. job runner๋Š” ์žก ์ด๋ฆ„๊ณผ ํŒŒ๋ผ๋ฏธํ„ฐ๋กœ ํ•ด๋‹น ์žก์„ ์‹คํ–‰์‹œํ‚ค๋Š” ์—ญํ• ์„ ํ•œ๋‹ค.

  • CommandLineJobRunner : ์Šคํฌ๋ฆฝํŠธ๋ฅผ ์ด์šฉํ•˜๊ฑฐ๋‚˜ ๋ช…๋ นํ–‰์—์„œ ์ง์ ‘ Job์„ ์ˆ˜ํ–‰ํ•  ๋•Œ ์‚ฌ์šฉ

    • ์Šคํ”„๋ง ๋ฐฐ์น˜ ์ œ๊ณต

    • ์Šคํ”„๋ง์„ ๋ถ€ํŠธ์ŠคํŠธ๋žฉํ•˜๋ฉฐ ์ „๋‹ฌ๋ฐ›์€ ํŒŒ๋ผ๋ฏธํ„ฐ๋ฅผ ์‚ฌ์šฉํ•ด ์š”์ฒญ๋œ ์žก ์‹คํ–‰

    • org.springframework.batch.core.launch.support.CommandLineJobRunner

  • JobRegistryBackgroundJobRunner : ์Šคํ”„๋ง์„ ๋ถ€ํŠธ์ŠคํŠธ๋žฉ์—์„œ ๊ธฐ๋™ํ•œ ์ž๋ฐ” ํ”„๋กœ์„ธ์Šค ๋‚ด์—์„œ Quartz๋‚˜ Jmx ํ›„ํฌ์™€ ๊ฐ™์€ ์Šค์ผ€์ค„๋Ÿฌ๋ฅผ ์‚ฌ์šฉํ•ด ์žก์„ ์‹คํ–‰ํ•˜๋ฉด JobReistry๋ฅผ ์ƒ์„ฑํ•˜๊ฒŒ ๋œ๋‹ค. JobRegistryBackgroundJobRunner๋Š” JobRegistry๋ฅผ ์ƒ์„ฑํ•˜๋Š” ๋ฐ ์‚ฌ์šฉ

    • ์Šคํ”„๋ง ๋ฐฐ์น˜ ์ œ๊ณต

    • org.springframework.batch.core.launch.support.JobRegistryBackgroundJobRunner

  • JobLauncherCommandLineRunner

    • ์Šคํ”„๋ง ๋ถ€ํŠธ ์ œ๊ณต

    • ๋ณ„๋„์˜ ๊ตฌ์„ฑ์ด ์—†๋‹ค๋ฉด ๊ธฐ๋ณธ์ ์œผ๋กœ ApplicationContext์— ์ •์˜๋œ Job ํƒ€์ž…์˜ ๋ชจ๋“  ๋นˆ์„ ๊ธฐ๋™์‹œ ์‹คํ–‰

Job Runner๋Š” ํ”„๋ ˆ์ž„์›Œํฌ๊ฐ€ ์ œ๊ณตํ•˜๋Š” ํ‘œ์ค€ ๋ชจ๋“ˆ์ด ์•„๋‹ˆ๋ฉฐ, ๊ฐ ์‹œ๋‚˜๋ฆฌ์˜ค๋งˆ๋‹ค ๋‹ค๋ฅธ ๊ตฌํ˜„์ฒด๊ฐ€ ํ•„์š”ํ•˜๊ธฐ ๋•Œ๋ฌธ์— ํ”„๋ ˆ์ž„์›Œํฌ๊ฐ€ JobRunner ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ๋ณ„๋„๋กœ ์ œ๊ณตํ•˜์ง€ ์•Š๋Š”๋‹ค.

์‹ค์ œ๋กœ ์žก ๋Ÿฌ๋„ˆ๊ฐ€ ์•„๋‹Œ org.springframework.batch.core.launch.JobLauncher ์ธํ„ฐํŽ˜์ด์Šค๊ฐ€ ํ”„๋ ˆ์ž„์›Œํฌ ์‹คํ–‰์‹œ์ž‘์ ์ด๋ฉฐ, ์Šคํ”„๋ง ๋ฐฐ์น˜๋Š” SimpleJobLauncher ๋งŒ ์ œ๊ณตํ•œ๋‹ค.

JobBuilderFactory

Job ๊ฐ์ฒด๋ฅผ ๋งŒ๋“œ๋Š” ๋นŒ๋”๋Š” ์—ฌ๋Ÿฌ๊ฐœ๊ฐ€ ์žˆ๋‹ค. JobBuilderFactory๋Š” ์—ฌ๋Ÿฌ ๋นŒ๋”๋ฅผ ํ†ตํ•ฉ ์ฒ˜๋ฆฌํ•˜๋Š” ๊ณต์žฅ์ด๋ฉฐ, ์›ํ•˜๋Š” Job์„ ์†์‰ฝ๊ฒŒ ๋งŒ๋“ค ์ˆ˜ ์žˆ๋‹ค.

package org.springframework.batch.core.configuration.annotation;

import org.springframework.batch.core.job.builder.JobBuilder;
import org.springframework.batch.core.repository.JobRepository;

public class JobBuilderFactory {
    private JobRepository jobRepository;

    public JobBuilderFactory(JobRepository jobRepository) {
        this.jobRepository = jobRepository;
    }

  	// JobBuilder ์ƒ์„ฑ ๋ฐ˜ํ™˜
    public JobBuilder get(String name) {
        JobBuilder builder = (JobBuilder)(new JobBuilder(name)).repository(this.jobRepository);
        return builder;
    }
}

JobBuilderFactory์˜ get ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•  ๋•Œ๋งˆ๋‹ค ์ƒˆ๋กœ์šด JobBuilder๋ฅผ ์ƒ์„ฑ๋˜๋ฉฐ, ์ƒˆ๋กœ์šด JobBuilder๋ฅผ ์ƒ์„ฑํ•  ๋•Œ๋งˆ๋‹ค JobBuilderFactory๊ฐ€ ์ƒ์„ฑ๋  ๋•Œ ์ฃผ์ž…๋ฐ›์€ JobRepository๋ฅผ ์„ค์ •ํ•˜๋Š” ๊ฒƒ์„ ๋ณผ ์ˆ˜ ์žˆ๋‹ค. ํ•ด๋‹น JobBuilderFactory์—์„œ ์ƒ์„ฑ๋˜๋Š” ๋ชจ๋“  JobBuilder๊ฐ€ ๋™์ผํ•œ JobRepository๋ฅผ ์‚ฌ์šฉํ•˜๋Š” ๊ฒƒ์ด๋‹ค.

์ฆ‰, JobBuilderFactory๋Š” JobBuilder๋ฅผ ์ƒ์„ฑํ•˜๋Š” ์—ญํ• ๋งŒ ์ˆ˜ํ–‰ํ•œ๋‹ค.

JobBuilder

public class JobBuilder extends JobBuilderHelper<JobBuilder> {
    public JobBuilder(String name) {
        super(name);
    }
		
    // 1. step์„ ์ถ”๊ฐ€ํ•˜์—ฌ ๊ฐ€์žฅ ๊ธฐ๋ณธ์ด ๋˜๋Š” SimpleJobBuilder ์ƒ์„ฑ
    public SimpleJobBuilder start(Step step) {
        return (new SimpleJobBuilder(this)).start(step);
    }

  	// 2. Flow๋ฅผ ์‹คํ–‰ํ•  JobFlowBuilder ์ƒ์„ฑ
    public JobFlowBuilder start(Flow flow) {
        return (new FlowJobBuilder(this)).start(flow);
    }

  	// 3. Step์„ ์‹คํ–‰ํ•  JobFlowBuilder ์ƒ์„ฑ
    public JobFlowBuilder flow(Step step) {
        return (new FlowJobBuilder(this)).start(step);
    }
}

JobBuilder๋Š” ์ง์ ‘์ ์œผ๋กœ Job์„ ์ƒ์„ฑํ•˜๋Š” ๊ฒƒ์ด ์•„๋‹ˆ๋ผ ๋ณ„๋„์˜ ๊ตฌ์ฒด์ ์ธ ๋นŒ๋”๋ฅผ ์ƒ์„ฑํ•ด ๋ฐ˜ํ™˜ํ•œ๋‹ค. ์™œ๋ƒํ•˜๋ฉด ๊ฒฝ์šฐ์— ๋”ฐ๋ผ Job ์ƒ์„ฑ ๋ฐฉ๋ฒ•์ด ๋ชจ๋‘ ๋‹ค๋ฅด๊ธฐ ๋–„๋ฌธ์— ๋ณ„๋„์˜ ๊ตฌ์ฒด์ ์ธ ๋นŒ๋”๋ฅผ ๊ตฌํ˜„ํ•˜๊ณ , ์ด๋ฅผ ํ†ตํ•ด Job ์ƒ์„ฑ์ด ์ด๋ฃจ์–ด์ง€๊ฒŒ ํ•œ๋‹ค.

์ค‘๊ฐ„์— ๋นŒ๋”๋ฅผ ํ•œ๋ฒˆ ๋” ๋ฐ˜ํ™˜ํ•˜์—ฌ ์‚ฌ์šฉํ•ด์•ผํ•˜์ง€๋งŒ, ๋ฉ”์„œ๋“œ ์ฒด์ธ ๋ฐฉ์‹์„ ํ™œ์šฉํ•˜๋ฉด ์†์‰ฝ๊ฒŒ ์ฒ˜๋ฆฌํ•  ์ˆ˜ ์žˆ๋‹ค. Job์€ Step / Flow ์ธ์Šคํ„ด์Šค์˜ ์ปจํ…Œ์ด๋„ˆ ์—ญํ• ์„ ํ•˜๊ธฐ ๋•Œ๋ฌธ์— ์ƒ์„ฑ ์ด์ „์— ์ธ์Šคํ„ด์Šค๋ฅผ ์ „๋‹ฌ ๋ฐ›๋Š”๋‹ค.

  • SimpleJobBuilder๋กœ Job ์ƒ์„ฑํ•˜๊ธฐ

		@Autowired
    private JobBuilderFactory jobBuilderFactory;

    @Bean
    public Job simpleJob(){
        return jobBuilderFactory.get("simpleJob")	// "simpleJob" ์ด๋ฆ„์„ ๊ฐ€์ง„ JobBuilder instance ๋ฐ˜ํ™˜
                                .start(simpleStep()) // step์„ ์ƒ์„ฑํ•˜๋Š” ๋ฉ”์„œ๋“œ๋กœ ์ƒ์„ฑ๋˜๋Š” SimpleJobBuilder
                                .build();						// build ๋ฉ”์„œ๋“œ ํ˜ธ์ถœ๋กœ Job ๋ฐ˜ํ™˜
    }

JobInstance

๋ฐฐ์น˜ ์ฒ˜๋ฆฌ์—์„œ Job์ด ์‹คํ–‰๋  ๋•Œ ํ•˜๋‚˜์˜ Job ์‹คํ–‰๋‹จ์œ„์ด๋‹ค. ์˜ˆ๋ฅผ๋“ค์–ด ํ•˜๋ฃจ์— ํ•œ๋ฒˆ ๋ฐฐ์น˜ Job์ด ์‹คํ–‰๋œ๋‹ค๋ฉด, ์–ด์ œ ์˜ค๋Š˜ ๊ฐ๊ฐ ์‹คํ–‰๋œ Job์„ JobInstance๋ผ ๋ถ€๋ฅธ๋‹ค.

JobInstance๋Š” ์ด๋ฆ„๊ณผ ๋…ผ๋ฆฌ์  ์‹คํ–‰์„ ์œ„ํ•ด ์ œ๊ณต๋˜๋Š” ๊ณ ์œ ํ•œ ์‹๋ณ„ ํŒŒ๋ผ๋ฏธํ„ฐ ๋ชจ์Œ์œผ๋กœ ์œ ์ผํ•˜๊ฒŒ ์กด์žฌํ•œ๋‹ค.

  • ExampleGenerator ์ด๋ฆ„์˜ Job์ด ๋‹ค๋ฅธ ํŒŒ๋ผ๋ฏธํ„ฐ๋กœ ์‹คํ–‰๋  ๋•Œ๋งˆ๋‹ค ์ƒˆ๋กœ์šด JobInstace๊ฐ€ ์ƒ์„ฑ๋œ๋‹ค.

  • BATCH_JOB_INSTANCE ํ…Œ์ด๋ธ”๋กœ ๊ด€๋ฆฌ

  • BATCH_JOB_EXECUTION_PARAMS ์—์„œ ์‹ค์ œ ์‹๋ณ„ ํŒŒ๋ผ๋ฏธํ„ฐ ๊ด€๋ฆฌ

Job์„ ์ฒ˜์Œ ์‹คํ–‰ํ•˜๋ฉด ์ƒˆ๋กœ์šด JobInstance๋ฅผ ์–ป๋Š”๋‹ค. ํ•˜์ง€๋งŒ ์‹คํ–‰์— ์‹คํŒจํ•œ ์ดํ›„ ๋‹ค์‹œ ์‹คํ–‰ํ•˜๋ฉด, ์—ฌ์ „ํžˆ ๋™์ผํ•œ ๋…ผ๋ฆฌ์  ์‹คํ–‰(ํŒŒ๋ผ๋ฏธํ„ฐ ๋™์ผ)์ด๋ฏ€๋กœ ์ƒˆ๋กœ์šด JobInstance๋ฅผ ์–ป์ง€ ๋ชปํ•˜๋ฉฐ, ์‹ค์ œ ์‹คํ–‰์„ ์ถ”์ ํ•˜๊ธฐ ์œ„ํ•œ ์ƒˆ๋กœ์šด JobExecution์„ ์–ป์„ ๊ฒƒ์ด๋‹ค.

์ฆ‰, JobInstance๋Š” ์‹คํŒจํ•œ JobExecution๊ณผ ์ƒˆ๋กœ ์ˆ˜ํ–‰ํ•œ JobExecution๊ณผ ๊ฐ™์ด JobExecution์„ ์—ฌ๋Ÿฌ ๊ฐœ ๊ฐ€์งˆ ์ˆ˜ ์žˆ๋‹ค.

JobExecution

JobInstance์— ๋Œ€ํ•œ ํ•œ ๋ฒˆ์˜ ์‹คํ–‰(์‹ค์ œ ์‹œ๋„)์„ ๋‚˜ํƒ€๋‚ด๋Š” ๊ฐ์ฒด์ด๋‹ค. JobExecution์€ Job ์‹คํ–‰์— ๋Œ€ํ•œ ์ •๋ณด๋ฅผ ๋‹ด๊ณ  ์žˆ๋Š” ๋„๋ฉ”์ธ ๊ฐ์ฒด์ด๋ฉฐ, JobInstance, ๋ฐฐ์น˜ ์‹คํ–‰ ์ƒํƒœ, ์‹œ์ž‘ ์‹œ๊ฐ„, ๋๋‚œ ์‹œ๊ฐ„, ์˜ค๋ฅ˜ ๋ฉ”์„ธ์ง€ ๋“ฑ์˜ ์ •๋ณด๋ฅผ ๋‹ด๊ณ  ์žˆ๋‹ค.

  • JobExecution์€ Job์ด ๊ตฌ๋™๋  ๋•Œ ๋งˆ๋‹ค ๋งค๋ฒˆ ์ƒˆ๋กœ์šด JobExecution์„ ์–ป๊ฒŒ๋œ๋‹ค.

  • BATCH_JOB_EXECUTION ํ…Œ์ด๋ธ”์— ๊ฐ ๋ ˆ์ฝ”๋“œ๋กœ ์ €์žฅ

  • BATCH_JOB_EXECUTION_CONTEXT ํ…Œ์ด๋ธ”์— ์ƒํƒœ ๊ฐ’ ์ €์žฅ

public class JobExecution extends Entity {
    private final JobParameters jobParameters; //Job ์‹คํ–‰์— ํ•„์š”ํ•œ ๋งค๊ฐœ ๋ณ€์ˆ˜ ๋ฐ์ดํ„ฐ
    private JobInstance jobInstance; // Job ์‹คํ–‰์˜ ๋‹จ์œ„๊ฐ€ ๋˜๋Š” ๊ฐ์ฒด
    private volatile Collection<StepExecution> stepExecutions; // StepExecution์„ ์—ฌ๋Ÿฌ๊ฐœ ๊ฐ€์งˆ ์ˆ˜ ์žˆ๋Š” Collection ํƒ€์ž…
    private volatile BatchStatus status; // Job์˜ ์‹คํ–‰ ์ƒํƒœ(COMPLETED, STARTING, STARTED ...)
    private volatile Date startTime; // Job์ด ์‹คํ–‰๋œ ์‹œ๊ฐ„(null์€ ์‹œ์ž‘ํ•˜์ง€ ์•Š์€ ๊ฒƒ)
    private volatile Date createTime; // JobExecution์ด ์ƒ์„ฑ๋œ ์‹œ๊ฐ„
    private volatile Date endTime; // JobExecution ์ข…๋ฃŒ ์‹œ๊ฐ„
    private volatile Date lastUpdated; // ๋งˆ์ง€๋ง‰ ์ˆ˜์ •์‹œ๊ฐ„
    private volatile ExitStatus exitStatus; // Job ์‹คํ–‰ ๊ฒฐ๊ณผ์— ๋Œ€ํ•œ ์ƒํƒœ๊ฐ’(UNKOWN, EXECUTING, COMPLETE, ...)
    private volatile ExecutionContext executionContext;// Job ์‹คํ–‰ ์‚ฌ์ด์— ์œ ์ง€ํ•ด์•ผํ•˜๋Š” ์‚ฌ์šฉ์ž ๋ฐ์ดํ„ฐ
    private transient volatile List<Throwable> failureExceptions; // Job ์‹คํ–‰ ์ค‘ ๋ฐœ์ƒํ•œ ์˜ˆ์™ธ 
    private final String jobConfigurationName; // Job ์„ค์ • ์ด๋ฆ„
  	...
}

JobParameters

Job์ด ์‹คํ–‰๋  ๋•Œ ํ•„์š”ํ•œ ํŒŒ๋ผ๋ฏธํ„ฐ๋“ค์„ Map ํƒ€์ž…์œผ๋กœ ์ง€์ •ํ•˜๋Š” ๊ฐ์ฒด๋กœ JobInstance(1:1 ๊ด€๊ณ„)๋ฅผ ๊ตฌ๋ถ„ํ•˜๋Š” ๊ธฐ์ค€์ด ๋˜๊ธฐ๋„ ํ•œ๋‹ค.

ํ•˜๋‚˜์˜ Job์„ ์ƒ์„ฑํ•  ๋•Œ ์‹œ์ž‘ ์‹œ๊ฐ„ ๋“ฑ์˜ ์ •๋ณด๋ฅผ ํŒŒ๋ผ๋ฏธํ„ฐ๋กœ ํ•ด์„œ ํ•˜๋‚˜์˜ JobInstance๋ฅผ ์ƒ์„ฑํ•œ๋‹ค. ์ฆ‰, 1:1 ๊ด€๊ณ„์ด๋‹ค.

  • ํŒŒ๋ผ๋ฏธํ„ฐ๋Š” key=value๋กœ ์ด๋ฃจ์–ด์ ธ์žˆ๋‹ค.

  • JobParameters๋Š” Map<String,JobParameter>์˜ wrapper์— ๋ถˆ๊ณผํ•˜๋‹ค.

    public class JobParameters implements Serializable {
    private final Map<String,JobParameter> parameters;
    ...
    }
  • ํŒŒ๋ผ๋ฏธํ„ฐ ํƒ€์ž…์€ String, Double, Date ํƒ€์ž…๋„ ๊ธฐ๋ณธ์ ์œผ๋กœ ์ œ๊ณตํ•œ๋‹ค.

  • ํŒŒ๋ผ๋ฏธํ„ฐ ์ด๋ฆ„ ๋’ค์— ๊ด„ํ˜ธ๋ฅผ ์“ฐ๊ณ  ํ•ด๋‹น ํŒŒ๋ผ๋ฏธํ„ฐ ํƒ€์ž…์„ ๋ช…์‹œํ•ด ์ง€์ •ํ•  ์ˆ˜ ์žˆ๋‹ค.

    • ์ด๋•Œ, ํŒŒ๋ผ๋ฏธํ„ฐ ํƒ€์ž…์˜ ์ด๋ฆ„์€ ๋ชจ๋‘ ์†Œ๋ฌธ์ž์—ฌ์•ผํ•œ๋‹ค.

      executionDate(date)=2021/11/27
    • BATCH_JOB_EXECUTION_PARAMS

  • ์‹๋ณ„์— ์‚ฌ์šฉํ•˜์ง€ ์•Š๋Š” ํŒŒ๋ผ๋ฏธํ„ฐ๋„ ์žˆ์„ ์ˆ˜ ์žˆ๋‹ค. ์‹๋ณ„์— ์‚ฌ์šฉํ•˜๊ณ  ์‹ถ์ง€ ์•Š๋Š” ํŒŒ๋ผ๋ฏธํ„ฐ๋Š” ์•ž์— -๋ฅผ ๋‹ค์Œ๊ณผ ๊ฐ™์ด ๋ถ™์—ฌ์ฃผ๋ฉด ๋œ๋‹ค.

    executionDate(date)=2021/11/27 -filename=test
    Caused by: org.springframework.batch.core.repository.JobInstanceAlreadyCompleteException: A job instance already exists and is complete for parameters={executionDate=1637938800000, filename=test}.  If you want to run this job again, change the parameters.
    	at org.springframework.batch.core.repository.support.SimpleJobRepository.createJobExecution(SimpleJobRepository.java:139) ~[spring-batch-core-4.3.3.jar:4.3.3]
    	at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) ~[na:na]
    	at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) ~[na:na]
    	at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) ~[na:na]
    	at java.base/java.lang.reflect.Method.invoke(Method.java:566) ~[na:na]
    	at org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:344) ~[spring-aop-5.3.12.jar:5.3.12]
    	at org.springframework.aop.framework.ReflectiveMethodInvocation.invokeJoinpoint(ReflectiveMethodInvocation.java:198) ~[spring-aop-5.3.12.jar:5.3.12]
    	at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:163) ~[spring-aop-5.3.12.jar:5.3.12]

    ๋‹ค์Œ๊ณผ ๊ฐ™์ด -๋ฅผ ๋ถ™์ธ ํŒŒ๋ผ๋ฏธํ„ฐ๋Š” ์‹๋ณ„ ํŒŒ๋ผ๋ฏธํ„ฐ๋กœ ์‚ฌ์šฉํ•˜์ง€ ์•Š์•„, ๊ธฐ์กด์— ์ด๋ฏธ ์ˆ˜ํ–‰๋œ Job์œผ๋กœ ์‹คํŒจํ•œ ๊ฒƒ์„ ์•Œ ์ˆ˜ ์žˆ๋‹ค.

ํŒŒ๋ผ๋ฏธํ„ฐ์— ์ ‘๊ทผํ•˜๋Š” ๋ฐฉ๋ฒ•

  1. ChunkContext ์ธ์Šคํ„ด์Šค

    • ์‹คํ–‰ ์‹œ์ ์˜ Job ์ƒํƒœ ์ œ๊ณต

    • tasklet ๋‚ด์—์„œ ์ฒ˜๋ฆฌ์ค‘์ธ chunk์™€ ๊ด€๋ จ๋œ ์ •๋ณด(์Šคํƒญ ๋ฐ ์žก๊ณผ ๊ด€๋ จ๋œ ์ •๋ณด ํฌํ•จ) ์ œ๊ณต

    • JobParametes๊ฐ€ ํฌํ•จ๋œ StepContext์ฐธ์กฐ๊ฐ€ ์žˆ์Œ

      public class ChunkContext extends AttributeAccessorSupport {
      
      	private final StepContext stepContext;
      	...
      }
      public class StepContext extends SynchronizedAttributeAccessor {
      
      	private StepExecution stepExecution;
      
      	private Map<String, Set<Runnable>> callbacks = new HashMap<>();
      
      	private BatchPropertyContext propertyContext = null;
      
      	...
          
      	/**
      	 * @return a map containing the items from the {@link JobParameters}
      	 */
      	public Map<String, Object> getJobParameters() {
      		Map<String, Object> result = new HashMap<>();
      		for (Entry<String, JobParameter> entry : stepExecution.getJobParameters().getParameters().entrySet()) 		{
      			result.put(entry.getKey(), entry.getValue().getValue());
      		}
      		return Collections.unmodifiableMap(result);
      	}
        ...
      }
    • ์˜ˆ์ œ

          @Bean Step step2(){
              return this.stepBuilderFactory.get("step2")
                      .tasklet(helloWorldTasklet())
                      .build();
          }
      
          @Bean
          public Tasklet helloWorldTasklet() {
              return ((contribution, chunkContext) -> {
                  String name = (String) chunkContext.getStepContext()
                                      .getJobParameters() // Map<String, Object>์ด๋ฏ€๋กœ ํƒ€์ž… ์บ์ŠคํŒ… ํ•„์š”
                                      .get("name");
      
                  System.out.println(String.format("Hello, %s", name));
                  return RepeatStatus.FINISHED;
              });
          }
  2. Late Binding : ์Šคํ”„๋ง ๊ตฌ์„ฑ์„ ์‚ฌ์šฉํ•ด ์ฃผ์ž…ํ•˜๋Š” ๋ฐฉ๋ฒ•์œผ๋กœ, JobParameters๋Š” ๋ณ€๊ฒฝํ•  ์ˆ˜ ์—†์œผ๋ฏ€๋กœ ๋ถ€ํŠธ์ŠคํŠธ๋žฉ์‹œ ๋ฐ”์ธ๋”ฉํ•˜๋Š” ๊ฒƒ์ด ์ข‹๋‹ค.

    		@StepScope
    		@Bean
        public Tasklet lateBindingParamTasklet(@Value("#{jobParameters['name']}") String name) {
            return ((contribution, chunkContext) -> {
                System.out.println(String.format("Hello, %s", name));
                return RepeatStatus.FINISHED;
            });
        }

ํŒŒ๋ผ๋ฏธํ„ฐ ์œ ํšจ์„ฑ ๊ฒ€์ฆ

  • org.springframework.batch.core.JobParametersValidator

    public interface JobParametersValidator {
    
    	/**
    	 * Check the parameters meet whatever requirements are appropriate, and
    	 * throw an exception if not.
    	 * 
    	 * @param parameters some {@link JobParameters} (can be {@code null})
    	 * @throws JobParametersInvalidException if the parameters are invalid
    	 */
    	void validate(@Nullable JobParameters parameters) throws JobParametersInvalidException;
    
    }

JobParametersValidator ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ๊ตฌํ˜„ํ•˜๊ณ , ํ•ด๋‹น ๊ตฌํ˜„์ฒด๋ฅผ ์žก ๋‚ด์— ๊ตฌ์„ฑํ•ด ํŒŒ๋ผ๋ฏธํ„ฐ ์œ ํšจ์„ฑ ๊ฒ€์ฆ์„ ํ•  ์ˆ˜ ์žˆ๋‹ค.

์Šคํ”„๋ง์€ ํ•„์ˆ˜ ํŒŒ๋ผ๋ฏธํ„ฐ๊ฐ€ ๋ˆ„๋ฝ์—†์ด ์ „๋‹ฌ๋๋Š”์ง€ ํ™•์ธํ•˜๋Š” DefaultJobParametersValidator๋ฅผ ๊ธฐ๋ณธ์ ์œผ๋กœ ์ œ๊ณตํ•ด์ค€๋‹ค.

  • DefaultJobParametersValidator ์‚ฌ์šฉํ•œ ์œ ํšจ์„ฑ ๊ฒ€์ฆ : ํŒŒ๋ผ๋ฏธํ„ฐ ์กด์žฌ ์—ฌ๋ถ€๋ฅผ ์ œ์™ธํ•œ ๋‹ค๋ฅธ ์œ ํšจ์„ฑ ๊ฒ€์ฆ์€ ์ˆ˜ํ–‰ํ•˜์ง€ ์•Š์Œ.

        @Bean
        public JobParametersValidator validator() {
            DefaultJobParametersValidator validator = new DefaultJobParametersValidator();
    
            validator.setRequiredKeys(new String[] {"executionDate", "fileName"}); // ํ•„์ˆ˜ ํŒŒ๋ผ๋ฏธํ„ฐ ํ™•์ธ
            validator.setOptionalKeys(new String[] {"name"}); // ์„ ํƒ ํŒŒ๋ผ๋ฏธํ„ฐ
            validator.afterPropertiesSet(); // ์„ ํƒ ํŒŒ๋ผ๋ฏธํ„ฐ์— ํ•„์ˆ˜ ํŒŒ๋ผ๋ฏธํ„ฐ๊ฐ€ ํฌํ•จ๋˜์ง€ ์•Š์•˜๋Š”์ง€ ํ™•์ธ
    
            return validator;
        }
    • ํ•„์ˆ˜ ํŒŒ๋ผ๋ฏธํ„ฐ ๋ฏธํฌํ•จ์‹œ ์˜ค๋ฅ˜

      Caused by: org.springframework.batch.core.JobParametersInvalidException: The JobParameters do not contain required keys: [fileName]
      	at org.springframework.batch.core.job.DefaultJobParametersValidator.validate(DefaultJobParametersValidator.java:120) ~[spring-batch-core-4.3.3.jar:4.3.3]
    • ์„ ํƒ ํŒŒ๋ผ๋ฏธํ„ฐ์™€ ํ•„์ˆ˜ ํŒŒ๋ผ๋ฏธํ„ฐ ๊ฒน์น˜๋Š” ๊ฒฝ์šฐ ์˜ค๋ฅ˜

      Caused by: java.lang.IllegalStateException: Optional keys cannot be required: fileName
      	at org.springframework.util.Assert.state(Assert.java:76) ~[spring-core-5.3.12.jar:5.3.12]
      	at org.springframework.batch.core.job.DefaultJobParametersValidator.afterPropertiesSet(DefaultJobParametersValidator.java:73) ~[spring-batch-core-4.3.3.jar:4.3.3]
    • ์„ ํƒ ํŒŒ๋ผ๋ฏธํ„ฐ์™€ ํ•„์ˆ˜ ํŒŒ๋ผ๋ฏธํ„ฐ์— ํฌํ•จ๋˜์ง€ ์•Š์€ ํŒŒ๋ผ๋ฏธํ„ฐ ์ „์†ก์‹œ ์˜ค๋ฅ˜

      Caused by: org.springframework.batch.core.JobParametersInvalidException: The JobParameters contains keys that are not explicitly optional or required: [displyYn]
      	at org.springframework.batch.core.job.DefaultJobParametersValidator.validate(DefaultJobParametersValidator.java:107) ~[spring-batch-core-4.3.3.jar:4.3.3]
  • ์ปค์Šคํ…€ ์œ ํšจ์„ฑ ๊ฒ€์ฆ : ํŒŒ๋ผ๋ฏธํ„ฐ ์กด์žฌ ์—ฌ๋ถ€ ์™ธ์˜ ์ถ”๊ฐ€ ์œ ํšจ์„ฑ ๊ฒ€์ฆ์ด ํ•„์š”ํ•œ ๊ฒฝ์šฐ ์ปค์Šคํ…€ JobParametersValidator ๊ตฌํ˜„ ํ•„์š”

    public class ParameterValidator implements JobParametersValidator {
        @Override
        public void validate(JobParameters parameters) throws JobParametersInvalidException {
            String fileName = parameters.getString("fileName");
    
            if (!StringUtils.hasText(fileName)) {
                throw new JobParametersInvalidException("fileName ํŒŒ๋ผ๋ฏธํ„ฐ๊ฐ€ ์กด์žฌํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค.");
            } else if (!StringUtils.endsWithIgnoreCase(fileName, ".csv")) {
                throw new JobParametersInvalidException("csv ํŒŒ์ผ์ด ์•„๋‹™๋‹ˆ๋‹ค.");
            }
        }
    }
  • CompositeJobParametersValidator : ์—ฌ๋Ÿฌ๊ฐœ์˜ ์œ ํšจ์„ฑ ๊ฒ€์ฆ

        @Bean
        public CompositeJobParametersValidator validator() {
    
            CompositeJobParametersValidator validator = new CompositeJobParametersValidator();
    
            // ํŒŒ๋ผ๋ฏธํ„ฐ ์œ ๋ฌด ๊ฒ€์ฆ
            DefaultJobParametersValidator defaultJobParametersValidator = new DefaultJobParametersValidator();
    
            defaultJobParametersValidator.setRequiredKeys(new String[] {"executionDate", "fileName"}); // ํ•„์ˆ˜ ํŒŒ๋ผ๋ฏธํ„ฐ ํ™•์ธ
            defaultJobParametersValidator.setOptionalKeys(new String[] {"name"}); // ์„ ํƒ ํŒŒ๋ผ๋ฏธํ„ฐ
            defaultJobParametersValidator.afterPropertiesSet(); // ์„ ํƒ ํŒŒ๋ผ๋ฏธํ„ฐ์— ํ•„์ˆ˜ ํŒŒ๋ผ๋ฏธํ„ฐ๊ฐ€ ํฌํ•จ๋˜์ง€ ์•Š์•˜๋Š”์ง€ ํ™•์ธ
    
          	// custom validator์™€ defaultJobParametersValidator List๋กœ ์ „์†ก
            validator.setValidators(Arrays.asList(new ParameterValidator(), defaultJobParametersValidator));
            return validator;
        }

์žก ํŒŒ๋ผ๋ฏธํ„ฐ ์ฆ๊ฐ€์‹œํ‚ค๊ธฐ

  • org.springframework.batch.core.JobParametersIncrementer

    public interface JobParametersIncrementer {
    
    	/**
    	 * Increment the provided parameters. If the input is empty, then this
    	 * should return a bootstrap or initial value to be used on the first
    	 * instance of a job.
    	 * 
    	 * @param parameters the last value used
    	 * @return the next value to use (never {@code null})
    	 */
    	JobParameters getNext(@Nullable JobParameters parameters);
    
    }

JobParametersIncrementer๋Š” ์‚ฌ์šฉํ•  ํŒŒ๋ผ๋ฏธํ„ฐ๋ฅผ ๊ณ ์œ ํ•˜๊ฒŒ ์ƒ์„ฑํ•  ์ˆ˜ ์žˆ๊ฒŒ ์Šคํ”„๋ง ๋ฐฐ์น˜๊ฐ€ ์ œ๊ณตํ•˜๋Š” ์ธํ„ฐํŽ˜์ด์Šค๋กœ ๋งค ์‹คํ–‰ ์‹œ timestamp๋ฅผ ์ถ”๊ฐ€ํ•˜๊ฑฐ๋‚˜ ํŒŒ๋ผ๋ฏธํ„ฐ๋ฅผ ์ฆ๊ฐ€์‹œ์ผœ์•ผํ•˜๋Š” ๊ฒฝ์šฐ์— ์‚ฌ์šฉํ•˜๊ธฐ ์ ํ•ฉํ•˜๋‹ค.

  • RunIdIncrementer : ํŒŒ๋ผ๋ฏธํ„ฐ ์ด๋ฆ„์ด run.id(long)์˜ ๊ฐ’์„ ์ฆ๊ฐ€

  • Custom Incrementer

    public class DailyJobTimestamper implements JobParametersIncrementer {
        @Override
        public JobParameters getNext(JobParameters parameters) {
            return new JobParametersBuilder(parameters)
                    .addDate("executionDate", new Date())
                    .toJobParameters();
        }
    }

jobBuilderFactory์—์„œ .incrementer() ๋กœ ์ˆ˜ํ–‰ํ•  ์ˆ˜ ์žˆ๋‹ค.

 @Bean
    public Job job() {
        // jobBuilderFactory.get("์žก์ด๋ฆ„")
        return this.jobBuilderFactory.get("basicJob")
                .start(step1())
                .validator(validator())
                .incrementer(new RunIdIncrementer())
                .next(step2())
                .build(); // ์‹ค์ œ job ์ƒ์„ฑ
    }

JobListener

๋ชจ๋“  Job์€ ์ƒ๋ช…์ฃผ๊ธฐ๋ฅผ ๊ฐ€์ง€๊ณ  ์žˆ์œผ๋ฉฐ, ์Šคํ”„๋ง ๋ฐฐ์น˜๋Š” ์ƒ๋ช…์ฃผ๊ธฐ์˜ ํŠน์ • ์‹œ์ ์—์„œ ๋กœ์ง์„ ์ถ”๊ฐ€ํ•  ์ˆ˜ ์žˆ๊ฒŒ ๊ธฐ๋Šฅ์„ ์ œ๊ณตํ•ด์ค€๋‹ค.

  • JobExecutionListener : Job ์‹คํ–‰๊ณผ ๊ด€๋ จ๋œ ๋ฆฌ์Šค๋„ˆ ์ธํ„ฐํŽ˜์ด์Šค

    public interface JobExecutionListener {
    
    	/**
    	 * Callback before a job executes.
    	 *
    	 * @param jobExecution the current {@link JobExecution}
    	 */
    	void beforeJob(JobExecution jobExecution);
    
    	/**
    	 * Callback after completion of a job. Called after both both successful and
    	 * failed executions. To perform logic on a particular status, use
    	 * "if (jobExecution.getStatus() == BatchStatus.X)".
    	 *
    	 * @param jobExecution the current {@link JobExecution}
    	 */
    	void afterJob(JobExecution jobExecution);
    
    }
    • beforeJob : Job ์ˆ˜ํ–‰ ์ด์ „์— ์ˆ˜ํ–‰

    • afterJob : Job ์ˆ˜ํ–‰์™„๋ฃŒ ํ›„ ์ˆ˜ํ–‰ํ•˜๋ฉฐ Job์˜ ์™„๋ฃŒ ์ƒํƒœ์™€ ์ƒ๊ด€ ์—†์ด ํ˜ธ์ถœ๋œ๋‹ค.

Job Listener๋ฅผ ์ž‘์„ฑํ•˜๋Š”๋ฐ ๋‘๊ฐ€์ง€ ๋ฐฉ๋ฒ•์ด ์žˆ๋‹ค.

  1. JobExecutionListener ์ธํ„ฐํŽ˜์ด์Šค ๊ตฌํ˜„

    public class JobLoggerListener implements JobExecutionListener {
    
        private static String START_MESSAGE = "%s is beginning execution";
        private static String END_MESSAGE = "%s has completed with the status %s";
    
        @Override
        public void beforeJob(JobExecution jobExecution) {
            System.out.println(String.format(START_MESSAGE, jobExecution.getJobInstance().getJobName()));
        }
    
        @Override
        public void afterJob(JobExecution jobExecution) {
            System.out.println(String.format(END_MESSAGE, jobExecution.getJobInstance().getJobName()
                                                        , jobExecution.getStatus()));
        }
    }

    JobBuilder ์˜ listener ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด ๋‹ค์Œ๊ณผ ๊ฐ™์ด Job์ˆ˜ํ–‰ ์ „ํ›„๋กœ ์ˆ˜ํ–‰๋œ ๊ฒƒ์„ ๋ณผ ์ˆ˜ ์žˆ๋‹ค.

    return this.jobBuilderFactory.get("basicJob")
                    .start(step1())
                    .validator(validator())
                    .incrementer(new DailyJobTimestamper())
                    .listener(new JobLoggerListener())
                    .next(step2())
                    .build(); // ์‹ค์ œ job ์ƒ์„ฑ
    2021-11-16 23:35:57.266  INFO 80890 --- [           main] o.s.b.c.l.support.SimpleJobLauncher      : Job: [SimpleJob: [name=basicJob]] launched with the following parameters: [{name=faker, executionDate=1637073357082, fileName=test4.csv, run.id=3}]
    basicJob is beginning execution
    ...
    2021-11-16 23:35:57.656  INFO 80890 --- [           main] o.s.batch.core.step.AbstractStep         : Step: [step2] executed in 129ms
    basicJob has completed with the status COMPLETED
  2. ์–ด๋…ธํ…Œ์ด์…˜ ์‚ฌ์šฉ(@BeforeJob, @AfterJob) : ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ๊ตฌํ˜„ํ•  ํ•„์š” ์—†์ด ์–ด๋…ธํ…Œ์ด์…˜๋งŒ์œผ๋กœ ๊ตฌํ˜„ํ•˜๋ฉด ๋œ๋‹ค.

    public class JobLoggerListener {
    
        private static String START_MESSAGE = "%s is beginning execution";
        private static String END_MESSAGE = "%s has completed with the status %s";
    
        @BeforeJob
        public void beforeJob(JobExecution jobExecution) {
            System.out.println(String.format(START_MESSAGE, jobExecution.getJobInstance().getJobName()));
        }
    
        @AfterJob
        public void afterJob(JobExecution jobExecution) {
            System.out.println(String.format(END_MESSAGE, jobExecution.getJobInstance().getJobName()
                                                        , jobExecution.getStatus()));
        }
    }

    ์–ด๋…ธํ…Œ์ด์…˜์œผ๋กœ ๊ตฌํ˜„ํ•˜๋Š” ๊ฒฝ์šฐ JobListenerFactoryBean ์œผ๋กœ ๋ฆฌ์Šค๋„ˆ๋ฅผ ์ฃผ์ž…ํ•  ์ˆ˜ ์žˆ๋‹ค.

    return this.jobBuilderFactory.get("basicJob")
                    .start(step1())
                    .validator(validator())
                    .incrementer(new DailyJobTimestamper())
                    .listener(JobListenerFactoryBean.getListener(new JobLoggerListener()))
                    .next(step2())
                    .build(); // ์‹ค์ œ job ์ƒ์„ฑ

ExecutionContext

ExecutionContext๋Š” ๋ฐฐ์น˜ ์žก์˜ ์„ธ์…˜์œผ๋กœ, ๊ฐ„๋‹จํ•œ ํ‚ค-๊ฐ’์„ ๋ณด๊ด€ํ•œ๋‹ค. ์ด๋•Œ Job์˜ ์ƒํƒœ๋ฅผ ์•ˆ์ „ํ•˜๊ฒŒ ๋ณด๊ด€ํ•  ์ˆ˜ ์žˆ๊ฒŒ ์ œ๊ณตํ•ด์ค€๋‹ค. Job์„ ์ˆ˜ํ–‰ํ•˜๋Š” ๊ณผ์ •์—์„œ ์—ฌ๋Ÿฌ๊ฐœ์˜ ExecutionContext๊ฐ€ ์กด์žฌํ•  ์ˆ˜ ์žˆ๋‹ค.

  • Job์— ๋Œ€ํ•œ ์ƒํƒœ : JobExecution์˜ ExecutionContext์— ์ €์žฅ

  • Step์— ๋Œ€ํ•œ ์ƒํƒœ : StepExecution์˜ ExecutionContext

์ด๋ ‡๊ฒŒ ๊ฐ Step์šฉ ๋ฐ์ดํ„ฐ์™€ Job ์ „์ฒด์šฉ ๋ฐ์ดํ„ฐ์™€ ๊ฐ™์ด ๋ฐ์ดํ„ฐ ์‚ฌ์šฉ ๋ฒ”์œ„๋ฅผ ์ง€์ •ํ•  ์ˆ˜ ์žˆ๋‹ค.

ExecutionContext๊ฐ€ ๋‹ด๊ณ ์žˆ๋Š” ๋ชจ๋“  ๋ฐ์ดํ„ฐ๋Š” JopRepository์— ์ €์žฅ๋œ๋‹ค.

ExecutionContext ๋‹ค๋ฃจ๊ธฐ

  1. Job์˜ ExecutionContext๋ฅผ ๊ฐ€์ ธ์˜ค๊ธฐ

      @Override
      public RepeatStatus execute(StepContribution contribution, ChunkContext chunkContext) throws Exception {
          String name = (String) chunkContext.getStepContext()
                          .getJobParameters()
                          .get("name");
    
          ExecutionContext jobContext = chunkContext.getStepContext()
                                              .getStepExecution()
                                              .getJobExecution()
                                              .getExecutionContext();
    
    
          jobContext.put("user.name", name);
    
          System.out.println(String.format(HELLO_WORLD, name));
    
          return RepeatStatus.FINISHED;
      }
  2. Step ExecutionContext

      @Override
      public RepeatStatus execute(StepContribution contribution, ChunkContext chunkContext) throws Exception {
          String name = (String) chunkContext.getStepContext()
                          .getJobParameters()
                          .get("name");
    
          // 2. Step ExecutionContext
          ExecutionContext stepContext = chunkContext.getStepContext()
                                              .getStepExecution()
                                              .getExecutionContext();
    
    
          jobContext.put("user.name", name);
    
          System.out.println(String.format(HELLO_WORLD, name));
    
          return RepeatStatus.FINISHED;
      }
  3. ExecutionContextPromotionListener : Step์˜ ExecutionContext์— ์žˆ๋Š” ํ‚ค๋ฅผ JobExecution์˜ ExecutionContext๋กœ ์Šน๊ฒฉํ•  ์ˆ˜ ์žˆ๋‹ค.

    @EnableBatchProcessing // ๋ฐฐ์น˜ ์ž‘์—…์— ํ•„์š”ํ•œ ๋นˆ์„ ๋ฏธ๋ฆฌ ๋“ฑ๋กํ•˜์—ฌ ์‚ฌ์šฉ, ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜ ๋‚ด ํ•œ๋ฒˆ๋งŒ ์ ์šฉํ•˜๋ฉด ๋จ.
    @SpringBootApplication
    public class JavaGradleApplication {
    
        @Autowired
        private JobBuilderFactory jobBuilderFactory;
    
        @Autowired
        private StepBuilderFactory stepBuilderFactory;
    
        /**
         * ์‹ค์ œ ์Šคํ”„๋ง ๋ฐฐ์น˜ Job ์ƒ์„ฑ
         */
        @Bean
        public Job job() {
            // jobBuilderFactory.get("์žก์ด๋ฆ„")
            return this.jobBuilderFactory.get("basicJob")
                    .start(step1())
                    .next(step2())
                    .build(); // ์‹ค์ œ job ์ƒ์„ฑ
        }
    
        /**
         * ์‹ค์ œ ์Šคํ”„๋ง ๋ฐฐ์น˜ step ์ƒ์„ฑ
         *
         * @return
         */
        @Bean
        public Step step1() {
            // stepBuilderFactory.get("์Šคํƒญ ์ด๋ฆ„")
            // tasklet ๊ตฌํ˜„์ฒด
            return this.stepBuilderFactory.get("step1")
                    .tasklet((contribution, chunkContext) -> {
                        System.out.println("Hello, world!");
                        return RepeatStatus.FINISHED;
                    })
                    .listener(promotionListener()) // step์ด ์™„๋ฃŒ ์ƒํƒœ๋กœ ์ข…๋ฃŒ๋œ ์ดํ›„ ์ˆ˜ํ–‰
                    .build();
        }
    
        @Bean
        Step step2() {
            return this.stepBuilderFactory.get("step2")
                    .tasklet(lateBindingParamTasklet("test"))
                    .build();
        }
    
        /**
         * step์ด ์™„๋ฃŒ ์ƒํƒœ๋กœ ์ข…๋ฃŒ๋œ ์ดํ›„ ์ˆ˜ํ–‰
         * "name" ํ‚ค๋ฅผ ์ฐพ์•„ Job์˜ ExecutionContext์— ๋ณต์‚ฌ
         *
         * @return
         */
        @Bean
        public StepExecutionListener promotionListener() {
            ExecutionContextPromotionListener listener = new ExecutionContextPromotionListener();
    
            listener.setKeys(new String[]{"name"});
    
            return listener;
        }
    
        public static void main(String[] args) {
            SpringApplication.run(JavaGradleApplication.class, args);
        }
    }

    ์œ„ ์˜ˆ์ œ์™€ ๊ฐ™์ด ์Šคํ…๊ฐ„์˜ ๊ณต์œ ํ•  ๋ฐ์ดํ„ฐ๊ฐ€ ์žˆ์ง€๋งŒ ์ฒซ๋ฒˆ์งธ step์ด ์„ฑ๊ณตํ–ˆ์„๋•Œ๋งŒ ๊ณต์œ ํ•  ๋•Œ ์œ ์šฉํ•˜๋‹ค.

ExecutionContext ์ €์žฅ

Job์ด ์ฒ˜๋ฆฌ๋˜๋Š” ๋™์•ˆ ์Šคํ”„๋ง ๋ฐฐ์น˜๋Š” ๊ฐ ์ฒญํฌ๋ฅผ ์ปค๋ฐ‹ํ•˜๋ฉฐ Job๊ณผ Step์˜ ์ƒํƒœ๋ฅผ ์ €์žฅํ•œ๋‹ค. (ExecutionContext ๋ฅผ ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค์— ์ €์žฅ)

์œ„์˜ Step ExecutionContext ์˜ˆ์ œ๋ฅผ ์ˆ˜ํ–‰ํ•˜๋ฉด BATCH_STEP_EXECUTION_CONTEXT์— ๋‹ค์Œ๊ณผ ๊ฐ™์ด "user.name":"dahyelele" ์ง€์ •ํ•œ ๋ฐ์ดํ„ฐ๊ฐ€ ํ‚ค-๊ฐ’์œผ๋กœ ๋“ค์–ด๊ฐ„ ๊ฒƒ์„ ์•Œ ์ˆ˜ ์žˆ๋‹ค.

{"@class":"java.util.HashMap","batch.taskletType":"io.spring.batch.javagradle.tasklet.HelloWorld","user.name":"dahyelele","batch.stepType":"org.springframework.batch.core.step.tasklet.TaskletStep"}

Spring Batch
org.springframework.batch.core.Job
image-20211115221805721
image-20211115221637438