Android面试题汇总-框架技术

一、OkHttp

OkHttp是一个流行的HTTP客户端库,用于发送和接收HTTP网络请求。以下是OkHttp的关键特性和工作原理的概述:

  1. 执行请求:
    • execute() 方法: 同步执行HTTP请求,返回Response对象。
    • enqueue() 方法: 异步执行HTTP请求,通过Callback接口处理响应。
  2. Dispatcher:
    • 调度器管理并发请求,维护三个队列:准备运行的异步请求、正在运行的异步请求和同步请求。
  3. 线程池:
    • OkHttp使用线程池来执行请求,使用SynchronousQueue作为阻塞队列,以实现快速响应。
  4. 拦截器:
    • 拦截器是OkHttp中的核心组件,用于在请求发送前和响应返回后进行数据处理和转换。
    • 主要拦截器包括RetryAndFollowUpInterceptorBridgeInterceptorCacheInterceptorConnectInterceptor
  5. 责任链模式:
    • OkHttp使用责任链模式来组织拦截器,每个拦截器按顺序处理请求,然后传递给下一个拦截器。
  6. 缓存:
    • OkHttp支持HTTP缓存,可以配置缓存策略和大小。
    • 可以通过自定义拦截器控制缓存行为。
  7. 连接池:
    • OkHttp的连接池可以复用Socket连接,减少连接建立和断开的开销。
    • 支持HTTP/1.x和HTTP/2协议,后者允许多路复用,提高效率。
  8. 自定义拦截器:
    • 用户可以通过addInterceptoraddNetworkInterceptor添加自定义拦截器,以实现特定的功能或逻辑。

同步GET请求:

OkHttpClient client = new OkHttpClient();
Request request = new Request.Builder().url("http://publicobject.com/helloworld.txt").build();try (Response response = client.newCall(request).execute()) {if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);// 打印响应头Headers responseHeaders = response.headers();for (int i = 0; i < responseHeaders.size(); i++) {System.out.println(responseHeaders.name(i) + ": " + responseHeaders.value(i));}// 打印响应体System.out.println(response.body().string());
}

异步GET请求:

client.newCall(request).enqueue(new Callback() {@Overridepublic void onFailure(Call call, IOException e) {e.printStackTrace();}@Overridepublic void onResponse(Call call, Response response) throws IOException {try (ResponseBody responseBody = response.body()) {if (!response.isSuccessful()) throw new IOException("Unexpected code " + response);// 打印响应头Headers responseHeaders = response.headers();for (int i = 0, size = responseHeaders.size(); i < size; i++) {System.out.println(responseHeaders.name(i) + ": " + responseHeaders.value(i));}// 打印响应体System.out.println(responseBody.string());}}
});

在这些示例中,我们首先创建了一个OkHttpClient实例,然后构建了一个Request对象,并通过调用execute()方法(对于同步请求)或enqueue()方法(对于异步请求)来执行请求。在异步请求的情况下,我们提供了一个Callback来处理成功和失败的响应。

二、Glide

Glide 是一个流行的图片加载库,它提供了丰富的功能来处理图片加载和显示。以下是 Glide 的关键特性和工作原理的概述:

  1. 生命周期绑定:
    • Glide 可以通过 Glide.with(Activity activity) 方法与 ActivityFragment 的生命周期绑定。
    • 它通过创建一个无UI的 Fragment 并将其绑定到当前 Activity 来实现这一点。
    • RequestManager 负责管理图片加载请求,并通过实现 LifecycleListener 接口与 Fragment 的生命周期同步。
  2. 缓存机制:
    • Glide 使用两种类型的缓存:内存缓存和磁盘缓存。
    • 内存缓存防止应用重复将图片读入内存,而磁盘缓存防止应用重复从网络或其他地方下载和读取数据。
    • 内存缓存使用弱引用和 LRU 缓存策略,而 LRU 缓存内部使用 LinkedHashMap 来存储数据。
  3. 与 Picasso 的区别:
    • Glide 和 Picasso 都是图片加载库,但它们在缓存机制和图片处理上有所不同。
    • Glide 通常比 Picasso 快,尤其是在处理大型图片或 GIF 动画时。
    • Glide 在缓存和内存管理方面更加高效,更有利于减少内存溢出的风险。

生命周期绑定:

public class MyActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_my);ImageView imageView = findViewById(R.id.my_image_view);// Glide与Activity的生命周期绑定Glide.with(this).load("http://example.com/image.jpg").into(imageView);}
}

在这个例子中,Glide.with(this)方法将Glide的请求与MyActivity的生命周期绑定。当Activity被销毁时,所有相关的图片加载请求也会被取消。

缓存机制: Glide的缓存机制包括内存缓存和磁盘缓存。以下是如何配置Glide的缓存策略的示例:

