Giter VIP home page Giter VIP logo

android-interview-questions-cn's Introduction

AndroidInterviewQuestions

关于

本项目 Fork 自 stormzhangandroid-interview-questions-cn 项目,为个人对这些问题的答案整理,该项目最新进展还请至原项目查看。

目录

内容

数据结构与算法

数据结构与算法问题的难度完全取决于你所申请的公司

数组

  • 数组由一组相同的数据类型组成。它存储在连续的内存空间内,使用索引可以找到元素的地址。数组包括一维数组和多维数组,一维数组是最简单的数据结构,也是最常用的。

    算法 平均 最坏
    空间(Space) O(n) O(n)
    查找(Search) O(n) O(n)
    插入(Insert) O(n) O(n)
    删除(Delete) O(n) O(n)

链表

  • 链表看起来更像树,而不是数组,它使用一组结点来表示一个序列。每一个结点都包含数据和一个指针。在链表中,结点中的数据可以为任意类型,而指针则是指向下一结点的引用。链表包含一个头结点和一个尾结点。头结点是链表中的第一个结点,尾结点是最后一个结点。链表不是一个循环数据结构,所以尾结点没有指向头结点的指针,指针为空。一些基础方法的时间复杂度如下:
| 算法          | 平均 | 最坏  |
|:------------:|:----:|:----:|
| 空间 (Space)  | O(n) | O(n) |
| 查找 (Search) | O(n) | O(n) |
| 插入 (Insert) | O(1) | O(1) |
| 删除 (Delete) | O(1) | O(1) |

双向链表

  • 一个双向链表首先是一个链表,但是在每个结点中有两个指针,前驱指针指向前驱结点,后继指针指向后继结点。双向链表也有一个头结点,头结点的后继指针指向第一个结点。最后一个结点的后继指针指向空,但是如果最后一个结点的后继指针指向第一个结点,这时称这个链表为双向循环链表。双向循环链表能非常方便地从每个结点查找它的前驱结点和后继结点。

    双向链表

    算法 平均 最坏
    空间 (Space) O(n) O(n)
    查找 (Search) O(n) O(n)
    插入 (Insert) O(1) O(1)
    删除 (Delete) O(1) O(1)

队列

优先队列

动态编程

字符串操作

二叉树

二叉搜索树

排序算法

哈希表与哈希图

广度优先搜索

深度优先搜索

贪心算法

Java 核心

解释一下 OOP(Object-oriented programming) 的概念

从对象和三大特性方面来讲:

  • 万物皆对象,对象是程序的基本单元
  • 类的定义包含数据的形式以及对数据的操作,对象是类的实例
  • 每个对象都有自己的空间,可以容纳其他对象
  • 面向对象编程三大特性:封装、继承、多态
  • 封装隐藏了类的内部实现机制,可以在不影响使用的情况下改变类的内部结构,同时也保护了数据
  • 继承可以重用父类代码,达到代码复用和可扩展的效果
  • 多态就是父类引用可以持有子类对象。非静态成员方法的调用:编译时看父类,运行时看子类
  • 多态优点:不用创建一堆子类对象的引用
  • 多态缺点:不能使用子类特有的成员属性和子类特有的成员方法

抽象类和接口的区别?Link

  • 抽象类可同时包含具体方法和抽象方法(方法未被实现)。
  • 抽象方法必须被该抽象类的子类实现。抽象类是可以继承的。
  • 接口像是描述类的一张蓝图或者说是类的一种契约,它包含了许多空方法,代表着它的所有的子类都应该拥有共同点。它的子类应该提供这些空方法的具体实现。
  • 一个类需要用 implements 来实现接口,接口可以用 extends 来继承其他接口。
  • 一个接口可以继承多个父接口,一个抽象类只可以继承一个父类

序列化是什么?如何实现它?

  • 序列化是一种将对象转换为字节流的过程,目的是为了将该对象存储到内存中,等后面再次构建该对象时可以获取到该对象先前的状态及数据信息。
  • Java 中,有两种方式可以实现序列化,既可以实现 Serializable 接口,也可以实现 Parcelable 接口。
  • 然而,在 Android 中,我们不应该使用 Serializable 接口。因为 Serializable 接口使用了反射机制,这个过程相对缓慢,而且往往会产生出很多临时对象,这样可能会触发垃圾回收器频繁地进行垃圾回收。相比而言,Parcelable 接口比 Serializable 接口效率更高,性能方面要高出 10x 多倍。

什么是单例?几种写法

  • 单例模式是一种对象创建型模式,指的是一个类只能被初始化一次,即只有一个实例。
  • 这个类只能有一个实例
  • 它必须自行创建这个实例
  • 它必须自行向整个系统提供这个实例

什么是匿名内部类?

  • 匿名内部类即没有名字的内部类,只可被使用一次
  • 使用匿名内部类前提条件:必须继承自一个父类或实现一个接口
  • 语法格式为 new SuperType(construction parameters){}
  • 语法格式为 new 父类构造器(参数列表)|实现接口(){}
  • 直接将抽象类的方法或者接口方法在大括号中实现,可以省略一个类的书写
  • 匿名内部类中不能定义构造方法,但可以使用初始化语块代替构造方法

对字符串进行 ==equals() 操作时有什么区别?

  • == 用来判断内存地址是否相同
  • equals()用来判断字符串的内容是否相同

hashCode()equals() 何时使用?

  • 相等(相同)的对象必须具有相等的哈希码(或者散列码)

  • 如果两个对象的 hashCode 相同,它们并不一定相同

  • 往 Set 集合中添加元素时,要保证元素不重复,先调用该元素的 hasCode 方法,定位到它应该放置的物理位置。 如果这个位置上没有元素,就可以直接存储在这个位置上,不用再进行任何比较;如果这个位置上已经有元素了,就调用它的 equals 方法与新元素进行比较,相同的话就不存了,不相同就散列其它的地址。

  • 集合查找时,hashcode 能大大降低对象比较次数,提高查找效率

Java 中的 final, finallyfinalize?

  • final:修饰基本数据类型时,代表常量(不可修改);修饰类或对象引用时,引用不可重新赋值,而类或者对象依然可以修改;修饰方法时,代表该方法不可被重写;修饰类时,代表该类不可被继承
  • finally:异常处理时 try/catch/finally 语句中的 finally 语句块无论有没有异常都会执行 finally 语句块,当然也有特殊情况不执行 finally
  • finalize:垃圾回收器要回收对象的时候,首先会调用这个类的 finalize 方法,可以在该方法中释放调用本地方法时分配的内存

什么是内存泄露,Java 是如何处理它的?

  • 存在下面的这种对象,这些对象不会被 GC 回收,却占用着内存,即为内存泄漏(简单说:存在已申请的无用内存无法被回收)

    • 该对象是可达的,即还在被引用着
    • 该对象是无用的,即程序以后不会再使用该对象
  • 长生命周期的对象持有短生命周期的引用,就很可能会出现内存泄露

  • 内存泄漏可能的情况:

    • 静态集合类引起的

垃圾收集器是什么?它是如何工作的

  • 垃圾收集是一种自动的内存管理机制

  • 所有的对象实例都在 JVM 管理的堆区域分配内存

  • 只要对象被引用,JVM 就会认为它还存活于进程中

  • 一旦对象不再被引用,垃圾收集器将删除它并重新声明为未使用的内存

比较 ArrayArrayList链接

不同点:

  • 实现上:Array 是本地的程序设计组件或者数据结构,ArrayList 是一个 Java 集合类的类
  • 性能上:Array 会优于 ArrayList
  • 类型安全方面:ArrayList 是类型安全的,支持编译时检查;Array 不是类型安全的,支持运行时检查
  • 灵活性:ArrayList 优于 Array,ArrayList 是动态的;Array 是静态的,创建了数组就无法更改它的大小
  • 基本类型:Array 既可以保存基本类型,也可以保存对象;ArrayList 不可以保存基本类型,可以保存封装类
  • 泛型:ArrayList 可以显示的支持泛型,Array 不可以
  • 维度:Array 可以是多维度的,ArrayList 并不支持指定维度

相同点:

  • 数据结构:都是基于 index 的数据结构
  • 空值:都可以存储空值(null),但只有 Object 的数组可以这样,基本类型会存储他们的默认值
  • 重复:都允许元素重复

