Android ANR 总结

工作之余,对之前学习到的和结合自己项目过程中的遇到的问题经验做一些总结,下面讲一讲Android开发过程中遇到的ANR的问题,做一下整理

一、概述

解决ANR一直是Android 开发者需要掌握的重要技巧,一般从三个方面着手。

开发阶段:通过工具检查各个方法的耗时,卡顿情况,发现一处修改一处。

线上阶段:这个阶段主要依靠监控工具发现ANR并上报,比如matrix。

分析阶段:如果线上用户发生ANR,并且你获取了一份日志,这就涉及了本文要分享的内容——ANR日志分析技巧

二、ANR产生机制

ANR——应用无响应,Activity是5秒,BroadCastReceiver是10秒,Service是20秒。

这句话说的很笼统,要想深入分析定位ANR,需要知道更多知识点,一般来说,ANR按产生机制,分为4类:

①输入事件超时(5s)

②广播类型超时(前台15s,后台60s)

③服务超时(前台20s,后台200s)

④ContentProvider 类型

2.1 输入事件超时(5s)

InputEvent Timeout

  1. a.InputDispatcher发送key事件给 对应的进程的 Focused Window ,对应的window不存在、处于暂停态、或通道(input channel)占满、通道未注册、通道异常、或5s内没有处理完一个事件,就会发生ANR
  2. b.InputDispatcher发送MotionEvent事件有个例外之处:当对应Touched Window的 input waitQueue中有超过0.5s的事件,inputDispatcher会暂停该事件,并等待5s,如果仍旧没有收到window的‘finish’事件,则触发ANR
  3. c.下一个事件到达,发现有一个超时事件才会触发ANR

 2.2 广播类型超时(前台15s,后台60s)

BroadcastReceiver Timeout

  1. a.静态注册的广播和有序广播会ANR,动态注册的非有序广播并不会ANR
  2. b.广播发送时,会判断该进程是否存在,不存在则创建,创建进程的耗时也算在超时时间里
  3. c.只有当进程存在前台显示的Activity才会弹出ANR对话框,否则会直接杀掉当前进程
  4. d.当onReceive执行超过阈值(前台15s,后台60s),将产生ANR
  5. e.如何发送前台广播:Intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND)

2.3 服务超时(前台20s,后台200s)

Service Timeout

  1. a.Service的以下方法都会触发ANR:onCreate(),onStartCommand(), onStart(), onBind(), onRebind(), onTaskRemoved(), onUnbind(),onDestroy().
  2. b.前台Service超时时间为20s,后台Service超时时间为200s
  3. c.如何区分前台、后台执行————当前APP处于用户态,此时执行的Service则为前台执行。
  4. d.用户态:有前台activity、有前台广播在执行、有foreground service执行

2.4 ContentProvider 类型

  1. a.ContentProvider创建发布超时并不会ANR
  2. b.使用ContentProviderclient来访问ContentProverder可以自主选择触发ANR,超时时间自己定
  3. client.setDetectNotResponding(PROVIDER_ANR_TIMEOUT);

三、导致ANR的原因

很多开发者认为,那就是耗时操作导致ANR,全部是app应用层的问题。实际上,线上环境大部分ANR由系统原因导致。

①应用层导致ANR(耗时操作)

② 系统导致ANR

3.1 应用层导致ANR(耗时操作)

  1. a. 函数阻塞:如死循环、主线程IO、处理大数据
  2. b. 锁出错:主线程等待子线程的锁
  3. c. 内存紧张:系统分配给一个应用的内存是有上限的,长期处于内存紧张,会导致频繁内存交换,进而导致应用的一些操作超时

3.2 系统导致ANR

  1. a. CPU被抢占:一般来说,前台在玩游戏,可能会导致你的后台广播被抢占CPU
  2. b. 系统服务无法及时响应:比如获取系统联系人等,系统的服务都是Binder机制,服务能力也是有限的,有可能系统服务长时间不响应导致ANR
  3. c. 其他应用占用的大量内存 

四、分析日志

发生ANR的时候,系统会产生一份anr日志文件(手机的/data/anr 目录下,文件名称可能各厂商不一样,业内大多称呼为trace文件),内含如下几项重要信息。

