Java에서 인터페이스 변수가 기본적으로 정적 및 최종적인 이유는 무엇입니까?
Java에서 인터페이스 변수가 기본적으로 정적 및 최종적인 이유는 무엇입니까?
답변:
Philip Shaw의 Java 인터페이스 디자인 FAQ에서 :
Java 인터페이스는 자체적으로 인스턴스화 할 수 없으므로 인터페이스 변수는 정적입니다. 변수의 값은 인스턴스가 존재하지 않는 정적 컨텍스트에서 할당되어야합니다. 최종 수정자는 인터페이스 변수에 지정된 값이 프로그램 코드로 재 할당 할 수없는 진정한 상수인지 확인합니다.
static
수정 자에 대한이 설명 은 완전히 허구입니다. 클래스의 퍼블릭 인스턴스 변수는 인터페이스의 일부이므로 interface
인스턴스 메소드와 마찬가지로 Java에서 추상화되지 않아야 할 이유가 없습니다 . Java interface
를 직접 인스턴스화 할 수 없다는 것은 중요하지 않습니다. 이를 구현하는 클래스 인스턴스를 계속 보유 할 수 interface
있으며 특정 공용 인스턴스 변수가 있어야하는 것이 합리적입니다. 에 대한 부분은 final
설명을 전혀 제공하지 않으며 final
의미가 무엇인지 설명 합니다.
인터페이스에는 직접적인 객체가 없으므로 클래스 / 인터페이스를 사용하는 것이 유일한 방법입니다. 따라서 인터페이스 변수가 존재하는 경우 정적이어야합니다. 그렇지 않으면 외부 세계에 전혀 액세스 할 수 없습니다. 정적이기 때문에 하나의 값만 가질 수 있으며 구현하는 모든 클래스는 값을 변경할 수 있으므로 모든 엉망이 될 것입니다.
따라서 인터페이스 변수가 있으면 암시 적으로 정적이며 최종적이며 명백하게 공개됩니다!
interface
. 클래스는 인터페이스에 필요한 인스턴스 변수를 선언하여 인터페이스를 구현합니다. 생성자 (또는 다른 메소드)는 인스턴스 변수를 설정합니다. 클래스의 인스턴스가 인스턴스화되면 인스턴스 변수에 액세스 할 수 있습니다.
public : 인터페이스에 존재하는 메소드와 마찬가지로 모든 클래스의 액세스 가능성
static : interface는 객체를 가질 수 없으므로 interfaceName.variableName을 사용하여이를 참조하거나이를 구현하는 클래스에서 variableName을 직접 사용할 수 있습니다.
final : 상수로 만듭니다. 두 클래스가 동일한 인터페이스를 구현하고 둘 다 값을 변경할 권한을 부여하면 var의 현재 값에서 충돌이 발생하므로 한 번만 초기화가 허용됩니다.
또한 이러한 모든 수정자는 인터페이스에 암시 적이므로 실제로 지정할 필요는 없습니다.
( 이것은 철학적 인 대답이 아니라 실용적인 것입니다 ). static
다른 사람들이 대답 한 수정 자 요구 사항 은 분명합니다. 기본적으로 인터페이스를 인스턴스화 할 수 없으므로 해당 필드에 액세스하는 유일한 방법은 인터페이스를 클래스 필드로 만드는 것 static
입니다.
interface
필드 뒤에서 자동으로 final
(일정하게) 되는 이유 는 다른 구현이 우연히 다른 구현의 동작에 영향을 줄 수있는 인터페이스 변수의 값을 실수로 변경하는 것을 방지하기위한 것입니다. 자바 interface
가 속성을 명시 적으로 만들지 않은 아래 시나리오를 상상해보십시오 final
.
public interface Actionable {
public static boolean isActionable = false;
public void performAction();
}
public NuclearAction implements Actionable {
public void performAction() {
// Code that depends on isActionable variable
if (isActionable) {
// Launch nuclear weapon!!!
}
}
}
이제 구현하는 다른 클래스가 Actionable
인터페이스 변수의 상태를 변경하면 어떻게 될지 생각해보십시오 .
public CleanAction implements Actionable {
public void performAction() {
// Code that can alter isActionable state since it is not constant
isActionable = true;
}
}
이러한 클래스는 클래스 로더에 의해 단일 JVM 내에로드 된 경우의 동작을 NuclearAction
다른 클래스에 의해 영향을받을 수는 CleanAction
, 그 때 performAction()
뒤에있는 invoke이다 CleanAction
의 최악이 경우 어떤 일 수있다 (그렇지 않으면 동일한 스레드 또는) 실행 (의미 적으로 그것은)입니다.
각 구현에서 interface
이러한 변수를 어떻게 사용할지 알 수 없으므로 암시 적으로이어야합니다 final
.
다른 것은 구현의 일부이므로 인터페이스에는 구현이 포함될 수 없습니다.
public interface A{
int x=65;
}
public interface B{
int x=66;
}
public class D implements A,B {
public static void main(String[] a){
System.out.println(x); // which x?
}
}
해결책은 다음과 같습니다.
System.out.println(A.x); // done
인터페이스 변수가 정적 인 이유 중 하나라고 생각합니다.
인터페이스 내에서 변수를 선언하지 마십시오.
static final
실제로 정적이고 최종적인 변수 앞에 쓰여지는 것을 보지 못할 수도 있습니다 .
static-인터페이스는 인스턴스를 가질 수 없기 때문입니다. 마지막으로 변경할 필요가 없기 때문입니다.
Java는 인터페이스에서 추상 변수 및 / 또는 생성자 정의를 허용하지 않습니다. 솔루션 : 인터페이스와 구현 사이에 추상 클래스를 연결하면 다음과 같이 추상 클래스 만 확장됩니다.
public interface IMyClass {
void methodA();
String methodB();
Integer methodC();
}
public abstract class myAbstractClass implements IMyClass {
protected String varA, varB;
//Constructor
myAbstractClass(String varA, String varB) {
this.varA = varA;
this.varB = VarB;
}
//Implement (some) interface methods here or leave them for the concrete class
protected void methodA() {
//Do something
}
//Add additional methods here which must be implemented in the concrete class
protected abstract Long methodD();
//Write some completely new methods which can be used by all subclasses
protected Float methodE() {
return 42.0;
}
}
public class myConcreteClass extends myAbstractClass {
//Constructor must now be implemented!
myClass(String varA, String varB) {
super(varA, varB);
}
//All non-private variables from the abstract class are available here
//All methods not implemented in the abstract class must be implemented here
}
나중에 다른 인터페이스와 함께 구현하지 않으려는 경우 인터페이스 없이 추상 클래스 를 사용할 수도 있습니다 . 추상 클래스의 인스턴스를 만들 수 없으므로 먼저 확장해야합니다.
"protected"키워드는 확장 클래스 만 이러한 메소드 및 변수에 액세스 할 수 있음을 의미합니다.
스파이로
인터페이스 : 시스템 요구 사항 서비스.
인터페이스에서 변수는 기본적으로 공개, 정적, 최종 액세스 수정 자에 의해 지정됩니다 . 때문에 :
public : 인터페이스가 다른 패키지에 배치 될 수도 있습니다. 따라서 프로젝트의 어느 곳에서나 변수에 액세스해야합니다.
static : 불완전한 클래스는 객체를 만들 수 없습니다. 따라서 프로젝트에서 객체없이 변수에 액세스해야합니다.interface_filename.variable_name
final : 하나의 인터페이스가 많은 클래스에 의해 구현되고 모든 클래스가 인터페이스 변수에 액세스하고 업데이트하려고한다고 가정하십시오. 따라서 데이터 변경이 일관되지 않고 다른 모든 클래스에 영향을 미칩니다. 따라서 final로 액세스 수정자를 선언해야합니다.
에서는 Java
, 인터페이스는 인스턴스 변수를 선언 할 수 없습니다. 인터페이스에서 선언 된 변수를 인스턴스 변수로 사용하면 컴파일 타임 오류가 반환됩니다.
static final
인스턴스 변수와 다른 상수 변수를 선언 할 수 있습니다 .
인터페이스는 모든 클래스에 의해 구현 될 수 있으며 구현 클래스 중 하나에 의해 값이 변경되면 다른 구현 클래스에 대한 오해가 발생할 수 있습니다. 인터페이스는 기본적으로 서로 관련이 있지만 서로 다른 두 엔티티를 결합하는 참조이므로 인터페이스 내부의 선언 변수는 암시 적으로 최종적이고 정적이며 인터페이스를 인스턴스화 할 수 없기 때문에 정적입니다.
Eclipse에서 방금 시도했지만 인터페이스의 변수는 기본적으로 최종 변수이므로 변경할 수 없습니다. 부모 클래스와 비교할 때 변수는 확실히 변경 가능합니다. 왜? 내 생각에 클래스의 변수는 어린이가 상속하는 속성이며 어린이는 실제 필요에 따라 변수를 변경할 수 있습니다. 반대로 인터페이스는 속성이 아닌 동작 만 정의합니다. 인터페이스에 변수를 넣는 유일한 이유는 해당 인터페이스와 관련된 const로 변수를 사용하는 것입니다. 그러나 이것은 다음 발췌에 따르면 좋은 습관이 아닙니다.
"인터페이스에 상수를 배치하는 것은 Java 초기에 널리 사용 된 기술 이었지만 이제는 인터페이스가 데이터가 아닌 객체가 제공하는 서비스를 처리해야하므로 인터페이스를 불쾌하게 사용하는 것으로 간주하고 있습니다. 클래스 별 구현은 일반적으로 구현 세부 사항이지만 인터페이스에 배치하면 클래스의 공용 API로 승격됩니다. "
또한 정적을 넣거나 전혀 차이를 만들지 않았습니다. 코드는 다음과 같습니다.
public interface Addable {
static int count = 6;
public int add(int i);
}
public class Impl implements Addable {
@Override
public int add(int i) {
return i+count;
}
}
public class Test {
public static void main(String... args) {
Impl impl = new Impl();
System.out.println(impl.add(4));
}
}