比较 HashSetTreeSet

  • TreeSet 会自动按自然排序法给元素排序,HashSet 不会
  • 如果不需要使用排序功能,应该使用 HashSet,因为其性能优于 TreeSet
  • 如果 TreeSet 传入的是自定义的对象,必须让该对象实现 Comparable 接口

Java 中的类型转换

Java 中的类型转换分为基本数据类型和引用数据类型

  • 基本数据类型

    基本数据类型的转换分为自动转换和强制转换

    • 自动转换是从位数低的类型向位数高的类型转换
    • 强制类型转是从位数高的类型向位数低的类型转换,需要转型的数据前加上“()”,然后在括号内加入需要转化的数据类型,转换会导致精度丢失
  • 引用数据类型

    • 子类可以转换成父类
    • 父类转换为子类不一定可行:当父类(引用)引用的为子类对象时,此时父类可以正常转换为子类;而当父类(引用)引用的确实为父类对象时,此时无法转换为子类,会抛出 ClassCastException 异常

方法重载和重写的区别链接

  • 重载是在同一个类中完成的,重写父类需要有子类。

  • 方法重载时返回值类型、参数列表可以不同;方法重写时返回值类型、参数列表必须相同

  • 重载的方法使用静态绑定完成,发生在编译时;重写的方法使用动态绑定完成,发生在运行时。性能:重载比重写更有效率

  • 静态方法可以重载,意味着一个类可以有多个同名的静态方法。静态方法不能被重写,即使在子类中声明了一个相同的静态方法,它与父类的相同方法无关

  • 私有方法和用 final 修饰的方法可以重载但不可重写。意味着一个类可以有多个同名的 private/final 方法,子类不能重写父类的 private/final 方法。

什么是访问修饰符?它们能做什么?

  • 访问修饰符是指能够控制类、成员变量、方法的使用权限的关键字
  • 可以使用它们来保护对类、变量、方法和构造方法的访问
  • public:共有的,对所有类可见
  • protected:受保护的,对同一包内的类和所有子类可见
  • private:私有的,在同一类内可见
  • 默认:在同一包内可见。默认不使用任何修饰符。

接口可以继承另一个接口吗?

可以,Java 允许一个接口继承多个父接口。

Java 中 static 关键字是什么意思?链接

  • static 表示“全局”或者“静态”的意思,用来修饰成员变量和成员方法,也可以修饰代码块

  • static 所蕴含“静态”的概念表示着它是不可恢复的,即在那个地方,你修改了,他是不会变回原样的,你清理了,他就不会回来了

  • 被 static 修饰的成员变量和成员方法是独立于该类的,它不依赖于某个特定的实例变量,也就是说它被该类的所有实例共享

  • 所有实例的引用都指向同一个地方,任何一个实例对其的修改都会导致其他实例的变化

  • 静态变量是随着类加载时被完成初始化的;静态方法也是类加载时就存在了,所以不可为 abstract,必须实现;静态代码块也会随着类的加载一块执行

Java 中静态方法可以被重写吗?

  • 不可以被重写,静态方法隶属于某个类,只和类相关

什么是多态?什么是继承?链接

多态:

  • 多态就是父类引用可以持有子类对象。非静态成员方法的调用:编译时看父类,运行时看子类
  • 多态优点:不用创建一堆子类对象的引用
  • 多态缺点:不能使用子类特有的成员属性和子类特有的成员方法

继承:

  • 继承是使用已存在的类的定义作为基础建立新类的技术,新类的定义可以增加新的数据或新的功能,也可以用父类的功能,但不能选择性地继承父类
  • 通过使用继承我们能够非常方便地复用以前的代码,大大提高开发的效率

Integerint 之间的区别自动装箱、自动拆箱

  • int 是基本数据类型,Integer 是包装类
  • Java中,会对 -128 到 127 的 Integer 对象进行缓存,当创建新的 Integer 对象时,如果符合这个这个范围,并且已有存在的相同值的对象,则返回这个对象,否则创建新的 Integer 对象

Java 中的对象是否会以引用传递或者值传递?详细说明。

Java 中的对象以按值传递的形式进行调用,所有方法得到的都是参数值的拷贝

  • 对于基本数据类型的参数,方法得到的是其值的拷贝
  • 对于引用数据类型的参数,方法得到的也是其值(所指向对象的内存地址)的拷贝
  • 方法均不可以修改参数变量的值,对于引用类型参数来说,就是不能让引用类型参数指向新的对象
  • 但是方法可以修改引用类型参数所指向对象的内容,因为方法得到的是该对象地址的拷贝,可以根据地址获取到该对象

什么是 ThreadPoolExecutor? Link

  • 线程池的目的是实现线程复用,减少频繁创建和销毁线程,提高系统性能

  • ThreadPoolExecutor 是线程池的核心类,用于创建线程池

  • ThreadPoolExecutor 的构造方法包括如下参数:

    • corePoolSize 核心线程池大小
    • maximumPoolSize 线程池最大容量
    • keepAliveTime 线程池空闲时,线程存活时间
    • TimeUnit 时间单位
    • ThreadFactory 线程工厂
    • BlockingQueue 任务队列
    • RejectedExecutionHandler 线程拒绝策略

本地变量、实例变量以及类变量之间的区别?

  • 类体由两部分组成,变量定义和方法定义
  • 本地变量即局部变量,定义在方法内部或者方法的形参中
  • 实例变量为为非静态变量,每一个对象的实例变量都不同
  • 类变量为静态变量,属于类所有

什么是反射? Link

在 Java 中什么是强引用、软引用、弱引用以及虚引用?参考链接

  • 强引用:是使用最普遍的引用。如果一个对象具有强引用,那垃圾回收器绝不会回收它。
  • 软引用:如果一个对象只具有软引用,且内存空间足够,垃圾回收器就不会回收它;如果内存空间不足了,就会回收这些对象的内存。
  • 弱引用:只具有弱引用的对象拥有比软引用更短暂的生命周期,在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。
  • 如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。

什么是依赖注入?能说几个依赖注入的库么?你使用过哪些?

关键字 synchronized 的作用是什么?参考链接

  • 当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。
  • 当两个并发线程访问同一个对象 object 中的这个 synchronized(this) 同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。
  • 然而,当一个线程访问 object 的一个 synchronized(this) 同步代码块时,另一个线程仍然可以访问该 object 中的非 synchronized(this) 同步代码块。
  • 尤其关键的是,当一个线程访问 object 的一个 synchronized(this) 同步代码块时,其他线程对 object 中 所有其它 synchronized(this) 同步代码块 的访问将被阻塞。
  • 上一个例子同样适用其它同步代码块。也就是说,当一个线程访问 object 的一个 synchronized(this) 同步代码块时,它就获得了这个 object 的对象锁。结果,其它线程对该 object 对象 所有同步代码部分 的访问都被暂时阻塞。

为什么说 String 不可变的?参考链接

  • 什么是不可变对象

    • 如果一个对象,在它创建完成之后,不能再改变它的状态,那么这个对象就是不可变的。
    • 不能改变状态的意思是,不能改变对象内的成员变量,包括基本数据类型的值不能改变,引用类型的变量不能指向其他的对象,引用类型指向的对象的状态也不能改变。
  • 在 Java 中 String 类其实就是对字符数组的封装

  • String 类中有一个 char 数组的 value 引用变量,value 指向真正的数组对象

  • JDK 1.6 及以前还有 offset 和 count 两个变量

  • 这三个变量都是 private final 的,即一旦这三个值被初始化,就不能再改变了

  • 但是可以通过反射改变 value 变量引用的对象,进而改变 String 对象

修饰符 transientvolatile 的作用是什么?

finalize() 方法的作用是什么?

异常捕获中的 try{} finally{} 块儿是如何工作的?

对象的实例化和初始化之间的区别是什么?参考链接1参考链接2参考链接3

  • 1.执行顺序不同:

    • 初始化执行顺序:静态成员变量/静态代码块 -> 成员变量/构造代码块 -> 构造方法
    • 实例化执行顺序:成员变量/构造代码块 -> 构造方法
    • 以上,静态成员变量/静态代码块的执行顺序为定义顺序,成员变量/构造代码块的执行顺序也为定义顺序
  • 2.执行时机不同

    类的初始化时机:

    • 类的一个实例被创建时
    • 类的一个静态方法被调用时
    • 类声明的一个静态变量被赋值时
    • 类声明的一个静态变量被使用且这个变量不是常量时
    • 一个类初始化时,如果父类没初始化则需要先初始化父类

    类的实例化时机

    • 当创建一个类的新的实例时(可以通过 new 关键字、反射机制、clone 方法、反序列化方式等)