① CPU 负载

②内存信息

③ 堆栈消息

4.1 CPU 负载

  1. Load: 2.62 / 2.55 / 2.25
  2. CPU usage from 0ms to 1987ms later (2020-03-10 08:31:55.169 to 2020-03-10 08:32:17.156):
  3.   41% 2080/system_server: 28% user + 12% kernel / faults: 76445 minor 180 major
  4.   26% 9378/com.xiaomi.store: 20% user + 6.8% kernel / faults: 68408 minor 68 major
  5. ........省略N行.....
  6. 66% TOTAL: 20% user + 15% kernel + 28% iowait + 0.7% irq + 0.7% softirq

如上所示:

  • 第一行:1、5、15 分钟内正在使用和等待使用CPU 的活动进程的平均数
  • 第二行:表明负载信息抓取在ANR发生之后的0~1987ms。同时也指明了ANR的时间点:2020-03-10 08:31:55.169
  • 中间部分:各个进程占用的CPU的详细情况
  • 最后一行:各个进程合计占用的CPU信息。

名词解释:

  1. a. user:用户态,kernel:内核态
  2. b. faults:内存缺页,minor——轻微的,major——重度,需要从磁盘拿数据
  3. c. iowait:IO使用(等待)占比
  4. d. irq:硬中断,softirq:软中断

注意:
iowait占比很高,意味着有很大可能,是io耗时导致ANR,具体进一步查看有没有进程faults major比较多。

  • 单进程CPU的负载并不是以100%为上限,而是有几个核,就有百分之几百,如4核上限为400%。

 4.2 内存信息

  1. Total number of allocations 476778  进程创建到现在一共创建了多少对象
  2. Total bytes allocated 52MB 进程创建到现在一共申请了多少内存
  3. Total bytes freed 52MB   进程创建到现在一共释放了多少内存
  4. Free memory 777KB    不扩展堆的情况下可用的内存
  5. Free memory until GC 777KB  GC前的可用内存
  6. Free memory until OOME 383MB  OOM之前的可用内存
  7. Total memory 当前总内存(已用+可用)
  8. Max memory 384MB 进程最多能申请的内存

从含义可以得出结论:Free memory until OOME 的值很小的时候,已经处于内存紧张状态。应用可能是占用了过多内存。

ps:如果ANR时间点前后,日志里有打印onTrimMemory,也可以作为内存紧张的一个参考判断

4.3 堆栈消息

堆栈信息是最重要的一个信息,展示了ANR发生的进程当前所有线程的状态。

  1. suspend all histogram: Sum: 2.834s 99% C.I. 5.738us-7145.919us Avg: 607.155us Max: 41543us
  2. DALVIK THREADS (248):
  3. "main" prio=5 tid=1 Native
  4.   | group="main" sCount=1 dsCount=0 flags=1 obj=0x74b17080 self=0x7bb7a14c00
  5.   | sysTid=2080 nice=-2 cgrp=default sched=0/0 handle=0x7c3e82b548
  6.   | state=S schedstat=( 757205342094 583547320723 2145008 ) utm=52002 stm=23718 core=5 HZ=100
  7.   | stack=0x7fdc995000-0x7fdc997000 stackSize=8MB
  8.   | held mutexes=
  9.   kernel: __switch_to+0xb0/0xbc
  10.   kernel: SyS_epoll_wait+0x288/0x364
  11.   kernel: SyS_epoll_pwait+0xb0/0x124
  12.   kernel: cpu_switch_to+0x38c/0x2258
  13.   native: #00 pc 000000000007cd8c /system/lib64/libc.so (__epoll_pwait+8)
  14.   native: #01 pc 0000000000014d48 /system/lib64/libutils.so (android::Looper::pollInner(int)+148)
  15.   native: #02 pc 0000000000014c18 /system/lib64/libutils.so (android::Looper::pollOnce(int, int*, int*, void**)+60)
  16.   native: #03 pc 0000000000127474 /system/lib64/libandroid_runtime.so (android::android_os_MessageQueue_nativePollOnce(_JNIEnv*, _jobject*, long, int)+44)
  17.   at android.os.MessageQueue.nativePollOnce(Native method)
  18.   at android.os.MessageQueue.next(MessageQueue.java:330)
  19.   at android.os.Looper.loop(Looper.java:169)
  20.   at com.android.server.SystemServer.run(SystemServer.java:508)
  21.   at com.android.server.SystemServer.main(SystemServer.java:340)
  22.   at java.lang.reflect.Method.invoke(Native method)
  23.   at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:536)
  24.   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:856
  25.   ........省略N行....
  26.   "OkHttp ConnectionPool" daemon prio=5 tid=251 TimedWaiting
  27.   | group="main" sCount=1 dsCount=0 flags=1 obj=0x13daea90 self=0x7bad32b400
  28.   | sysTid=29998 nice=0 cgrp=default sched=0/0 handle=0x7b7d2614f0
  29.   | state=S schedstat=( 951407 137448 11 ) utm=0 stm=0 core=3 HZ=100
  30.   | stack=0x7b7d15e000-0x7b7d160000 stackSize=1041KB
  31.   | held mutexes=
  32.   at java.lang.Object.wait(Native method)
  33.   - waiting on <0x05e5732e> (a com.android.okhttp.ConnectionPool)
  34.   at com.android.okhttp.ConnectionPool$1.run(ConnectionPool.java:103)
  35.   - locked <0x05e5732e> (a com.android.okhttp.ConnectionPool)
  36.   at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
  37.   at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
  38.   at java.lang.Thread.run(Thread.java:764)

