Dieser Artikel ist der Artikel zum 18. Tag von MicroAd Adventskalender 2017 .
Ist es etwas, woran ich normalerweise nicht denke? Wenn ich jetzt nicht darüber spreche, werde ich in Zukunft nicht mehr darüber sprechen. Lassen Sie uns also über Statik in Java sprechen.
Nur statisch aber statisch, Lassen Sie uns beim Schreiben eines einfachen Quiz noch einmal statische Dinge in Java organisieren.
Statisch bedeutet in erster Linie "statisch". ~~ ~~
Wenn Sie "nicht dynamisch" sagen, ist das Feld nur eines in der Klasse, unabhängig davon, wie viele Instanzen Sie erstellen.
Fügen Sie den Feldern, für die Sie Ressourcen und Informationen freigeben möchten, statische Elemente hinzu, die weiterhin von mehreren Instanzen gemeinsam genutzt werden sollen. Nennen Sie es mit `class name.field name
`. Sie können auf dasselbe zugreifen, indem Sie eine Instanz erstellen und aufrufen.
Wenn Sie beim Definieren einer Konstante mit `` `final static``` deklarieren, wird verhindert, dass bei jedem neuen Vorgang derselbe Wert in der Instanz dupliziert wird, wodurch Speicherplatz gespart wird.
public final static String KEY = "happy-toilet";
Es ist ein fester Wert von Happy Toilet, der überall aufgerufen werden kann.
Die statische Methode selbst gehört zur Klasse, nicht zu einer Instanz der Klasse.
Aufrufe können direkt mit `class name.method name
und für Methodenreferenzen in funktionalen Schnittstellen mit
class name :: method name`
erfolgen.
Zusammen mit dem statischen Feld wird es als statisches Element bezeichnet.
Allgemeine Methoden ohne statische Methoden werden als nicht statische Methoden und Instanzmethoden bezeichnet.
Nachfolgend sind die Unterschiede zwischen statischen und Instanzmethodenaufrufen in Methodenreferenzen für funktionale Schnittstellen seit Java8 aufgeführt.
MethodReference.java
public class MethodReference {
public static void main(String[] args) {
//statische Methode
wakeUp(MethodReference::printWakeUp);
System.out.print(" and ");
//Instanzmethode
MethodReference mr = new MethodReference();
mr.goToilet(mr::printGoToilet);
}
interface Human {
void doSomething();
}
static void wakeUp(Human human) {
human.doSomething();
}
static void printWakeUp() {
String wakeUp = "wake up";
System.out.print(wakeUp);
}
void goToilet(Human human) {
human.doSomething();
}
void printGoToilet() {
String toilet = "happy toilet";
System.out.print(toilet);
}
}
Das Ergebnis ist Aufwachen und fröhliche Toilette </ font>.
public class StaticPractice {
public static void main(String[] args) {
Human developer = new Developer();
System.out.println(developer.getMorning());
}
}
class Human {
static String morning = "Happy Toilet";
static String getMorning() {
return morning;
}
}
class Developer extends Human {
static String morning = "Happy Coding";
static String getMorning() {
return morning;
}
}
Das Ergebnis ist natürlich Happy Toilet </ font>.
Wenn Sie es in "Developer developer = new Developer ();" ändern, können Sie vorerst Happy Coding durchführen.
Hier erstelle ich jedoch eine Instanz, um nur "kann nicht überschrieben werden" darzustellen und von dieser Instanz aus aufzurufen, aber in erster Linie ist dies die statische Methode
Human.getMorning();
Developer.getMorning();
Und es ist wünschenswert, von der Klasse selbst zu verweisen, ohne die Instanz zu durchlaufen.
In der Tat, wenn Sie nur den obigen Code an IntelliJ übertragen
Wird als statische Methode bezeichnet, auf die die Klasse selbst nicht verweist.
(Besonderer Dank geht an @kawamnv)
public static void main(String [] args) Dies liegt daran, dass beim Aufrufen der Hauptmethode die Instanz noch nicht im Speicher vorhanden ist. Daher muss die Hauptmethode ausgeführt werden, auch wenn die Klasse, die sie enthält, nicht instanziiert wird.
Es gibt drei Arten von Klassen: innere Klassen, Mitgliedsklassen, lokale Klassen und anonyme Klassen. Die statische Elementklasse ist eine der Elementklassen, und der Deklarationsspeicherort befindet sich im Klassenblock (dieselbe Position wie das Feld und die Methode). Genau genommen ist es jedoch schwierig, eine statische Elementklasse als innere Klasse zu bezeichnen, und es ist korrekter, sie als eine völlig andere Klasse zu beschreiben.
Wenn Sie die Klasse, die die innere Klasse einschließt, als äußere Klasse bezeichnen, Es hat eine relativ schwache Beziehung zur äußeren Klasse und ihren Instanzen und kann auch auf die statischen Mitglieder der äußeren Klasse zugreifen.
Wie andere statische Dinge wird es mit `externem Klassennamen.member Klassenname
`verwendet.
Outer.java
class Outer { //Externe Klasse
//Mitgliedsklasse ➀ statische Klasse
static class StaticInner {
private Integer in;
public Integer getIn() {
return in;
}
public void setIn(Integer in) {
this.in = in;
}
}
//Mitgliedsklasse ➁ Strenge Mitgliedsklasse
class Inner {
private String str;
public String getStr() {
return str;
}
public void setStr(String str) {
this.str = str;
}
}
}
Aufrufe in irrelevanten Klassen und Hauptmethoden sind:
Execute.java
class Execute {
public static void main(String[] args) {
//statische Klasse
Outer.StaticInner staticInner = new Outer.StaticInner();
//Strenge Mitgliederklasse
Outer outer = new Outer();
Outer.Inner inner = outer.new Inner();
}
}
Wie ich im obigen Beispiel aufgenommen habe, ist eine strenge Mitgliedsklasse eng mit den einzelnen Instanzen verbunden, die von der äußeren Klasse erstellt wurden. Daher kann eine neue Klasse nicht ohne eine Instanz der äußeren Klasse erstellt werden. Da es sich nicht um eine statische Sache handelt, können auch nicht statische Mitglieder darauf zugreifen.
static {
//wird bearbeitet
}
Ein statischer Initialisierer ist ein Block, der nur einmal ausgeführt wird, wenn eine Klasse geladen wird (.class-Datei wird geladen). Beschreiben Sie den Prozess, den Sie aufrufen und ausführen möchten, bevor Sie eine Klasse oder die Hauptmethode instanziieren. Natürlich können Sie nur statische Dinge beschreiben.
Gibt es eine Verwendung dafür? Ich dachte,
public final static String VALUE;
static {
VALUE = makeValue();
}
/**
*Generieren Sie unter bestimmten Bedingungen dynamisch Werte
* @Rückgabewert generierter Wert
*/
private static String makeValue() {
// omit
}
Neulich konnte ich diese Art der Nutzung versehentlich ausprobieren. Ich denke, es war genau richtig, weil der Zweck darin bestand, "zu Beginn nur einmal vor der Instanziierung einen neuen Wert zu schaffen" und ihn wiederzuverwenden.
class Something {
void printA() {
System.out.print("t");
}
void printB() {
System.out.print("e");
}
Something () {
System.out.print("l");
printB();
}
static {
System.out.print("i");
}
}
class StaticTest {
static {
System.out.print("T");
}
public static void main(String [] args) {
System.out.print("o");
new Something().printA();
}
}
Das Ergebnis ist Toilette </ font>. Sie können sehen, dass der statische Initialisierer früher aufgerufen wird als der Konstruktor, der schließlich von new aufgerufen wird (Instanziierung).
Eingeführt seit java5.
Die folgende Quelle wird beim Erstellen der Einschränkungsanmerkung für vorherigen Beitrag verwendet.
CustomSize.java
import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.RetentionPolicy.RUNTIME;
@Target({FIELD})
@Retention(RUNTIME)
@Constraint(validatedBy = {CustomSizeValidator.class})
public @interface CustomSize {
// omit
}
Tatsächlich sind `ElementType``` und`
RetentionPolicy``` Aufzählungen mit langen Paketnamen, sodass Sie sie wie oben ordentlich schreiben können.
Wenn es ein statisches Mitglied einer externen Klasse ist, kann es verwendet werden, auch wenn es nicht enum ist.
import static java.lang.System.*;
import static java.lang.Math.*;
Es ist sehr wichtig, die Grundlagen der Grammatik ebenso wie die erste Toilette am Morgen zu bestätigen.
das ist alles
Recommended Posts