WedX - журнал о программировании и компьютерных науках

ListView getView - если (convertView == null) не вызывается при первой прокрутке

Когда записи впервые появляются на ListView, они выглядят нормально, а getView if (convertView == null) вызывается правильно, поскольку записи видны впервые.
Однако, если я прокручиваю ListView, даже в первый раз, if (convertView == null) не вызывается. Из-за чего дочерний элемент в верхнем элементе повторяется в первом элементе, который появляется при прокрутке. Насколько я понимаю, если ListView прокручивается в первый раз, следует вызвать if (convertView == null).
Пожалуйста, поправьте меня, если я ошибаюсь.
getView :

@Override
    public View getView(int position, View convertView, ViewGroup parent) {

        try {
            ViewHolder holder = null;


            int rowType = getItemViewType(position);
            if (convertView == null) {


                holder = new ViewHolder();

                switch (rowType) {
                    case TYPE_ITEM:
                        convertView = inflater.inflate(R.layout.activity_item, null);

                        activityModel = (truegroups.model.Activity) arrActivity.get(position);

                        holder.activityLayout = (RelativeLayout) convertView.findViewById(R.id.activity_item_layout);
                        holder.startDate = (TextView) convertView.findViewById(R.id.lblStartDate);
                        holder.line = (TextView) convertView.findViewById(R.id.lblLine);
                        holder.endDate = (TextView) convertView.findViewById(R.id.lblEndDate);
                        holder.groupName = (TextView) convertView.findViewById(R.id.lblGroupName);
                        holder.activity = (TextView) convertView.findViewById(R.id.lblActivity);
                        holder.location = (TextView) convertView.findViewById(R.id.lblLocation);
                        holder.topInviteeLayout = (RelativeLayout) convertView.findViewById(R.id.invitationTopPanel);

                        int iCount = 0;
                        for (Invitation invitation : activityModel.getArrInvitations()) {

                            ImageView imgTopStatus = new ImageView(activity);

                            RelativeLayout.LayoutParams imgStatusParams = new RelativeLayout.LayoutParams(40, 40);
                            imgStatusParams.setMarginStart(160);
                            imgStatusParams.setMargins(160, 20, 0, 0);

                            if (iCount > 0)
                                imgStatusParams.addRule(RelativeLayout.BELOW, holder.arrImageStatus.get(iCount - 1).getId());

                            imgTopStatus.setId(iCount * iCount + iCount + 1);
                            holder.topInviteeLayout.addView(imgTopStatus, imgStatusParams);
                            holder.arrImageStatus.add(imgTopStatus);

                            TextView topInviteeName = new TextView(activity);

                            RelativeLayout.LayoutParams topInviteeParams = new RelativeLayout.LayoutParams(300, ViewGroup.LayoutParams.WRAP_CONTENT);
                            topInviteeParams.addRule(RelativeLayout.RIGHT_OF, imgTopStatus.getId());
                            if (iCount > 0)
                                topInviteeParams.addRule(RelativeLayout.BELOW, holder.arrlblTopInviteeName.get(iCount - 1).getId());


                            topInviteeParams.setMargins(15, 5, 5, 15);
                            topInviteeName.setTextColor(Color.BLACK);
                            topInviteeName.setTextSize(14);
                            topInviteeName.setBackgroundColor(Color.WHITE);
                            topInviteeName.setId(iCount * iCount + 2 * iCount + 2);
                            //topInviteeName.setText(invitation.getChildName());
                            holder.topInviteeLayout.addView(topInviteeName, topInviteeParams);
                            holder.arrlblTopInviteeName.add(topInviteeName);

                            //topInviteeParams.addRule(RelativeLayout.ALIGN_TOP, imgTopStatus.getId());
                           // imgStatusParams.addRule(RelativeLayout.ALIGN_BASELINE, topInviteeName.getId());

                            Button decline = new Button(activity);

                            RelativeLayout.LayoutParams paramsDecline = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, 80);
                            paramsDecline.addRule(RelativeLayout.ALIGN_PARENT_RIGHT, RelativeLayout.TRUE);
                            if (iCount > 0)
                                paramsDecline.addRule(RelativeLayout.BELOW, holder.arrDecline.get(iCount - 1).getId());

                            decline.setTextColor(Color.BLACK);
                            decline.setTextSize(9);
                            decline.setText("Decline");
                            decline.setId(iCount * iCount + 3 * iCount + 3);
                            holder.topInviteeLayout.addView(decline, paramsDecline);
                            holder.arrDecline.add(decline);

                            Button accept = new Button(activity);

                            RelativeLayout.LayoutParams paramsAccept = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, 80);

                            if (iCount > 0)
                                paramsAccept.addRule(RelativeLayout.BELOW, holder.arrAccept.get(iCount - 1).getId());
                            paramsAccept.addRule(RelativeLayout.RIGHT_OF, topInviteeName.getId());

                           // paramsAccept.addRule(RelativeLayout.ALIGN_BOTTOM, decline.getId());
                            paramsAccept.setMarginStart(155);

                            accept.setTextColor(Color.BLACK);
                            accept.setTextSize(9);
                            accept.setId(iCount * iCount + 8 * iCount + 8);
                            accept.setText("Accept");
                            holder.topInviteeLayout.addView(accept, paramsAccept);
                            holder.arrAccept.add(accept);

                            paramsDecline.addRule(RelativeLayout.ALIGN_TOP, accept.getId());

                            iCount++;
                        }

                        break;
                    case TYPE_SEPARATOR:
                        convertView = inflater.inflate(R.layout.activity_header, null);

                        holder.header = (TextView) convertView.findViewById(R.id.textSeparator);
                        holder.headerWeek = (TextView) convertView.findViewById(R.id.textWeekNumber);
                        break;
                }

                convertView.setTag(holder);
            } else {
                holder = (ViewHolder) convertView.getTag();
            }

            if (arrActivity.size() <= 0) {
                Log.d("Data", "No data found !!");

            } else {
                switch (rowType) {
                    case TYPE_ITEM:
                        activityModel = (truegroups.model.Activity) arrActivity.get(position);
//
                        holder.groupName.setText(activityModel.getGroupName());
                        holder.activity.setText(activityModel.getTitle());
                        holder.location.setText(activityModel.getLocation());
                        holder.startDate.setText(getHours(activityModel.getStartDate()));
                        holder.endDate.setText(getHours(activityModel.getEndDate()));



                        int iCount = 0;
                        for (Invitation invitation : activityModel.getArrInvitations()) {
                           ImageView imgStatus = holder.arrImageStatus.get(iCount);
                            //ImageView imgStatus = (ImageView) holder.topInviteeLayout.getChildAt(0);
                            if (invitation.getResponse().equals("3"))
                                imgStatus.setImageResource(R.drawable.notresponded);
                            else if (invitation.getResponse().equals("1"))
                                imgStatus.setImageResource(R.drawable.attending);
                            else
                                imgStatus.setImageResource(R.drawable.decline);

                           TextView inviteeName = holder.arrlblTopInviteeName.get(iCount);
                            //TextView inviteeName = (TextView) holder.topInviteeLayout.getChildAt(1);
                            inviteeName.setText(invitation.getChildName());

                            iCount++;
                        }


                        convertView.setOnClickListener(new OnItemClickListener(position));
                        break;
                    case TYPE_SEPARATOR:

                        holder.header.setText(getHeaderDate(String.valueOf(arrActivity.get(position))));

                        Calendar calendarGivenDate = Calendar.getInstance();
                        calendarGivenDate.set(Calendar.DAY_OF_MONTH, Integer.parseInt(String.valueOf(arrActivity.get(position)).split("/")[1]));
                        calendarGivenDate.set(Calendar.MONTH, Integer.parseInt(String.valueOf(arrActivity.get(position)).split("/")[0]));
                        calendarGivenDate.set(Calendar.YEAR, Integer.parseInt(String.valueOf(arrActivity.get(position)).split("/")[2]));

                        int weekNumber = calendarGivenDate.get(Calendar.WEEK_OF_YEAR);
                        holder.headerWeek.setText(StringUtils.join("WEEK ", String.valueOf(weekNumber)));

                        break;
                }

            }

            return convertView;
        } catch (Exception e) {
            e.printStackTrace();
            Toast.makeText(activity, e.getMessage(), Toast.LENGTH_LONG).show();
            Log.d("Exception : ", e.getMessage());
            return convertView;
        }
    }

  • Это на самом деле не связано с вашим вопросом, но когда вы помещаете цикл for в свой код getView, вы потенциально немного снижаете производительность. Подумайте о том, чтобы вычислить данные из этого цикла for и сохранить их где-нибудь вместо того, чтобы вычислять их во время прокрутки. 29.07.2015