静态块何时运行?链接

  • 静态块在 JVM 加载类时执行,仅执行一次
  • 一个类中可以有多个静态代码块
  • 类调用时,先执行静态代码块,然后才执行主函数
  • 执行静态代码块 -> 执行构造代码块 -> 执行构造函数
  • 静态代码块其实就是给类初始化的,而构造代码块是给对象初始化的
  • 静态代码块中的变量是局部变量,与普通函数中的局部变量性质没有区别

解释一下 Java 中的泛型?参考链接

  • 泛型即宽泛的数据类型
  • 允许在定义类和接口的时候使用类型参数(type parameter)
  • 声明的类型参数在使用时用具体的类型来替换
  • 使用泛型的时候加上的类型参数,会被编译器在编译的时候去掉,这个过程就称为类型擦除

StringStringBufferStringBuilder 的区别在哪里?

  • String 是不可变对象,因此每次对 String 对象进行更改时,都会生成一个新的对象,然后将指针指向新的对象。
  • 使用 StringBuffer 类时,每次都是对 StringBuffer 对象本身进行操作,并不是生成新的对象并改变引用。
  • StringBuilder 和 StringBuffer 非常类似,但是是非线程安全的,在单线程中性能比 StringBuffer 高

StringBuilder 是怎么避免不可变字符串分配的问题?

什么是自动装箱和拆箱?

  • 自动装箱就是 Java 自动将原始类型值转换成对应的对象,比如将 int 的变量转换成 Integer 对象
  • 反之将 Integer 对象自动转换成 int 类型值,这个过程叫做拆箱
  • 自动装箱时编译器调用 valueOf 将原始类型值转换成对象
  • 自动拆箱时,编译器通过调用类似 intValue(), doubleValue() 这类的方法将对象转换成原始类型值
  • 自动装箱、拆箱主要发生在
    • 赋值时
    • 方法调用时

枚举和迭代器有什么区别?

Java 中 fail-fastfail-safe 的区别?

什么是 Java 优先级队列?

什么是设计模式 Link

Android 核心

阐述一下 Activity 的生命周期。参考链接

Activity 生命周期的顺序为:

onCreate() -> onContentChanged() -> onStart() -> onPostCreate()-> onResume() -> onPostResume -> onPause() -> onStop() -> onDestroy()

  • onCreate() 方法:首次创建 Activity 时调用,应该在此方法中执行所有正常的静态设置 — 创建视图、将数据绑定到列表等等。
  • onRestart() 方法:在 Activity 已停止(调用 onStop())并即将再次启动前调用,始终后接 onStart()
  • onStart() 方法:在 Activity 即将对用户可见之前调用,可后接 onStop()进入隐藏状态
  • onResume() 方法:在 Activity 即将开始与用户进行交互之前调用,始终后接 onPause()
  • onPause() 方法:当系统即将开始继续另一个 Activity 时调用。用于确认对持久性数据的未保存更改、停止动画以及其他可能消耗 CPU 的内容。应该非常迅速地执行所需操作,因为它返回后,下一个 Activity 才能继续执行。
  • onStop() 方法:在 Activity 对用户不再可见时调用。可后接 onRestart()。
  • onDestroy() 方法:在 Activity 被销毁前调用。

方法调用后终止问题:

  • onCreate(),onRestart(),onStart(),onResume() 调用后,系统不能随时终止 Activity 进程。
  • 而 onPause(),onStop(),onDestroy() 调用后,可以终止。
  • Activity 创建后,onPause() 必定成为最后调用的方法,然后才能终止进程 — 如果系统在紧急情况下必须恢复内存,则可能不会调用 onStop() 和 onDestroy()。
  • 所以,在 onPause() 调用时,向存储设备写入至关重要的持久性数据(例如用户编辑)。
  • 不过,应该对 onPause() 调用期间必须保留的信息有所选择,因为该方法中的任何阻止过程都会妨碍向下一个 Activity 的转变并拖慢用户体验。

谈谈 Android 的四大组件。参考链接1参考链接2

四大组件:Activity、Service、BroadcastReceiver、ContentProvider

  • Activity:

    • 一个 Activity 通常就是一个单独的屏幕,它上面可以显示一些控件也可以监听并处理用户的事件做出响应
    • Activity 之间通过 Intent 进行通信,应用内使用显示 Intent,跨应用使用隐式 Intent
    • 在隐式 Intent 中,有三个最重要的部分:动作(Action)、类别(category)以及动作对应的数据(data)
    • Action、category 和 Uri 分别对应着 intent-filter 中的 action、category 和 data
  • Service:

    • Service 是实现程序后台运行的解决方案,非常适合去执行不需要和用户交互(即没有界面)并且还要求长期执行的任务
    • 通过 Context 的 startService 和 stopService 可以启动、停止服务,启动服务后启动者和服务就没有关系了,启动者结束生命周期,服务依然在
    • 通过 Context 的 bindService 和 unBindService 可以绑定、解绑服务,绑定服务后服务会返回给启动者一个 IBind 接口实例,启动者可以通过这个接口实例回调服务的方法。此时 Context 退出,服务也会跟着解绑退出
  • BroadcastReceiver:

    • 应用可以使用广播接收器对感兴趣的外部事件或内部事件进行注册监听,可以静态注册、动态注册
    • 广播接收器没有用户界面。但它们可以启动一个 Activity 或 Service 来响应它们收到的信息,或者用 NotificationManager 来通知用户
    • 广播分为普通广播、有序广播、粘性广播。注册有序广播接收器可以设定接收优先级按序接收或拦截;发送粘性广播后注册广播接收器也可以接收到该粘性广播
  • ContentProvider:可参考《第二行代码》Page 270

    • 内容提供器用于提供在不同程序之间实现数据共享的功能,允许一个程序访问另一个程序中的数据,同时还能保证被访数据的安全性

    • 每个 ContentProvider 都用一个 Uri 作为独立的标识

    • 访问其他程序中的数据

      • 可以通过 Context 的 getContentResolver 方法获取到 ContentResolver 的实例,通过该实例的一系列方法对目标数据进行 CRUD 操作
      • 通过内容 Uri 表明想要访问的目标程序
    • 创建自己的内容提供器

      • 继承 ContentProvider,实现抽象方法
      • 通过 UriMacther 类的 match 方法判断其他程序想要访问的数据地址

Service 与 IntentService 的区别。Link

  • 相同点:

    • 都是在后台运行,不可见
    • 都可在任何线程、系统组件进行开启
  • 不同点:

    • Service 默认运行在主线程,不可直接做耗时操作;
    • IntentService 默认运行在子线程,可直接执行耗时操作
    • Service 不会自动退出,需要手动 stopSelf、stopService 或者 unbindService;
    • IntentService onHandleIntent 方法执行完毕后会自动退出。当有多次请求(即多次调用 startService)时,因只有一个工作线程,所以会一个一个进行处理,处理完毕后退出

Android 应用的结构是什么?

Android 应用中如何保存数据。

分为保存键值集、保存文件(分为内部存储和外部存储)、在 SQL 数据库中保存数据以及在网络中使用网络存储。

其中:

  • 键值集中适合保存私有原始轻量化数据
  • 内部存储适合保存私有数据
  • 外部存储适合保存公共数据
  • 私有数据库中适合保存结构化数据
  • 网络存储需使用自己的网络服务器存储数据

如何在 Android 应用中执行耗时操作。

将耗时操作放到非 UI 线程执行,常用的包括:AsyncTask、Handler、Thread 等。

两个 Fragment 之间如何通信。参考链接

  • 通过宿主 Activity 拿到 FragmentManager,进而再 findFragmentById 获取到另一个 Fragment
  • 使用接口:在一个 Fragment 中定义接口,并在 onAttach 方法中拿到 Activity 初始化接口对象。Activity 来实现该接口,并在实现中调用另一个 Fragment 的相应方法
  • 使用 Activity 的公共方法:直接 getActivity 并调用 Activity 的公共方法
  • 使用广播
  • 使用 EventBus
  • 使用 Fragment 的 setArguments?