如上日志所示,本文截图了两个线程信息,一个是主线程main,它的状态是native。

另一个是OkHttp ConnectionPool,它的状态是TimeWaiting。众所周知,教科书上说线程状态有5种:新建、就绪、执行、阻塞、死亡。而Java中的线程状态有6种,6种状态都定义在:java.lang.Thread.State中

问题来了,上述main线程的native是什么状态,哪来的?其实trace文件中的状态是是CPP代码中定义的状态

由此可知,main函数的native状态是正在执行JNI函数。堆栈信息是我们分析ANR的第一个重要的信息,一般来说:

main线程处于 BLOCK、WAITING、TIMEWAITING状态,那基本上是函数阻塞导致ANR;

如果main线程无异常,则应该排查CPU负载和内存环境。

 五、典型案例分析

 5.1 主线程执行耗时操作

  1. "main" prio=5 tid=1 Runnable
  2.   | group="main" sCount=0 dsCount=0 flags=0 obj=0x72deb848 self=0x7748c10800
  3.   | sysTid=8968 nice=-10 cgrp=default sched=0/0 handle=0x77cfa75ed0
  4.   | state=R schedstat=( 24783612979 48520902 756 ) utm=2473 stm=5 core=5 HZ=100
  5.   | stack=0x7fce68b000-0x7fce68d000 stackSize=8192KB
  6.   | held mutexes= "mutator lock"(shared held)
  7.   at com.example.test.MainActivity$onCreate$2.onClick(MainActivity.kt:20)——关键行!!!
  8.   at android.view.View.performClick(View.java:7187)
  9.   at android.view.View.performClickInternal(View.java:7164)
  10.   at android.view.View.access$3500(View.java:813)
  11.   at android.view.View$PerformClick.run(View.java:27640)
  12.   at android.os.Handler.handleCallback(Handler.java:883)
  13.   at android.os.Handler.dispatchMessage(Handler.java:100)
  14.   at android.os.Looper.loop(Looper.java:230)
  15.   at android.app.ActivityThread.main(ActivityThread.java:7725)
  16.   at java.lang.reflect.Method.invoke(Native method)
  17.   at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:526)
  18.   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1034)

上述日志表明,主线程正处于执行状态,看堆栈信息可知不是处于空闲状态,发生ANR是因为一处click监听函数里执行了耗时操作。