Ответы:


1

Поскольку ListView реализует перезапуск представления, вы никогда не должны получить случай, когда convertView == null после его инициализации.

Когда вы инициализируете ListView, он создает строки и вызывает для них getView до тех пор, пока экран не заполнится строками. Предположим, это число — 10 рядов, пронумерованных от 0 до 9, а 10-й ряд виден лишь частично. Когда вы прокручиваете вниз, строка 0 прокручивается вне поля зрения, а новая строка должна отображаться внизу. Вместо того, чтобы инициализировать новый блок памяти для строки 10, он берет строку 0 и отдает ее вам в вашей функции getView(). Если вы не измените значения, они снова появятся в нижней части списка со всеми исходными значениями строки 0. В этом примере единственный раз, когда вы получите null convertView, это если вы вы используете несколько типов объектов (например, разделитель), и при прокрутке на самом деле необходимо создать новую строку до того, как предыдущая строка исчезнет с экрана.

Таким образом, вам нужно сначала создать строку, если converView имеет значение null, но отдельно от этого вам нужно (повторно) инициализировать все значения внутри представления впоследствии.

Лучшая стратегия для написания функции getView следует следующей базовой структуре:

if (convertView == null) {
    convertView = new WhateverViewTypeYouNeed();
}

if (convertView != SEPARATOR) {
    initializeValues();
}
return convertView;