阐述一下 Android 的通知系统。参考链接1参考链接2

通知是可以在应用常规 UI 外部可以向用户显示的消息。

  • 创建通知:

    • 在 NotificationCompat.Builder 对象中指定通知的相关信息
    • 调用 notify 方法将通知发送出去
    • 通知操作由 PendingIntent 定义
  • 更新通知:

    • 调用 notify 方法时调用带有 id 的通知
    • 更新或重新创建 NotificationCompat.Builder 对象,重新发送相同 id 的通知对象即可
  • 删除通知:

    • 用户手动清除(如果可以清除的话)
    • 创建通知时调用了 setAutoCancle 方法,且用户点击了通知
    • 调用 cancle 方法,cancleAll 方法

两个不同的 app 之间如何交互。

  • Activity
  • Broadcast
  • ContentProvider
  • Service(AIDL)
  • 利用 ShareUserId 共享数据:两个应用的 ShareUserId 相同,则共享对方 data 目录下的文件

什么是 Fragment?

  • 碎片是一种可以嵌入在活动中的 UI 片段
  • 可以将多个碎片组合在一个 Activity 中,也可以在多个 Activity 中重复使用一个碎片
  • 可以在 Activity 运行时动态添加、删除碎片
  • 可以把碎片当作 Activity 的模块化组成部分,具有自己的生命周期,能接收自己的输入事件

为什么建议只使用默认的构造方法来创建 Fragment?Link

  • 应当使用默认构造方法创建 Fragment,然后使用 setArguments 添加 Bundle 来传递参数
  • 因为系统在储存 Fragment 的状态时(比如说屏幕旋转导致的重新加载),系统会为我们自动存储 Bundle

为什么 Bundle 被用来传递数据,为什么不能使用简单的 Map 数据结构?链接1链接2链接3

原因一:速度和内存占用问题

  • 使用 Bundle 传递数据一般用于启动 Activity,启动 Fragment 时,数据量一般都不大
  • Bundle 内部由 ArrayMap 实现。ArrayMap 的内部实现是两个数组,一个数组存放 key 的 hashCode 值,一个数组存放 key 与 value 值。在查找、添加、删除数据时,会根据二分查找法查找 key 的 hashCode 值所对应的 index 值,再根据 index 去查找 key 所对应的 value 值
  • HashMap 的实现则是采用数组加链表的结构,默认使用一个容量为 16 的数组来存储,数组中每一个元素又是一个链表的头节点
  • 所以在小数据量的情况下,ArrayMap(Bundle) 比 HashMap 在速度和内存上都更占优势

原因二:序列化问题

  • Android 中使用 Intent 来传递数据时,需要保证数据是基本数据类型或可序列化类型
  • Bundle 使用 Parcelable 序列化,HashMap 使用 Serialable 序列化
  • 在 Android 中,更推荐使用 Parcelable 序列化,因为 Serializable 接口使用了反射机制,这个过程相对缓慢,而且往往会产生出很多临时对象,可能会触发垃圾回收器频繁地进行垃圾回收。相比而言,Parcelable 接口比 Serializable 接口效率更高,性能方面要高出 10x 多倍。

阐述一下 Fragment 的生命周期。参考链接

  • Fragment 的生命周期与 Activity 的生命周期协调一致
  • Activity 的每次生命周期回调都会引发每个片段的类似回调
  • 片段还有几个额外的生命周期回调,用于处理与 Activity 的唯一交互,以执行构建和销毁片段 UI 等操作,包括:
    • onAttach:在片段已与 Activity 关联时调用
    • onCreateView:在这可创建片段的视图层次结构
    • onActivityCreated:在 Activity 的 onCreate() 方法已返回时调用
    • onDestroyView:在移除与片段关联的视图层次结构时调用
    • onDetach:在取消片段与 Activity 的关联时调用

下图为 Activity 的每个连续状态如何决定片段可以收到的回调方法

如何理解 Android 的 Dialog ?参考链接

  • Dialog 是对话框的基类,使用时应实例化其子类:AlertDialog、DatePickerDialog 或 TimePickerDialog

  • 应使用 DialogFragment 作为对话框的容器

  • 创建对话框:

    • 通过扩展 DialogFragment 并在 onCreateDialog 方法中 创建 AlertDialog
    • AlertDialog 包括:标题、内容区域、操作按钮
    • 其中内容区域可以为:单选列表、单选框、复选框
    • 按钮可以为:肯定、否定、中性
    • 可以自定义布局
  • 事件回传给对话框的宿主

    • 通过 interface 实现,在 DialogFragmen 的 onAttach 方法
  • 显示对话框:

    • 调用 DialogFragment 实例的 show 方法
  • 清除对话框

    • 用户做出选择后会被清除
    • 调用 dismiss 方法手动清除
  • 取消对话框

    • 代表用户显式离开,而不完成任务(点击返回键、触摸对话框外部区域)
    • 调用 cancel 方法手动取消

解释下 Android 的 View 。

  • View 是 UI 组件最基本的构建块
  • 一个 View 占据屏幕的一块区域,负责绘制和事件处理

你能创建自定义 View 吗?具体是如何创建的?参考链接

  • 自定义 View 包括自定义 View 和自定义 ViewGroup

  • 自定义 View 一般继承自 View、SurfaceView 或其他 View,不包含子 View

  • 自定义 ViewGroup 是利用现有的组件,根据特定的布局方式,来组成新的组件。一般继承自 ViewGroup 或各种 Layout,包含有子 View

  • 自定义 View 的流程:

    • 构造函数:初始化一些内容,获取自定义属性
    • 测量 View 的大小(onMeasure):View 大小不仅由自身决定,也会受父控件的影响,我们自己进行测量可以适应各种情况
    • 确定 View 的大小(onSizeChanged):在视图大小发生改变时调用
    • 确定子 View 布局(onLayout):在自定义 ViewGroup 时用到,用于确定子 View 的位置。循环取出子 View,计算出各个子 View 的位置,调用子 View 的 layout 方法设置其位置
    • 绘制内容(onDraw)
    • 对外提供操作方法和监听回调:控制 View 的状态,监听 View 的变化等

什么是 ViewGroup ,它与 View 的区别在哪里?

  • ViewGroup 是一个放置 View 的容器,继承自 View,是一种特殊的 View
  • ViewGroup 的职能为:给 childView 计算出建议的宽、高和测量模式;决定 childView 的位置;
  • View 的职能为:根据测量模式和 ViewGroup 建议的宽高计算出自己的宽和高;在 ViewGroup 为其指定的区域内绘制自己的形态

Fragment 和 Activity 有什么区别?它们之间又有什么关系?

  • Fragment 在 Android 3.0 后引入,可以翻译为碎片、片段
  • Activity 为活动,代表着当前屏幕显示的一个窗口
  • Fragment 可以重用,必须嵌套在 Activity 中使用,由 Activity 的 FragmentManager 管理,生命周期受 Activity 影响

谈谈 Serializable 接口和 Parcelable 接口的区别。在 Android 中最好使用哪种接口?

Activity 的启动模式有哪些?参考链接

  • standard 模式:创建新实例,放入当前任务中

  • singleTop 模式:

    • 当前任务中已存在,且在栈顶,不创建
    • 当前任务中已存在,不在栈顶,创建新实例
    • 当前任务中不存在,创建新实例
  • singleTask 模式:

    • 首先检测要启动的 Activity 的 affinity 和当前任务的 affinity 是否相同
    • affinity 相同则查看当前任务是否有实例。有则调用其 onNewIntent 方法,实例上面的 Activity 全部出栈;无则创建新实例
    • affinity 不相同则创建新任务、新实例
  • SingleInstance 模式:

    • 启动一个新的任务,创建新实例
    • 并且新的任务中不会再添加其他 Activity

解释一下 Android 中的 Intent 。Link

  • Intent 是一个消息传递对象,在启动其他应用组件时使用
  • 可以启动 Activity,启动 Service,发送广播
  • Intent 可以描述目标组件的一些特性,并可以携带一些必要数据

什么是隐式 Intent ?

  • 不指定特定的组件,而是声明要执行的常规操作(Action),从而允许其他应用中的组件来处理它
  • 如声明发送功能,那么拥有发送 Action 的组件都可进行响应