5.2 主线程被锁阻塞
 

  1. "main" prio=5 tid=1 Blocked
  2.   | group="main" sCount=1 dsCount=0 flags=1 obj=0x72deb848 self=0x7748c10800
  3.   | sysTid=22838 nice=-10 cgrp=default sched=0/0 handle=0x77cfa75ed0
  4.   | state=S schedstat=( 390366023 28399376 279 ) utm=34 stm=5 core=1 HZ=100
  5.   | stack=0x7fce68b000-0x7fce68d000 stackSize=8192KB
  6.   | held mutexes=
  7.   at com.example.test.MainActivity$onCreate$1.onClick(MainActivity.kt:15)
  8.   - waiting to lock <0x01aed1da> (a java.lang.Object) held by thread 3 ——————关键行!!!
  9.   at android.view.View.performClick(View.java:7187)
  10.   at android.view.View.performClickInternal(View.java:7164)
  11.   at android.view.View.access$3500(View.java:813)
  12.   at android.view.View$PerformClick.run(View.java:27640)
  13.   at android.os.Handler.handleCallback(Handler.java:883)
  14.   at android.os.Handler.dispatchMessage(Handler.java:100)
  15.   at android.os.Looper.loop(Looper.java:230)
  16.   at android.app.ActivityThread.main(ActivityThread.java:7725)
  17.   at java.lang.reflect.Method.invoke(Native method)
  18.   at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:526)
  19.   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1034)
  20.    
  21.   ........省略N行.....
  22.    
  23.   "WQW TEST" prio=5 tid=3 TimeWating
  24.   | group="main" sCount=1 dsCount=0 flags=1 obj=0x12c44230 self=0x772f0ec000
  25.   | sysTid=22938 nice=0 cgrp=default sched=0/0 handle=0x77391fbd50
  26.   | state=S schedstat=( 274896 0 1 ) utm=0 stm=0 core=1 HZ=100
  27.   | stack=0x77390f9000-0x77390fb000 stackSize=1039KB
  28.   | held mutexes=
  29.   at java.lang.Thread.sleep(Native method)
  30.   - sleeping on <0x043831a6> (a java.lang.Object)
  31.   at java.lang.Thread.sleep(Thread.java:440)
  32.   - locked <0x043831a6> (a java.lang.Object)
  33.   at java.lang.Thread.sleep(Thread.java:356)
  34.   at com.example.test.MainActivity$onCreate$2$thread$1.run(MainActivity.kt:22)
  35.   - locked <0x01aed1da> (a java.lang.Object)————————————————————关键行!!!
  36.   at java.lang.Thread.run(Thread.java:919)

 这是一个典型的主线程被锁阻塞的例子; 

  1. waiting to lock <0x01aed1da> (a java.lang.Object) held by thread 3

 其中等待的锁是<0x01aed1da>,这个锁的持有者是线程 3。进一步搜索 “tid=3” 找到线程3, 发现它正在TimeWating。

那么ANR的原因找到了:线程3持有了一把锁,并且自身长时间不释放,主线程等待这把锁发生超时。在线上环境中,常见因锁而ANR的场景是SharePreference写入

5.3 CPU被抢占 

  1. CPU usage from 0ms to 10625ms later (2020-03-09 14:38:31.633 to 2020-03-09 14:38:42.257):
  2.   543% 2045/com.alibaba.android.rimet: 54% user + 89% kernel / faults: 4608 minor 1 major ————关键行!!!
  3.   99% 674/android.hardware.camera.provider@2.4-service: 81% user + 18% kernel / faults: 403 minor
  4.   24% 32589/com.wang.test: 22% user + 1.4% kernel / faults: 7432 minor 1 major
  5.   ........省略N行.....

如上日志,第二行是钉钉的进程,占据CPU高达543%,抢占了大部分CPU资源,因而导致发生ANR。

5.4 内存紧张导致ANR 

