0

在提交MapFragment並獲取對其地圖的引用(getMap())後嘗試設置地圖類型時,出現空指針異常。空指針異常使用Android Google Maps API設置MapType

我認爲錯誤的原因是片段尚未初始化,因此我無法設置地圖類型。

我怎麼知道這個片段什麼時候被初始化了,我可以調用它的公共方法?是否有我可以在我的MainActivity中實現的接口來知道片段何時加載?

此外,爲什麼我可以在mMapFragment上調用getMap()(如果尚未初始化)?它實際上只是GoogleMap未正確初始化的對象嗎?

這裏是我的代碼:

public class MainActivity extends Activity { 

    private static final String LOG_TAG = MainActivity.class.getSimpleName(); 
    private FragmentManager fm; 
    private MapFragment mMapFragment; 
    private GoogleMap mGoogleMap; 

    @Override 
    protected void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.main_container); 

     fm = getFragmentManager(); 
     mMapFragment = MapFragment.newInstance(); 

     fm.beginTransaction().add(R.id.fragment_container, mMapFragment).commit(); 
     mGoogleMap = mMapFragment.getMap(); 

     mGoogleMap.setMapType(GoogleMap.MAP_TYPE_TERRAIN); 
    } 
} 

和XML爲main_container

<?xml version="1.0" encoding="utf-8"?> 
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" 
    android:id="@+id/fragment_container" 
    android:layout_width="match_parent" 
    android:layout_height="match_parent" 
    android:orientation="vertical" > 

</FrameLayout> 

而這裏的logcat的輸出:

10-09 11:17:16.457: E/AndroidRuntime(31679): FATAL EXCEPTION: main 
10-09 11:17:16.457: E/AndroidRuntime(31679): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.matthewlogan.loopfinder/com.matthewlogan.loopfinder.MainActivity}: java.lang.NullPointerException 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2248) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2298) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.app.ActivityThread.access$600(ActivityThread.java:142) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1285) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.os.Handler.dispatchMessage(Handler.java:99) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.os.Looper.loop(Looper.java:137) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.app.ActivityThread.main(ActivityThread.java:5270) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at java.lang.reflect.Method.invokeNative(Native Method) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at java.lang.reflect.Method.invoke(Method.java:525) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:974) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:790) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at dalvik.system.NativeStart.main(Native Method) 
10-09 11:17:16.457: E/AndroidRuntime(31679): Caused by: java.lang.NullPointerException 
10-09 11:17:16.457: E/AndroidRuntime(31679): at com.matthewlogan.loopfinder.MainActivity.onCreate(MainActivity.java:36) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.app.Activity.performCreate(Activity.java:5133) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1098) 
10-09 11:17:16.457: E/AndroidRuntime(31679): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2212) 
10-09 11:17:16.457: E/AndroidRuntime(31679): ... 11 more 

回答

3

MapFragmentGoogleMap通過代碼創建準備時:安裝

  1. onCreateView返回
  2. 谷歌Play服務的應用程序,並擁有正確的版本

如果你想獲得非-null GoogleMap在你的Activity,做到這一點onResume

if (mGoogleMap == null) { 
    mGoogleMap = mMapFragment.getMap(); 
    if (mGoogleMap != null) { 
     initMap(); 
    } 
} 

雙重ifs確保initMap僅在Activity實例中被調用一次,並且僅在GoogleMap準備就緒時調用。這是一種正確的處理方式,因爲在用戶安裝Google Play服務並返回到您的應用(onResume第二次)後,GoogleMap可以準備就緒。

1

我怎麼能知道什麼時候該片段已經初始化,我被允許打電話給它ublic方法?

在其生命週期方法之一中配置地圖片段,如onActivityCreated()

或者,不使用FragmentTransaction,而是在佈局中使用<fragment>標記。