2015. 3. 24. 00:38

▶ 직렬화 기술 이해

  • 직렬화란 무엇인가?
    컴퓨터 과학, 그중에서 데이터 저장소(data storage)의 맥락에서, 직렬화는 객체의 상태 혹은 데이터 구조를 기록할 수 있는 포맷(예를 들면 파일 또는 메모리 버퍼,  또는 네트워크 연결 링크를 통해 전송될 수 있는 형태)으로 변환하며, 나중에 동일 혹은 다른 컴퓨터 환경에서 재구성할 수 있게끔 하는 절차이다.

    직렬화의 결과로 생성된 연속된 비트 데이터는 원래의 객체와 동일한 의미를 가진 객체를 생성하는데 사용된다. 광범위한 참조를 사용하는 복잡한 객체의 경우,  직렬화 과정이 간단하지 않다. 객체지향 프로그램의 객체들을 직렬화할 경우 객체와 불가분의 관계를 맺는 메소드들은 직렬화 대상에 포함되지 않는다.

    참조 링크 : 위키피디아 - serialization

  • 직렬화는 왜 중요한가?
    직렬화는 인터넷이 보편화된 시대에 빼놓을 수 없는 기술이 되었다. 거의 모든 소프트웨어는 네트워크를 통해 데이터를 주고 받고 있다. 따라서, 프로그램의 구현하는데 필요한 기반 기술 중에서 빼놓을 수 없게 되었다. 게다가 직렬화는 네크워크 의존성이 높은 소프트웨어일 경우, 전체 성능을 좌우할 수도 있는 중요한 기술이다. 

  • 마샬링(marshalling)과의 차이점
    마샬링과 직렬화는 원격 프로시저 호출(remote procedure call)이라는 맥락에서는 대략적으로 비슷한 단어이다, 의도적인 측면에서는 그 의미가 다르다. 
    깊이 얘기해 보자면, 마샬링은 이곳에서 저곳으로 인자(parameter)들을 전달하는 작업이고, 직렬화는 구조적인 데이터를 원시형(primitive) 형태에서 바이트 스트림과 같은 형식을 복사하는 작업이다. 이러한 의미에서 직렬화는 마샬링의 pass-by-value 개념을 구현하는 수단이다.

    참조 링크 : 스택오버플로우 - 직렬화와 마샬링은 어떻게 다른가?

  • 분산 컴퓨팅의 흑역사 (COM and CORBA)
    COM (Component Object Model)컴포넌트 객체 모델 (COM)는 마이크로소프트에 의해 1993 년에 도입된 소프트웨어 컴포넌트에 대한 바이너리 인터페이스(binary interface) 표준이다. 다양한 프로그래밍 언어로 작성된 프로세스 간 통신 및 동적 객체 생성을 가능하도록 하는 기술이다. COM 표준은 OLE, OLE 자동화(automation), ActiveX, COM+, DCOM, 윈도우 쉘, DirectX, UMDF 및 윈도우 런타임 등 여러 가지 다양한 Microsoft 기술 및 프레임 워크의 기초가 된다.

    참조 링크 : 위키피디아 - COM

    CORBA (Common Object Request Broker Architecture)
    Common Object Request Broker Architecture (CORBA)는 Object Management Group(OMG)에 의해 정의된 다양한 플랫폼 상에서 동작하는 시스템 간의 통신을 용이하게 하기 위한 표준이다. CORBA는 서로 다른 운영 체제, 프로그래밍 언어, 컴퓨팅 하드웨어 위에서 동작하는 시스템들이 협업(collaboration)할 수 있게 해준다. CORBA는 캡슐화와 재사용 등 객체지향 프로그래밍과 동일한 설계 목표를 가진다. CORBA를 사용하는 시스템은 객체지향 프로그램일 필요는 없지만, CORBA 자체는 객체지향 모델을 사용한다. CORBA는 분산 객체 패러다임의 사례이다.

    참조 링크 : 위키피디아 - CORBA

  • 직렬화 적용 분야
    직렬화가 적용되는 분야는 구체적으로 다음과 같다.

    파일 저장소 (File storage) :  프로그램 실행 중에 생성된 데이터를 영구 저장소(파일 시스템) 등에 저장한 후, 이후에 프로그램이 다시 실행되었을 때 저장된 데이터를 메로리 상에 객체(혹은 구조체) 형태로 복구해 사용한다.

    네트워크 통신 (Network communication)네크워크 상에 떨어져 있는 프로그램 간에 데이터를 주고 받기 위해 데이터를 직렬화한 후, 패킷(packet)에 담아 전송한다.

    데이터베이스 (Database)복잡한 형태의 객체를 데이터베이스에 저장할 때 직렬화한 문자열 형태로 테이블의 컬럼에 저장하기도 한다.

    웹 환경 (Web environment) : 웹 서버에서 브라우저(클라이언트)로 구조화된 데이터를 전송할 때 직렬화 한 후 -JSON 형식 등- 전달하는 방식이 점차 많이 사용되고 있다.

  • 직렬화 기법 선택 시 고려할 점
    다양한 직렬화 기법(프레임워크)가 존재하며 특정 기술을 선택해서 적용해야만 한다. 직렬화 기법을 선택하는데 활용할 수 있는 기준들은 다음과 같다. 

    단순성 (simple) : 사용하기가 복잡하지 않아야 한다. 직렬화를 위해 추가해야 할 코드가 적거나, 이상적인 것은 기존 코드를 변경하는 작업이 아예 없어야 한다.

    경량 (compact) : 프레임워크(혹은 라이브러리)의 규모가 작아야 한다. 라이브러리 혹은 프레임워크의 크기가 작다는 것(small foot-print)은 시스템에 대한 추가 부담(overhead)가 적어지므로 소규모의 시스템 - 이상적인 것은 안드로이드 같은 모바일 시스템까지-에서 구동될 수 있는 확률을 높다는 것을 의미한다.

    유연성 (flexible) : 다양한 데이터 타입을 직렬화할 수 있어야 한다. 한정된 자료형(숫자 타입, 문자열 타입 등)만 직렬화할 수 있다면, 도입 자체가 불가능하거나, 혹은 개발/테스트 단계에서 뒤늦게 오류를 유발하여 프로젝트 실패의 원인이 될 수도 있다.

    버전지원 (viersioning) : 객체의 데이터 구조는 설계 및 개발, 나아가 유지보수 단계에서 변화할 수 있다. 객체의 구조가 변경되더라도 클래스의 명칭을 변경하지 않고 새로운 버전을 부여할 수 있고, 서로 다른 버전의 객체들을 자동으로 변환(직렬화 및 역직렬화)해주는 기능을 프레임워크에서 제공해 줄 경우, 시스템의 유지관리가 용이해진다.

    속도 (fast) : 처리 속도가 빠르면 빠를수록 좋다!

    확장성 (scalable) : 복잡하거나, 거대한 형태의 데이터를 직렬화 할 수 있어야 한다. 예를 들어 메모리 사용량이 많아, 직렬화 대상 객체가 증가함에 따라 메모리 부족(out of memory)를 유발한다면 심각한 수준의 문제로 인해 직렬화 프레임워크를 사용할 수 없게 된다.

  • 직렬화 데이터 형식
    직렬화된 데이터는 다양한 형식(foramat)으로 표현될 수 있다.

    Binary : 메모리에 저장된 데이터를 최소한의 가공 혹은 가공 없이 바이트의 연속된 형태로 저장하는 방식이나, 사람이 읽기 불편하기 때문에, 데이터의 무결성(integrity)을 검사하기 어렵다.

    JSON (JavaScript Object Notation) : 텍스트 형식이므로 사람과 기계 모두 읽기 가능하다. 다양한 프로그래밍 언어에서 읽고 쓸 수 있기 때문에 널리 사용된다. XML 에 비해 파싱(parsign) 속도가 빠르다. 참조 링크 : 위키피디아 - JSON

    XML (eXtensible Markup Language) : 텍스트 형식이며, JSON에 비해 복잡하다. JSON에 대해 가지는 장점은 스키마(schema)를 적용할 수 있어, 데이터 구조의 무결성을 검사할 수 있다는 점이다. 참조 링크 : 위키피디아 - XML

    YAML (YAML Ain't Markup Language) :  XML에 비해 사람이 읽고 쓰기 쉽도록 고안된 마크업(markup) 언어이다. 문법이 상대적을 단순하고, 가독성이 높게 설계되어 있다. 참조 링크 : 위키피디아 - YAML

  • 구현 기법에 따른 성능 차이
    직렬화는 내부적으로 사용하는 구현 기법에 따라 속도가 달라진다. 다음에 나열된 순서대로 점차 처리 속도가 느리다.

    Native memory copying using C operations : 객체가 할당되어 있는 메모리 영역 자체를 복사하기 위해 C 함수를 사용한다.

    “Unsafe” operationsUnsafe Java 는 본래 자바 코어 개발자들이 low-level 프로그래밍 하기 위해 만든 API이다. JNI(Java Native Interface) 와 비슷하거나 유사한 수준의 성능을 내는 자바 코딩 기술이며 자바의 캡슐화 기반의 보안을 무시할 수 있다는 장점 등이 있어서 몇몇 라이브러리에서 사용되었다. 참조 링크 : Unsafe Java

    Ignore object introspectionintrospection 은 reflection과 유사한 기술이다. 다만, introspection 은 자바의 instanceof 연산자 처럼 객체의 타입 정보만 조회하는 것이고, reflection 은 메타 정보 조회 뿐 아니라, 객체에 대한 조작 기술까지를 의미한다.

    Direct object-object copying : 자바 코드로 객체 내의 모든 멤버 변수를 복사하는 로직을 구현하는 것이다.


  • "성능"! 왜 성능이 가장 중요한가?
    앞서 다양한 직렬화 프레임워크 중에서 한가지를 선택하기 위한 다양한 지표를 제시하였으나 , 가장 중요한 평가 기준은 성능이다. 성능은 왜 중요한가? 성능이 바로 '비용(cost)'에 직결되기 때문이다. 비용은 소프트웨어의 성패를 좌우하는 매우 중요한 요소이다.

    CPU 비용 : 메모리에 존재하는 바이너리 형태의 객체를 디스크 등에 저장할 수 있는 형태 (텍스트 등)로 저장하기 위해서는 변환(transform) 처리 과정이 필요하며, 반대의 처리 또한 필요하다. 객체를 스캔(scan)하고, 데이터 타입을 문자열로 변환하거나 반대의 작업은 CPU 자원을 소모하는데다가 데이터 형식(format)이 복잡할수록 긴 시간을 소모한다.

    메모리 비용 : 변환 작업을 수행하는 과정에서 임시 버퍼(temporary buffer)를 할당하고, 네트워크를 통한 송수신 과정에서 스트림 처리 등에 따른 공간 할당이 필요하다. 메모리 및 각종 통신 자원을 할당하고 해제하는 일은 상당히 많은 자원을 소모하는 작업이므로 보다 적은 메모리를 사용하는 것이 좋다. 

    네크워크 비용 : 직렬화를 수행하는 대다수의 프로그램 혹은 시스템은 네트워크를 통해 데이터를 주고 받게 된다. 네트워크 송수신에 있어서 패킷(packet)의 크기가 커질수록 전체 성능은 떨어진다.


▶ 직렬화 프레임워크 개요

  • JDK (Serializable)
    • 프로그래밍하기 가장 쉽고, Serializable 인터페이스를 이용해 별도의 라이브러리 없이 즉시 사용할 수 있다.
    • 클래스를 릴리즈(release)한 후에는 구현을 변경하기 어려워 유연성(flexibility)을 감소시킨다.
    • C++, 파이썬(python)등 다른 언어로 구현된 프로그램과 데이터를 교환(exchange)할 수 없다.
    • 기본 연산자의 취약점(hole)으로 인해 불변 값이 손상되거나, 비정상적인 접근이 발생할 수 있다.
      (invariant corruption and illegal access)
    • 커스터마이징(customization)이 불가능하고, 소스 코드를 수정할 수 있어야 한다.

  • Java externalization
    • 객체를 저장(persist) 및 복구(restore)하는 Externalizable 인터페이스를 구현해 직접 직렬화를 구현한다.
    • 인스턴스의 컨텐츠를 저장하고 복구하는 역할을 수행하는 클래스를 구현해야 한다.
    • 클래스의 구조가 변경될 때 마다, 읽고 쓰는 코드를 수정해야 한다.

      

  • Google GSON
    • 자바 객체를 JSON으로 변환하거나 반대의 작업을 수행하는 자바 라이브러리.
    • 직렬화된 객체의 소스 코드를 필요로 하지 않는다.
    • 커스텀 표현(custom representatives)을 지원한다.

       

  • Jackson
    • 고성능, 인간공학적 JSON 프로세서 자바 라이브러리
    • 광범위한 커스터마이징 툴 지원
    • 혼합 어노테이션 (Mix-in annotations)
    • 실체화된 인터페이스 (Materialized interfaces)
    • 다양한 데이터 포맷 : JSON, CSV, Smile(binary JSON), XML, YAML

  • BSON for Jackson
    • 바이너리 인코딩된 JSON (Binary encoded JSON)
    • 몽고 DB의 주된 데이터 교환 포맷 (Main data exchange format for MongoDB)
    • 확장 프로그램 작성 가능 (Allows writing custom extensions)

  • Protocol buffers
    • 구조적인 데이터를 확장가능하며 효율적인 포맷을 변환하는 방법 제공
    • 구글 내부에서 대부분의 내부 RPC 프로토콜과 파일 포맷에 Protocol Buffers를 사용 중.
    • Java, C++, Python 지원

  • Kryo
    • 빠르고 효율적인 객체 그래프 직렬화 자바 프레임워크
    • 구글 코드 상의 오픈 소스 프로젝트
    • 자동화된 깊고 얕은 복사/복제 (Automatic deep and shallow copying/cloning)
    • 소스 클래스에 대한 코드 작성 요건이 거의 없음
      (Doesn’t put requirements on the source classes in most cases)

▶ 성능 비교 분석


Posted by 곽중선
2015. 3. 22. 13:17

▶ Apache Tika 예제 테스트

  • 다양한 문서에서 메타 정보와 컨텐츠를 추출하기 위해 사용할 수 있는 유용한 라이브러리는 아파치 Tika이다.
  • Docar 프로젝트에 Tika를 적용을 위한 사전 검증으로서 프로젝트 환경 설정 및 예제 테스트를 수행한다.

▶ 프로젝트 환경 설정 및 예제 작성

  • Maven 환경 설정
    : 프로젝트 내에서 Tika를 사용하기 위해 의존성(dependencies)을 설정한다. 2015년 3월 기준, 아파치 Tika의 최신 버전은 1.7 이다.

     	<!-- [end] 아파치 Tika 라이브러리 -->
      <properties>
     	<tika.version>1.7</tika.version>
      </properties>
    
            <!-- [start] 아파치 Tika 라이브러리 -->  	
     	<dependency>
     		<groupId>org.apache.tika</groupId>
     		<artifactId>tika-core</artifactId>
     		<version>${tika.version}</version>
     	</dependency>
     	<dependency>
     		<groupId>org.apache.tika</groupId>
     		<artifactId>tika-parsers</artifactId>
     		<version>${tika.version}</version>
     	</dependency>
    


  • 예제 코드 작성
    : MS Word 문서 파일 읽어들여, 메타 데이터를 검출하고 컨텐츠를 추출하는 예제는 다음과 같다.

    package docar.archive.extraction;
    
    import java.io.IOException;
    import java.io.InputStream;
    
    import org.apache.tika.exception.TikaException;
    import org.apache.tika.metadata.Metadata;
    import org.apache.tika.parser.AutoDetectParser;
    import org.apache.tika.parser.ParseContext;
    import org.apache.tika.parser.Parser;
    import org.apache.tika.sax.BodyContentHandler;
    import org.xml.sax.SAXException;
    
    public class AutoDetectionExample {
    	public static void main(final String[] args) throws IOException,
    			SAXException, TikaException {
    		Parser parser = new AutoDetectParser();
    
    		System.out.println("------------ Parsing an Office Document:");
    		extractFromFile(parser, "/demo.doc");
    	}
    
    	private static void extractFromFile(final Parser parser,
    			final String fileName) throws IOException, SAXException,
    			TikaException {
    		long start = System.currentTimeMillis();
    		BodyContentHandler handler = new BodyContentHandler(10000000);
    		Metadata metadata = new Metadata();
    		InputStream content = AutoDetectionExample.class
    				.getResourceAsStream(fileName);
    		parser.parse(content, handler, metadata, new ParseContext());
    		for (String name : metadata.names()) {
    			System.out.println(name + ":\t" + metadata.get(name));
    		}
    		System.out.println(String.format(
    				"------------ Processing took %s millis\n\n",
    				System.currentTimeMillis() - start));
    		System.out.println("------------ content of document\n" + handler.toString());
    	}
    }
    


▶ 추가 고려 사항

  • 다양한 문서 형식에서 추출할 수 있는 메타 데이터에서 '문서관리 시스템'에서 활용할 수 있는 유용한 후보(candidates)들은 다음과 같다. : "Content-Type", "Author", "Last-Modified", "Creation-Date", "title"
  • "Content-Type" : 파일 형식 별 검색을 위해 필요한 정보이므로 필수 항목이다.
  • "Author" : "작성자" 혹은 "저자" 정보는 MS-Office 등의 문서 형식에서 부정확한 경우가 많다. 예를 들어 타인이 작성한 문서를 복사 및 편집한 후 저장한 문서일 경우에는 원래 파일의 작성자 정보만이 남아 있게 된다. 따라서, 부수적인 정보로 취급해야 한다.
  • "Creation-Date" : 문서의 생성일자는 특정 기간에 생성된 문서를 검색하는데 유용한 정보이다.
  • "Last-Modified" : 문서의 최종 변경일자는 버전 관리를 위해 필요한 정보이다. 문서 저장소에 저장된 파일과 입력된 파일을 비교해서 입력된 파일의 변경일자가 저장소의 파일보다 나중인 경우에는 새로운 버전으로 문서 저장소에 추가해야 한다.
  • "title" : 제목 또한 부정확한 정보가 될 수 있다. 앞서 작성자 정보와 마찬가지로 복사한 파일을 편집한 후에 저장하면 이전 작성자의 정보가 그대로 남아있게 된다.

▶ 참조 링크 및 이클립스 프로젝트 (소스 다운로드)


Posted by 곽중선
2015. 3. 21. 13:21

▶ Non Functional Requirement

  • "성능 측정(performance estimation) 혹은 프로파일링(profiling)"은 서비스 혹은 어플리케이션이 '실용적인지 여부'를 판단하는 기준이 된다. 아무로 기능 면에서 좋은 소프트웨어라고 할지라도 필요한 시스템 사양(specification)이 과다하다면 써먹을 수 없다. 혹은 실행 속도가 느리다면 환영 받지 못할 것이다. 
  • 성능 측정을 모든 기능을 개발한 후에 수행하게 되면 2가지 문제가 발생할 수 있다.

    • 성능 문제를 해결하는데 필요한 시간이 부족해서, 최초에 수립한 일정을 지키기 못하거나,
    • 문제를 해결할 수 없어서 실패한 프로젝트가 되고 마는 것이다. 

  • 따라서, 성능 측정 준비는 기능 설계를 진행하면서 함께 준비하고, 기능 구현 과정 중에 몇 차례에 걸쳐 성능을 측정함으로써 중간 과정에서 취약점(weak point)가 어느 부분인지, 어떤 기능으로 인해 급작스럽게 성능이 하락하는지 검사하는 것이 현명하다.


▶ 성능에 대한 핵심 개념과 사전 지식

  • "성능(performance)"를 좌우하는 2가지 요소는 "공간"과 "시간"이다. 프로그래밍에서 "공간"이라는 개념은 프로그램이 소모하는 메모리(작게는 메인 메모리, 크게 보면 디스크와 네트워크)이며, "시간" 개념은 얼마나 CPU 자원을 오래 사용하는가 이다.
  • 따라서, 소프트웨어 성능은 "메모리 사용량(공간")과 "CPU 사용량(시간)" 등 2가지 측정 지표(metrics)가 존재한다.
  • 개발자가 작성하는 "코드"가 얼마나 많은 "공간"과 "시간"을 필요로 하는가에 대해서는 "알고리즘"과 "자료구조" 지식을 통해 이론적으로 예측할 수는 있으나, 다음과 같은 조건에 따라 "같은 문제"라고 하더라도 실질적으로 상당한 차이가 발생한다.

    • 어떤 언어로 개발하는가?
      : 하위 수준 언어일수록 빠르고, 컴파일러(compiler) 방식의 언어가 인터프리터(interpreter) 언어보다 빠르다.
    • 개발자가 구체적으로 어떤 방법으로 구현했는가?
      : 구현 방식은 API 호출, 라이브러리 활용, 직접 구현 등 방식을 한정할 수 없다.
    • 어떤 컴파일러를 사용해서 컴파일 했는다?
      : 컴파일러에 따라서 기계어를 생성하는 로직이 다르다. 컴파일러 자체에 성능 최적화 기능이 포함되어 있는 경우도 많다.
    • 어떤 머신(machine)에서 실행하는가?
      : 게임과 같은 실수형 계산 및 그래픽 처리를 많이 하는 프로그램은 GPU 성능에 의존하는 경향이 있다.

  • 위와 같은 이유로 성능은 정적인 "코드"만으로 예측할 수 없기 때문에 실행하면서 측정하는 것이 가장 '정확하다'! 프로그램을 실행하면서 성능을 측정하고, 분석하는 행위를 "프로파일링(profiling)"이라고 하며, 프로그램의 성능을 측정하는 도구를 "프로파일러(profiler)"라고 한다. 프로그램의 성능을 측정한 후, 개선하는 작업을 "최적화(optimizing)"라고 한다.