如果有一份日志,CPU和堆栈都很正常(不贴出来了),仍旧发生ANR,考虑是内存紧张。 

 系统日志里搜索am_meminfo, 这个没有搜索到。再次搜索onTrimMemory,果然发现了很多条记录;

  1. 10-31 22:37:19.749 20733 20733 E Runtime : onTrimMemory level:80,pid:com.xxx.xxx:Launcher0
  2. 10-31 22:37:33.458 20733 20733 E Runtime : onTrimMemory level:80,pid:com.xxx.xxx:Launcher0
  3. 10-31 22:38:00.153 20733 20733 E Runtime : onTrimMemory level:80,pid:com.xxx.xxx:Launcher0
  4. 10-31 22:38:58.731 20733 20733 E Runtime : onTrimMemory level:80,pid:com.xxx.xxx:Launcher0
  5. 10-31 22:39:02.816 20733 20733 E Runtime : onTrimMemory level:80,pid:com.xxx.xxx:Launcher0

 可以看出,在发生ANR的时间点前后,内存都处于紧张状态,level等级是80,

可知80这个等级是很严重的,应用马上就要被杀死,被杀死的这个应用从名字可以看出来是桌面,连桌面都快要被杀死,那普通应用能好到哪里去呢?

一般来说,发生内存紧张,会导致多个应用发生ANR,所以在日志中如果发现有多个应用一起ANR了,可以初步判定,此ANR与你的应用无关。

5.5 系统服务超时导致ANR 

系统服务超时一般会包含BinderProxy.transactNative关键字,请看如下日志:

  1. "main" prio=5 tid=1 Native
  2.   | group="main" sCount=1 dsCount=0 flags=1 obj=0x727851e8 self=0x78d7060e00
  3.   | sysTid=4894 nice=0 cgrp=default sched=0/0 handle=0x795cc1e9a8
  4.   | state=S schedstat=( 8292806752 1621087524 7167 ) utm=707 stm=122 core=5 HZ=100
  5.   | stack=0x7febb64000-0x7febb66000 stackSize=8MB
  6.   | held mutexes=
  7.   kernel: __switch_to+0x90/0xc4
  8.   kernel: binder_thread_read+0xbd8/0x144c
  9.   kernel: binder_ioctl_write_read.constprop.58+0x20c/0x348
  10.   kernel: binder_ioctl+0x5d4/0x88c
  11.   kernel: do_vfs_ioctl+0xb8/0xb1c
  12.   kernel: SyS_ioctl+0x84/0x98
  13.   kernel: cpu_switch_to+0x34c/0x22c0
  14.   native: #00 pc 000000000007a2ac /system/lib64/libc.so (__ioctl+4)
  15.   native: #01 pc 00000000000276ec /system/lib64/libc.so (ioctl+132)
  16.   native: #02 pc 00000000000557d4 /system/lib64/libbinder.so (android::IPCThreadState::talkWithDriver(bool)+252)
  17.   native: #03 pc 0000000000056494 /system/lib64/libbinder.so (android::IPCThreadState::waitForResponse(android::Parcel*, int*)+60)
  18.   native: #04 pc 00000000000562d0 /system/lib64/libbinder.so (android::IPCThreadState::transact(int, unsigned int, android::Parcel const&, android::Parcel*, unsigned int)+216)
  19.   native: #05 pc 000000000004ce1c /system/lib64/libbinder.so (android::BpBinder::transact(unsigned int, android::Parcel const&, android::Parcel*, unsigned int)+72)
  20.   native: #06 pc 00000000001281c8 /system/lib64/libandroid_runtime.so (???)
  21.   native: #07 pc 0000000000947ed4 /system/framework/arm64/boot-framework.oat (Java_android_os_BinderProxy_transactNative__ILandroid_os_Parcel_2Landroid_os_Parcel_2I+196)
  22.   at android.os.BinderProxy.transactNative(Native method) ————————————————关键行!!!
  23.   at android.os.BinderProxy.transact(Binder.java:804)
  24.   at android.net.IConnectivityManager$Stub$Proxy.getActiveNetworkInfo(IConnectivityManager.java:1204)—关键行!
  25.   at android.net.ConnectivityManager.getActiveNetworkInfo(ConnectivityManager.java:800)
  26.   at com.xiaomi.NetworkUtils.getNetworkInfo(NetworkUtils.java:2)
  27.   at com.xiaomi.frameworkbase.utils.NetworkUtils.getNetWorkType(NetworkUtils.java:1)

从堆栈可以看出获取网络信息发生了ANR:getActiveNetworkInfo。

