【问题标题】:Google Maps v2 lag after popping their fragment back from stack谷歌地图 v2 在从堆栈中弹出他们的片段后滞后
【发布时间】:2013-04-28 17:00:11
【问题描述】:

我有一个Activity 和一个MapFragment,我使用FragmentTransaction 以编程方式将其添加到Activity

private static final String MAP_FRAGMENT_TAG = "map";
private MapFragment mapFragment = null;

...

protected void onCreate(Bundle savedInstanceState) {

    ...

    mapFragment = (MapFragment) getFragmentManager().findFragmentByTag(MAP_FRAGMENT_TAG);
    if (mapFragment == null) {
        mapFragment = MapFragment.newInstance();
        FragmentTransaction fragmentTransaction = getFragmentManager().beginTransaction();
        fragmentTransaction.add(R.id.fragment_wrapper, mapFragment, MAP_FRAGMENT_TAG);
        fragmentTransaction.commit();
    }

    ...

}

标准方式。然后我从mapFragment 获取GoogleMap 实例并设置它的设置,设置监听器,用它做一些事情。一切正常。

然后当用户完成地图时,AsyncTask 被触发以显示 ProgressDialog,执行一些操作,将不同的片段放入 fragment_wrapper 并再次关闭 ProgressDialog

private class GetFlightsTask extends AsyncTask<Double, Void, String> {

@Override
protected void onPreExecute() {
    super.onPreExecute();
    // the activity context has been passed to the AsyncTask through its constructor
    loadingFlightsSpinner = new ProgressDialog(context);
    // setting the dialog up
    loadingFlightsSpinner.show();
}

@Override
protected String doInBackground(Double... params) {
    // some pretty long remote API call
    // (loading a JSON file from http://some.website.com/...)
}

@Override
protected void onPostExecute(String flightsJSON) {
    super.onPostExecute(flightsJSON);
    // here I do stuff with the JSON and then I swtich the fragments like this
    FragmentTransaction fragmentTransaction = getFragmentManager().beginTransaction();
    FlightsFragment fragment = new FlightsFragment();
    fragmentTransaction.replace(R.id.fragment_wrapper, fragment);
    fragmentTransaction.addToBackStack(null);
    fragmentTransaction.commit();
    loadingFlightsSpinner.dismiss();
}

一切仍然正常。用户在FlightsFragment 中做了一些事情,然后可能决定返回地图。按下返回按钮,地图再次弹出。这是地图变得滞后的时候。它上面的国家/城市名称加载非常缓慢,它在移动地图时严重滞后......我不知道为什么,我没有做任何事情来弹出 MapFragment 回来。

有趣的是,它会在例如按下 home 按钮然后再次返回应用程序时得到修复...

我做错了什么?

感谢您的任何想法。

【问题讨论】:

  • 修改后的代码没有问题。问题可能出在 AsyncTask 执行中。
  • 请检查新的编辑,谢谢。
  • 顺便说一句,将ProgressDialog 完全移动到AsyncTask 也无济于事。
  • 我非常感谢给问题投反对票的人告诉我它有什么问题,以便我可以编辑它......

标签: android android-fragments android-maps-v2


【解决方案1】:

我运行了一个简单的测试:

public class MapFragmentOnBackStackExample extends FragmentActivity {

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

        FragmentManager fm = getSupportFragmentManager();
        Fragment f = fm.findFragmentById(R.id.fragment_container);
        if (f == null) {
            f = SupportMapFragment.newInstance();
            FragmentTransaction transaction = fm.beginTransaction();
            transaction.add(R.id.fragment_container, f);
            transaction.commit();
        }
    }

    public void onAddFragmentClick(View view) {
        FragmentManager fm = getSupportFragmentManager();
        FragmentTransaction transaction = fm.beginTransaction();
        transaction.replace(R.id.fragment_container, new MyFragment());
        transaction.addToBackStack(null);
        transaction.commit();
    }

    public static class MyFragment extends Fragment {

        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
            TextView textView = new TextView(getActivity());
            textView.setText("MyFragment: " + hashCode());
            return textView;
        }
    }
}

并且看不到任何问题。

