Android Studio - Почему может потребоваться, чтобы ViewById (R.id.recyclerview) не работал в API 22 (в 28) - PullRequest
0 голосов
/ 05 апреля 2019

Короче говоря , API 28 все хорошо; Приложение API 22 вылетает в MainActivity.onCreate с

java.lang.NoSuchMethodError: No virtual method requireViewById

в следующей строке:

RecyclerView recyclerView = requireViewById(R.id.recyclerview);

Теперь длинное объяснение:

Я настраиваю приложение Android Studio. У меня есть один MVVM-набор, включающий RecyclerView (в пределах android.support.design.widget.CoordinatorLayout) в моем activity_main.xml; android.support.v7.widget.RecyclerView имеет tools:listitem="@layout/parcel_item"; parcel_item.xml является android.support.v7.widget.CardView.

Все выглядит нормально под API 28. Когда я работаю с API 22 - либо на моем физическом планшете Samsung Galaxy Note (Android 5.1.1 API 22), либо на виртуальном Nexus 7 (2012) API 22 - приложение вылетает почти сразу же. В MainActivity.onCreate:

RecyclerView recyclerView = requireViewById(R.id.recyclerview);

с:

E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.zazzem.thatsthat, PID: 5613
    java.lang.NoSuchMethodError: No virtual method requireViewById(I)Landroid/view/View; in class Lcom/zazzem/thatsthat/MainActivity; or its super classes (declaration of 'com.zazzem.thatsthat.MainActivity' appears in /data/app/com.zazzem.thatsthat-1/split_lib_slice_7_apk.apk)
        at com.zazzem.thatsthat.MainActivity.onCreate(MainActivity.java:26)
        at android.app.Activity.performCreate(Activity.java:5990)
        at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1106)
        at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2278)
        at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2387)
        at android.app.ActivityThread.access$800(ActivityThread.java:151)
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1303)
        at android.os.Handler.dispatchMessage(Handler.java:102)
        at android.os.Looper.loop(Looper.java:135)
        at android.app.ActivityThread.main(ActivityThread.java:5254)
        at java.lang.reflect.Method.invoke(Native Method)
        at java.lang.reflect.Method.invoke(Method.java:372)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:698)

Проблема, No virtual method requireViewById, по-видимому, подразумевает виртуальный метод, который где-то не был переопределен, но - как я упоминал ранее - он прекрасно работает в API 28.

В Интернете я не нашел ничего конкретного, касающегося "Нет виртуального метода requireViewById", и я попробовал все, что было предложено в различных публикациях, относящихся к "java.lang.NoSuchMethodError", включая File | Отменить кеширование / перезапустить и проверить мои скрипты Gradle (насколько я могу; я очень новичок в Android Studio и Java). Все безрезультатно.

MainActivity.java

//MainActivity.java
package com.zazzem.thatsthat;

import android.arch.lifecycle.Observer;
import android.arch.lifecycle.ViewModelProviders;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.View;
import android.widget.Toast;

import java.util.List;

public class MainActivity extends AppCompatActivity {
    private ParcelViewModel parcelViewModel;

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

        Log.d("##","before"); //this I get <<====================
        RecyclerView recyclerView = requireViewById(R.id.recyclerview);
        Log.d("##","after"); //this I don't <<====================
        recyclerView.setLayoutManager(new LinearLayoutManager(this));
        recyclerView.setHasFixedSize(true);

        final ParcelAdapter adapter = new ParcelAdapter();
        recyclerView.setAdapter(adapter);

        parcelViewModel = ViewModelProviders.of(this).get(ParcelViewModel.class);
        parcelViewModel.getAllParcels().observe(this, new Observer<List<Parcel>>() {
            @Override
            public void onChanged(@Nullable List<Parcel> parcels) {
                adapter.setParcels(parcels);
            }
        });
    }

    public void menuButtonClick(View view) {
        Toast.makeText(this, "menuButtonClick", Toast.LENGTH_SHORT).show();
    }
} //MainActivity

Любая помощь с благодарностью.

Также: это мой первый в истории Android Studio / java пост здесь (где угодно). Если что-то я делаю неправильно (или не совсем правильно) в отношении публикации, пожалуйста, дайте мне знать.

Помощь | Проверить наличие обновлений ... все текущие

версии и такие:

Android Studio 3.3.2

SDK Платформы: -Android 9.0 (Pie) 28 (Revision 6) установлен -Android 8.1 (Oreo) 27 (Revision 3) частично установлен -Android 5.1 (Lollipop) 22 (Редакция 2) Частично установлен

SDK Инструменты: -Android SDK Build-Tools 29-RC2 Android-эмулятор b28.0.25 -Android SDK Platform-Tools v28.0.2 -Android SDK Tools v26.1.1 -Поддержка репозитория (все 4 установлены: -, -, 47.0.0,58)

Спасибо.

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">
    <android.support.v7.widget.RecyclerView
        android:id="@+id/recyclerview"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="@color/facadeLight"
        tools:listitem="@layout/parcel_item" />

    <android.support.design.widget.FloatingActionButton
        android:id="@+id/button_menu"
        ...
        android:tint="@color/facadeLight" />

</android.support.design.widget.CoordinatorLayout>

parcel_item.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.v7.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_marginStart="24dp"
    android:layout_marginTop="12dp"
    android:layout_marginEnd="24dp">
    <RelativeLayout
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:background="@drawable/parcel_theme">

        <TextView
            android:id="@+id/textview_Descr"
            ...
            android:textSize="60sp" />
        <TextView
            android:id="@+id/textview_ParcelID"
            ...
            android:textSize="12sp" />
        <TextView
            android:id="@+id/textview_NrPuzzles"
            ...
            android:textSize="20sp" />
        <TextView
            android:id="@+id/textview_AdCopy"
            ...
            android:text="Ad Copy" />
    </RelativeLayout>
</android.support.v7.widget.CardView>

наконец, скрипты Gradle:

build.gradle (проект: ThatsThat)

buildscript {
    repositories {
        google()
        jcenter()
    }
    dependencies {
        classpath 'com.android.tools.build:gradle:3.3.2'
    }
}

allprojects {
    repositories {
        google()
        jcenter()
    }
}

task clean(type: Delete) {
    delete rootProject.buildDir
}

и build.gradle (Модуль: приложение)

apply plugin: 'com.android.application'

android {
    compileSdkVersion 28
    defaultConfig {
        applicationId "com.zazzem.thatsthat"
        minSdkVersion 19
        targetSdkVersion 28
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }
}

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation 'com.android.support:appcompat-v7:28.0.0'
    implementation 'com.android.support:design:28.0.0'
    implementation 'com.android.support:cardview-v7:28.0.0'
    implementation 'com.android.support.constraint:constraint-layout:1.1.3'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'

    //Lifecycle components
    implementation 'android.arch.lifecycle:extensions:1.1.1'
    annotationProcessor 'android.arch.lifecycle:compiler:1.1.1'

    //Room components
    implementation 'android.arch.persistence.room:runtime:1.1.1'
    annotationProcessor 'android.arch.persistence.room:compiler:1.1.1'
}

Ответы [ 2 ]

1 голос
/ 05 апреля 2019

Это findViewById, не требующий ViewViewById

RecyclerView recyclerView = requireViewById(R.id.recyclerview);

должно быть:

RecyclerView recyclerView = findViewById(R.id.recyclerview);

Также, если вы используете MVVM, вы можете воспользоваться библиотекой привязки данных:)

0 голосов
/ 05 апреля 2019
RecyclerView recyclerView = findViewById(R.id.recyclerview);

Вы должны обновить вышеуказанную строку.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...