前文有讲过:系统的服务都是Binder机制(16个线程),服务能力也是有限的,有可能系统服务长时间不响应导致ANR。如果其他应用占用了所有Binder线程,那么当前应用只能等待。

可进一步搜索:blockUntilThreadAvailable关键字:
 

  1. at android.os.Binder.blockUntilThreadAvailable(Native method)

如果有发现某个线程的堆栈,包含此字样,可进一步看其堆栈,确定是调用了什么系统服务。此类ANR也是属于系统环境的问题,如果某类型机器上频繁发生此问题,应用层可以考虑规避策略。

参考链接:ANR日志分析全面解析_android anr 日志-CSDN博客

后面分享使用 Matrix工具去监控app卡顿

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/633879.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CMake TcpServer项目 生成静态库.a / 动态库.so

CMake 实战构建TcpServer项目 静态库/动态库-CSDN博客https://blog.csdn.net/weixin_41987016/article/details/135608829?spm1001.2014.3001.5501 在这篇博客的基础上&#xff0c;我们把头文件放在include里边&#xff0c;把源文件放在src里边&#xff0c;重新构建 hehedali…

区间预测 | Matlab实现LSTM-Adaboost-ABKDE的集成学习长短期记忆神经网络自适应带宽核密度估计多变量回归区间预测

区间预测 | Matlab实现LSTM-Adaboost-ABKDE的集成学习长短期记忆神经网络自适应带宽核密度估计多变量回归区间预测 目录 区间预测 | Matlab实现LSTM-Adaboost-ABKDE的集成学习长短期记忆神经网络自适应带宽核密度估计多变量回归区间预测效果一览基本介绍程序设计参考资料 效果一…

五个常见的 jQuery 面试题

1. 如何使用 jQuery 选择器选中所有 class 为 “example” 的元素? $(".example");2. 如何使用 jQuery 遍历一个数组&#xff0c;并将每个元素添加到一个 <ul> 列表中? <ul id"list"></ul>var array ["Item 1", "It…

Plane Geometry (Junior High School)

初中平面几何&#xff0c; ACBD, ∠CAD60&#xff0c;∠C40&#xff0c;求∠B Vertical Calculation-CSDN博客 Rectangular Area-CSDN博客

开发安全之:Password Management: Hardcoded Password

Overview Hardcoded password 可能会削弱系统安全性&#xff0c;并且无法轻易修正出现的安全问题。 Details 使用硬编码方式处理密码绝非好方法。这不仅是因为所有项目开发人员都可以使用通过硬编码方式处理的密码&#xff0c;而且还会使解决这一问题变得极其困难。在代码投…

kafka系列(二)

本章承接kafka一内容&#xff0c;文章在本人博客主页都有&#xff0c;可以自行点击浏览。 幂等性 请求执行多次&#xff0c;但执行的结果是一致的。 如果&#xff0c;某个系统是不具备幂等性的&#xff0c;如果用户重复提交了某个表格&#xff0c;就可能会造成不良影响。例如…

街机模拟游戏逆向工程(HACKROM)教程:[10]68K汇编add指令

我们之前已经介绍了move指令&#xff0c;从本章开始&#xff0c;我们会一步步介绍更多的M68K指令。 简介&#xff1a; add :加法指令 该指令的作用是[源操作数]加[目的操作数]&#xff0c;结果传递至[目的操作数]&#xff0c;[源操作数]保持不变。 例子&#xff1a;…

基于改进蝙蝠算法的三维航线规划算法

matlab2020a可正常运行 基于改进蝙蝠算法的三维航线规划资源-CSDN文库

Oracle 数据库备份与恢复的重要性与最佳实践

文章目录 一、备份的重要性二、备份工具-RMAN四、比较备份策略五、实例恢复六、完全恢复与不完全恢复七、备份与恢复脚本 引言&#xff1a; 在现代信息时代&#xff0c;数据已成为组织和企业最重要的资产之一。保护和恢复数据的能力对于确保业务连续性和减少潜在风险至关重要。…

Flink1.17 基础知识

Flink1.17 基础知识 来源&#xff1a;B站尚硅谷 目录 Flink1.17 基础知识Flink 概述Flink 是什么Flink特点Flink vs SparkStreamingFlink的应用场景Flink分层API Flink快速上手创建项目WordCount代码编写批处理流处理 Flink部署集群角色部署模式会话模式&#xff08;Session …

