Android技术栈(一)从Activity迁移到Fragment,面试安卓开发工程师应该问什么

requireFragmentManager()
.beginTransaction()
.add(id, fragment)
.runOnCommit(()->{/TODO/})
.addToBackStack(null)
.commit();

当然FragmentTransaction不止可以执行add操作,同样也可以执行remove,show,hide等操作.

这里插入一个简短的题外话作为上面知识的补充。如何在Android Studio中启用Java8?在你模块的build.gradle

android{

//省略…

//加上下面的脚本代码,然后sync你的项目

compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
}

onBackPressed在哪?我知道第一次使用Fragment的人肯定都超想问这个问题.众所周知Fragment本身是没有onBackPressed的.不是Google不设计,而是真的没法管理啊!!!,如果一个界面上有三四个地方都有Fragment存在,一按回退键,谁知道要交给哪个Fragment处理呢?所以Fragment是"没有"onBackPressed的.

在这里我的“没有”打了引号,因为实际上给Fragment添加类似onBackPressed的功能的办法是存在的,只是Google把它设计成交给开发者自行管理了.

要想使用FragmentonBackPressed,你可能需要先升级到AndroidX.

这里可能有人会问AndroidX是什么?

简单来讲AndroidX就是一个与平台解绑的appcompat(低版本兼容高版本功能)库,也就是说在build.gradle中不需要再与compileSdkVersion写成一样,例如之前这样的写法:

compile ‘com.android.support:appcompat-v7:24.+’

(注:使用24.+则表明使用 24. 开头的版本的最新版本,若直接使用+号则表明直接使用该库的最新版本。

现在可以写成:

implementation ‘androidx.appcompat:appcompat:1.1.0-alpha02’

(注:新的依赖方式implementationcompile功能相同,但是implementation无法在该模块内引用依赖的依赖,但compile可以,这么做的好处是可以加快编译速度。新的依赖方式apicompile完全相同,只是换了名字而已)

Android Studo中的Refactor->Migrate to AndroidX的选点击之后即可将项目迁移到AndroidX,在确认的时会提示你将项目备份以免迁移失败时丢失原有项目,通常情况下不会迁移失败,只是迁移的过程会花费很多的时间,如果项目很大,迁移时间会很长,这时即使Android StudioCPU利用率为0也不要关闭, 但是如果发生迁移失败,这时候就需要手动迁移了。

一些使用gradle依赖的一些第三方库中的某些类可能继承了android.support.v4包下的Fragment,但迁移到AndroidXappcompatFragment变成了androidx.fragment.app包下,原有的代码下会画红线,Android Studio也会警告你出现错误,但是不用担心,依然可以正常编译,Android Studio在编译的时候会自动完成基类的替换,但前提是你要确保你项目里的gradle.properties进行了如下设置。

android.useAndroidX=true

android.enableJetifier=tru

为了消除这些难看的红线,你可以直接将新的Fragment使用这种方式强制转换成原有的Fragment

TextureSupportMapFragment mapFragment = TextureSupportMapFragment
.class.cast(getChildFragmentManager()
.findFragmentById(R.id.map_view));

同理,也可以将旧的Fragment强制类型转换成新的Fragment.

Fragment f = Fragment.class.cast(mapFragment);

(注:上面的TextureSupportMapFragment是一个典型案例,他是高德地图SDK中的Fragment,本身已经继承了v4包下的Fragment,可以用过上面的转换来使他兼容AndroidX

差点扯远了,搞定AndroidX后,我们就可以使用FragmentActivityaddOnBackPressedCallback方法为你的Fragment提供拦截OnBackPressed的功能了.

public void addOnBackPressedCallback(@NonNull LifecycleOwner owner,
@NonNull OnBackPressedCallback onBackPressedCallback)

OnBackPressedCallback#handleOnBackPressed需要返回一个boolean值。如果你在这个回调里拦截了onBackPressed应该返回true,说明你自己已经处理了本次返回键按下的操作,这样你的Fragment就不会被弹出返回栈了。

值得注意的是,这个函数的第一个参数,一个LifecycleOwner,ActivityFragment都是LifecycleOwner,用于提供组件的生命周期,这个参数可以帮我们自动管理OnBackPressedCallback回调,你无需手动将他从Activity中移除,在LifecycleOwnerON_DESTROY事件来到的时候,他会被自动移除列表,你无需担心内存泄漏,框架会帮你完成这些事情。

/**

  • Interface for handling {@link ComponentActivity#onBackPressed()} callbacks without
  • strongly coupling that implementation to a subclass of {@link ComponentActivity}.
  • @see ComponentActivity#addOnBackPressedCallback(LifecycleOwner, OnBackPressedCallback)
  • @see ComponentActivity#removeOnBackPressedCallback(OnBackPressedCallback)
    /
    public interface OnBackPressedCallback {
    /
    *
  • Callback for handling the {@link ComponentActivity#onBackPressed()} event.
  • @return True if you handled the {@link ComponentActivity#onBackPressed()} event. No
  • further {@link OnBackPressedCallback} instances will be called if you return true.
    */
    boolean handleOnBackPressed();
    }

我们可以看到Activity内管理的OnBackPressedCallback的执行循序与添加时间有关.最后被添加进去的能最先得到执行.

public void addOnBackPressedCallback(@NonNull LifecycleOwner owner,
@NonNull OnBackPressedCallback onBackPressedCallback) {
Lifecycle lifecycle = owner.getLifecycle();
if (lifecycle.getCurrentState() == Lifecycle.State.DESTROYED) {
// Already destroyed, nothing to do
return;
}
// Add new callbacks to the front of the list so that
// the most recently added callbacks get priority
mOnBackPressedCallbacks.add(0, new LifecycleAwareOnBackPressedCallback(
lifecycle, onBackPressedCallback));
}

可以看到它是添加到mOnBackPressedCallbacks这个List的最前面的.

Fragment是必须要有id的,使用getId可以返回自身的id,通常用这个方法返回它所在的容器的id,供其他Fragment添加进FragmentManager时使用。(比如说你使用了一个FrameLayout作为Fragment的容器,那么它就会返回那个FrameLayoutid)

/**

  • Return the identifier this fragment is known by. This is either
  • the android:id value supplied in a layout or the container view ID
  • supplied when adding the fragment.
    */
    final public int getId() {
    return mFragmentId;
    }

startFragmentForResult方法在哪?对不起和OnBackPressed一样,Google没有直接为我们实现这个方法,但这并不代表Fragment没有这个功能,你当然可以直接用定义getter的方式来获取Fragment上内容,但这并不是最佳实践,为了规范编码我们最好还是使用公共的API

Fragment#setTargetFragment可以给当前Fragment设置一个目标Fragment和一个请求码

public void setTargetFragment(@Nullable Fragment fragment, int requestCode)

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

最后

目前已经更新的部分资料:



一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
img
mg-JX7fZ7ko-1712619100861)]

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-CCvgGHjv-1712619100861)]

  • 26
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值