深入分析Android“卡顿掉帧”问题
创始人
2025-05-28 04:18:11
0

前言

img

全球手机市场中,安卓和iOS一直占着主流市场,iPhone手机给人的感觉就是流畅,而安卓手机却一直是卡顿的代名词

其实,安卓机刚上手时还是速度飞快的,并且基于它开放性的原则,受到多数用户的喜爱,但通病是:运行一段时间后,反应变慢、容易卡顿,这也是iOS用户不肯换安卓最主要的原因

Android出现使用不流畅,卡顿的主要原因

img

CPU使用率过高

● 手机固件有缺陷,导致CPU使用率始终过高,这时您刷一个稳定点的ROM就好了

● 开启了过多的程序;这时您可以使用进程管理程序清理一下后台进程

● 某个程序由于设计不当或者不兼容导致占用大量CPU资源,这时您可以使用手机安全卫士体检里的运行监测(只勾选这个)查看当前所有正在运行程序的CPU占用,找到消耗资源特别多的,结束或者卸载它

● 您执行的某一个操作可能导致CPU过高(有时候也可能是查看CPU占用这个操作)

● CPU使用率查看软件不准确,这时您可以用多个软件查看

系统内存使用率过高

系统内存分为物理内存和虚拟内存

● 物理内存就是系统硬件提供的内存大小,是真正的内存;在linux下还有一个虚拟内存的概念,虚拟内存就是为了满足物理内存的不足而提出的策略,它是利用磁盘空间虚拟出的一块逻辑内存,用作虚拟内存的磁盘空间被称为 交换空间(Swap Space)

● linux的内存管理采取的是分页存取机制,为了保证物理内存能得到充分的利用,内核会在适当的时候将物理内存中不经常使用的数据块自动交换到虚拟内存中,而将经常使用的信息保留到物理内存。而进行这种交换所遵循的依据是“LRU”算法(Least Recently Used,最近最少使用算法)

● 无论是使用桌面操作系统还是移动操作系统,很多人都喜欢随时关注内存,一旦发现内存使用率过高就难受,忍不住的要杀进程以释放内存。这种习惯很大程度上都是源自Windows系统,当然这在Windows下也确实没错。然而很多人在使用Linux系统时仍然有这个习惯,甚至到了Android系统下,也改不掉(尤其是Android手机刚出现的几年),Clean Master等各种清理软件铺天盖地。毫不客气的说,Windows毒害了不少人!当然,这也不能怪Windows,毕竟Windows的普及率太高了,而大部分普通用户(甚至一些计算机相关人员)又不了解Windows和Linux在内存管理方面的差别

屏幕刷新丢帧

刷新频率

● 屏幕的刷新频率(Refresh Rate),就是一秒内屏幕刷新的次数

● 我们知道,在某一个时刻,将图像数据涂到屏幕上我们就能直观地看到一幅静态的画面,但这显然不能满足用户需求。我们需要看到的是屏幕上的动画——即不断切换的连续衔接的画面。在动画术语中,每一张这样的衔接画面被称作帧。也就是说,为了看到动画,我们需要以恒定的速度取到连续的帧,并将帧涂到屏幕上

如上,要显示屏幕动画,我们要设置两个时机

● 时机一:生成帧,产生了新的画面(帧),将其填充到 FrameBuffer 中,这个过程由 CPU(计算绘制需求)和 GPU(完成数据绘制)完成

● 时机二:显示帧,显示屏显示完一帧图像之后,等待一个固定的间隔后,从 FrameBuffer 中取下一帧图像并显示,这个过程由 GPU 完成

对于设备而言,其屏幕的刷新频率 就相当于显示帧的时机和速度,可以看做是额定不变的(而生成帧速度对应我们通常说的帧率)

刷新频率这个参数是手机出厂就决定的,取决于硬件的固定参数;目前大多数设备的刷新率是 60Hz,也就是一秒刷新60次,所以每次屏幕刷新的过程占用时间就是16ms(1000/60)左右,这个是固定参数,运行过程中,不会发生改变

UI线程阻塞

当一个应用程序启动之后,android系统会为这个应用创建一个主线程;这个线程非常重要,它负责渲染视图,分发事件到响应监听器并执行,对界面进行轮询监听;因此,一般也叫做“UI线程”

● android系统不会给应用程序的多个元素组件,建立多个线程来执行;一个视图Activity中的多个view组件运行在同一个UI线程中;因此,多个view组件的监听器的执行可能会相互影响

● 例如:当在UI线程中执行耗时操作,比如访问网络,访问数据库等,则会导致UI线程阻塞;当UI线程阻塞,则屏幕就会出现卡死情况;这样用户体验非常差;当线程阻塞超过5秒以后,android系统有可能进行干预,弹出对话框询问是否关闭应用程序

