Niektórzy pytają: jak singleton może zwrócić wskaźnik zerowy?
Odpowiadam na to pytanie. (Nie mogę odpowiedzieć w komentarzu, ponieważ muszę wysłać kod).
Może zwracać wartość null pomiędzy dwoma zdarzeniami: (1) klasa jest ładowana i (2) tworzony jest obiekt tej klasy. Oto przykład:
class X {
static X xinstance;
static Y yinstance = Y.yinstance;
X() {xinstance=this;}
}
class Y {
static X xinstance = X.xinstance;
static Y yinstance;
Y() {yinstance=this;}
}
public class A {
public static void main(String[] p) {
X x = new X();
Y y = new Y();
System.out.println("x:"+X.xinstance+" y:"+Y.yinstance);
System.out.println("x:"+Y.xinstance+" y:"+X.yinstance);
}
}
Uruchommy kod:
$ javac A.java
$ java A
x:X@a63599 y:Y@9036e
x:null y:null
Drugi wiersz pokazuje, że instancja Y.xinstance i X.yinstance ma wartość NULL ; są one puste, ponieważ zmienne X.xinstance i Y.yinstance zostały odczytane, gdy były puste.
Czy można to naprawić? Tak,
class X {
static Y y = Y.getInstance();
static X theinstance;
static X getInstance() {if(theinstance==null) {theinstance = new X();} return theinstance;}
}
class Y {
static X x = X.getInstance();
static Y theinstance;
static Y getInstance() {if(theinstance==null) {theinstance = new Y();} return theinstance;}
}
public class A {
public static void main(String[] p) {
System.out.println("x:"+X.getInstance()+" y:"+Y.getInstance());
System.out.println("x:"+Y.x+" y:"+X.y);
}
}
a ten kod nie wykazuje anomalii:
$ javac A.java
$ java A
x:X@1c059f6 y:Y@152506e
x:X@1c059f6 y:Y@152506e
ALE nie jest to opcja dla Application
obiektu z Androidem : programista nie kontroluje czasu jego utworzenia.
Jeszcze raz: różnica między pierwszym przykładem a drugim polega na tym, że drugi przykład tworzy instancję, jeśli wskaźnik statyczny ma wartość NULL. Ale programista nie może tworzyć z Androidem obiektu aplikacji zanim system zdecyduje się to zrobić.
AKTUALIZACJA
Jeszcze jeden zagadkowy przykład, w którym znajdują się zainicjowane pola statyczne null
.
Main.java :
enum MyEnum {
FIRST,SECOND;
private static String prefix="<", suffix=">";
String myName;
MyEnum() {
myName = makeMyName();
}
String makeMyName() {
return prefix + name() + suffix;
}
String getMyName() {
return myName;
}
}
public class Main {
public static void main(String args[]) {
System.out.println("first: "+MyEnum.FIRST+" second: "+MyEnum.SECOND);
System.out.println("first: "+MyEnum.FIRST.makeMyName()+" second: "+MyEnum.SECOND.makeMyName());
System.out.println("first: "+MyEnum.FIRST.getMyName()+" second: "+MyEnum.SECOND.getMyName());
}
}
I dostajesz:
$ javac Main.java
$ java Main
first: FIRST second: SECOND
first: <FIRST> second: <SECOND>
first: nullFIRSTnull second: nullSECONDnull
Pamiętaj, że nie można przenieść deklaracji zmiennej statycznej o jeden wiersz wyżej, kod nie zostanie skompilowany.
<application>
węzeł pliku AndroidManifest.xml zawierać następującą definicję atrybutu:android:name="MyApp"
. Moja aplikacja musi znajdować się w tym samym pakiecie, do którego odwołują się twoje manifesty.