기본 콘텐츠로 건너뛰기

스레드 동기화2 - volatile, Atomic


참고 : 이펙티브자바  (규칙66 변경 가능 공유 데이터에 대한 접근은 동기화하라)

이펙티브자바에서는
스레드를 중지할때 Thread.stop() 함수를 사용하지 말라고 강조한다.
뭔가 안전하지 않다는 이유다.
대신 반복문으로 이를 해결하라고 당부한다.


반복문으로 실행하는 스레드

예를들어,
스레드를 하나 생성하여 실행한다.
외부에서 스레드의 실행조건(반복문)을 변경하고자 한다.
스레드는 실행조건이 변경을 감지하고 실행을 종료한다.

public class D  {

    private static boolean stopRequested=false;

    private static void requestStop(){
        stopRequested = true;
    }

    private static boolean stopRequested(){
        return stopRequested;
    }

    public static void main(String[] args) throws InterruptedException {
        Thread backThread = new Thread(new Runnable(){
            public void run(){
                int i=0;
                System.out.println(2);
                while(!stopRequested)
                    i++;
                System.out.println(4+"/"+i);
            }
        });

        backThread.start();
        System.out.println(1);
        TimeUnit.SECONDS.sleep(1);
        System.out.println(3);
        stopRequested = true;

    }
}

결과
1
2
3

4와 오지게 호출됬던 i가 몇까지 올라갔는지 확인하지 못했다.
main스레드가 변경한값을 backThread가 보지 않고있다는 증거이다.


동기화

이문제는 동기화로 해결 할 수 있다.
자원에 접근하는 함수들을 동기화 한다.
public class E {

    private static boolean stopRequested;

    private static synchronized void requestStop(){
        stopRequested = true;
    }
    private static synchronized boolean stopRequested(){
        return stopRequested;
    }

    public static void main(String[] args) throws InterruptedException {
        Thread backThread = new Thread(new Runnable(){
            public void run(){
                int i=0;
                System.out.println(2);
                while(!stopRequested())
                    i++;
                System.out.println(4+"/"+i);
            }
        });

        backThread.start();
        System.out.println(1);
        TimeUnit.SECONDS.sleep(1);
        System.out.println(3);
        requestStop();
    }
}


결과
1
2
3
4/36251944

책에서는
쓰기메서드(requestStop)만 동기화하는것은 충분하지 않고,
두 함수 모두 동기화해야된다고 한다.
하지만 실제로 읽기메서드(stopRequested)만 동기화해도
정상적으로 실행은 된다.
단 쓰기메서드만 동기화 했을시에는 원하는 결과를 얻지 못한다.


volatile

이번예제에서 가장 적합한 방법이다.
공유자원을 volatile로 선언해주면
스레드는 무조건 가장 최근의 값을 읽도록 보장한다.
syncronized를 사용하지 않고 동기화를 할 수 있다는 것이다.

private static boolean stopRequested;

1
2
3
4/1607704186

결과는 나오는데, i값이 동기화를 사용했을때보다 45배정도 증가했다.
속도에서는 volatile 이라는 놈이 분명한 단점을 가지고 있을듯 하다.

volatile을 사용할때 주의해야 할것이 있다.
volatile int a = 0;

public work(){
   a--;
}

volatile을 가장 최근의 값을 가져온다.
그런데 만약 '--' 나 '++' 같은 원자적이지 않는 함수를 실행할때 문제가 생길 수 있다.
원자적이라는 말은한번에 한가지일만 하고, 부가적일은 하지 않는 다는 얘기다.

여기서 'a--'는 내부적으로 두가지일을 한다.
1. 현재 a의 값을 읽는다.
2. 읽은값에 1을 빼서 다시 기록한다.

즉 첫번째 스레드가 --작업을 채 마치기전에
두번째 스레드가 들어와서 a를 읽으면
두스레드는 같은 값을 가지게 되어 원하는 결과를 얻지 못한다.

이처럼 원자적이지 못한 일들을 처리할떄는 반드시
명시적으로 동기화선언을 해주어야 한다.

Atomic클래스

이럴떄 synchronized를 사용하지 않고 동기화를 할 수 있는 방법이 또 있다.
이전(스레드 동기화1 - syncronized)에 작성했던
공유자원클래스였던 A를 조금 수정해봤다.

int를 없애고 대신에 AtomicInteger를 사용했다.
눈치챘을수도 있지만 Integer말고도 타입은 Long, Array 등등 여러가지가 있다.

public class A {

    private static AtomicInteger a ;
    A(){
        a=new AtomicInteger(15);
    }
    public void work(){
        //a--;
        a.decrementAndGet();
    }
    public AtomicInteger getA(){
        return a;
    }

}

이전 예제로
결과는 10번정도 돌려봤는데
Thread0/13
Thread1/13
Thread3/12
Thread4/11
Thread2/10
Thread1/8
Thread0/8
Thread3/7
Thread4/6
Thread2/5
Thread0/3
Thread3/2
Thread1/4
Thread2/1
Thread4/0

매번 이렇게 나오지는 않았지만,
정상적으로 0이 매번 출력되었다.


결론

