前言
近期用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 ,还真是挺让人烦心的。
有过错或许不对的当地,欢迎指出