Android 绘制UI方式

把图形直接绘制到画布上(Canvas对象),这种方法可以通过独立的线程来管理surfaceView对象,并由独立线程来管理绘制过

● Android中的图形系统采用 Client/Server 架构。Server (即SurfaceFlinger)主要由 C++ 代码编写而成。Client 端代码分为两部分,一部分是由 Java 提供的供应用程序使用的 API,另一部分则是用 C++ 写成的底层实现

● 每个应用可能有一个或多个surface(含surface的情况下),surfaceFlinger是本地服务,用于管理surface的创建、销毁、zorder合成。View及其子类(如TextView, Button)要画在surface上。每个surface创建一个Canvas对象 (但属性时常改变),用来管理view在surface上的绘图操作,如画点画线。每个canvas对象对应一个bitmap,存储画在surface上的内容。当然这里还有个Layer的概念,在后面创建surface流程中我们再介绍

surface 创建

SurfaceControl surfaceControl = winAnimator.createSurfaceLocked();if (surfaceControl != null){outSurface.copyFrom(surfaceControl);if (SHOW_TRANSACTIONS) Slog.i(TAG,"  OUT SURFACE " + outSurface + ": copied");}
else {// For some reason there isn't a surface.  Clear the// caller's object so they see the same state.
outSurface.release();}

Surface的绘制

在Android系统刷新过程中ViewRoot会调用performTraversals方法并依次调用performMeasure、performLayout、performDraw。在performDraw中会区分是否支持硬件加速,如果支持直接通过OPENGL做硬件加速绘制,如果不支持则走软件绘制。因为我们在独立线程绘制过程中一般走的是软件绘制。这里对软件绘制的方法做介绍以掌握如何在独立线程中绘制UI

   private boolean drawSoftware(Surface surface, AttachInfo attachInfo, int yoff,boolean scalingRequired, Rect dirty) {// Draw with software renderer.Canvas canvas;try {int left = dirty.left;int top = dirty.top;int right = dirty.right;int bottom = dirty.bottom;canvas = mSurface.lockCanvas(dirty);// The dirty rectangle can be modified by Surface.lockCanvas()//noinspection ConstantConditionsif (left != dirty.left || top != dirty.top || right != dirty.right ||bottom != dirty.bottom) {attachInfo.mIgnoreDirtyState = true;}// TODO: Do this in nativecanvas.setDensity(mDensity);} catch (Surface.OutOfResourcesException e) {handleOutOfResourcesException(e);return false;} catch (IllegalArgumentException e) {Log.e(TAG, "Could not lock surface", e);// Don't assume this is due to out of memory, it could be// something else, and if it is something else then we could// kill stuff (or ourself) for no reason.
mLayoutRequested = true;    // ask wm for a new surface next time.return false;}try {if (DEBUG_ORIENTATION || DEBUG_DRAW) {Log.v(TAG, "Surface " + surface + " drawing to bitmap w="+ canvas.getWidth() + ", h=" + canvas.getHeight());//canvas.drawARGB(255, 255, 0, 0);}// If this bitmap's format includes an alpha channel, we// need to clear it before drawing so that the child will// properly re-composite its drawing on a transparent// background. This automatically respects the clip/dirty region// or// If we are applying an offset, we need to clear the area// where the offset doesn't appear to avoid having garbage// left in the blank areas.if (!canvas.isOpaque() || yoff != 0) {canvas.drawColor(0, PorterDuff.Mode.CLEAR);}dirty.setEmpty();mIsAnimating = false;attachInfo.mDrawingTime = SystemClock.uptimeMillis();mView.mPrivateFlags |= View.PFLAG_DRAWN;if (DEBUG_DRAW) {Context cxt = mView.getContext();Log.i(TAG, "Drawing: package:" + cxt.getPackageName() +", metrics=" + cxt.getResources().getDisplayMetrics() +", compatibilityInfo=" + cxt.getResources().getCompatibilityInfo());}ry {canvas.translate(0, -yoff);if (mTranslator != null) {mTranslator.translateCanvas(canvas);}canvas.setScreenDensity(scalingRequired ? mNoncompatDensity : 0);attachInfo.mSetIgnoreDirtyState = false;mView.draw(canvas);drawAccessibilityFocusedDrawableIfNeeded(canvas);} finally {if (!attachInfo.mSetIgnoreDirtyState) {// Only clear the flag if it was not set during the mView.draw() callattachInfo.mIgnoreDirtyState = false;}}} finally {try {
surface.unlockCanvasAndPost(canvas);} catch (IllegalArgumentException e) {Log.e(TAG, "Could not unlock surface", e);mLayoutRequested = true;    // ask wm for a new surface next time.//noinspection ReturnInsideFinallyBlockreturn false;}if (LOCAL_LOGV) {
Log.v(TAG, "Surface " + surface + " unlockCanvasAndPost");}}
return true;}

