我知道我能给
setOffscreenPageLimit(int)
的最低数字是1。但我需要一次加载一页,因为内存问题。
我是否必须使用旧式 tabhost 等?或者有什么方法/技巧可以让我的 viewPager 一次加载一页?
我的适配器使用 ViewHolder 模式扩展了 BaseAdapter。
我遇到了同样的问题,我找到了解决方案:
步骤:
1)首先从
this链接下载
CustomViewPager
类。
2) 如下所述使用该类:
在 Java 中:
CustomViewPager mViewPager;
mViewPager = (CustomViewPager) findViewById(R.id.swipePager);
mViewPager.setOffscreenPageLimit(0);
在 XML 中:
<com.yourpackagename.CustomViewPager
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/swipePager"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
现在一次只会加载一页。
P.S:根据问题的要求,我已经发布了
Viewpager
的解决方案。我还没有用 TabLayout
尝试过同样的方法。如果我能找到任何解决方案,我会更新答案。
在此文件中,使用了 KeyEventCompat Android studio 可能无法找到它,因为 KeyEnentCompat 类在 API 级别 26.0.0 中已弃用,因此您需要将 KeyEventCompat 替换为事件以获取更多详细信息,您可以查看 https://developer.android.com/sdk/support_api_diff/26.0.0-alpha1/changes/android.support.v4.view.KeyEventCompat
据我所知,这是 使用 ViewPager 时不可能的。至少不是,当你希望你的页面可以滑动时。
因此解释很简单:
当你在两个页面之间滑动时,有一个点需要两个页面都可见,因为 你不能在两个东西之间滑动,而其中一个在那个时候甚至不存在。
有关更多信息,请参阅此问题:ViewPager.setOffscreenPageLimit(0) doesn't work as expected
CommonsWare 在他的answer 的评论中提供了很好的解释。
但是因为内存问题,我需要一次加载一页。
假设你得到
OutOfMemoryError
s.
我是否必须使用旧式 tabhost 等?
是的,或
FragmentTabHost
,或操作栏选项卡。
或者有什么方法/技巧可以让我的 viewPager 一次加载一页?
没有,原因很简单
ViewPager
一次需要不止一页的滑动动画。您可以使用 ViewPager
并滑动来查看。
或者,您可以着手解决您感知到的记忆问题。假设此应用与您今天早些时候报告的应用相同,您只使用了 7MB 的堆空间。如果您剩余的堆高度碎片化,那只会导致
OutOfMemoryError
s。内存管理策略(例如,inBitmap
上的 BitmapOptions
用于从外部源创建位图)有助于解决此类碎片问题。
我的适配器使用 ViewHolder 模式扩展了 BaseAdapter。
BaseAdapter
与 AdapterView
一起使用,而不是 ViewPager
.
我有一个答案。上述方法 setUserVisibleHint() 已弃用,您可以使用 setMaxLifecycle() 方法。要仅加载可见片段,您必须在 viewpager 适配器中将行为设置为 BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT。 IE;在构造函数中。为了处理片段,请在片段中使用 onResume() 方法。
通过这种方式,您一次只能在 viewpager 中加载一个片段。
public static class MyAdapter extends FragmentStatePagerAdapter {
public MyAdapter(FragmentManager fm) {
super(fm, BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT);
}
@Override
public int getCount() {
return NUM_ITEMS;
}
@Override
public Fragment getItem(int position) {
return ArrayListFragment.newInstance(position);
}
}
在科特林中:
class MyAdapter(fm: FragmentManager) : FragmentStatePagerAdapter(fm,BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT )
也以同样的方式与
FragmentPagerAdapter
(现已弃用)一起使用
通过使用此方法,您可以使用视图寻呼机在选项卡布局中一次加载一页`
@Override
public void onResume() {
super.onResume();
if (getUserVisibleHint() && !isVisible) {
Log.e("~~onResume: ", "::onLatestResume");
//your code
}
isVisible = true;
}
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
if (isVisibleToUser && isVisible) {
Handler handler = new Handler();
handler.postDelayed(new Runnable() {
@Override
public void run() {
//your code
}
}, 500);
}
}
`
覆盖
setUserVisibleHint
并在每个片段中添加postDelayed
如下所示。
override fun setUserVisibleHint(isVisibleToUser: Boolean) {
if (isVisibleToUser)
Handler().postDelayed({
if (activity != null) {
// Do you stuff here
}
}, 200)
super.setUserVisibleHint(isVisibleToUser)
}
我可以通过这种方式进行管理,现在它对我来说工作正常。
首先,复制 SmartFragmentStatePagerAdapter.java,它在我们的 ViewPager 中提供已注册片段的智能缓存。它通过覆盖 instantiateItem() 方法并在内部缓存任何创建的片段来实现。这解决了需要访问 ViewPager 中的当前项的常见问题。
现在,我们想在声明我们的适配器时从上面复制的 SmartFragmentStatePagerAdapter 进行扩展,这样我们就可以利用状态分页器更好的内存管理:
public abstract class SmartFragmentStatePagerAdapter extends FragmentStatePagerAdapter {
// Sparse array to keep track of registered fragments in memory
private SparseArray<Fragment> registeredFragments = new SparseArray<Fragment>();
public SmartFragmentStatePagerAdapter(FragmentManager fragmentManager) {
super(fragmentManager);
}
// Register the fragment when the item is instantiated
@Override
public Object instantiateItem(ViewGroup container, int position) {
Fragment fragment = (Fragment) super.instantiateItem(container, position);
registeredFragments.put(position, fragment);
return fragment;
}
// Unregister when the item is inactive
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
registeredFragments.remove(position);
super.destroyItem(container, position, object);
}
// Returns the fragment for the position (if instantiated)
public Fragment getRegisteredFragment(int position) {
return registeredFragments.get(position);
}
}
// Extend from SmartFragmentStatePagerAdapter now instead for more dynamic ViewPager items
public static class MyPagerAdapter extends SmartFragmentStatePagerAdapter {
private static int NUM_ITEMS = 3;
public MyPagerAdapter(FragmentManager fragmentManager) {
super(fragmentManager);
}
// Returns total number of pages
@Override
public int getCount() {
return NUM_ITEMS;
}
// Returns the fragment to display for that page
@Override
public Fragment getItem(int position) {
switch (position) {
case 0: // Fragment # 0 - This will show FirstFragment
return FirstFragment.newInstance(0, "Page # 1");
case 1: // Fragment # 0 - This will show FirstFragment different title
return FirstFragment.newInstance(1, "Page # 2");
case 2: // Fragment # 1 - This will show SecondFragment
return SecondFragment.newInstance(2, "Page # 3");
default:
return null;
}
}
// Returns the page title for the top indicator
@Override
public CharSequence getPageTitle(int position) {
return "Page " + position;
}
}
您实际上不需要自定义 ViewPager。
我有同样的问题,我喜欢这个。
setOffscreenPageLimit()
为1.onResume
和onPause
生命周期方法。在这些生命周期方法上初始化和释放内存。
我知道这是一篇旧帖子,但我偶然发现了这个问题,并找到了一个很好的解决方案,如果你的加载片段。简单地说,通过重写 setUserVisibleHint() 来检查用户是否看到片段。之后加载数据。
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
if (isVisibleToUser) {
getData(1, getBaseUrl(), getLink());
}
}