【问题标题】:Android Viewpager Fragment Network Call handlingAndroid Viewpager Fragment 网络调用处理
【发布时间】:2021-01-12 08:51:45
【问题描述】:

在 viewpager 片段内部有一个网络调用来加载数据。由于此网络调用,它会为用户造成延迟。我如何处理驻留在 viewpager 内的片段内的网络调用。我只是不希望我的用户看到延迟,因为加载片段需要 4-7 秒。什么是通过网络调用 Viewpager 加载片段的有效方法

下面是 Activity 的代码:

import androidx.appcompat.app.AppCompatActivity;
import androidx.viewpager.widget.ViewPager;

import android.os.Bundle;

import com.test.androidtest20202.R;
import com.test.androidtest20202.adapter.ViewPagerAdapter;

import java.util.ArrayList;
import java.util.List;

public class ViewPagerActivity extends AppCompatActivity {
    ViewPagerAdapter adapter;
    ViewPager viewPager;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_view_pager);
        viewPager = findViewById(R.id.viewpager);

        ArrayList<Integer> task_ids = new ArrayList<>();
        for(int i=0;i<5;i++){
            task_ids.add(i);
        }
        adapter = new ViewPagerAdapter(this, getSupportFragmentManager(),task_ids);
        viewPager.setAdapter(adapter);
    }
}

活动xml布局下方:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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=".activity.ViewPagerActivity">

    <androidx.viewpager.widget.ViewPager
        android:id="@+id/viewpager"
        android:layout_width="0dp"
        android:layout_height="0dp"
        android:background="@color/white"
        android:largeHeap="true"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>

下面是我的 Viewpager 适配器:

import android.content.Context;
import android.os.Bundle;
import android.util.SparseArray;
import android.view.View;
import android.view.ViewGroup;

import androidx.annotation.NonNull;
import androidx.fragment.app.Fragment;
import androidx.fragment.app.FragmentManager;
import androidx.fragment.app.FragmentStatePagerAdapter;

import com.test.androidtest20202.fragments.ViewPagerFragment;

import java.util.ArrayList;

public class ViewPagerAdapter   extends FragmentStatePagerAdapter {
    private ArrayList<Integer> taskid;
    private SparseArray<Fragment> registeredFragments = new SparseArray<>();

    public ViewPagerAdapter(Context context, @NonNull FragmentManager fm, ArrayList<Integer> taskid) {
        super(fm);
        this.taskid = taskid;
    }

    @NonNull
    @Override
    public Fragment getItem(int position) {
        Fragment fragment = null;
        Bundle bundle = new Bundle();
        bundle.putInt("COMPLETED_TASKID", taskid.get(position));
        fragment = new ViewPagerFragment();
        registeredFragments.put(position, fragment);

        fragment.setArguments(bundle);
        return fragment;
    }


    @Override
    public void destroyItem(ViewGroup container, int position, Object object) {
        registeredFragments.remove(position);
        super.destroyItem(container, position, object);
    }

    @Override
    public int getCount() {
        return taskid.size();
    }


}

下面是我的 Viewpager 片段代码:

import android.os.AsyncTask;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.fragment.app.Fragment;

import com.test.androidtest20202.R;
import com.test.androidtest20202.pojo.SaleskenResponse;
import com.test.androidtest20202.util.RestApiClient;
import com.test.androidtest20202.util.RestUrlInterface;

import butterknife.BindView;
import butterknife.ButterKnife;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;

public class ViewPagerFragment  extends Fragment {
    private static final String TAG = "ViewPagerFragment";
    private ViewGroup container;
    private LayoutInflater inflater;
    @BindView(R.id.textView2)
    TextView textView;
    private AsyncTask mAsyncTask;
    public RestUrlInterface restUrlInterface;
    Integer taskid = -1;


    public View onCreateView(LayoutInflater inflater,
                             ViewGroup container, Bundle savedInstanceState) {
        this.container = container;
        this.inflater = inflater;
        return initializeView();
    }
    private View initializeView() {

        final View view;
        view = inflater.inflate(
                R.layout.fragment_laout, container, false);
        ButterKnife.bind(this, view);
        restUrlInterface = RestApiClient.getClient(getContext()).create(RestUrlInterface.class);
        if (getArguments() != null) {
            taskid = getArguments().getInt("COMPLETED_TASKID");
        }
        return view;
    }