其中关键就是canvas = mSurface.lockCanvas(dirty) 与 surface.unlockC anvasAndPost(canvas);先lockCanvas,绘制UI,最后通过unlockCanvasAndPost通知surfaceFlinger先做zorder组合显示。

lockCanvas(dirty) 就是通过JNI调用nativeLockCanvas返回一个Canvas下面看nativeLockCanvas的实现。

sttic void nativeLockCanvas(JNIEnv* env, jclass clazz, jint nativeObject, jobject canvasObj, jobject dirtyRectObj) { sp surface(reinterpret_cast(nativeObject));

if (!isSurfaceValid(surface)) {doThrowIAE(env);return;
}// get dirty region
Region dirtyRegion;
if (dirtyRectObj) {Rect dirty;dirty.left = env->GetIntField(dirtyRectObj, gRectClassInfo.left);dirty.top = env->GetIntField(dirtyRectObj, gRectClassInfo.top);dirty.right = env->GetIntField(dirtyRectObj, gRectClassInfo.right);dirty.bottom = env->GetIntField(dirtyRectObj, gRectClassInfo.bottom);if (!dirty.isEmpty()) {dirtyRegion.set(dirty);}
} else {dirtyRegion.set(Rect(0x3FFF, 0x3FFF));
}ANativeWindow_Buffer outBuffer;
Rect dirtyBounds(dirtyRegion.getBounds());
status_t err = surface->lock(&outBuffer, &dirtyBounds);
dirtyRegion.set(dirtyBounds);
if (err < 0) {const char* const exception = (err == NO_MEMORY) ?OutOfResourcesException :"java/lang/IllegalArgumentException";jniThrowException(env, exception, NULL);return;
}// Associate a SkCanvas object to this surface
env->SetIntField(canvasObj, gCanvasClassInfo.mSurfaceFormat, outBuffer.format);SkBitmap bitmap;
ssize_t bpr = outBuffer.stride * bytesPerPixel(outBuffer.format);
bitmap.setConfig(convertPixelFormat(outBuffer.format), outBuffer.width, outBuffer.height, bpr);
if (outBuffer.format == PIXEL_FORMAT_RGBX_8888) {bitmap.setIsOpaque(true);
}
if (outBuffer.width > 0 && outBuffer.height > 0) {bitmap.setPixels(outBuffer.bits);
} else {// be safe with an empty bitmap.bitmap.setPixels(NULL);
}SkCanvas* nativeCanvas = SkNEW_ARGS(SkCanvas, (bitmap));
swapCanvasPtr(env, canvasObj, nativeCanvas);SkRegion clipReg;
if (dirtyRegion.isRect()) { // very common caseconst Rect b(dirtyRegion.getBounds());clipReg.setRect(b.left, b.top, b.right, b.bottom);
} else {size_t count;Rect const* r = dirtyRegion.getArray(&count);while (count) {clipReg.op(r->left, r->top, r->right, r->bottom, SkRegion::kUnion_Op);r++, count--;}
}nativeCanvas->clipRegion(clipReg);if (dirtyRectObj) {const Rect& bounds(dirtyRegion.getBounds());env->SetIntField(dirtyRectObj, gRectClassInfo.left, bounds.left);env->SetIntField(dirtyRectObj, gRectClassInfo.top, bounds.top);env->SetIntField(dirtyRectObj, gRectClassInfo.right, bounds.right);env->SetIntField(dirtyRectObj, gRectClassInfo.bottom, bounds.bottom);
}
}

在JNI层实现的就是通过surface获取到Layer中的buffer,并生成一个skiabitmap, Android 2D软件绘图使用skia作为核心引擎,这个bitmap的存储空间为Layer buffer。绘制的UI就是写入到这个buffer中,绘制好后通过 unlockCanvasAndPost通知surfaceflinger输出显示。如果是独立线程绘制UI,那么流程与上描述基本一致。但需要注意的是如果独立线程绘制的话,surface可通过surfaceView来获取

private void draw() {  
try {  //  步骤1  
canvas = sfh.lockCanvas(); // 得到一个canvas实例    
//  步骤2  
canvas.drawColor(Color.WHITE);// 刷屏                   
canvas.drawText("test", 100, 100, paint);// 画文字文本  
} catch (Exception ex) {  
} finally {   // 步骤3  
if (canvas != null)  sfh.unlockCanvasAndPost(canvas); // 将画好的画布提交  
}  
}  

总结

Android 原生系统是一个不断进化的过程 , 每个版本都会解决非常多的性能问题 , 同时也会引进一些问题 ; 到了手机厂商这里 , 由于硬件差异和软件定制 , 会在系统中加入大量的自己的代码 , 这无疑也会影响系统的性能 . 同样由于 Android 的开放 , App 的质量和行为也影响着整机的用户体验.