Вы можете просмотреть эту ссылку для дальнейшего чтения.

28.07.2015
  • Где именно я должен повторно инициализировать все значения внутри представления? 29.07.2015
  • Я обновил свой ответ. Надеюсь, теперь это имеет больше смысла. 29.07.2015
  • Новые материалы

    Как создать диаграмму градиентной кисти с помощью D3.js
    Резюме: Из этого туториала Вы узнаете, как добавить градиентную кисть к диаграмме с областями в D3.js. Мы добавим градиент к значениям SVG и применим градиент в качестве заливки к диаграмме с..

    Я хотел выучить язык программирования MVC4, но не мог выучить его раньше, потому что это выглядит сложно…
    Просто начните и учитесь самостоятельно Я хотел выучить язык программирования MVC4, но не мог выучить его раньше, потому что он кажется мне сложным, и я бросил его. Это в основном инструмент..

    Лицензии с открытым исходным кодом: руководство для разработчиков и создателей
    В динамичном мире разработки программного обеспечения открытый исходный код стал мощной парадигмой, способствующей сотрудничеству, инновациям и прогрессу, движимому сообществом. В основе..

    Объяснение документов 02: BERT
    BERT представил двухступенчатую структуру обучения: предварительное обучение и тонкая настройка. Во время предварительного обучения модель обучается на неразмеченных данных с помощью..

    Как проанализировать работу вашего классификатора?
    Не всегда просто знать, какие показатели использовать С развитием глубокого обучения все больше и больше людей учатся обучать свой первый классификатор. Но как только вы закончите..

    Работа с цепями Маркова, часть 4 (Машинное обучение)
    Нелинейные цепи Маркова с агрегатором и их приложения (arXiv) Автор : Бар Лайт Аннотация: Изучаются свойства подкласса случайных процессов, называемых дискретными нелинейными цепями Маркова..

    Crazy Laravel Livewire упростил мне создание электронной коммерции (панель администратора и API) [Часть 3]
    Как вы сегодня, ребята? В этой части мы создадим CRUD для данных о продукте. Думаю, в этой части я не буду слишком много делиться теорией, но чаще буду делиться своим кодом. Потому что..


    Для любых предложений по сайту: [email protected]