    @Override
    public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        Call<SaleskenResponse> task_details = restUrlInterface.getTaskDetail(getString(R.string.token), taskid );
        task_details.enqueue(new Callback<SaleskenResponse>() {
            @Override
            public void onResponse(Call<SaleskenResponse> call, Response<SaleskenResponse> response) {
                switch (response.code()) {
                    case 200:
                        textView.setText( response.body().toString());
                }
            }

            @Override
            public void onFailure(Call<SaleskenResponse> call, Throwable t) {

            }
        });
    }
}

下面是片段布局:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
    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">

    <TextView
        android:id="@+id/textView2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="TextView"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>

【问题讨论】:

  • 一些带有问题的代码会更好地解释您的问题。至少你是如何调用网络服务的以及你如何处理响应
  • 你能贴一些代码吗?
  • 用代码更新了问题
  • 这取决于从网络中检索到的数据的动态程度以及检索到的数据是否取决于用户在该片段或浏览器中的其他片段中输入的内容以及数据需要多新鲜是。查看代码请求看起来是静态的,但不知道它需要多新鲜。
  • 有114个片段,数据自动加载。用户没有为此片段输入任何输入。如果网络调用在片段上需要 3-7 秒,您对此有何建议。我希望缓存该网络调用或缓存片段

标签: android performance android-fragments android-viewpager


【解决方案1】:

因此,只需在您创建 task_ids ArrayList 之后,我将在活动 onCreate 中执行此操作后,才需要加载数据。

您似乎正在使用自定义 RestAPI 包,但大多数包都异步执行网络请求,因此当您收到每个请求的响应时,您将使用唯一的 task_id 作为索引将它们存储在共享视图模型中。

然后可以从片段中观察共享视图模型,当数据可用时,片段将被通知使用它。

所以基本上只要 Activity 启动就开始网络加载,以便在设置 viewpager 和加载第一个 Fragment 之前先启动它,并独立于 Activity 和 Fragment 存储您的数据。

请注意,这并不能保证 viewpager 中的第一个片段在显示时将具有可用的数据,但至少在用户切换到第二页时它应该是使其可用的一部分在 viewpager 中数据应该已经可用。

如何使用共享视图模型在 Fragment 之间共享数据的示例位于 https://blog.mindorks.com/shared-viewmodel-in-android-shared-between-fragments(此示例在 2 个 Fragment 之间,但相同的概念可用于 Activity 中的后台 RestAPI 调用与 viewpager 中的每个 Fragment 之间的通信。

在并行请求所有 task_id 的数据与按顺序从第一页开始,然后一旦加载请求下一个页面,在第一页加载时间上可能需要权衡。

抱歉,没有代码示例(除了关于如何使用共享视图模型的链接),但更多的是架构答案,因为您似乎使用的是我不熟悉的自定义 RestAPI 包。

这基本上是在 Activity 开始时缓存每个 Fragment 的数据,而 Fragments 只显示缓存的数据。

如果您可以在每次启动 Activity 之间进行缓存,您可以使用像 Android Rooms 这样的数据库,而不是共享视图模型,那么当它启动 Activity 时,它会包含上次启动 Activity 的数据,并且真的可以在后台只需通过 RestAPI 请求检查存储在数据库中的数据的新鲜度。

【讨论】:

  • 我有 143 个网络调用,所以你说我应该并行进行所有这些网络调用并存储结果/将数据传递给片段?
  • 对于 143,正如我所提到的,在并行请求和获得第一页的快速响应之间进行权衡,这一点更为重要。对于 143 并行,我认为这会减慢第一个响应。因此,我会按顺序执行该数字,或者您可能会发现更好的性能权衡是分批执行,例如对前 4 个页面进行请求,当每个页面返回时,加载队列中的下一个页面。通常一些 restAPI 库具有限制并行请求数量的功能。
猜你喜欢
  • 2017-06-02
  • 2019-12-19
  • 2013-10-23
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2014-05-03
  • 1970-01-01
相关资源
最近更新 更多