当前位置: 首页 > news >正文

面试题:Android 中 Intent 采用了什么设计模式?

在这里插入图片描述

答案是采用了原型模式。原型模式的好处在于方便地拷贝某个实例的属性进行使用、又不会对原实例造成影响,其逻辑在于对 Cloneable 接口的实现。

话不多说看下 Intent 的关键源码:

// frameworks/base/core/java/android/content/Intent.java
public class Intent implements Parcelable, Cloneable {...private static final int COPY_MODE_ALL = 0;private static final int COPY_MODE_FILTER = 1;private static final int COPY_MODE_HISTORY = 2;@Overridepublic Object clone() {return new Intent(this);}public Intent(Intent o) {this(o, COPY_MODE_ALL);}private Intent(Intent o, @CopyMode int copyMode) {this.mAction = o.mAction;this.mData = o.mData;this.mType = o.mType;this.mIdentifier = o.mIdentifier;this.mPackage = o.mPackage;this.mComponent = o.mComponent;this.mOriginalIntent = o.mOriginalIntent;...if (copyMode != COPY_MODE_FILTER) {...if (copyMode != COPY_MODE_HISTORY) {...}}}...
}

可以看到 Intent 实现的 clone() 逻辑是直接调用了 new 并传入了自身实例,而非调用 super.clone() 进行拷贝。

默认的拷贝策略是 COPY_MODE_ALL,顾名思义,将完整拷贝源实例的所有属性进行构造。其他的拷贝策略是 COPY_MODE_FILTER 指的是只拷贝跟 Intent-filter 相关的属性,即用来判断启动目标组件的 actiondatatypecomponentcategory 等必备信息。无视启动 flagbundle 等数据。

// frameworks/base/core/java/android/content/Intent.java
public class Intent implements Parcelable, Cloneable {...public @NonNull Intent cloneFilter() {return new Intent(this, COPY_MODE_FILTER);}private Intent(Intent o, @CopyMode int copyMode) {this.mAction = o.mAction;...if (copyMode != COPY_MODE_FILTER) {this.mFlags = o.mFlags;this.mContentUserHint = o.mContentUserHint;this.mLaunchToken = o.mLaunchToken;...}}
}

还有中拷贝策略是 COPY_MODE_HISTORY,不需要 bundle 等历史数据,保留 action 等基本信息和启动 flag 等数据。

// frameworks/base/core/java/android/content/Intent.java
public class Intent implements Parcelable, Cloneable {...public Intent maybeStripForHistory() {if (!canStripForHistory()) {return this;}return new Intent(this, COPY_MODE_HISTORY);}private Intent(Intent o, @CopyMode int copyMode) {this.mAction = o.mAction;...if (copyMode != COPY_MODE_FILTER) {...if (copyMode != COPY_MODE_HISTORY) {if (o.mExtras != null) {this.mExtras = new Bundle(o.mExtras);}if (o.mClipData != null) {this.mClipData = new ClipData(o.mClipData);}} else {if (o.mExtras != null && !o.mExtras.isDefinitelyEmpty()) {this.mExtras = Bundle.STRIPPED;}}}}
}

总结起来:

Copy Modeaction 等数据flags 等数据bundle 等历史
COPY_MODE_ALLYESYESYES
COPY_MODE_FILTERYESNONO
COPY_MODE_HISTORYYESYESNO

除了 Intent,Android 源码中还有很多地方采用了原型模式。

  • Bundle 也实现了 clone(),提供了 new Bundle(this) 的处理:

    public final class Bundle extends BaseBundle implements Cloneable, Parcelable {...@Overridepublic Object clone() {return new Bundle(this);}
    }
    
  • 组件信息类 ComponentName 也在 clone() 中提供了类似的实现:

    public final class ComponentName implements Parcelable, Cloneable, Comparable<ComponentName> {...public ComponentName clone() {return new ComponentName(mPackage, mClass);}
    }
    
  • 工具类 IntArray 亦是如此:

    public class IntArray implements Cloneable {...@Overridepublic IntArray clone() {return new IntArray(mValues.clone(), mSize);}
    }
    

原型模式也不一定非得实现 Cloneable,提供了类似的实现即可。比如:

  • Bitmap 没有实现该接口但提供了 copy(),内部将传递原始 Bitmap 在 native 中的对象指针并伴随目标配置进行新实例的创建:

    public final class ComponentName implements Parcelable, Cloneable, Comparable<ComponentName> {...public Bitmap copy(Config config, boolean isMutable) {...noteHardwareBitmapSlowCall();Bitmap b = nativeCopy(mNativePtr, config.nativeInt, isMutable);if (b != null) {b.setPremultiplied(mRequestPremultiplied);b.mDensity = mDensity;}return b;}
    }
    
http://www.lryc.cn/news/20202.html

相关文章:

  • Java数据类型与变量
  • Python为CANoe工程添加/删除DBC文件
  • 不同的产品经理特征和需要的能力
  • webpack之处理样式资源
  • Golang 接口笔记
  • [计算机网络(第八版)]第二章 物理层(章节测试/章节作业)
  • [iOS 理解] Swift Runtime (1) 类
  • ASEMI低压MOS管20N06参数,20N06体积,20N06大小
  • 常见前端基础面试题(HTML,CSS,JS)(四)
  • RabbitMQ发布确认模式
  • 零基础的人如何入门 Python ?看完这篇文章你就懂了
  • Atcoder abc257 E
  • 模拟退火算法改进
  • SpringBoot+HttpClient+JsonPath提取A接口返回值作为参数调用B接口
  • JUC 之 CompletableFuture
  • 7-vue-1
  • OpenAPI SDK组件介绍
  • 【Java】Synchronized锁原理和优化
  • 西北工业大学2020-2021学年大物(I)下期末试题选填解析
  • PHP - ChatGpt API 接入 ,代码,亲测!(最简单!)
  • 物联网MQTT协议简单介绍
  • Dubbo 源码解读:负载均衡策略
  • 吃瓜教程笔记—Task04
  • 进程地址空间(虚拟地址空间)
  • 【项目精选】基于Vue + ECharts的数据可视化系统的设计与实现(论文+源码+视频)
  • JavaScript Window Screen
  • 【双重注意机制:肺癌:超分】
  • 各种中间件的使用
  • Systemverilog覆盖率的合并和计算方式
  • (周末公众号解读系列)2000字-视觉SLAM综述