스레드를 사용할때 공유자원이 있다면 반드시 동기화작업이 선행되어야 한다.
Synchromized는 가장 확실한 방법일 수는 있지만
상황에 맞춰서 volatile이나 Atomic 클래를 활용해서
보다 유연하게 사용할 수 있다.


예제파일 : https://gist.github.com/ch7895/39b13b8d80128457fdb8

댓글

이 블로그의 인기 게시물

메일서버가 스팸으로 취급받을때

설치한 메일서버를 통해 발송되는 메일이 스팸으로 들어가는 경우가 더러 있다. 이게 한번 들어가기는 쉬운데, 빠져나오기는 드럽게 힘든것 같다... 본인의 경우에는 우선 국내서비스에는 별 무리 없이 들어간다. (naver,daum 등) 그런데 해외메일 그중 Gmail, Hotmail 에는 에누리없이 스팸으로 간주되고 있었다. Gmail같은 경우에는 그래도 스팸함으로 발송은 제대로 되는반면에 Hotmail같은경우에는 아예 수신자체가 안되는 경우도 더러있다.. ㅡ,.ㅡ; 제일 좋은 방법은 Gmail,Hotmail에 전화걸어서 우리 메일서버 IP white Ip로 등록해달라!!! 하면 좋지만, 얘네들은 걸어봤자 자동응답기고, 문의채널은 구글 그룹스 게시판이 전부다.. 본론으로 들어가서. 해외 메일이 차단될 경우 내 매일서버ip가 스팸ip로 등록되 버린 경우일 수 있다. (본인처럼. ㅎ) 이것부터 조회 해보고 싶으면 RBL(real-time blocking List) 체크를 해야 하는데, RBL체크 해주는 사이트는 꽤 많이 있고, 그중 좀 깔끔해 보이는곳 하나 소개. http://www.anti-abuse.org/ 메일서버ip 입력하고 조회해보면 쭈루룩 리스트가 나온다. 그 중 빨간불이 들어온 부분이 메일 서버가 스팸서버가 된 각종 이유들이다.ㅋ 본인의 경우 CBL 때문에 걸렸는데, 내용은 아래와 같다. This IP address is HELO'ing as  "localhost.localdomain"  which violates the relevant standards (specifically: RFC5321). 메일서버 도메인에 별다른 작업을 안해놓아서 "localhost.localdomain" 으로 설정되어있었다. 만약 CBL만 바로 테스트 해보고 싶으면 http://cbl.abusea...

[javascript] 특정시간에만 함수 실행

특정시간에만 팝업을 띄우려면?? 특정시간에만 로그인을 막으려면?? 특정시간에만 할일은 의외로 참 많다. 방법? 딱히 없다. 현재시간 구해서 시작시간, 종료시간 사이에 있을때 시작하는 수밖엔. if ((현재시간 > 시작시간) && (현재시간 < 종료시간)){ .. 팝업노출(); 공사페이지 리다이렉트(); 기타등등(); .. } 자바스크립트로 작성하면 다음과 같다. var startdate = "2014012008" ; var enddate = "2014012418" ; var now = new Date (); //현재시간 year = now. getFullYear (); //현재시간 중 4자리 연도 month = now. getMonth () + 1 ; //현재시간 중 달. 달은 0부터 시작하기 때문에 +1 if ((month + "" ). length < 2 ){ month = "0" + month; //달의 숫자가 1자리면 앞에 0을 붙임. } date = now. getDate (); //현재 시간 중 날짜. if ((date + "" ). length < 2 ){ date = "0" + date; } hour = now. getHours (); //현재 시간 중 시간. if ((hour + "" ). length < 2 ){ hour = "0" + hour; } today = year + "" + month + "" + date + "" + hour; //오늘 날짜 완성. / / 시간비교 i...

[spring] 인터셉터 와 필터

스프링에서 전후처리기(?)를 담당하는 인터셉터 와   필터 라는게 있다.  둘다 알고 계신 훌륭한 개발자 분들도 많으시겠지만 ㅋ  본인처럼 인터셉터만 알고 있는 경우에는  가볍게라도 필터에 대해서 알아두시라고 몇자 적어 본다. ㅎ  사실  기능만 보면  인터셉터와 필터는  무척 비슷해 보인다.  게다가 filter로 해야 되는 일들은  사실   interceptor로 해결할 수 있는 듯 하다.  (적어도 본인이 하고 있는 프로젝트에서는.. ㅋ) 구글 창에다가  'difference interceptor and filter' 의 검색결과가 여러페이지 나오는 걸 보면 많은 사람들이 궁금해 하는게  다 이러한 이유인것 같다.  본론으로 들어가서  우선 인터셉터 설정파일을 보면  < mvc:interceptors > < mvc:interceptor > < mvc:mapping path = "/api1/*" /> < mvc:mapping path = "/api2/*" /> < mvc:mapping path = "/api3/*" /> < bean class = "com.changpd.test.interceptor.인터셉터클래스" /> </ mvc:interceptor > </ mvc:interceptors > 보통 이런식으로 선언 되어있을듯 하다.  해석은  api1,2,3 호출되기전에 인터셉터클래스를 먼저 호출하겠다.  뭐 이런...