什么是显式 Intent ?

  • 按名称指定要启动的组件,完全限定类名
  • 通常会在自己的应用中使用显式 Intent 来启动组件,因为知道想要启动的 Activity 或服务的类名

解释一下 AsyncTask。参考链接

  • AsyncTask 是 Android 提供的一个助手类,对 Thread 和 Handler 进行了封装,方便我们在后台线程执行耗时操作,在主线程更新 UI 等
  • AsyncTask 有四个回调方法:onPreExecute、doInBackground、onProgressUpdate、onPostExecute。
    • onPreExecute 在执行后台操作之前调用,运行在主线程
    • doInBackground 是必须实现的一个核心方法,可以执行后台耗时操作,运行在自子线程
    • onProgressUpdate 在 doInBackground 方法中调用publishProgress 方法时会进行回调,可以进行后台操作状态的展示更新,运行在主线程
    • onPostExecute 在后台操作完成后调用,运行在主线程

如何理解 Android 中的广播。Link

  • 广播接收器是一种用于响应系统范围广播通知的组件
  • 广播接收器不会显示用户界面,但可以创建状态栏通知,在发生广播事件时提醒用户
  • 广播接收器更常见的用途是作为通向其他组件的“通道”,比如开启一个服务,发起一个通知等,设计用于执行极少量的工作

如何理解 Android 的 LocalBroadcastManager 。参考链接

  • LocalBroadcastManager 为本地广播
  • 使用这个机制发出的广播仅在应用内部传递
  • 广播接收器也只能接收本应用发出的广播
  • 本地广播比系统全局广播更加安全、有效

什么是 JobScheduler ?参考链接

  • 当一定的条件(网络、时间、电量等)被满足时,JobScheduler 会为应用执行某项任务
  • 与 AlarmManager 不同,JobScheduler 的执行时间是不确定的
  • JobScheduler 允许同时执行多个任务

什么是 DDMS ?你可以用它来做什么?参考链接

  • DDMS 是 Android 应用程序调试和分析工具,可以截屏、查看进程、线程、堆栈信息、Logcat、广播状态信息、模拟呼叫、短信、位置等
  • 但是现在已经不推荐使用
  • Android Studio 3.0 以后推荐使用 Android Profile 来分析网络、内存、cpu 使用状态
  • 推荐使用 ADB 来发送命令、传文件、截图等

解释一下什么是 support libary ,以及为什么要引入 support library ?Link

Android 支持库提供了许多没有内置到框架中的功能。

  • 向后兼容:提供了许多没有内置到框架中的功能、UI 元素,可以保证使用了新系统功能的 app 可以向后兼容旧版本系统
  • 提供布局模式、界面元素:可以避免再做一些重复性工作
  • 支持不同形态的设备:可以通过支持库为各个平台(手机、电视、手表等)提供功能

如何理解 Android 中的 ContentProvider 。它通常用来干什么?参考链接

  • 内容提供器管理一组共享的应用数据
  • 其他应用可以通过我们应用程序的内容提供程序查询、甚至修改我们应用的数据
  • 我们也可以通过其他应用程序的内容提供器查询、修改该应用的数据
  • 也适用于读取和写入应用不共享的私有数据
  • 定义内容提供器需要继承自 ContentProvider,并实现相应的方法

什么是 Data Binding ?Link参考链接2

数据绑定指:将数据源和 UI 进行绑定,并使其同步

在 Android 中:

  • 使用 Android 官方提供的 Data binding 库,将 xml 布局和数据源进行绑定
  • UI 布局接收用户事件并同步更新数据源,修改数据源也可同步展示到 UI 上

数据绑定和 MVVM 的区别:

  • MVVM 是一种架构模式
  • 数据绑定是一种实现数据和 UI 绑定的框架
  • 数据绑定是构建 MVVM 模式的一个工具

Android 的核心组件具体都有什么?

  • Activity
  • Service
  • BroadcastReceiver
  • ContentProvider

什么是 ADB ?参考链接

ADB 是 Android Debug Bridge 的简称,即 Android 调试桥。是一个通用命令行工具,允许与模拟器或连接的 Android 设备进行通信。包括三个组件:

  • 客户端:运行在开发计算机上。用来发送命令,可以通过 adb 命令在命令行终端调用到客户端
  • 服务器:以后台进程形式运行在开发计算机上。用来管理客户端(开发计算机)和后台程序(Android 终端)之间的通信
  • 后台程序(adbd):以后台进程形式运行在模拟器或 Android 设备上。在设备上运行命令

什么是 ANR ?如何避免发生 ANR ?参考链接1参考链接2参考链接3

  • ANR 是 Application Not Responding,即应用程序未响应

  • 以下情况会引起 ANR:

    • 主线程被 IO 操作堵塞
    • 主线程存在耗时计算
    • 主线程存在错误操作,如 Thread.sleep 或 Thread.wait 等
  • 以下情况会弹出 ANR 对话框

    • 应用在 5 秒内未响应用户的输入事件(触摸、按键)
    • BroadcastReceiver 未在 10 秒内完成相应的处理
    • Activity、Application 回调方法超时时间:5 秒
    • Service 回调方法超时时间:20 秒
    • BroadcastReceiver 回调方法超时时间:前台 App 10 秒,后台 App 60 秒
  • 如何避免发生 ANR:

    • 使用 AsyncTask 处理耗时 IO 操作
    • 使用 Thread 或 HandlerThread 时,要调用 Process.setThreadPriority(PROCESS_THREAD_PRIORITY_BACKGROUND)设置线程优先级,因为默认 Thread 优先级和主线程相同
    • 使用 Handler 处理工作线程结果,不要使用 Thread.sleep,Thread.wait
    • Activity 的 onCreate 和 onResume 回调中,避免耗时代码
    • BroadcastReceiver 的 onReceive 方法也应避免耗时代码,可以开启 IntentService 代替
    • 可以通过查看 /data/anr/traces.txt 文件定位 ANR 情况

AndroidManifest.xml 是什么?参考链接

  • AndroidManifest.xml 是应用清单,用来向 Android 系统提供必要的应用信息,系统必须得有这些信息才可以运行应用的代码

  • 具体功能如下:

    • 为应用的 Java 软件包命名,软件包名称充当应用的唯一标志符
    • 描述四大组件,以及它们可以处理的 Intent 信息,启动这些组件的条件信息等
    • 确定托管应用组件的进程
    • 声明权限,权限是用于限制对部分代码或设备上数据的访问,为了保护可能被误用以致破坏或损害用户体验的关键数据和代码
    • (声明应用所需的最低 Android API 级别)
    • (列出应用必须链接到的库)

解释一下 broadcast 和 intent 在 app 内传递消息的工作流程。

当 Bitmap 占用较多内存时,你是怎么处理的?

  • 及时释放 Bitmap 内存

  • 捕获 OutOfMemory 异常

  • 将图片放置于合适的 drawable 文件夹下

  • 图片压缩

    • 质量压缩,通过 Bitmap.compress()
    • 内存压缩,通过设置 BitmapFactory.Options.inSampleSize 值
  • 图片复用

    • 内存缓存
    • 磁盘缓存
    • Bitmap 复用

Android 应用有哪些不同的存储数据的方式?

  • 键值集(SharedPreferences)
  • SQL 数据库
  • 文件
  • 网络存储

什么是 Dalvik 虚拟机?

  • Dalvik 虚拟机是 Google 设计用于 Android 平台的虚拟机
  • 支持已转换为 .dex 格式的 Java 程序的运行,.dex 格式是专门为 Dalvik 虚拟机设计的一种压缩格式
  • 允许在有限内存中运行多个虚拟机实例,并且每一个都作为独立的 Linux 进程运行

Dalvik 与 JVM 的关系:

  • Dalvik 基于寄存器,JVM 基于堆栈
  • Dalvik 有自己的字节码,并不使用 Java 字节码
  • 从 Android 2.2 开始,Dalvik 开始支持即时编译
  • Dalvik 会通过 Zygote 进行类的预加载和资源的预加载,完成虚拟机的初始化

AsyncTask 的生命周期和(它所属的) Activity 的生命周期有什么关系?这种关系可能会导致什么样的问题? 如何避免这些问题发生?参考链接

  • AsyncTask 并不会随着 Activity 的销毁而销毁,而是会一直执行 doInBackground 方法直到方法结束
  • doInBackground 方法执行结束后,如果 cancel(boolean) 方法调用了,执行 onCancelled 方法;cancel(boolean) 没调用时,执行 onPostExecute 方法

