面试常客:谈谈你对抽象类和接口的理解,2024年网易Android岗面试必问

抽象类 即包含抽象方法的类。

如果一个类包含一个或者多个抽象方法,该类必须被限定为抽象的。抽象类可以不包含抽象方法。

public abstract class BaseActivity {
private final String TAG = this.getClass().getSimpleName(); //抽象类可以有成员

void log(String msg){ //抽象类可以有具体方法
System.out.println(msg);
}

// abstract void initView(); //抽象类也可以没有抽象方法
}

接口 是抽象类的一种特殊形式,使用 interface 修饰。

public interface OnClickListener {
void onClick(View v);
}

特点与区别

抽象类的特点

抽象类的初衷是“抽象”,即规定这个类“是什么”,具体的实现暂不确定,是不完整的,因此不允许直接创建实例。

  • 抽象类是由子类具有相同的一类特征抽象而来,也可以说是其基类或者父类
  • 抽象方法必须为 public 或者 protected(因为如果为 private,则不能被子类继承,子类便无法实现该方法),缺省情况下默认为 public
  • 抽象类不能用来创建对象
  • 抽象方法必须由子类来实现
  • 如果一个类继承于一个抽象类,则子类必须实现父类的抽象方法,如果子类没有实现父类的抽象方法,则必须将子类也定义为抽象类
  • 抽象类还是很有用的重构工具,因为它们使得我们可以很容易地将公共方法沿着继承层次结构向上移动

接口的特点

Java 为了保证数据安全性是不能多继承的,也就是一个类只有一个父类。

但是接口不同,一个类可以同时实现多个接口,不管这些接口之间有没有关系,所以接口弥补了抽象类不能多继承的缺陷。

接口是抽象类的延伸,它可以定义没有方法体的方法,要求实现者去实现。

  • 接口的所有方法访问权限自动被声明为 public
  • 接口中可以定义“成员变量”,会自动变为 public static final 修饰的静态常量
  • 可以通过类命名直接访问:ImplementClass.name
  • 不推荐使用接口创建常量类
  • 实现接口的非抽象类必须实现接口中所有方法,抽象类可以不用全部实现
  • 接口不能创建对象,但可以申明一个接口变量,方便调用
  • 完全解耦,可以编写可复用性更好的代码

栗子

前面说了太多,我们直接上代码。

假设我们新开始一个项目,需要写大量的 Activity,这些 Activity 会有一些通用的属性和方法,于是我们会创建一个基类,把这些通用的方法放进去:

public class BaseActivity extends Activity {
private final String TAG = this.getClass().getSimpleName();

void toast(String msg) {
Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
}
//其他重复的工作,比如设置标题栏、沉浸式状态栏、检测网络状态等等
}

这时 BaseActivity 是一个基类,它的作用就是:封装重复的内容

写着写着,我们发现有的同事代码写的太烂了,一个方法里几百行代码,看着太痛苦。于是我们就本着“职责分离”的原则,在 BaseActivity 里创建了一些抽象方法,要求子类必须实现:

public abstract class BaseActivity extends Activity {
private final String TAG = this.getClass().getSimpleName();

@Override
protected void onCreate(final Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(getContentViewLayoutId());

initView(); //这里初始化布局
loadData(); //这里加载数据
}

/**

  • 需要子类实现的方法
  • @return
    */
    protected abstract int getContentViewLayoutId();
    protected abstract void initView();
    protected abstract void loadData();

void toast(String msg) {
Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
}
}

定义的抽象方法访问权限修饰符可以是 public protecteddefault,但不能是 private,因为这样子类就无法实现了。

这时 BaseActivity 因为有了抽象方法,变成了一个抽象类。它的作用就是:定义规范,强制子类符合标准;如果有调用抽象方法,也会制定执行顺序的规则。

继承 BaseActivity 的类只要实现这些方法,同时为父类提供需要的内容,就可以和父类一样保证代码的整洁性。

public class MainActivity extends BaseActivity{

private TextView mTitleTv;

@Override
protected int getContentViewLayoutId() {
return R.layout.activity_main;
}

@Override
void initView() {
mTitleTv = (TextView) findViewById(R.id.main_title_tv);
mTitleTv.setOnClickListener(this);
}

@Override
protected void loadData() {
//这里加载数据
}
}

以后如果发现有某些功能在不同 Activity 中重复出现的次数比较多,就可以把这个功能的实现提到 BaseActivity 中。但是注意不要轻易添加抽象方法,因为这会影响到之前的子类。

项目写着写着,发现很多页面都有根据定位信息改变而重新请求数据的情况,为了方便管理,再把这样的代码放到 BaseActivity? 也可以,但是这样一来,那些不需要定位相关的代码不也被“污染”了么,而且冗余逻辑太多 BaseActivity 不也成了大杂烩了么。

我们想要把位置相关的放到另一个类,但是 Java 只有单继承,这时就可以使用接口了。

我们创建一个接口表示对地理位置的监听:

interface OnLocationChangeListener {
void onLocationUpdate(String locationInfo);
}

接口默认是 public,不能使用其他修饰符。

然后在一个位置观察者里持有这个接口的引用:

public class LocationObserver {

List mListeners;

public LocationObserver setListeners(final List listeners) {
mListeners = listeners;
return this;
}

public List getListeners() {
return mListeners;
}

public void notify(String locationInfo) {
if (mListeners != null) {
for (OnLocationChangeListener listener : mListeners) {
listener.onLocationUpdate(locationInfo);
}
}
}

interface OnLocationChangeListener {
void onLocationUpdate(String locationInfo);
}
}

这样我们在需要定位的页面里实现这个接口:

public class MainActivity extends BaseActivity implements View.OnClickListener,
LocationObserver.OnLocationChangeListener {

private TextView mTitleTv;

@Override
protected int getContentViewLayoutId() {
return R.layout.activity_main;
}

@Override
public void onClick(final View v) {
int id = v.getId();
if (id == R.id.main_title_tv) {
toast(“你点击了 title”);
}
}

@Override
void initView() {
mTitleTv = (TextView) findViewById(R.id.main_title_tv);
mTitleTv.setOnClickListener(this);
}

@Override
protected void loadData() {
//这里加载数据
}

@Override
public void onLocationUpdate(final String locationInfo) {
mTitleTv.setText(“现在位置是:” + locationInfo);
}
}

这样 MainActivity 就具有了监听位置改变的能力。

如果 MainActivity 中需要添加其他功能,可以再创建对应的接口,然后予以实现。

小结

小结

有了这么多优秀的开发工具,可以做出更高质量的Android应用。

当然了,“打铁还需自身硬”,想要写出优秀的代码,最重要的一点还是自身的技术水平,不然用再好的工具也不能发挥出它的全部实力。

在这里我也分享一份大佬自己收录整理的Android学习PDF+架构视频+面试文档+源码笔记,还有高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料这些都是我闲暇还会反复翻阅的精品资料。在脑图中,每个知识点专题都配有相对应的实战项目,可以有效的帮助大家掌握知识点。

总之也是在这里帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习

如果你有需要的话,可以点击这里领取

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级安卓工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Android移动开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Android)
img

-1711031543382)]
[外链图片转存中…(img-s3EPKagp-1711031543383)]
[外链图片转存中…(img-179fPnKZ-1711031543383)]

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Android)
[外链图片转存中…(img-2VIx5gTv-1711031543383)]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值