Увеличение кучи (случай фрагмента) на фрагменте ListView с использованием настраиваемого ParseQueryAdapter при быстрой прокрутке

Я использую Parse.com Android SDK. Внутри моего TabsActivity.java, У меня есть SearchFragment который расширяется ListFragment для того, чтобы заполнить ListView с пользовательским ParseQueryAdapter. Внутри моего пользовательского адаптера я объявляю пользовательский макет строки списка search_list_item.xml, Этот макет содержит только ParseImageView,

Моя проблема в том, что когда я быстро прокручиваю список вниз, мой logcat заполняется

I/dalvikvm-heap﹕ Увеличить кучу (случай фрагмента) до... МБ для выделения байтов

и listView возвращается в исходное положение (это означает, что он возвращается к первому элементу). Если, с другой стороны, я медленно прокручиваю список, который могу получить в конце пунктов без этой ошибки. Как я мог это исправить??

Кроме того, если я использую ParseQueryAdapter по умолчанию без настройки строк с search_list_item.xml У меня нет такой проблемы.

Ниже я публикую некоторый код, который, я думаю, будет полезен:

Код search_list_item.xml

<?xml version="1.0" encoding="utf-8"?>

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content">

    <com.parse.ParseImageView
        android:id="@+id/ProfileImage"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />



</RelativeLayout>

Код TabsActivity.Java

public class TabsActivity extends Activity implements SearchFragment.OnFragmentInteractionListener, ActionBar.TabListener {

    /**
     * The {@link android.support.v4.view.PagerAdapter} that will provide
     * fragments for each of the sections. We use a
     * {@link FragmentPagerAdapter} derivative, which will keep every
     * loaded fragment in memory. If this becomes too memory intensive, it
     * may be best to switch to a
     * {@link android.support.v13.app.FragmentStatePagerAdapter}.
     */
    SectionsPagerAdapter mSectionsPagerAdapter;

    /**
     * The {@link ViewPager} that will host the section contents.
     */
    ViewPager mViewPager;

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

        // Set up the action bar.
        final ActionBar actionBar = getActionBar();
        actionBar.setLogo(R.drawable.logo_white);
        actionBar.setDisplayShowTitleEnabled(false);
        actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);

        // Create the adapter that will return a fragment for each of the three
        // primary sections of the activity.
        mSectionsPagerAdapter = new SectionsPagerAdapter(getFragmentManager());

        // Set up the ViewPager with the sections adapter.
        mViewPager = (ViewPager) findViewById(R.id.tabs_pager);
        mViewPager.setAdapter(mSectionsPagerAdapter);

        // When swiping between different sections, select the corresponding
        // tab. We can also use ActionBar.Tab#select() to do this if we have
        // a reference to the Tab.
        mViewPager.setOnPageChangeListener(new ViewPager.SimpleOnPageChangeListener() {
            @Override
            public void onPageSelected(int position) {
                actionBar.setSelectedNavigationItem(position);
            }
        });

        // For each of the sections in the app, add a tab to the action bar.

        actionBar.addTab(
                actionBar.newTab()
                        .setText("Search")
                        .setTabListener(this));

        actionBar.addTab(
                actionBar.newTab()
                        .setText("Secind")
                        .setTabListener(this));

        actionBar.addTab(
                actionBar.newTab()
                        .setText("Third")
                        .setTabListener(this)
        );

        actionBar.addTab(
                actionBar.newTab()
                        .setText("Fourth")
                        .setTabListener(this));


    }


    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.tabs, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();
        if (id == R.id.action_settings) {

        }
        return super.onOptionsItemSelected(item);
    }

    @Override
    public void onTabSelected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
        // When the given tab is selected, switch to the corresponding page in
        // the ViewPager.
        mViewPager.setCurrentItem(tab.getPosition());
    }

    @Override
    public void onTabUnselected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
    }

    @Override
    public void onTabReselected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
    }

    /**
     * A {@link FragmentPagerAdapter} that returns a fragment corresponding to
     * one of the sections/tabs/wizard1.
     */
    public class SectionsPagerAdapter extends FragmentPagerAdapter {

        public SectionsPagerAdapter(FragmentManager fm) {
            super(fm);
        }


        @Override
        public Fragment getItem(int index) {

            switch (index) {
                case 0:
                    // Search fragment activity
              return new SearchFragment();
                case 1:
                    // Flirts fragment activity
                    return new SecondFragment();
                case 2:
                    // Explore fragment activity
                    return new ThirdFragment();
                case 3:
                    //Profile fragment activity
                    return new FourthFragment();
        }

        return null;
    }


        @Override
        public int getCount() {
            // Show 4 total wizard1.
            return 4;
        }

    }


    public void onFragmentInteraction(String id) {
        // The user selected the headline of an article from the HeadlinesFragment
        // Do something here to display that article
    }

}

Код SearchFragment.java

public class SearchFragment extends ListFragment {

    private OnFragmentInteractionListener mListener;
    private CustomDogAdapter mainAdapter;

