Android开发:kotlin封装 Intent 跳转Activity,报ActivityNotFoundException 问题

抬头图片

Android开发:kotlin封装 Intent 跳转Activity,报ActivityNotFoundException 问题

前言

近期用kotlin进行项目开发,写了挺多次跳转Activity页面代码,发现和Java有一点不一致,但是可不可以和Java一样封装起来方便调用呢?对此,我找了chatGPT聊了聊,他给出了答案,然后我就 CV大法这么用了,结果报出了一个不可思议的问题:ActivityNotFoundException,经过一系列检查,最终解决,特此记录一下。如果你很急,但是你先别急,直接点击目录 解决段落 可查看解决方法,起因段落问题段落 都是吹水,只贪图一乐。

起因

  • 起因段落

在一个阴雨绵绵的下午,在版本迭代之后,暂时没有开发任务的我。看了看之前迭代的代码。找找有没有可以优化的点。然后想起来了之前跳转Activity还没有封装。这怎么能行呢。然后就打开了 chatGPT,问了问他:

在这里插入图片描述
好家伙,不看不知道,一看吓一跳,不得使用我们的CV大法搞一下? 一顿操作之后。把它集成到项目中了,并且优化了一下一下:


/**
 * 带参数跳转的Activity  如:startActivity<TestActivity>()
 *
 * @param [T]跳转的Activity
 * @param [data]携带的数据  可为空
 */
inline fun <reified T : Activity> Context.startActivity(data: Bundle? = null) {
    
    
    val intent = Intent(this, T::class.java)
    if (data != null) {
    
    
        intent.putExtras(data)
    }
    this.startActivity(intent)

}

OK,CV大法已经搞定了,那么我们就可以正式开始使用了。

问题

  • 问题段落

不得不说的是,这确实很简洁,看起来一目了然,下面示例是在fragment页面中使用的,所以会有 requireContext() 的前缀,如果是在Activity,可直接使用 startActivity<目标Activity>() 方法,注意的是因为命名和安卓自带的一样,需要区分。


    override fun onClick(view: View?) {
    
    
        view ?: return
        when (view.id) {
    
    
            R.id.lin_my_about -> {
    
    
            
                //仅打开
                requireContext().startActivity<TestActivity>()

                //传值
                val bundle = Bundle()
                bundle.putString("test","test")
                requireContext().startActivity<TestActivity>(bundle)
            }
        }
    }

正当我兴高采烈的构建项目,打包到虚拟机的时候,不出意外的话,意外就要发生了。当我小心翼翼的点击跳转按钮······>

在这里插入图片描述
what? App居然闪退了,这是什么情况?这仅仅是一个跳转代码哇,而且也就这么几行,也没有写什么东西哇,妈耶,怎么它就闪退了。这个东西这么不争气的吗······ 脑子里不断翻滚,但是既然闪退了,就有异常抛出,俺们去log日志看看:

在这里插入图片描述

wtf? 找不到Activity异常?没有在AndroidManifest.xml 里面注册?这么低级的错误?不会吧不会吧,真的不会是没有注册吧?

在这里插入图片描述
查看代码,发现AndroidManifest.xml 文件里面是有的,what? 那究竟是什么问题?难道是系统出故障了?仔细想想,应该不会是系统的问题,应该我我们的代码问题,可是就这么几行代码,咋就不行了呢?为此,我们用最基本的系统自带方法试试。

    startActivity(Intent(requireContext(),TestActivity::class.java))

当我们使用系统方法的时候,页面正常跳转了。这能百分之百确定,就是我们写的代码有问题了。我们再来仔细看看报错的问题,

android.content.ActivityNotFoundException: Unable to find explicit activity class {leo.study.kotlin_mvp_demo/int}; have you declared this activity in your AndroidManifest.xml?

仔细一看,这个 {leo.study.kotlin_mvp_demo/int} 好像不对,都不是一个Activity的路径。

在这里插入图片描述
报错路径后面接了一个 Int 一个整型,确实很离谱。然后继续问chatGPT,给的答案并不符合我们的预期,果然它也并不是万能的,可能我们问的不够细致。

