近期用kotlin进行项目开发,写了挺多次跳转Activity页面代码,发现和Java有一点不一致,但是可不可以和Java一样封装起来方便调用呢?对此,我找了chatGPT聊了聊,他给出了答案,然后我就 CV大法这么用了,结果报出了一个不可思议的问题:ActivityNotFoundException,经过一系列检查,最终解决,特此记录一下。如果你很急,但是你先别急,直接点击目录 解决段落 可查看解决方法,起因段落、问题段落 都是吹水,只贪图一乐。
在一个阴雨绵绵的下午,在版本迭代之后,暂时没有开发任务的我。看了看之前迭代的代码。找找有没有可以优化的点。然后想起来了之前跳转Activity还没有封装。这怎么能行呢。然后就打开了 chatGPT,问了问他:
好家伙,不看不知道,一看吓一跳,不得使用我们的CV大法搞一下? 一顿操作之后。把它集成到项目中了,并且优化了一下一下:
/**
* 带参数跳转的Activity 如:startActivity()
*
* @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()
*
* @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()
*
* @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 ,还真是挺让人烦心的。