导致的问题:

  • 当使用非静态内部 AsyncTask 类时,AsyncTask 类会持有外部类 Activity 的引用,当 Activity 销毁时,如果 AsyncTask 还在执行,会造成 Activity 对象无法回收,内存泄漏

解决方案:

  • 如果 doInBackground 方法内有循环操作时,应使用 isCancelled 来进行判断,避免后续无用的循环操作
  • 销毁 Activity 实例时,调用 AsyncTask 的 cancel 方法
  • 使用静态内部类,持有外部类的弱引用

Intent filter 是用来做什么的?

  • Intent filter 即 Intent 过滤器,是应用清单文件中的表达式,用来指定该组件要接收的 Intent 类型
  • 比如给 Activity 声明了 Intent filter 后,可以使其他应用使用某一特定类型的 Intent 启动该 Activity
  • Intent filter 可以包括 action(必须有一个)、category 和 data

什么是 Sticky Intent?Link

  • Sticky Intent 是发送粘性广播时的消息传递对象

什么是 AIDL ?列举一下通过 AIDL 创建被绑定的服务(bounded service)的步骤。参考链接

Android 的权限有多少个不同的保护等级?参考链接

  • 正常权限:应用需要访问沙盒外部数据或资源,但对用户隐私或其他应用风险很小的区域。系统会自动授予应用正常权限
  • 危险权限:应用需要涉及用户的隐私信息或资源,或对用户数据、其他应用产生影响。用户需手动向应用授予此类权限
  • 特殊权限:SYSTEM_ALERT_WINDOW、WRITE_SETTINGS 等,大多数应用不该使用它们,需要的话需要发送请求用户授予的 Intent

在转屏时你如何保存 Activity 的状态?参考《Android 开发艺术探索》Page 8

  • 资源相关的系统配置发生改变时会导致 Activity 被杀死并重新创建

保存 Activity 状态的方法:

  • 系统会在 Activity 即将被销毁且有机会重新显示的情况下,会在 onStop 方法之前调用 onSaveInstanceState 方法
  • 系统会默认帮我们保存当前 Activity 的视图结构,通过查看每个 View 的 onSaveInstanceState 和 onRestoreInstanceState 方法可以查看系统会自动为 View 保存哪些状态
  • 我们可以手动在 onSaveInstanceState 方法中保存数据

恢复 Activity 状态:

  • Activity 启动时,会回调 onCreate 和 onRestoreInstanceState 方法(onStart 方法之后调用),我们可以在这两个方法里拿到 Activity 销毁时保存的数据
  • 在 onCreate 方法中取 Bundle 对象时,需要先判断是否为空,因为正常启动的 Activity 时系统无保存状态

相对布局和线性布局的区别。

如何实现 XML 命名空间?参考链接1参考链接2

  • 命名空间里存放的是特定属性的集合,可以避免元素命名冲突
  • 为布局文件的根元素增加 xmlns 属性,即可通过不同的命名空间调用相应的属性

View.GONE 和 View.INVISIBLE 之间的区别。

  • View.GONE:控件不可见,在布局中不占据空间
  • View.INVISIBLE:控件不可见,但在布局中还占据空间

Bitmap 和 .9(nine-patch)图片之间有什么区别?

  • 点 9 图是一种可伸缩的位图,可以指定图片哪些部分可以拉伸,哪些不可以拉伸。避免图片拉伸变形,在不同分辨率下可以达到较好的适配效果。
  • 而 Bitmap 展示出来后在不同的屏幕分辨率下可能会被异常拉伸

谈谈位图池。Link

在 Android 中如何避免内存泄漏?参考链接

  • 将非静态内部类改为静态内部类
  • 通过程序逻辑切段非静态内部类所持有的外部类引用
  • 单例模式使用生命周期更长的 Context
  • 及时关闭各种连接(数据库、网络、cursor 等),释放资源

Android 桌面的小部件是什么?

  • AppWidgetProvider 的本质是一个广播接收器,继承自 BroadcastReceiver
  • AppWidgetProvider 能接收 Widget 的相关广播,如 Widget 的更新、删除、开启、禁用等
  • 桌面小部件的加载以及更新采用的都是 RemoteViews

什么是 AAPT ?参考链接

  • AAPT 是 Android Asset Packaging Tool 的缩写,为 SDK 自带的工具
  • 可以查看、创建、更新 zip 格式的文档附件(zip、jar、apk),也可以将资源文件编译成二进制文件

你是如何在 Android 应用程序中发现内存泄漏的?参考链接

发现内存泄漏的方法:

  • 使用 LeakCanary
  • 使用 Android Monitor
  • 启用 StrictMode

你如何排查应用崩溃的原因?

  • Java Crash:

    • 通过 Logcat 查看堆栈信息
    • 通过 UncaughtExceptionHandler 来记录异常日志并查看
    • ANR 可通过查看系统 /data/anr/ 文件夹下的 traces.txt 查看分析
    • 第三方 SDK 如腾讯的 Bugly,友盟等
  • Native Crash:

    • 通过 Logcat 分析
    • 通过第三方 SDK

为什么你应该避免在主线程上运行非用户界面相关的代码?

  • 如果主线程执行耗时操作的话,当 UI 事件发生时,让用户等待时间超过 5 秒而未处理就会出现 ANR
  • 主线程主要负责把 UI 事件分发给合适的 View 或 Widget

你是如何适配不同分辨率的手机的?参考链接

如何理解 Doze 模式。如何理解应用程序待机模式(App Standby)。

在 Android 中,你可以使用什么来进行后台操作?

  • 采用 Service
  • 采用子线程搭配 Handler

什么是 ORM ?它是如何工作的?

  • ORM 即 Object Relational Mapping,对象关系映射
  • 目前的数据库是关系型数据库,ORM 的工作就是把数据库中的关系数据映射为程序中的对象

什么是 Loader ?参考链接

  • Loader 即加载器,可以在 Activity 和 Fragment 中轻松实现异步加载数据
  • 对数据源变化进行监听,实时更新数据

什么是 NDK ,为什么它是有用的?参考链接

  • NDK 即原生开发工具包,是一组允许在 Android 应用使用原生代码语言(C、C++)的工具

优势:

  • 可以从设备获取卓越的性能用于计算密集型应用,如游戏或物理模拟
  • 可以复用 C 或 C++ 库
  • 可以在平台之间移植应用

如何理解严格模式(StrictMode)。 参考链接

StrictMode 是用来检测程序中违例情况的开发者工具。最常用的场景是检查主线程中文件读写和网络读写等耗时操作。

StrictMode 主要检测两大问题:线程策略(ThreadPolicy)和 VM 策略(VMPolicy)

线程策略检查等内容有:

  • 自定义的耗时调用
  • 磁盘读取
  • 磁盘写入
  • 网络操作

VM 策略检查内容有:

  • Activity 泄漏
  • 未关闭的 Closable 对象泄漏
  • Sqlite 对象泄漏
  • 检测实例数量

什么是 Lint ?它的用途是什么?

  • Lint 是 Android Studio 提供的一个静态代码检查工具
  • 帮助我们发现代码中的潜在 bug、安全、性能、国际化、辅助性、错误拼写等问题

什么是 SurfaceView ?参考链接

  • SurfaceView 拥有独立的绘图表面,即不与宿主窗口共享一个绘图表面
  • 由于拥有独立的绘图表面,SurfaceView 的 UI 就可以在一个独立的线程中进行绘制
  • 由于不占用主线程资源,SurfaceView 一方面可以实现复杂而高效的 UI,另一方面又会不会导致用户输入不能及时响应

ListView 和 RecyclerView 有什么区别?参考链接1参考链接2

效率上:

  • RecyclerView 效率较于 ListView 更高
  • RecyclerView 的 ViewHolder 模式更加规范

功能上:

  • RecyclerView 新增了 LayoutManager,布局效果更丰富,包括线性布局、网格布局、瀑布流布局等

使用上:

  • ListView 自带空数据处理:setEmptyView 方法
  • ListView 自带 HeaderView、FooterView
  • RecyclerView 提供局部刷新方法:notifyItemChanged
  • RecylerView 默认封装了一些动画