本篇主要列出了自身的实现问题导致的流畅性问题 , Android 最大的问题就是质量良莠不齐 , 不同于 App Store 这样的强力管理市场 , Android App 不仅可以在 Google Play 上面进行安装 , 也可以在其他的软件市场上面安装 , 甚至可以下载安装包自行安装 , 可以说上架的门槛非常低 , 那么质量就只能由 开发者自己来把握了

有需要文中代码的同学,可以顺手给我点赞评论支持一下

获取方式点击 《Android底层源码+Android学习笔记》

技术是无止境的,你需要对自己提交的每一行代码、使用的每一个工具负责,不断挖掘其底层原理,才能使自己的技术升华到更高的层面

Android 架构师之路还很漫长,与君共勉

PS:有问题欢迎指正,可以在评论区留下你的建议和感受;

欢迎大家点赞评论,觉得内容可以的话,可以转发分享一下

相关内容

热门资讯

蓝筹股持续发力 沪指再创年内新... 7月30日,A股三大股指高位分化。蓝筹股持续发力,带动沪指再创今年以来新高;中小市值风格则整体走弱,...
小红书 KOS(企业员工号)全... 在小红书生态中,企业运营正迎来新的工具助力 ——KOS(企业员工号 )。它以全新的绑定及管理模式,为...
协鑫光电再获近2亿元融资,协鑫... 作者 | 柯阳明 编辑 | 杨锐 排版校对 | 甘惠淇 近日,协鑫集团旗下协鑫光电再次宣布完成近2...
多家连锁餐饮品牌发声:欢迎多平... 2025年夏日暑期,城市烟火气持续升腾,消费活力加速释放。街头巷尾,从社区食堂到商圈地标,从老字号焕...
吉和昌IPO控制权稳定性遭问询... 瑞财经 王敏 7月28日,武汉吉和昌新材料股份有限公司(以下简称“吉和昌”)发布关于公开发行股票并在...
智能体穿透企业生产与消费场景,... 近日,IDC发布报告称,2025年,AI Agent将迎来规模化落地浪潮,其通过智能化任务处理重构标...
德邦物流总经理黄华波辞职,上半... 7月31日,德邦物流(603056)公告:公司董事兼总经理黄华波因个人原因申请辞去公司第六届董事会董...
迪瑞医疗上半年归母净利润最高降... 近日,迪瑞医疗科技股份有限公司(以下简称“迪瑞医疗”)发布2025年半年度业绩预告显示,期内预计实现...
欣旺达递表港交所 高盛和中信证... 欣旺达已向港交所主板提交上市申请,高盛和中信证券担任联席保荐人。 公司致力于锂电池的研发、设计、制造...
上半场保险机构吃罚单1.83亿... 来源:市场资讯 来源:中保新知 一直以来,监管罚单都是透视保险业经营主体合规性的重要窗口,通过分析罚...
白酒在胖东来、山姆、盒马找到“... (图片系AI生成) 新型商超卷新业态,白酒从中看到了机会。 去年奥乐齐9.9元白酒爆红网络,今年胖东...
创新药热度停不下来!医药基金7... 财联社7月31日讯(记者 吴雨其)创新药板块的强劲行情已成为当前资本市场最受关注的话题之一,热度从年...
中国石化等在浙江成立易电科技公... 天眼查App显示,近日,浙江中石化易电科技有限公司成立,法定代表人为岑纪卫,注册资本2900万人民币...
亿道信息上半年预盈最高增123... 瑞财经 钟鸣辰 亿道信息(001314)近日发布2025年半年度业绩预告,预计上半年净利增长88.9...
投资5亿!国内首条兆瓦级AEM... 2025年7月28日,深圳稳石氢能科技有限公司正式签约落户安徽省合肥市蜀山区,稳石氢能将于蜀山经济技...
金融Agent落地,谁能「敲开... 生成式AI,将给全球银行业带来超过2000亿美元(约合人民币超过1.4万亿元)的商业增量。这是中国商...
扎克伯格下血本:Meta 计划... IT之家 7 月 31 日消息,Meta 在其第二季度财报中透露,为全面推进其人工智能宏图,公司正计...
人民币升值受益板块7月31日跌... 证券之星消息,7月31日人民币升值受益板块较上一交易日下跌2.69%,凯撒旅业领跌。当日上证指数报收...
A股龙虎榜丨长城军工涨停创新高... 格隆汇7月31日|(601606.SH)今日涨停,股价创历史新高,换手率20.56%,成交额49.6...
扎克伯格放话:未来不戴 AI ... IT之家 7 月 31 日消息,Meta 首席执行官马克・扎克伯格在公司第二季度财报电话会议上再次强...