月薪20+的Android面试都问这些问题系列一(含答案)

作者 : 开心源码 本文共5109个字,预计阅读时间需要13分钟 发布时间: 2022-05-12 共221人阅读

金三银四跳槽季,相信大家一定急需一套Android面试宝典,今天给大家准备了我珍藏已久的Android高阶面试宝典,供大家学习 !【文末有干货】

一、面试题:

1.自己设置Handler时如何避免内存泄漏
2.onNewIntent的调用时机
3.RecyclerView相比ListView有哪些优势
4.谈一谈Proguard混淆技术
5.ANR出现的场景及处理方案

二、详细解析:

1.自己设置Handler时如何避免内存泄漏

一般非静态内部类持有外部类的引用的情况下,造成外部类在使用完成后不能被系统回收内存,从而造成内存泄漏。为了避免这个问题,我们可以自己设置的Handler公告为静态内部类形式,而后通过弱引用的方式,让Handler持有外部类的引用,从而可避免内存泄漏问题。

以下是代码实现

private WeakReference<MainActivity> activityWeakReference;private MyHandler myHandler;static class MyHandler extends Handler {private MainActivity activity;MyHandler(WeakReference<MainActivity> ref) {this.activity = ref.get;}@Overridepublic void handleMessage(Message msg) {super.handleMessage(msg);switch (msg.what) {case 1://需要做判空操作if (activity != ) {activity.mTextView.setText("new Value");}break;default:Log.i(TAG, "handleMessage: default ");break;}@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);//在onCreate中初始化activityWeakReference = new WeakReference<MainActivity>(this);myHandler = new MyHandler(activityWeakReference);myHandler.sendEmptyMessage(1);mTextView = (TextView) findViewById(R.id.tv_test);}

2.onNewIntent的调用时机

在Android应用程序开发的时候,从一个Activity启动另一个Activity并传递少量数据到新的Activity上非常简单,但是当您需要让后端运行的Activity回到前端并传递少量数据可能就会存在一点点小问题。

首先,在默认情况下,当您通过Intent启到一个Activity的时候,就算已经存在一个相同的正在运行的Activity,系统都会创立一个新的Activity实例并显示出来。为了不让Activity实例化屡次,我们需要通过在AndroidManifest.xml配置activity的加载方式(launchMode)以实现单任务模式,如下所示:

<activity android:label="@string/app_name"android:launchmode="singleTask"android:name="Activity1"></activity>

launchMode为singleTask的时候,通过Intent启到一个Activity,假如系统已经存在一个实例,系统就会将请求发送到这个实例上,但这个时候,系统就不会再调用通常情况下我们解决请求数据的onCreate方法,而是调用onNewIntent方法

前提:ActivityA已经启动过,处于当前应用的Activity堆栈中;当ActivityA的LaunchMode为SingleTop时,假如ActivityA在栈顶,且现在要再启动ActivityA,这时会调用onNewIntent方法

当ActivityA的LaunchMode为SingleInstance,SingleTask时,假如已经ActivityA已经在堆栈中,那么此时会调用onNewIntent方法

当ActivityA的LaunchMode为Standard时,因为每次启动ActivityA都是启动新的实例,和原来启动的没关系,所以不会调用原来ActivityA的onNewIntent方法,依然调用的是onCreate方法

以下是代码实例

  • 设置MainActivity的启动模式为SingleTask(栈内复用)
<activityandroid:name=".MainActivity"android:launchMode="singleTask"><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity>
  • MainActivity中重写onNewIntent方法
public class MainActivity extends AppCompatActivity {private static final String TAG = "MainActivity";private Button mButton;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mButton = (Button) findViewById(R.id.forward_btn);mButton.setOnClickListener(new View.OnClickListener {@Overridepublic void onClick(View view) {startActivity(new Intent(MainActivity.this, Main2Activity.class));}});@Overrideprotected void onNewIntent(Intent intent) {Toast.makeText(this, "onnewIntent", Toast.LENGTH_SHORT).show;Log.i(TAG, "onNewIntent: i done....");}
  • Main2Actvity执行点击跳转,MainActivity被复用,执行onNewIntent方法
public class Main2Activity extends AppCompatActivity {private Button mButton;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main2);mButton = (Button)findViewById(R.id.btn);mButton.setOnClickListener(new View.OnClickListener {@Overridepublic void onClick(View view) {startActivity(new Intent(Main2Activity.this,MainActivity.class));finish;}});

3.RecyclerView相比ListView有哪些优势

首先需要解释下RecyclerView的这个名字了,从它类名上看,RecyclerView代表的意义是,我只管Recycler View,也就是说RecyclerView只管回收与复用View,其余的你可以自己去设置。可以看出其高度的解耦,给予你充分的定制自由(所以你才可以轻松的通过这个控件实现ListView,GirdView,瀑布流等效果)

其次RecyclerView提供了增加、删除item的动画 效果,而且可以自己设置

RecyclerView相比ListView优势在于可以轻松实现:

  • ListView的功能
  • GridView的功能
  • 横向ListView的功能
  • 横向ScrollView的功能
  • 瀑布流效果
  • 便于增加Item添加和移除动画

不过一个挺郁闷的地方就是,系统没有提供ClickListener和LongClickListener。

不过我们也可以自己去增加,只是会多了些代码而已。

实现的方式比较多,你可以通过mRecyclerView.addOnItemTouchListener去监听而后去判断手势,

当然你也可以通过adapter中自己去提供回调


4.谈一谈Proguard混淆技术

Proguard技术有如下功能:

  • 压缩 –检查并移除代码中无用的类
  • 优化–对字节码的优化,移除无用的字节码
  • 混淆–混淆定义的名称,避免反编译
  • 预监测–在java平台对解决后的代码再次进行检测

代码混淆只在上线时才会用到,debug模式下会关闭,是一种可选的技术。

那么为什么要使用代码混淆呢?

由于Java是一种跨平台的解释性开发语言,而java的源代码会被编译成字节码文件,存储在.class文件中,因为跨平台的需要,java的字节码中包含了很多源代码信息,诸如变量名、方法名等等。并且通过这些名称来访问变量和方法,这些变量很多是无意义的,但是又很容易反编译成java源代码,为了防止这种现象,我们就需要通过proguard来对java的字节码进行混淆,混淆就是对发布的程序进行重新组织和解决,使得解决后的代码与解决前的代码有相同的功能,和不同的代码展现,即便被反编译也很难读懂代码的含义,哪些混淆过的代码仍能按照之前的逻辑执行得到一样的结果。

但是,某些java类是不能被混淆的,比方实现了序列化的java类是不能被混淆的,否则反序列化时会出问题。

下面这类代码混淆的时候要注意保留,不能混淆。

  • Android系统组件,系统组件有固定的方法被系统调用。
  • 被Android Resource 文件引用到的。名字已经固定,也不能混淆,比方自己设置的View 。
  • Android Parcelable ,需要使用android 序列化的。

其余Anroid 官方建议 不混淆的,如

  • android.app.backup.BackupAgentHelper
  • android.preference.Preference
  • com.android.vending.licensing.ILicensingService
  • Java序列化方法,系统序列化需要固定的方法。
  • 枚举 ,系统需要解决枚举的固定方法。
  • 本地方法,不能修改本地方法名
  • annotations 注释
  • 数据库驱动
  • 有些resource 文件

5.ANR出现的场景及处理方案

在Android中,应用的响应性被活动管理器(Activity Manager)和窗口管理器(Window Manager)这两个系统服务所监视。当客户触发了输入事件(如键盘输入,点击按钮等),假如应用5秒内没有响应客户的输入事件,那么,Android会认为该应用无响应,便弹出ANR对话框。而弹出ANR异常,也主要是为了提升客户体验。

处理方案是对于耗时的操作,比方访问网络、访问数据库等操作,需要开拓子线程,在子线程解决耗时的操作,主线程主要实现UI的操作


这些是小编根据上面的高级工程师技术大纲整理的一套系统全面而且非常深入的Android进阶资料
高级进阶技术大纲
Android系统进阶资料
这些资料都可以免费分享给大家!QQ群:【Android技术开发交流②】979045005:https://jq.qq.com/?_wv=1027&k=5gc0B9E
欢迎大家进群,领取资料,一起学习交流!

命运永远会眷顾那些努力付出的人。假如你还没达到你的目标,那是由于你努力的程度还不够!

希望看到最后的朋友们,通过一年到两年的努力,都能实现追求的目标。

说明
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是摆设,本站源码仅提供给会员学习使用!
7. 如遇到加密压缩包,请使用360解压,如遇到无法解压的请联系管理员
开心源码网 » 月薪20+的Android面试都问这些问题系列一(含答案)

发表回复