java 의존성 충돌 테스트 (use Gradle)

gradle을 이용한 java 프로젝트에서 문뜩 내가 사용하고 있는 라이브러리와, 내 프로젝트가 사용하는 라이브러리가 동일한 라이브러리를 사용하고 있으면 어떻게 될까? 궁금하여 테스트하였습니다.

의존성 충돌을 위한 구조

picture 0

  • RootProject : 최상위 프로젝트입니다.
  • libSample : 의존성 충돌을 일으키기 위해 1.0.0과 2.0.0으로 나눠진 두개의 버전을 준비합니다.
  • dependency-test : root 프로젝트와는 다른 버전의 libSample 을 의존하도록 하여 의존성 충돌을 발생시킵니다.

RootProject 의존관계

// build.gradle
dependencies {

    // ~~~

    implementation files('./libs/lib-sample-1.0.0.jar')
    implementation files('./libs/dependency-test-1.0.0.jar')

    // ~~~
}

dependency-test-1.0.0 의존관계

// build.gradle
dependencies {

    // ~~~

    implementation files('./libs/lib-sample-2.0.0.jar')

    // ~~~
}

RootProjectlib-sample-1.0.0.jar 을 의존 dependency-testlib-sample-2.0.0.jar 을 의존


의존성 충돌을 발생시키는 코드 구성

먼저 lib-sample-1.0.0 과 lib-sample-2.0.0 의 호출되는 함수는 같으나 내부 기능이 다르도록 만들어 보겠습니다.

lib-sample-1.0.0

public class LibSample {

    private String value;

    public LibSample() {
        this.value = "test";
    }

    public String getValue() {
        return value;
    }

    public void setValue(String value) {
        this.value = value;
    }
}

lib-sample-2.0.0

public class LibSample {

    private int value;

    public LibSample() {
        this.value = 1;
    }

    public int getValue() {
    return value;
    }

    public void setValue(int value) {
    this.value = value;
    }
}

위와 같이 value 라는 변수명, LibSample이라는 클래스명, getValue, setValue등 이름은 동일하나

2.0.0 버전은 int 타입을, 1.0.0 버전은 String 타입을 사용하도록 차이점을 두었습니다.

자 이제 빌드를 하여 lib-sample-2.0.0.jar 를 사용하는 dependency-test 의 코드를 작성해보겠습니다.

dependency-test-1.0.0


public class DependencyTest {

    public LibSample getLibSample() {

        LibSample libSample = new LibSample();
        libSample.setValue(100);

        return libSample;
    }
}

dependency-test-1.0.0lib-sample-2.0.0.jar 을 의존하기 때문에 int setValue(int value) 와 같이 int 타입을 기준으로 동작합니다.

그렇기 때문에 libSample.setValue(100); 과 같이 value100으로 셋팅된 libSample 객체를 반환하도록 하겠습니다

자 그럼 이제 String 타입을 기준으로 동작하는 lib-sample-1.0.0.jardependency-test-1.0.0 를 동시에 의존하는 RootProject 의 코드를 작성해보겠습니다.

RootProject

public static void main(String[] args) {

    DependencyTest dependencyTest = new DependencyTest();

    LibSample libSample = dependencyTest.getLibSample();

    System.out.println(libSample.getValue());
}

자 위와 같이 RootProject의 코드를 작성하고 동작을 시킬 때 어떻게 되는지 알아봅시다.

예상

먼저 이상적인 결과를 예상을 해보자면 에러 없이 “100” 이라는 결과가 찍히길 원합니다.

그렇게 동작할려면 DependencyTest 아무 문제 없이 lib-sample-2.0.0.jar 를 사용할 수 있으면 됩니다.

하지만 현실은…

동작시키면 아래와 같은 에러를 마주하게 됩니다.

...
Exception in thread "main" java.lang.NoSuchMethodError: 'void libsample.LibSample.setValue(int)'
...

왜 setValue(int) 를 못찾지??

int 타입 기반의 setValue() 를 찾지 못하여 에러가 발생했습니다.

그렇다면 lib-sample-2.0.0.jar 를 사용하지 않고 lib-sample-1.0.0.jar 을 사용했다는 것인데 왜 이렇게 되었을까요?

동일한 패키지에 동일한 클래스명은 단 하나만 가능

java에서는 동일한 패키지 경로에 동일한 클래스명은 단 하나만 존재할 수 있습니다.

위 예시의 경우를 보면 lib-sample-1.0.0.jar 에도 LibSample 이라는 클래스가, lib-sample-2.0.0.jar 에도 LibSample 이라는 클래스가 동일한 패키지 경로에 존재합니다.

그렇기 때문에 둘중 단 하나의 클래스만 선택될 수 있으며, 이 선택의 기준은 최상위 프로젝트와 가까운 클래스 가 선택됩니다.

lib-sample-1.0.0.jar 의 경우 최상위 프로젝트인 RootProject가 바로 의존하고 있기 때문에 채택된 것 입니다.

순서가 반대라면?

만약 반대로 RootProject가 lib-sample-2.0.0.jar 를 의존하고 DependencyTestlib-sample-1.0.0.jar 을 의존했을 경우에는 lib-sample-2.0.0.jar 가 선택되었을겁니다.


해결 방법은?

사실 위 예시와 같을 경우 해결 방법이 없습니다…

강제로 공통된 버전을 사용하도록 하는 방법 이 존재하나 둘 중 하나는 오류가 발생할 가능성이 높기 때문에 추천하지 못하는 방식입니다.

결국은 호환되는 버전의 라이브러리를 공통적으로 사용하는 버전을 찾아서 다른 하나를 다운그레이드 하던지, 아니면 업그레이드 하던지 하여 안전하게 버전을 맞춰야합니다.