如何使用上下文调用另一个 class 的函数 (android, java)

How to call a function from another class with its context (android, java)

每当用户在 viewPager 的项目内按下按钮时,我都会尝试重新加载 viewPager。这种情况下的相关 class 是 SetupActivityAdapter。在 activity class 我已经做了两个函数来重新加载 viewPager:

public void deleteSession(String title) {
        dbHelper = new DatabaseHelper(this);
        ((DatabaseHelper) dbHelper).deleteRow(title);
    }

public void refreshAdapter () {

        dbHelper = new DatabaseHelper(this); 
        Cursor dataCursor = ((DatabaseHelper) dbHelper).getData();
        models = new ArrayList<>(); 

        while (dataCursor.moveToNext()) { 
            models.add(new Model(
                    dataCursor.getString(1),
                    dataCursor.getInt(2),
                    dataCursor.getInt(3),
                    dataCursor.getInt(4)
            ));
        }
        adapter = new Adapter(models, this);
        viewPager.setAdapter(adapter);
        viewPager.setCurrentItem(0, true);
    }

所以基本上,deleteSession() 从数据库中删除特定项目。 refreshAdapter() 从数据库中读取更新的数据,并为viewPager 设置一个新的适配器。 如果我从 SetupActivity class 中的 OnClick 事件中调用它们,这些函数工作正常。但是,如果我尝试从 Adapter class 中的 OnClick 事件调用它们,应用程序会崩溃并出现以下错误:

java.lang.NullPointerException: Attempt to invoke virtual method 'java.io.File android.content.Context.getDatabasePath(java.lang.String)' on a null object reference
        at android.content.ContextWrapper.getDatabasePath(ContextWrapper.java:327)
        at android.database.sqlite.SQLiteOpenHelper.getDatabaseLocked(SQLiteOpenHelper.java:352)
        at android.database.sqlite.SQLiteOpenHelper.getReadableDatabase(SQLiteOpenHelper.java:322)
        at com.example.android.slink.DatabaseHelper.getData(DatabaseHelper.java:69)
        at com.example.android.slink.SetupActivity.refreshAdapter(SetupActivity.java:436)
        at com.example.android.slink.Adapter.onClick(Adapter.java:111)
        at android.view.View.performClick(View.java:6669)
        at android.view.View.performClickInternal(View.java:6638)
        at android.view.View.access00(View.java:789)
        at android.view.View$PerformClick.run(View.java:26145)
        at android.os.Handler.handleCallback(Handler.java:873)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:193)
        at android.app.ActivityThread.main(ActivityThread.java:6863)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:537)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:858)

这是我的SetupActiviy class(减少到相关部分):

package com.example.android.slink;

public class SetupActivity extends AppCompatActivity {

    private SQLiteOpenHelper dbHelper;
    private SQLiteDatabase database;

    ViewPager viewPager;
    Adapter adapter;
    static ArrayList<Model> models;

    public void deleteSession(String title) {
        dbHelper = new DatabaseHelper(this);
        ((DatabaseHelper) dbHelper).deleteRow(title);
    }

    public void refreshAdapter () {
        dbHelper = new DatabaseHelper(this);
        Cursor dataCursor = ((DatabaseHelper) dbHelper).getData();
        models = new ArrayList<>();
        models.add(new Model("", 0,0,0));
        while (dataCursor.moveToNext()) {
            models.add(new Model(
                    dataCursor.getString(1),
                    dataCursor.getInt(2),
                    dataCursor.getInt(3),
                    dataCursor.getInt(4)
            ));
        }
        adapter = new Adapter(models, this);
        viewPager.setAdapter(adapter);
        viewPager.setCurrentItem(0, true);
    }

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

        editTextScreenTitle = findViewById(R.id.editTextScreenTitle);
        relativeLayoutCard = findViewById(R.id.relativeLayoutCard);
        viewPager = findViewById(R.id.viewPager);

        dbHelper = new DatabaseHelper(this);
        Cursor dataCursor = ((DatabaseHelper) dbHelper).getData();

        models = new ArrayList<>();
        models.add(new Model("", 0,0,0));
        while (dataCursor.moveToNext()) {
            models.add(new Model(
                    dataCursor.getString(1),
                    dataCursor.getInt(2),
                    dataCursor.getInt(3),
                    dataCursor.getInt(4)
            ));
        }
        adapter = new Adapter(models, this);
        viewPager.setAdapter(adapter);
    }
}

这里是我的Adapterclass(也删减了相关部分):

package com.example.android.slink;

public class Adapter extends PagerAdapter {

    private List<Model> models;
    private LayoutInflater layoutInflater;
    private Context context;

    SetupActivity setupActivity = new SetupActivity();

    public Adapter(List<Model> models, Context context) {
        this.models = models;
        this.context = context;
    }

    @NonNull
    @Override
    public Object instantiateItem(@NonNull final ViewGroup container, final int position) {
        layoutInflater = LayoutInflater.from(context);
        final View view = layoutInflater.inflate(R.layout.item, container, false);

        TextView buttonDeleteCard = view.findViewById(R.id.textViewDeleteCard);

        buttonDeleteCard.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                setupActivity.deleteSession([...some title...]);
                setupActivity.refreshAdapter();
            }
        });

        container.addView(view, 0);
        return view;
    }
}

我认为这与上下文有关。 我是否必须将上下文从 SetupActiity 传递到 Adapter class,如果是这样,我该怎么做? 还是不可能,我需要从 SetupActivity class 中捕获 Adapter class 中的 OnClick 事件? 我是 java 编程新手,如果有人能帮助我解决这个问题,我将不胜感激!

像这样更改您的 Adapter

public class Adapter extends PagerAdapter {

    private List<Model> models;
    private LayoutInflater layoutInflater;
    private Context context;

    private SetupActivity setupActivity;

    public Adapter(List<Model> models, Context context) {
        this.models = models;
        this.context = context;
        setupActivity = (SetupActivity) context;
    }

    @NonNull
    @Override
    public Object instantiateItem(@NonNull final ViewGroup container, final int position) {
        layoutInflater = LayoutInflater.from(context);
        final View view = layoutInflater.inflate(R.layout.item, container, false);

        TextView buttonDeleteCard = view.findViewById(R.id.textViewDeleteCard);

        buttonDeleteCard.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                setupActivity.deleteSession([...some title...]);
                setupActivity.refreshAdapter();
            }
        });

        container.addView(view, 0);
        return view;
    }
}

为什么会报错?

您将 ContextSetupActivity 传递给您的 Adapter,您将该引用保存在字段 (context) 中,但错误地创建了 SetupActivity

这就是您获得 NullPointerException 的原因。 activity 是你的 Context。它继承自它。所以只需将它投射到你的构造函数中,稍后再使用它。