什么是 ViewHolder 模式?为什么我们应该使用它?

  • 使用 ListView 或 RecyclerView 时,每个 item 在进入用户视野时会从 item 的 xml 文件中创建 view 对象,这一步可以利用 ConvertView 对创建好的布局进行缓存,直接从 ConvertView 中取即可。
  • 拿到 item 布局后,再 findViewById 找到 item 的子 view 的控件对象,这一步是树查找操作极其耗时。
  • 可以创建一个 ViewHolder 对象,将 item 子控件的实例都存放在 ViewHolder 对象中,这样就不用每次 findViewById 了,提高效率

什么是 PendingIntent ?

PendingIntent 可以理解为延迟执行的 Intent,在某个特定条件下才会执行该 Intent。

Flags 的类型:

  • FLAG_ONE_SHOT:得到的 PendingIntent 只能使用一次,使用一次后自动调用 cancel 方法解除 PendingIntent 和 Intent 的关联
  • FLAG_NO_CREATE:当 PendingIntent 不存在时,不进行创建,直接返回 null
  • FLAG_CANCEL_CURRENT:当 PendingIntent 已存在时,执行 cancel 进行解除,再创建一个新的
  • FLAG_UPDATE_CURRENT:不存在时就进行创建,创建后每次使用会对数据进行更新
  • FLAG_IMMUTABLE:创建好 PendingIntent 后就保持一成不变

send 方法:

调用 send 方法时会启动包装的 Intent

cancel 方法:

调用 cancel 方法时会解除 PendingIntent 和被包装 Intent 之间的关联,只有创建该 PengdingIntent 的程序才有权解除

你能手动调用垃圾回收吗?

周期地更新页面的最好方式是什么?

  • 使用 Alarm 机制
  • 使用定时器 Timer 类

Alarm 机制简介:参考《第一行代码》Page469

  • AlarmManager 设置定时任务的常用方法包括:

    • set(@AlarmType int type, long triggerAtMillis, PendingIntent operation)
    • setRepeating(@AlarmType int type, long triggerAtMillis, long intervalMillis, PendingIntent operation)
    • setExact(@AlarmType int type, long triggerAtMillis, PendingIntent operation)
  • 顾名思义,set 方法设置的是执行一次的定时任务,setRepeating 方法设置的是循环执行的定时任务

  • 从 Android 4.4 开始,系统会对 Alarm 唤醒对齐,所以 set 方法设置的 Alarm 任务执行可能会有一段时间的误差;而 setExact 方法可以保证唤醒时间的精准。

  • 第一个参数 AlarmType,有以下值可选:

    • ELAPSED_REALTIME:定时任务的触发时间从系统开机算起,不会唤醒 CPU
    • ELAPSED_REALTIME_WAKEUP:会唤醒 CPU
    • RTC:定时任务的出发时间从 1970 年 1 月 1 日 0 点算起,不会唤醒 CPU
    • RTC_WAKEUP:会唤醒 CPU
    • 系统开机时间可通过 SystemClock.elapsedRealtime 获得
    • 从 1970 年 1 月 1 日 0 点起的时间可通过 System.currentTimeMillis 获得 intervalMillis 参数代表
  • 第二个参数 triggerAtMillis 代表定时任务触发的时间

  • 第三个参数是一个 PendingIntent 代表任务触发时的执行动作

  • setRepeating 方法中的 intervalMillis 代表定时任务的执行时间间隔

  • 另外有 setInexactRepeating 方法用于设定不精确时间的循环定时任务,比 setRepeating 方法更加节能

有哪些类型的广播?

按执行顺序分为:

  • 标准广播:完全异步执行,所有广播接收器一起接到,无法被某个接收器截断
  • 有序广播:同步执行,优先级高的广播接收器先接收,并可以截断该条广播

按广播定义类型分为:

  • 系统广播:Android 系统发出的广播,比如电池电量改变、时区改变、网络状态改变等
  • 自定义广播:自己定义的广播

按广播范围分为:

  • 全局广播:可以被任何应用程序接收到的广播
  • 本地广播:只能在本应用程序内部传递的广播

按注册后是否可接收分为:

  • 粘性广播:粘性广播发送后,再进行注册的广播接收器也可接收到最后发出的一条该广播(注:粘性广播在 API 21 中已被 deprecated)
  • 非粘性广播:广播发送后注册的广播接收器无法接收到该广播

你开发过组件吗?请描述一下。Link

如何理解上下文(Context)。怎么使用它?参考链接1参考链接2

  • Android 程序要有一个完整的 Android 工程环境,这个工程环境下有四大组件,四大组件需要有各自的上下文
  • 因此 Context 是维持 Android 程序中各个组件正常工作的一个重要功能类

使用中注意点:

  • getApplication 和 getApplicationContext 都可以获取到 Application 实例,并且获取的是同一个对象
  • getApplication 只可以在 Activity 和 Service 中调用,而任何一个 Context 实例都可以调用 getApplicationContext
  • 和 UI 相关的不建议使用 Application Context,单例模式下需要考虑内存泄漏问题
  • 上图数字1:启动 Activity 在这些类中是可以的,但是需要创建一个新的 task。一般情况不推荐。
  • 上图数字2:在这些类中去 layout inflate 是合法的,但是会使用系统默认的主题样式,如果你自定义了某些样式可能不会被使用。
  • 注:ContentProvider、BroadcastReceiver 之所以在上述表格中,是因为在其内部方法中都有一个 context 用于使用

Context 的继承关系:

由图可见:

  • ContextWrapper 是 Context 的封装类,ContextImpl 是 Context 的实现类
  • 我们平时需要 Context 调用的方法都在 ContextWrapper 中可以看到声明,均是调用了成员变量 mBase 的相应方法
  • mBase 为 ContextImpl 对象,在合适的时机由系统调用 attachBaseContext 方法传递进来

你知道什么是视图树(View Tree)吗?怎样优化它的深度?

onTrimMemory() 方法是什么?参考链接1参考链接2

  • Android 4.0 之后加入的一个回调,任何实现了 CompomentCallbacks2 接口的类都可以重写实现该方法
  • 主要作用是指导应用程序在不同的情况下进行自身的内存释放,以避免被系统直接杀掉
  • 系统会根据不同等级的内存使用情况,调用这个回调方法,并传入相应的等级

等级分类如下:

  • TRIM_MEMORY_UI_HIDDEN(20):常用的一个等级,在 UI 界面被隐藏时回调。此时应释放一些 UI 占用的大块内存

程序正常运行时的回调:

  • TRIM_MEMORY_RUNNING_MODERATE(5):应用程序正常运行,进程不会被杀掉。但内存已经有点低了,系统可能会开始通过 LRU 缓存规则去杀死缓存进程了
  • TRIM_MEMORY_RUNNING_LOW(10):应用程序正常运行,进程不会被杀掉。但手机内存已经非常低了,此时应该释放不需要的资源
  • TRIM_MEMORY_RUNNING_CRITICAL(15):应用程序正常运行,但系统已经根据 LRU 缓存规则杀掉了大部分缓存进程。此时应尽可能释放不必要的资源,否则系统会继续杀死缓存进程,并可能开始杀死后台运行服务了

程序是缓存时的回调:

  • TRIM_MEMORY_BACKGROUND(40):手机内存已经很低了,系统准备开始根据 LRU 缓存规则来杀死进程了。此时我们的进程已经被加入 LRU 列表中了,此时释放一些资源可以使手机内存保持充足,从而使我们程序更长时间保存在缓存中
  • TRIM_MEMORY_MODERATE(60):手机内存已经很低了,此时我们的程序进程处于 LRU 缓存列表的中间位置,如果手机内存资源不能得到释放,我们的缓存进程就有可能被杀死
  • TRIM_MEMORY_COMPLETE(80):手机内存已经很低了,此时我们的程序进程处于 LRU 缓存列表的最边缘位置,系统将会优先考虑杀死我们的程序进程,此时应该释放所有能释放的资源

Android 应用可以使用多进程吗?怎样使用?参考链接

如何使用:

  • 依赖于 android:process 属性
  • 如果该属性值以 : 开头,代表这个进程是应用私有的,无法跨应用共用
  • 如果该属性值以小写字母开头,代表这个进程是全局进程,可以被多个应用共用
  • 适用于:Application、Activity、Service、Broadcast、ContentProvider