在这里插入图片描述在这里插入图片描述
不过 what ever ,已经无所谓了。其实chatGPT也说明白了,应该就是我们封装的那段代码出现了问题,我们再来仔细看下我们写的封装代码。

在这里插入图片描述
咋一看是没有什么问题,我只是修改了一下,增加了 Bundle() 传值,将 Aactivity 修改为了 Context其他基本就没有变化了啊。到底是什么问题呢?
秉承着“事出必有因”,“邪乎到家必有鬼”,“皮裤套棉裤,必定有缘故” 的原则,我们再仔细看看这报错和封装代码,发挥24K黄金单生狗眼的功力,好像真的发现了一点点蛛丝马迹。各位看官们发现了吗?

在这里插入图片描述

第一个 T 是灰色的,意味着没有调用,第二个 T 是斜体。貌似这两个货,不是同一个东西哇。为了确定这个观点,我 “command+单击” 进入了这个斜体 T 里,发现了一个新大陆。

在这里插入图片描述

原来斜体 T 指代的是这个玩意儿,那就怪不得会报错啦~~~~~~ 造成这样子的原因就是俺的 安卓死丢丢 开了自动导包,然后复制chatGPT的代码,自动导包了这个类当中的 T

解决

  • 解决段落

方法一:

俺们知道,我们所写的这个 T 是一个泛型,指代的是将要跳转的目标 Activity 那么其实解决这个问题,就十分简单了。我们只需要把指代泛型的这个 T 修改成别的字母就好啦。


/**
 * 带参数跳转的Activity  如:startAct<TestActivity>()
 *
 * @param [A]            跳转的 Activity
 * @param [data]         携带的数据  可为空
 * @param [requestCode]  请求 code 当不为0时,startActivityForResult
 */
inline fun <reified A : Activity> Context.startActivity(
    data: Bundle? = null,
    requestCode: Int? = 0
) {
    
    
    val intent = Intent(this, A::class.java)
    if (data != null) {
    
    
        intent.putExtras(data)
    }
    if (requestCode != 0) {
    
    
        requestCode?.let {
    
     (this as Activity).startActivityForResult(intent, it) }
    } else {
    
    
        this.startActivity(intent)
    }

}

我们这时候在看看这个 泛型所指代的 A 已经变成同一样的红色了。那么到这里,这个开篇的问题,就解决了。赶快给自己鼓个掌,毕竟,“又过了一天,已经很不错了。”

在这里插入图片描述

方法二:

既然我们已经知道,出现这个问题主要是导错包,而导致的。那么我们将 import 导包的那一行删除,也是可行的。

在这里插入图片描述
删除之后代码:


/**
 * 带参数跳转的Activity  如:startAct<TestActivity>()
 *
 * @param [T]            跳转的 Activity
 * @param [data]         携带的数据  可为空
 * @param [requestCode]  请求 code 当不为0时,startActivityForResult
 */
inline fun <reified T : Activity> Context.startActivity(
    data: Bundle? = null,
    requestCode: Int? = 0
) {
    
    
    val intent = Intent(this, T::class.java)
    if (data != null) {
    
    
        intent.putExtras(data)
    }
    if (requestCode != 0) {
    
    
        requestCode?.let {
    
     (this as Activity).startActivityForResult(intent, it) }
    } else {
    
    
        this.startActivity(intent)
    }

}

总结

以上就是本片文章的全部内容啦。纵观全文,本篇文章也就是解决段落有一点点干货。其他都是吹水。可能别人同学没有出现这样的问题。但是,正所谓吃一堑长一智,还是那句话:“邪乎到家,必有鬼” 起因和问题的编写,主要是记录俺发现问题,解决问题的思路历程。现在来看虽然好像仅仅只是两三段的文字描写完的,但是俺在当时的时候,心里不知道喊过多少次的 “wtf”。特别是当俺在使用系统的 startActivity() 方法又正常跳转的时候,俺直接怀疑人生。

一锅老鼠屎,坏了一锅粥。就好比这个小小的大写泛型 T ,还真是挺让人烦心的。

有错误或者不对的地方,欢迎指出
抬头图片

猜你喜欢

转载自blog.csdn.net/weixin_43683367/article/details/129819022