Glide.with(context).load("http://example.com/image.jpg").diskCacheStrategy(DiskCacheStrategy.ALL) // 使用磁盘缓存所有版本的图片.memoryCache(new LruResourceCache(yourSizeInBytes)) // 设置内存缓存的大小.into(imageView);

在这个例子中,diskCacheStrategy(DiskCacheStrategy.ALL)指示Glide缓存所有版本的图片(原始尺寸和转换后的尺寸)。memoryCache(new LruResourceCache(yourSizeInBytes))设置了内存缓存的大小。

这些代码片段展示了如何在Android应用中使用Glide进行图片加载,并与Activity的生命周期绑定以及配置缓存策略。

三、LruCache

LruCache的原理基于最近最少使用(LRU)算法,它使用双向链表和哈希表来存储和管理缓存数据。以下是LruCache工作原理的详细解释和代码示例:

原理:

  • 双向链表: 用于维护缓存项的顺序,链表头部存放最久未使用的缓存项,尾部存放最近使用的缓存项。

  • 哈希表: 用于快速查找缓存项,确保get和put操作的时间复杂度为O(1)。

操作逻辑:

  • get操作: 使用key从哈希表中查找元素。如果找到,将其移动到链表尾部,表示最近被使用。

  • put操作: 检查哈希表中是否存在key。

    • 如果存在,更新值并移动到链表尾部。
    • 如果不存在,检查当前缓存大小。
      • 如果超过最大容量,移除链表头部元素(最久未使用的元素)。
      • 将新元素插入链表尾部。

为什么使用LinkedHashMap:

  • LinkedHashMap维护了元素的插入顺序或访问顺序,这对于实现LRU缓存至关重要。

代码示例:

int cacheSize = 4 * 1024 * 1024; // 4MiB
LruCache<String, Bitmap> cache = new LruCache<String, Bitmap>(cacheSize) {protected int sizeOf(String key, Bitmap value) {return value.getByteCount();}
};// 使用get方法获取缓存项
Bitmap bitmap = cache.get("key");
if (bitmap == null) {// 缓存项不存在,加载新的缓存项bitmap = loadImage("key");// 使用put方法添加缓存项cache.put("key", bitmap);
}// 使用put方法更新缓存项
cache.put("newKey", newBitmap);

在这个示例中,我们创建了一个LruCache实例,指定了缓存大小,并重写了sizeOf方法来计算每个缓存项的大小。使用get方法来检索缓存项,如果不存在,则加载新的缓存项并使用put方法将其添加到缓存中。这样,最近最少使用的缓存项会被自动移除以保持缓存大小不超过设定的限制。

四、Retrofit