Java:选择哪个Java IDE好?

Java&#xff1a;选择哪个Java IDE好? 在开始前我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「java的资料从专业入门到高级教程」&#xff0c; 点个关注在评论区回复“888”之后私信回复“888”&#xff0c;全部无偿共享给大家&#xff01;&#xff01;&…

线程基础知识点

1. 线程和进程的区别&#xff1f; 程序由指令和数据组成&#xff0c;但这些指令要运行&#xff0c;数据要读写&#xff0c;就必须将指令加载至 CPU&#xff0c;数据加载至内存。在指令运行过程中还需要用到磁盘、网络等设备。进程就是用来加载指令、管理内存、管理 IO 的。 当…

深度学习(1)--基础概念

目录 一.计算机视觉(CV) 二.神经网络基础 三.神经网络整体架构 一.计算机视觉(CV) (1).计算机视觉中图像表示为三位数组&#xff0c;其中三维数组中像素的值为0~255&#xff0c;像素的值越低表示该点越暗&#xff0c;像素的值越高表示该点越亮。 (2).图像表示 A*B*C&#xf…

【Go面试向】defer与return的执行顺序初探

defer与return常考点 简述 defer 和 return 返回值 的执行顺序&#xff1f;如果要返回一个 defer 执行后的值, 可以通过哪些方式&#xff1f; 大家好 我是寸铁&#x1f44a; 总结了一篇defer 和 return 返回值 的执行顺序探讨的文章✨ 喜欢的小伙伴可以点点关注 &#x1f49d;…

list上

文章目录 初步了解list面试题&#xff1a;为什么会有list&#xff1f;vector的缺点&#xff1a;vector、list优点 list结构迭代器的分类list的简单运用insert、erase、迭代器失效&#xff08;和vector的区别&#xff09;erase class和structlist的迭代器为什么这个迭代器的构造…

借钱的原则

我个人经验&#xff1a;远离赌博的人&#xff0c;原理没有信用的人&#xff0c;远离频繁借钱的人 潮哥的经验&#xff1a;我在对待财富方面&#xff0c;定下了几条原则&#xff1a;一&#xff0c;助出现危机的病患&#xff0c;不助出现危机的生活与事业。二&#xff0c;若助公益…

【办公类-21-02】20240118育婴员操作题word打印2.0

作品展示 把12页一套的操作题批量制作10份&#xff0c;便于打印 背景需求 将昨天整理的育婴师操作题共享&#xff0c; 因为题目里面有大量的红蓝颜色文字&#xff0c;中大班办公室都是黑白单面手动翻页打印。只有我待的教务室办公室有彩色打印机打印&#xff08;可以自动双面…

深入详解使用 RabbitMQ 过程中涉及到的多个细节问题(面试可用)

目录 1、基础类问题 2、cluster 相关问题 3、综合性问题 4、参考资料 C软件异常排查从入门到精通系列教程&#xff08;专栏文章列表&#xff0c;欢迎订阅&#xff0c;持续更新...&#xff09;https://blog.csdn.net/chenlycly/article/details/125529931C/C基础与进阶&…

Nginx配置ssl访问 443端口

来一篇简单的。 Nginx服务配置ssl访问&#xff0c;使用默认的443端口。 Nginx ssl module 配置之前首先需要确认Nginx是否已经启用了ssl模块&#xff1a; [rooterpTest sbin]# ./nginx -V nginx version: nginx/1.19.9 built by gcc 4.8.5 20150623 (Red Hat 4.8.5-44) (GC…

CVer从0入门NLP(二)———LSTM、ELMO、Transformer模型

&#x1f34a;作者简介&#xff1a;秃头小苏&#xff0c;致力于用最通俗的语言描述问题 &#x1f34a;专栏推荐&#xff1a;深度学习网络原理与实战 &#x1f34a;近期目标&#xff1a;写好专栏的每一篇文章 &#x1f34a;支持小苏&#xff1a;点赞&#x1f44d;&#x1f3fc;、…