    /**
     * Mandatory empty constructor for the fragment manager to instantiate the
     * fragment (e.g. upon screen orientation changes).
     */
    public SearchFragment() {
    }

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


        mainAdapter = new CustomAdapter(this.getActivity());


        // Set the ListActivity's adapter to be the PQA
        mainAdapter.setAutoload(false);
        mainAdapter.loadObjects();
        setListAdapter(mainAdapter);


    }



    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        try {
            mListener = (OnFragmentInteractionListener) activity;
        } catch (ClassCastException e) {
            throw new ClassCastException(activity.toString()
                + " must implement OnFragmentInteractionListener");
        }
    }

    @Override
    public void onDetach() {
        super.onDetach();
        mListener = null;
    }


    @Override
    public void onListItemClick(ListView l, View v, int position, long id) {
        super.onListItemClick(l, v, position, id);

        if (null != mListener) {
            // Notify the active callbacks interface (the activity, if the
            // fragment is attached to one) that an item has been selected.
        }
    }

    /**
    * This interface must be implemented by activities that contain this
    * fragment to allow an interaction in this fragment to be communicated
    * to the activity and potentially other fragments contained in that
    * activity.
    * <p>
    * See the Android Training lesson <a href=
    * "http://developer.android.com/training/basics/fragments/communicating.html"
    * >Communicating with Other Fragments</a> for more information.
    */
    public interface OnFragmentInteractionListener {
        // TODO: Update argument type and name
        public void onFragmentInteraction(String id);
    }



}

Код CustomAdapter.java

public class CustomAdapter extends ParseQueryAdapter<ParseObject> {

    public CustomAdapter(Context context) {
        // Use the QueryFactory to construct a PQA that will only show
        // Todos marked as high-pri
        super(context, new ParseQueryAdapter.QueryFactory<ParseObject>() {
            public ParseQuery create() {
                ParseQuery query = new ParseQuery("Photo");
                query.whereEqualTo("imageName", "profileImage");
//                query.setCachePolicy(ParseQuery.CachePolicy.CACHE_ELSE_NETWORK);
                return query;
            }
        });
    }

    // Customize the layout by overriding getItemView
    @Override
    public View getItemView(ParseObject object, View v, ViewGroup parent) {
        if (v == null) {
            v = View.inflate(getContext(), R.layout.search_list_item, null);
        }

        super.getItemView(object, v, parent);

        // Add and download the image
        ParseImageView image = (ParseImageView) v.findViewById(R.id.profileImage);
        ParseFile photoFile = object.getParseFile("imageFile");
        if (photoFile != null) {
            image.setParseFile(photoFile);
            image.loadInBackground();
//                    (new GetDataCallback() {
//                @Override
//                public void done(byte[] data, ParseException e) {
//                    // nothing to do
//                }
//            });

        }


//        // Add the title view
//        TextView titleTextView = (TextView) v.findViewById(R.id.text1);
//        titleTextView.setText(object.getString("title"));
//
//        // Add a reminder of how long this item has been outstanding
//        TextView timestampView = (TextView) v.findViewById(R.id.timestamp);
//        timestampView.setText(object.getCreatedAt().toString());
        return v;
    }

}

Я много раз изучал и следовал этой документации от Parse.com

1) UI-ParseQueryAdapter

2) MealSpotting

Большое спасибо за ваше время и извините за длинный пост, просто постарался быть как можно более объяснительным.

1 ответ

Решение

После более общего поиска по списку просмотров и их производительности, я наткнулся на сообщение в блоге, в котором есть очень интересный последний абзац, в котором говорится:

Никогда не устанавливайте высоту ListView равным wrap_content. Если все ваши данные доступны локально, это может показаться не таким уж плохим, но это становится особенно проблематичным, когда вы этого не делаете. Если вы используете wrap_content для своего ListView, вот что происходит: первый вызов getView завершен, convertView равен нулю и позиция 0 загружена. Теперь позиция 1 загружена, но ей передается только что сгенерированное представление для позиции 0 в качестве его convertView. Затем позиция 2 загружается с тем же видом, и так далее. Это сделано для того, чтобы выложить ListView, так как он должен выяснить, насколько высоким он должен быть, а вы явно не сказали этого. После прохождения всех этих позиций это представление передается обратно в положение 0 для еще одного вызова getView, а затем в положение 1 и далее загружается getView, а не convertView. В итоге вы увидите, что getView вызывается в два или три раза чаще, чем вы ожидали. Это не только отстой в производительности, но вы можете получить некоторые действительно запутанные проблемы.

источник: прочитайте последний абзац

Я не совсем понимаю, так как я новичок в разработке, но после прочтения решил изменить высоту и ширину макета с помощью предопределенных значений dp и установить android:scaleType="centerCrop", Это решило проблему роста кучи, и прокрутка стала действительно плавной.

Я надеюсь, что кто-то найдет это полезным, как и я, и если кто-нибудь сможет объяснить это более технически, мне будет очень интересно его прочитать. Заранее большое спасибо!

Другие вопросы по тегам