当然,我可以帮助您了解Retrofit的动态代理机制。Retrofit是一个类型安全的HTTP客户端,它的动态代理功能允许在运行时创建代理类,而不是在编译时。这样做的好处是可以在不知道原始类和接口的具体实现的情况下,动态地确定代理类的行为。这里是一个简化的过程说明:

  1. 建造者模式构建Retrofit实例: 使用Retrofit.Builder类,可以设置基础URL、转换器等,最终调用build()方法来创建一个Retrofit实例。

    Retrofit retrofit = new Retrofit.Builder().baseUrl("https://api.example.com").addConverterFactory(GsonConverterFactory.create()).build();
    
  2. 返回Service的动态代理对象: 通过调用Retrofit实例的create()方法,传入API接口的Class对象,Retrofit内部使用动态代理生成该接口的实现。

    MyApiService service = retrofit.create(MyApiService.class);
    
  3. 解析接口注解: 当调用service的方法时,Retrofit解析方法上的注解,并根据这些注解动态构建HTTP请求。

    @GET("users/{user}/repos")
    Call<List<Repo>> listRepos(@Path("user") String user);
    
  4. 调用OkHttp的网络请求方法: Retrofit内部使用OkHttp来执行实际的网络请求。通过回调执行器,可以将结果从子线程切换到主线程。

    service.listRepos("octocat").enqueue(new Callback<List<Repo>>() {@Overridepublic void onResponse(Call<List<Repo>> call, Response<List<Repo>> response) {// 处理响应,如更新UI}@Overridepublic void onFailure(Call<List<Repo>> call, Throwable t) {// 处理失败情况}
    });
    

动态代理的核心是InvocationHandler接口和Proxy类。InvocationHandler负责定义代理逻辑,而Proxy类在运行时创建代理实例。这种机制允许Retrofit在不直接依赖具体实现的情况下,动态地处理对API接口方法的调用。

五、LeakCanary

LeakCanary 是一个内存泄漏检测库,它利用了 JVM 的垃圾回收机制中的引用类型来监测内存泄漏。

  1. 四种引用:
    • 强引用(Strong Reference): 最常见的引用类型,只要强引用还存在,垃圾回收器永远不会回收被引用的对象。
    • 软引用(Soft Reference): 内存不足时,垃圾回收器可能会回收这些对象。
    • 弱引用(Weak Reference): 垃圾回收器在任何时候都可能回收这些对象。
    • 虚引用(Phantom Reference): 最弱的引用关系,无法通过虚引用来取得对象实例,必须和引用队列(ReferenceQueue)联合使用。
  2. ReferenceQueue: 当软引用、弱引用或虚引用指向的对象被垃圾回收器回收时,JVM 会将这些引用加入到与之关联的引用队列中。这允许程序监控哪些对象已被回收。
  3. LeakCanary 的工作流程:
    • 注册 ActivityLifecycleCallbacks 回调,以监控所有 Activity 的生命周期事件。
    • ActivityFragment 被销毁时,将它们放入一个弱引用 WeakReference 中,并将该弱引用与 ReferenceQueue 关联。
    • 延时检测 ReferenceQueue 中是否存在当前弱引用对象,如果不存在,可能发生了内存泄漏。
    • 如果检测到内存泄漏,手动触发 GC,并检查 ReferenceQueue
    • 如果对象仍未被回收,获取内存快照(hprof 文件)。
    • 使用 Shark 库解析 hprof 文件,转换成 Hprof 对象。
    • 通过对象关系图 HprofHeapGraph 获取泄漏对象的 objectIds
    • 找出内存泄漏对象到 GC roots 的最短路径。
    • 输出分析结果并展示到页面。

LeakCanary 的优点在于它能够自动检测内存泄漏,并提供详细的泄漏路径,帮助开发者快速定位问题。

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

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

相关文章

STM32的独立看门狗详解

目录 1.独立看门狗是什么&#xff1f; 2.独立看门狗的作用 3.独立看门狗的实现原理 4.独立看门狗用到的寄存器 4.1 IWDG_KR &#xff08;关键字计时器&#xff09; 4.2 IWDG_PR&#xff08;预分频寄存器&#xff09; 4.3 IWDG_RLR&#xff08;重装载寄存器&#xff09…

云原生存储:使用MinIO与Spring整合

在现代云原生应用开发中&#xff0c;高效、可靠的存储解决方案是至关重要的。MinIO是一个高性能、分布式的对象存储系统&#xff0c;它与Amazon S3兼容&#xff0c;非常适合在Kubernetes等云原生环境中使用。本文将详细介绍如何在Spring Boot应用中整合MinIO&#xff0c;并提供…

接口调用的三种方式

例子&#xff1a; curl --location http://110.0.0.1:1024 \ --header Content-Type: application/json \ --data {"task_id": 1 }方式一&#xff1a;postman可视化图形调用 方式二&#xff1a;Vscode中powershell发送请求 #powershell (psh) Invoke-WebRequest -U…

熟悉Realsense和机械臂的控制库(如MoveIt!)的使用,以及基本的PID控制和其他控制算法

项目1&#xff1a;基础Realsense数据采集与处理 目标&#xff1a;了解Realsense摄像头的基本使用&#xff0c;数据采集和处理。 步骤&#xff1a; 安装并配置Realsense SDK。使用Realsense摄像头采集深度图像和RGB图像。实现基本的图像处理操作&#xff0c;如边缘检测、物体识…

温州海经区管委会主任、乐清市委书记徐建兵带队莅临麒麟信安调研

7月8日上午&#xff0c;温州海经区管委会主任、乐清市委书记徐建兵&#xff0c;乐清市委常委、副市长叶序锋&#xff0c;乐清市委办主任郑志坚一行莅临麒麟信安调研&#xff0c;乐清市投资促进服务中心及湖南省浙江总商会相关人员陪同参加。麒麟信安董事长杨涛、总裁刘文清热情…

超图Environment.initialization报空指针

如果库没倒错&#xff0c;许可证也在&#xff0c;一般是权限问题&#xff0c;要确保以下三个读写权限都在&#xff0c;制定版本要动态申请&#xff0c;官方Demo的代码里动态申请权限少一个。 /*** 需要申请的权限数组*/protected String[] needPermissions {Manifest.permiss…

Linux - VIM 全面教程

Linux - VIM 全面教程 前言 VIM 是一个强大的文本编辑器&#xff0c;被广泛用于 Linux 系统上。对于许多程序员和系统管理员来说&#xff0c;熟练掌握 VIM 是一项非常重要的技能。本教程将全面介绍 VIM 的基础知识和高级功能&#xff0c;帮助你更好地利用这一工具。 目录 安…

elasticsearch集群模式部署

系统版本&#xff1a;CentOS Linux release 7.9.2009 (Core) es版本&#xff1a; elasticsearch-7.6.2 本次搭建es集群为三个节点 添加启动用户 添加之前用户要是创建好了的(这里的es用户并不是绝对要求&#xff0c;你可以根据具体的需要命名创建) visudo 修改配置文件 sys…

【吊打面试官系列-MyBatis面试题】使用 MyBatis 的 mapper 接口调用时有哪些要求?

大家好&#xff0c;我是锋哥。今天分享关于 【使用 MyBatis 的 mapper 接口调用时有哪些要求&#xff1f;】面试题&#xff0c;希望对大家有帮助&#xff1b; 使用 MyBatis 的 mapper 接口调用时有哪些要求&#xff1f; 1、Mapper 接口方法名和 mapper.xml 中定义的每个 sql 的…

数据结构(初阶1)

文章目录 一、复杂度概念 二、时间复杂度 2.1 大O的渐进表示法 2.2 时间复杂度计算示例 2.2.1. // 计算Func2的时间复杂度&#xff1f; 2.2.2.// 计算Func3的时间复杂度&#xff1f; 2.2.3.// 计算Func4的时间复杂度&#xff1f; 2.2.4.// 计算strchr的时间复杂度&#xff1f; …

Sharding-JDBC分库分表之SpringBoot分片策略

Sharding-JDBC系列 1、Sharding-JDBC分库分表的基本使用 2、Sharding-JDBC分库分表之SpringBoot分片策略 前言 前一篇以一个示例分享了Sharding-JDBC的基本使用。在进行分库分表时&#xff0c;可以设置分库分表的分片策略&#xff0c;在示例中&#xff0c;使用的是最简单的…

面对数据不一致性的解决方案:

polarDB是读写分离和计算存储分离的分布式数据库&#xff0c;并且副本的log replicate是基于Parallel-Raft协议来实现的。所以在瞬时进行写和读的操作时&#xff0c;是不可避免会存在数据一致性问题&#xff0c;导致这个数据一致性问题的原因不是事务&#xff0c;而是多副本日志…

springboot篮球馆管理系统-计算机毕业设计源码21945

目 录 摘要 1 绪论 1.1选题背景 1.2研究意义 1.3论文结构与章节安排 2 篮球馆管理系统系统分析 2.1 可行性分析 2.1.1 技术可行性分析 2.1.2 经济可行性分析 2.1.3 法律可行性分析 2.2 系统功能分析 2.2.1 功能性分析 2.2.2 非功能性分析 2.3 系统用例分析 2.4 …

数据结构--二叉树收尾

1.二叉树销毁 运用递归方法 分类&#xff1a; 根节点左子树右子树&#xff08;一般都是这个思路&#xff0c;不断进行递归即可&#xff09; 选择方法&#xff08;分析)&#xff1a; 前序&#xff1a;如果直接销毁根就无法找到左子树右子树 中序&#xff1a;也会导致丢失其…

【算法】(C语言):快速排序(递归)、归并排序(递归)、希尔排序

快速排序&#xff08;递归&#xff09; 左指针指向第一个数据&#xff0c;右指针指向最后一个数据。取第一个数据作为中间值。右指针指向的数据 循环与中间值比对&#xff0c;若大于中间值&#xff0c;右指针往左移动一位&#xff0c;若小于中间值&#xff0c;右指针停住。右…

红酒的奇幻之旅:从葡萄园到酒杯的魔法

在世界的某个角落&#xff0c;隐藏着一场关于红酒的奇幻之旅。这是一场从葡萄园到酒杯的魔法变幻&#xff0c;将大自然的馈赠与人类的智慧很好结合&#xff0c;最终呈现在我们眼前的&#xff0c;是一杯散发着迷人香气的雷盛红酒。 一、葡萄园的魔法启幕 当清晨的第一缕阳光洒落…

windows server 2019 更新补丁

1 点击 搜索按键--windows 更新设置 2 点击 安装补丁 3 安装完成&#xff0c;重新启动服务器

四、Python日志系统之日志文件的备份和删除

import os import datetime import logging from watchdog.events import FileSystemEventHandler from watchdog.observers import Observer """实现代码中处理日志文件的备份和删除""" class UserLog:def __init__(self):self.logger logging…

简述 Kamailio 的多租户

如题&#xff0c;大家猜下我会怎么写 过几天公布答案 首先&#xff0c;是认证表 如果单租户, insert into subscriber(username, password) values (1000, 1234)&#xff0c;这样就行 多租户则不然, insert into subscriber(username, domain, password) values (1000, abc.…

WebGIS面试题

文章目录 1. 前端1.1. 选择器的优先级1.2. CSS 中它的布局有哪些&#xff1f;1.3. CSS3 的新特性1.4. CSS 的两种盒子模型1.5. CSS 的伪元素选择器和伪类选择器有哪些&#xff1f;1.6. ES6 的新特性1.7. 谈谈你对 promise 的理解1.8. 简单说一下原型链1.9. 简单说一下深浅拷贝1…