我可以在评论 if (f == null) { 时看到问题,让它总是在旋转时创建新片段,这显然是错误的,但这会带来一些怀疑。

你能在内存中同时看到超过 1 个 MapFragment 吗?尝试使用 Eclipse 内存分析器 (MAT)。

【讨论】:

  • 嗯,我以前从未使用过 MAT,所以我不知道获得问题答案的最佳方法是什么。但我尝试转储 HPROF 文件,然后在 MAT 的直方图视图中查找 SupportMapFragment,它找到了一个 SupportMapFragment,然后是一个 SupportMapFragment$a 和一个 SupportMapFragment$b,我真的不知道是什么。我这样做了 3 次——当应用程序启动时,当我切换片段时,当我回到地图时。总是得到相同的结果。
  • 请注意,由于兼容性原因,我现在使用SupportMapFragment 而不是MapFragment,但自从提出问题后代码并没有太大变化,它仍然做同样的事情.也感谢您的时间和回答。
  • @Zabri 我不确定在没有更多信息的情况下可以回答这个问题。您可以创建一个重现问题的简单示例或尝试使用我的代码吗?我不希望您从应用程序中粘贴代码,因为我经常忽略超过 100 行代码的问题。除了 Maps API v2 之外,您还使用任何其他库吗?
  • 我一直在尝试提出一个重现问题的简单示例,但我发现问题可能出在我最初想象的其他地方。请检查我对问题所做的编辑。谢谢。
【解决方案2】:

只有按下后退键才会延迟?

如果这是问题尝试阻止后退按钮或使其退出应用程序试试这个代码:

    @Override
public void onBackPressed(){
        AlertDialog.Builder builder = new AlertDialog.Builder(this);
        builder.setMessage("You wanna leave the aplication?").setPositiveButton("Yes", dialogClickListener)
            .setNegativeButton("No", dialogClickListener).show();

}

或者试试这个代码,它是一种将地图片段放入另一个片段(嵌套地图片段)的方法,它在几周前对我有用:

Java 类:

public class Yourfragment extends Fragment {

    private MapView mMapView;
    private GoogleMap mMap;
    private Bundle mBundle;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View inflatedView = inflater.inflate(R.layout.map_fragment, container, false);

        try {
            MapsInitializer.initialize(getActivity());
        } catch (GooglePlayServicesNotAvailableException e) {
            // TODO handle this situation
        }

        mMapView = (MapView) inflatedView.findViewById(R.id.map);
        mMapView.onCreate(mBundle);
        setUpMapIfNeeded(inflatedView);

        return inflatedView;
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mBundle = savedInstanceState;
    }

    private void setUpMapIfNeeded(View inflatedView) {
        if (mMap == null) {
            mMap = ((MapView) inflatedView.findViewById(R.id.map)).getMap();
            if (mMap != null) {
                setUpMap();
            }
        }
    }

    private void setUpMap() {
        mMap.addMarker(new MarkerOptions().position(new LatLng(0, 0)).title("Marker"));
    }

    @Override
    public void onResume() {
        super.onResume();
        mMapView.onResume();
    }

    @Override
    public void onPause() {
        super.onPause();
        mMapView.onPause();
    }

    @Override
    public void onDestroy() {
        mMapView.onDestroy();
        super.onDestroy();
    }
}

XML:

<com.google.android.gms.maps.MapView
android:id="@+id/map"
android:layout_width="match_parent"
android:layout_height="match_parent" />

将此代码放在执行后:

View inflatedView = inflater.inflate(R.layout.map_fragment, container, false);

try {
    MapsInitializer.initialize(getActivity());
} catch (GooglePlayServicesNotAvailableException e) {
    // TODO handle this situation
}

mMapView = (MapView) inflatedView.findViewById(R.id.map);
mMapView.onCreate(mBundle);
setUpMapIfNeeded(inflatedView);

return inflatedView;

并在 oncreateview 上调用 assynctask

试试这个:

public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    //Call assyncTask
}

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    mBundle = savedInstanceState;
}

private void setUpMapIfNeeded(View inflatedView) {
    if (mMap == null) {
        mMap = ((MapView) inflatedView.findViewById(R.id.map)).getMap();
        if (mMap != null) {
            setUpMap();
        }
    }
}

private void setUpMap() {
    mMap.addMarker(new MarkerOptions().position(new LatLng(0, 0)).title("Marker"));
}

@Override
public void onResume() {
    super.onResume();
    mMapView.onResume();
}

@Override
public void onPause() {
    super.onPause();
    mMapView.onPause();
}

@Override
public void onDestroy() {
    mMapView.onDestroy();
    super.onDestroy();
}
private class GetFlightsTask extends AsyncTask<Double, Void, String> {

@Override
protected void onPreExecute() {
    super.onPreExecute();
    // if I remove the next line, everything gets fixed
    loadingFlightsSpinner.show();
}

@Override
protected String doInBackground(Double... params) {
    // some pretty long remote API call
    // (loading a JSON file from http://some.website.com/flights?...)
    // works fine
    String flightsJSON = loadJSON("flights?flyFrom=CZ&to=...");
}

@Override
protected void onPostExecute(String flightsJSON) {
    super.onPostExecute(flightsJSON);
    loadingFlightsSpinner.dismiss();
    // here I do stuff with the JSON and then replace the fragment
    dohardwork()
}
public view dohardwork(){
    View inflatedView = inflater.inflate(R.layout.map_fragment, container, false);

    try {
        MapsInitializer.initialize(getActivity());
    } catch (GooglePlayServicesNotAvailableException e) {
        // TODO handle this situation
    }

    mMapView = (MapView) inflatedView.findViewById(R.id.map);
    mMapView.onCreate(mBundle);
    setUpMapIfNeeded(inflatedView);

    return inflatedView;
}

【讨论】:

  • 首先感谢您的回答。在按下 back 按钮时结束应用程序不是一种选择。当显示FlightsFragment 并且用户按下返回按钮时,我需要返回地图。
  • 至于第二个建议的解决方案 - 我很快就尝试过了,它似乎有效。每次从堆栈中弹出片段时,地图都会重新加载,并且不再滞后。但它会丢失相机位置、缩放、标记等。我认为如果我实现了保存这些东西并在片段回到前面时恢复它们,它会很好地工作。问题是我的论文的截止日期快到了,现在我没有时间再实施它了。无论如何,除非出现更好的答案(例如用一行代码或其他东西修复它),否则赏金是你的。
  • 对代码做一些测试也许你可以让它像你想要的那样工作
  • 如果您愿意,请查看 Maciej 答案下的 cmets。问题似乎出在其他地方。只是为了赏金的竞争是公平的。谢谢。
  • 我从Activity调用AsyncTask,然后我替换onPostExecute()方法中的片段。您的代码似乎不相关...
【解决方案3】:

我已经通过在 AsyncTaskdoInBackground() 方法的末尾而不是在 onPostExecute() 方法的开头解除 ProgressDialog 来修复它。

这有点奇怪,因为我实际上认为我不应该在 doInBackground() 方法中触摸 UI 中的东西...如果有人想详细说明一下,我很高兴知道它为什么会这样工作这个。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-12-15
    • 2014-07-26
    • 1970-01-01
    • 2013-08-14
    • 1970-01-01
    相关资源
    最近更新 更多