▶ 성능에 대한 정의와 이해 at slideshare.net



▶ DIY Profiling at slideshare.net



▶ 성능 측정 유형

  • CPU 측정
    • Times/ Calls 
      : 함수 혹은 메소드가 얼마나 오랜동안 CPU를 사용하는가 혹은 얼마나 많이 호출되는가를 측정하는 것이다.
    • Sampling
      : 아주 짧은 시간동안 호출되는 함수의 수행 시간을 측정하면 해당 함수의 수행 시간을 왜곡하게 된다. (함수가 수행되는 시간보다 더 많은 시간을 측정하는데 소모하거나, 수행시간이 느린 것으로 잘못 판단할 수 있다.) 혹은 초당 10만번 이상 호출되는 함수들을 측정하고자 할 경우 샘플링(sampling) 기법을 사용한다. 샘플링 기법은 프로그램 수행 중 일정 주기 마다 현재 실행 상태 (실행 중인 위치 정보)를 기록하는 기법이다.

  • 메모리 측정
    • Usage
      : 메모리를 얼마나 "자주" 할당 및 해제하는가를 측정한다. 메모리의 할당 및 해제는 비용이 많이 드는 '비싼' 작업이다.
    • Allocation
      : 메모리를 얼마나 "많이" 할당하느냐를 측정한다. 메모리를 과다하게 할당할 경우 "메모리 부족(Out of memory)" 오류를 발생 시키거나, 스왑(swap) 메모리를 사용하게 되어 급격한 성능 저하를 유발하게 된다.