多进程的好处:

  • 增加 App 可用内存
  • 独立于主进程,确保某些任务的执行与完成

多进程的缺点:

  • 数据共享问题
  • SQLite 被锁
  • Application.onCreate() 不必要的初始化

内存溢出(OutOfMemory)是怎么发生的?参考链接

内存溢出指程序在申请内存空间时,系统没法提供足够的内存空间供其使用

  • 内存泄漏导致
  • 保存了多个占用内存过大的对象(如 bitmap)或加载单个超大图片

加载 bitmap 导致的内存溢出解决方案:

  • 加载多图使用软引用、弱引用
  • 图不再使用时,使用 Bitmap.recycle 加速回收
  • 使用文件缓存

文本样式接口(Spannable)是什么?

什么是过度绘制(overdraw)?参考链接参考链接2

  • GPU 过度绘制指:屏幕上的某个像素,在同一帧时间内,被绘制了多次

产生过度绘制原因:

  • 多层 View 叠加绘制导致

解决过度绘制方法:

  • 移除默认的 Window 背景
  • 移除不必要的背景
  • 写合理且高效的布局,减少层级嵌套
    • 使用 ViewStub 来加载一些不常用的布局
    • 使用 merge 标签减少布局嵌套层次
    • 可复用的组件抽取出来使用 include 引入
  • 自定义控件进行优化(clipRect、quickReject)

什么是渲染脚本(renderscript)?Link

Dalvik 虚拟机模式和 ART(Android Runtime)虚拟机模式的区别。即时编译ART

  • Dalvik 采用 JIT 即时编译技术,ART 采用 Ahead-of-time AOT 预编译技术
  • Dalvik 虚拟机在应用程序启动时,JIT 通过进行连续的性能分析来优化程序代码的执行,在程序运行的过程中,Dalvik 虚拟机在不断的将字节码编译成机器码
  • ART 虚拟机在应用程序安装的过程中,ART 就已经将所有的字节码重新编译成了机器码。应用程序运行过程中无需进行实时的编译工作,只需要进行直接调用
  • 所以 ART 虚拟机提高了程序运行效率,减少手机电量消耗
  • 机器码占用空间更大,所以 ART 下应用占用内部空间更大,首次安装因需要预编译所以时间相比 Dalvik 会略长

FlatBuffers 和 JSON 的区别。Link

谈谈 Android 的注解。Link1, Link2

描述一下约束布局(Constraint Layout)。参考链接

  • 根据布局中其他元素或视图,确定 View 在布局中的位置
  • View 的位置受到三类约束:其他 View、父容器、基准线
  • 支持设置比例

阐述一下 Android 中的 HashMap , ArrayMap 和 SparseArray 。Link

阐述一下 Looper, Handler 和 HandlerThread 。参考链接

一句话:Looper 不断从 MessageQueue 中取出 Message 交给相应的 Handler 处理。

以上称为消息处理机制(消息循环)。

  • Android 中消息循环和消息队列都是针对具体线程的,除了 UI 线程之外,默认创建的工作线程是没有消息循环的
  • Handler 用来将消息压入消息队列以及处理消息
  • 普通工作线程想具有消息循环机制的话,先调用 Looper.prepare 创建消息队列、构造 Looper,再调用 Looper.loop 开启消息循环。此时该线程为 LooperThread
  • 在 LooperThread 中创建 Handler 对象,此时 Handler 对象会自动关联到当前线程的 Looper 对象
  • (构造 Handler 时如果不传 Looper,则会自动调用 mLooper = Looper.myLooper(),Handler 对象会自动关联到当前线程的 Looper 对象)
  • 使用 HandlerThread 可以很方便的开启一个包含 Looper 的子线程,也就是 HandlerThread 自动帮我们 Looper.prepare,Looper.loop。我们只要调用 HandlerThread.start 开启线程后,通过该线程的 Looper 对象去构建相应的 Handler 对象即可。
  • HandlerThread 提供了 quit 和 quitSafely 方法,可以很方便的终止线程消息队列

关于如何将 Message 压入 MessageQueue?

  • 调用 Handler 的 send(Message message) 方法发送一个 Message,最终会调用到 MessageQueue 的 enqueueuMessage 方法,将消息放入消息队列
  • 调用 Handler 的 post(Runnable r) 方法发送一个 Runnable,Runnable 先被封装为 Message 的 callback,再发送该 Message
  • 调用 View 的 post(Runnable r) 方法发送一个 Runnable,和上一条类似。不过调用的是 UI 线程的 Handler 发送的 Message
  • 调用 Activity 的 runOnUiThread(Runnable r) 方法,在 UI 线程调用时,直接执行 Runnable,在非 UI 线程调用时,调用 UI 线程的 Handler 将该 Runnable 发送出去

关于 Looper 从 MessageQueue 中取出消息后的分发?

  • Looper 从消息队列取出消息后
  • 首先调用 Handler 的 dispatchMessage 进行分发,我们可以重写此方法更改逻辑。
  • dispatchMessage 的默认策略如下:
    • Message 的 callback 不为空时,优先调用 Message 的 callback
    • Handler 的 mCallback 不为空时,调用 Handler 的 mCallback
    • 上面俩都为空时,才调用 handleMessage,也就是我们经常重写的那个方法

如何降低 Android 应用的耗电量?Link

SnapHelper 是什么?Link

在 Android 中怎么处理多点触控?link

谈谈对 RxJava 的理解

  • Rxjava 是一个实现异步操作的库
  • RxJava 基于事件,事件在整个过程中进行流动,流动的同时可以进行线程切换、事件转换等等
  • 最常见的使用情景是:在子线程进行数据计算、网络请求,然后回到主线程展示结果

LayoutInflater 的 inflate 方法的几个参数分别代表什么?参考链接

  • 获取到 LayoutInflater 对象可以通过 LayoutInflater.from(Context context)方法或者 activity.getLayoutInflater()方法
  • resouse(int)代表:欲加载的 xml 布局文件的 ID
  • root(ViewGroup)代表:当 attachToRoot 为 true 时,root 为 inflate 方法加载出来的 view 的根布局;attach 为 false 时,root 仅仅为将要加载出来的 View 提供一组 LayoutParams 参数
  • attachToRoot(boolean)代表:是否要将加载出来的 View 附加到上面的 root 中
  • return(View)代表:inflate 方法的返回值是一个 View,为即将加载出来的视图结构的根布局。如果 attachToRoot 为 true,则返回的 View 为刚才的 root;否则返回加载出来 View 的根布局

Maven 和 Gradle 的区别

OkHttp、Retrofit 的区别

介绍下观察者模式

架构

请介绍一下你做的上一个 App 的架构。

请介绍一下 MVP。 Link

Presenter 是什么?

什么是模型?

请介绍一下 MVC。

Controller 是什么?

请介绍一下 MVVM。 Link

谈谈你对代码整洁之道(clean code)的理解。Link

设计问题

请设计 Uber App。

请设计 Facebook App。

请设计 Facebook Near-By Friends App。

请设计 WhatsApp。

请设计 SnapChat。

基于地理位置 App 的设计问题。

工具和技能

Git. Link

RxJava. Link

Dagger 2. Link

Android 开发实用工具。 Link

Firebase. Link

Android 测试驱动开发

Espresso 是什么? Link

Robolectric 是什么? Link

UI-Automator 是什么? Link

请解释一下单元测试。

请解释一下设备化测试。

你是否做过单元测试或者自动测试?

为什么要使用 Mockito? Link

请描述一下 JUnit 测试。

其他

描述一下 REST APIs 如何工作 ?

描述一下 SQLite 。

描述一下 数据库 。

项目管理工具 - trello ,basecamp ,kanban ,jira ,asana 。

关于构建系统 - gradle , ant , buck 。

APK 逆向工程 。

混淆器用于什么 ?

什么是混淆? 用于什么? 如何压缩 ?

如何构建你的发布版本的 APP ?

如何面向特定用户群体更新应用程序版本 ?

可以识别卸载我们的应用程序的用户吗 ?

缩小 APK 的体积 。Link

Android 开发最佳实践 。Link

Android 代码风格和指南 。Link

你尝试使用过 Kotlin 吗 ?Link

开发 Android 应用程序时应该连续测量哪些指标 ?Link

License

   Copyright (C) 2017 xcy396

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

       http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.