▶ 스스로 하는 성능 측정 (DIY Java Profiling)

  • 앞서 Romon Elizrov 의 자료에서 제시하는 프로파일링 기법들은 다음과 같다.

  • 자바 코드 내에서 성능 측정 코드 삽입 (Just code it in Java)
    : 개발자가 직접 성능을 측정하는 코드를 자바 코드 내에 삽입하는 방법이다.

  • 자바 가상 머신의 기능 활용 (Know your JVM features)
    : JVM 에서는 메모리 사용 상태 및 스레드(thread)의 실행 상태를 확인할 수 있는 기능(옵션)이 제공된다.

  • 바이트 코드 조작 활용 (Use byte code manipulation)
    : 자신이 작성한 프로그램이 아니거가 - 소스 코드가 없다거나 -, 이미 완성된 프로그램의 성능을 측정하고자 하는 경우에 사용되는 기법이다. 컴파일된 클래스 파일 (확장자가 .class)을 조작해서 성능 로그를 생성하는 코드를 삽입하는 기법이며, 자칫 JVM의 비정상 종료(crush)를 유발할 수 있기 때문에 조심스럽게 적용해야 한다.

▶ 프로파일링 클래스 구현

  • docar.util.profile.Profiler
    : 성능 측정 정보를 기록하는 싱글턴(singleton) 패턴의 클래스. addCallEvent() 메소드를 호출할 때마다, 특정 기능의 수행 횟수 및 누적 수행 시간을 기록한다.
  • docar.util.profile.StopWatch
    : 특정 기능(메소드, 클래스)의 수행 시간을 측정하는 클래스.
  • docar.uti.Report
    : 성능 측정 결과를 출력하는 클래스. "기능 명칭 및 호출 횟수", "총 수행 시간", "평균 수행 시간" 등을 출력한다.

  • 사용 방법 예시
    // 메소드 시작 부분에서 "기능 명칭"을 인자로 스톱워치 생성
    StopWatch stopWatch = new StopWatch("DuplicationFinder.main");
    
    // 메소드 종료 부분에서 수행 시간 기록(누적)
    Profiler.getInstance().addCallEvent(stopWatch.stop());
    
    // 수행 시간 측정 결과 출력
    Report.report(System.out);
    
    
  • 측정 결과 예시



Posted by 곽중선