# JAVA之旅(七)——final關鍵字 , 抽象類abstract,模板方法模式,接口interface,implements,特點,擴展
* * *
> OK,我們繼續學習JAVA,美滋滋的
## 一.final
> 我們來聊聊final這個關鍵字
* final可以修飾類,方法和變量
* final修飾的類不可以被繼承
* final修飾的方法不可以被覆蓋
* final修飾的變量是一個常量,只能被賦值一次
* 內部類只能訪問被final修飾的局部變量
> final,故名思意,就是最終的意思,由以上的五種特性,不過final的出現,也是有弊端的,他破壞了封裝性,對繼承有了挑戰,為了避免被繼承,被子類復寫功能,還有,當你描述事物時,一些數據的出現值是固定的,那么,這時為了增強閱讀行,都給這個值起個名字方便閱讀,而這值不需要改變,就會用到final去修飾,作為常量,常量的書寫規范是所有字母都大寫,如果由多個單詞組成,單詞間通過下劃線鏈接!而且內部類定義在類中的局部位置只能訪問該局部被final修飾的局部變量
>
> final大家只要記住他的特性就行
## 二.抽象類
> 這個抽象類應該是屬于繼承的下半部分的,我們看一個標注的代碼
~~~
//公共的 類 類名
public class HelloJJAVA {
// 公共的 靜態 無返回值 main方法 數組
public static void main(String[] str) {
}
}
/**
* 學生
*
* @author LGL
*
*/
class student extends Base {
void speak() {
System.out.println("學習");
}
}
/**
* 工人
*
* @author LGL
*
*/
class worker extends Base {
void speak() {
System.out.println("工作");
}
}
/**
* 基類
*
* @author LGL
*
*/
class Base {
void speak() {
System.out.println("Hello");
}
}
~~~
> 這里,學生和工人都是要說話,所以我們可以抽取,但是這里,他們說話的內容卻是不同的,當多個類出現相同功能,但是功能主體不同,這個時候就可以進行向上抽取,這時只抽取功能主體;這個時候就得用到我們的抽象類了abstract;所以我們的基類是這樣的
~~~
/**
* 基類
*
* @author LGL
*
*/
abstract class Base {
abstract void speak();
}
~~~
> 什么叫抽象?
* 看不懂
> 特點
* 1.抽象方法一定定義在抽象類中
* 2.抽象方法和抽象類都必須被abstract關鍵字修飾
* 3.抽象類不可以用new創建對象,因為調用抽象方法沒意義
* 4.抽象類中的方法要被使用必須由子類復寫其所有的抽象方法后建立子類對象調用?
* 如果子類只覆蓋了部分抽象方法,那么該子類還是一個抽象類
> 其實抽象類和一般的類沒有什么太大的不同,只是要注意該怎么描述事物就怎么描述事物,只不過該事物中出現了一些看不懂的東西,這些不確定的部分也是該事物的功能,需要明確出來,但是無法定義主體,通過抽象方法來表示!
* 抽象類比其他類多了抽象函數,就是在類中可以定義抽象方法
* 抽象類不可以實例化
> 特殊:抽象類中可以不頂用抽象方法,看上去很沒有意義,但是這樣做可以做到不讓該類建立對象,不是很多見
>
> 抽象方法文字部分說了這, 多,我們做一個小練習
* 題目:假如我們在開發一個系統時需要對員工進行建模,員工包含三個屬性 姓名,工號和工資,經理也是員工,除了含有員工的屬性外,另外還 有一個獎金屬性,請使用繼承的思路設計出員工類和經理類,要求類 種提供必要的方法進行屬性訪問!
> 我們實現之前可以簡單的分析一下,我們的員工類應該有三個屬性,name,id,pay,而經歷類,理論上是繼承了員工類并且有資金的獎金屬性,行,這樣的話我們可以用代碼去測試一下
~~~
//公共的 類 類名
public class HelloJJAVA {
// 公共的 靜態 無返回值 main方法 數組
public static void main(String[] str) {
/**
* 假如我們在開發一個系統時需要對員工進行建模,員工包含三個屬性 姓名,工號和工資,經理也是員工,除了含有員工的屬性外,另外還
* 有一個獎金屬性,請使用繼承的思路設計出員工類和經理類,要求類 種提供必要的方法進行屬性訪問!
*/
}
}
/**
* 員工類
*
* @author LGL
*
*/
abstract class Employee {
// 姓名
private String name;
// 工號
private String id;
// 工資
private double pay;
// 這個員工一生成,這三個屬性必須有
public Employee(String name, String id, double pay) {
this.name = name;
this.id = id;
this.pay = pay;
}
// 員工做什么是不確定的
public abstract void work();
}
/**
* 經理類
*
* @author LGL
*
*/
class Manager extends Employee {
// 獎金
private int bonus;
public Manager(String name, String id, double pay, int bonus) {
super(name, id, pay);
this.bonus = bonus;
}
// 復寫
@Override
public void work() {
System.out.println("管理");
}
}
~~~
> 這代碼很清晰的就表現了抽象的關系
## 三.模板方法模式
> 這是一個小案例,獲取一段程序的運行時間,這個需求應該很簡單吧,計時,
* 原理:獲取程序開始和結束的時間并相減
> 獲取時間的方法:System.currentTimeMillis()
>
> 代碼邏輯是這樣寫的
~~~
//公共的 類 類名
public class HelloJJAVA {
// 公共的 靜態 無返回值 main方法 數組
public static void main(String[] str) {
GetTime gt = new GetTime();
gt.getTime();
}
}
/**
* 時間類
*
* @author LGL
*
*/
class GetTime {
// 獲取時間
public void getTime() {
long start = System.currentTimeMillis();
// 耗時
for (int i = 0; i < 10000; i++) {
System.out.print("" + i);
}
long end = System.currentTimeMillis();
System.out.println("耗時:" + (end - start));
}
}
~~~
> 我們就可以得到你代碼運行的毫秒數了

> 但是我們發現,其實這個耗時的部分是不確定的,對吧,那既然這樣,我們復寫的話,就有點多余了,我們可以使用使用模板方法模式,也就是抽成一個方法公用
~~~
import org.ietf.jgss.Oid;
//公共的 類 類名
public class HelloJJAVA {
// 公共的 靜態 無返回值 main方法 數組
public static void main(String[] str) {
GetTime gt = new GetTime();
gt.getTime();
}
}
/**
* 時間類
*
* @author LGL
*
*/
class GetTime {
// 獲取時間
public void getTime() {
long start = System.currentTimeMillis();
runCode();
long end = System.currentTimeMillis();
System.out.println("耗時:" + (end - start));
}
/**
* 耗時方法
*/
private void runCode() {
// 耗時
for (int i = 0; i < 10000; i++) {
System.out.print("" + i);
}
}
}
~~~
> 這個時候,要是其他類想使用的話,就只要復寫一個方法就行了,Test想使用的話,就可以直接繼承復寫了
~~~
//公共的 類 類名
public class HelloJJAVA {
// 公共的 靜態 無返回值 main方法 數組
public static void main(String[] str) {
// GetTime gt = new GetTime();
Test t = new Test();
t.getTime();
}
}
/**
* 時間類
*
* @author LGL
*
*/
class GetTime {
// 獲取時間
public void getTime() {
long start = System.currentTimeMillis();
runCode();
long end = System.currentTimeMillis();
System.out.println("耗時:" + (end - start));
}
/**
* 耗時方法
*/
public void runCode() {
// 耗時
for (int i = 0; i < 10000; i++) {
System.out.print("" + i);
}
}
}
class Test extends GetTime {
@Override
public void runCode() {
// 耗時
for (int i = 0; i < 50000; i++) {
System.out.print("" + i);
}
}
}
~~~
> 這樣,輸出的內容就是我們隨便改的了

> 我們還可以用抽象去做,你就一定要去做這件事兒
~~~
//公共的 類 類名
public class HelloJJAVA {
// 公共的 靜態 無返回值 main方法 數組
public static void main(String[] str) {
// GetTime gt = new GetTime();
Test t = new Test();
t.getTime();
}
}
/**
* 時間類
*
* @author LGL
*
*/
abstract class GetTime {
// 獲取時間
public void getTime() {
long start = System.currentTimeMillis();
runCode();
long end = System.currentTimeMillis();
System.out.println("耗時:" + (end - start));
}
/**
* 耗時方法
*/
public abstract void runCode();
}
class Test extends GetTime {
@Override
public void runCode() {
// 耗時
for (int i = 0; i < 50000; i++) {
System.out.print("" + i);
}
}
}
~~~
> 我把這個延時的操作留給子類,你愛咋地就咋滴,這樣是不是更方便?當然,我們可以給getTime加一個final修飾,這樣就讓程序更加的健壯;
>
> 當代碼完成優化之后,就可以解決這類問題了,我們把這種方式叫做:模板方法設計模式
* 什么是模板方法??
> 在定義功能的時候,功能的一部分是不確定的,而確定的部分在使用不確定的部分的時候,那么這時就將不確定的 部分暴露出去讓子類去完成,這就是嗎,嗎,模板方法模式了
## 四.接口
> 接口的關鍵字是interface,接口中的成員修飾符是固定的
* 成員常量:public static final
* 成員函數:public abstract
> 接口的出現將“多繼承”通過另一種形勢體現,即“多實現”
>
> 上面的都是概念。我還是通俗易懂的來說吧,接口,初期理解,你可以認為是一個特殊的抽象類,當抽象類中的方法都是抽象的,那么該類可以通過接口的形式表示,interface,class用于定義類,定義接口
>
> 接口定義時,格式特點在于?
> - 1.接口中常見的定義一個是常量,一個是抽象方法?
> - 2.接口中的成員都有固定修飾符?
> - 常量:public static final?
> - 方法:public abstract
>
> 具體格式:
~~~
/**
* 人的接口
*
* @author LGL
*
*/
interface Person {
public static final int AGE = 20;
/**
* 說話
*/
public abstract void speak();
}
~~~
> 這里注意,接口中的成員都是public,我們要想使用這個接口,需要用到另一個關鍵字了**implements**
>
> 為什么我們類不能繼承類呢?因為接口,是不可以創建對象的,因為有抽象方法,需要被子類去實現,子類對接口中的抽象方法全都覆蓋過后子類才可以實例化,否則子類是一個抽象類
~~~
//公共的 類 類名
public class HelloJJAVA {
// 公共的 靜態 無返回值 main方法 數組
public static void main(String[] str) {
Student s = new Student();
System.out.println(s.AGE);
System.out.println(Student.AGE);
System.out.println(Person.AGE);
}
}
class Student implements Person {
@Override
public void speak() {
}
}
/**
* 人的接口
*
* @author LGL
*
*/
interface Person {
public static final int AGE = 20;
/**
* 說話
*/
public abstract void speak();
}
~~~
> 這樣執行的后,我們就等得到數據了

> 不過接口不僅僅是這么簡單,接口是可以被類多實現了,什么叫做多實現?就是一個類單繼承,但是可以實現多個接口,對繼承不支持的形式,java支持多實現
~~~
class Student implements Person, Person2 {
@Override
public void speak() {
}
@Override
public void work() {
}
}
/**
* 人的接口
*
* @author LGL
*
*/
interface Person {
public static final int AGE = 20;
/**
* 說話
*/
public abstract void speak();
}
interface Person2 {
public abstract void work();
}
~~~
> 可以看到實現了;兩個接口,但是他為什么沒有繼承的弊端呢?因為他沒有方法主體,子類愛怎么著就怎么著了
>
> 接口間的關系時繼承
### 接口的特點
> 這個接口的重點比較實在,所以單獨提取出來講一下,首先我們來連接一下接口的特點
* 接口誰對外暴露的規則
* 接口是程序的功能擴展
* 接口可以用力多實現
* 類和接口之間是實現關系,而且類可以繼承一個類的同時實現多個接口
* 接口與接口之間可以有繼承關系
> 就好比筆記本,我們擴展一樣,這就是接口的概念,說不如做,我們寫個例子來展現:
~~~
//公共的 類 類名
public class HelloJJAVA {
// 公共的 靜態 無返回值 main方法 數組
public static void main(String[] str) {
}
}
/**
* 學生類
*
* @author LGL
*
*/
abstract class Student {
// 學習不確定
abstract void study();
// 都要睡覺
void Sleep() {
System.out.println("sleep");
}
//都抽煙
abstract void smoke();
}
/**
* 我
* @author LGL
*
*/
class lgl extends Student{
@Override
void study() {
System.out.println("lgl學習");
}
@Override
void smoke() {
System.out.println("lgl 抽煙");
}
}
~~~
> 我這里定義了一個人,他有抽煙,睡覺,學習的方法,學習的方法不確定,所以要抽象,睡覺都要,抽煙,有的人抽,有的不抽,品牌也不一樣,所以也得弄成這樣,現在我去繼承這個人,機會有學習和睡覺的方法,但是強制性的抽煙了,這就是這個例子的問題,那我們要怎么改善?還得使用接口了,我們可以把抽煙的方法抽寫成接口,有需要接實現這個接口
~~~
//公共的 類 類名
public class HelloJJAVA {
// 公共的 靜態 無返回值 main方法 數組
public static void main(String[] str) {
lgl l = new lgl();
l.smoke();
l.study();
}
}
/**
* 學生類
*
* @author LGL
*
*/
abstract class Student {
// 學習不確定
abstract void study();
// 都要睡覺
void Sleep() {
System.out.println("sleep");
}
}
/**
* 我
*
* @author LGL
*
*/
class lgl extends Student implements smoke{
@Override
void study() {
System.out.println("lgl學習");
}
@Override
public void smoke() {
System.out.println("lgl 抽煙");
}
}
/**
* 抽煙的接口
* @author LGL
*
*/
interface smoke {
void smoke();
}
~~~
> 這樣就可以輸出

> 好的,我們本篇幅就先到這里,如果有不明白的地方,還是要多溫習幾遍哦,想學好android,java功底是必不可少的呢!
### 我創建一個很有意思的群:555974449,要是感興趣的可以進來一起交流一下哦!
版權聲明:本文為博主原創文章,博客地址:http://blog.csdn.net/qq_26787115,未經博主允許不得轉載。
- 0-發現
- AndroidInterview-Q-A
- Android能讓你少走彎路的干貨整理
- LearningNotes
- temp
- temp11
- 部分地址
- 0-待辦任務
- 待補充列表
- 0-未分類
- AndroidView事件分發與滑動沖突處理
- Spannable
- 事件分發機制詳解
- 1-Java
- 1-Java-01基礎
- 未歸檔
- 你應該知道的JDK知識
- 集合框架
- 1-Java-04合集
- Java之旅0
- Java之旅
- JAVA之旅01
- JAVA之旅02
- JAVA之旅03
- JAVA之旅04
- JAVA之旅05
- JAVA之旅06
- JAVA之旅07
- JAVA之旅08
- JAVA之旅09
- java之旅1
- JAVA之旅10
- JAVA之旅11
- JAVA之旅12
- JAVA之旅13
- JAVA之旅14
- JAVA之旅15
- JAVA之旅16
- JAVA之旅17
- JAVA之旅18
- JAVA之旅19
- java之旅2
- JAVA之旅20
- JAVA之旅21
- JAVA之旅22
- JAVA之旅23
- JAVA之旅24
- JAVA之旅25
- JAVA之旅26
- JAVA之旅27
- JAVA之旅28
- JAVA之旅29
- java之旅3
- JAVA之旅30
- JAVA之旅31
- JAVA之旅32
- JAVA之旅33
- JAVA之旅34
- JAVA之旅35
- 1-Java-05辨析
- HashMapArrayMap
- Java8新特性
- Java8接口默認方法
- 圖解HashMap(1)
- 圖解HashMap(2)
- 2-Android
- 2-Android-1-基礎
- View繪制流程
- 事件分發
- AndroidView的事件分發機制和滑動沖突解決
- 自定義View基礎
- 1-安卓自定義View基礎-坐標系
- 2-安卓自定義View基礎-角度弧度
- 3-安卓自定義View基礎-顏色
- 自定義View進階
- 1-安卓自定義View進階-分類和流程
- 10-安卓自定義View進階-Matrix詳解
- 11-安卓自定義View進階-MatrixCamera
- 12-安卓自定義View進階-事件分發機制原理
- 13-安卓自定義View進階-事件分發機制詳解
- 14-安卓自定義View進階-MotionEvent詳解
- 15-安卓自定義View進階-特殊形狀控件事件處理方案
- 16-安卓自定義View進階-多點觸控詳解
- 17-安卓自定義View進階-手勢檢測GestureDetector
- 2-安卓自定義View進階-繪制基本圖形
- 3-安卓自定義View進階-畫布操作
- 4-安卓自定義View進階-圖片文字
- 5-安卓自定義View進階-Path基本操作
- 6-安卓自定義View進階-貝塞爾曲線
- 7-安卓自定義View進階-Path完結篇偽
- 8-安卓自定義View進階-Path玩出花樣PathMeasure
- 9-安卓自定義View進階-Matrix原理
- 通用類介紹
- Application
- 2-Android-2-使用
- 2-Android-02控件
- ViewGroup
- ConstraintLayout
- CoordinatorLayout
- 2-Android-03三方使用
- Dagger2
- Dagger2圖文完全教程
- Dagger2最清晰的使用教程
- Dagger2讓你愛不釋手-終結篇
- Dagger2讓你愛不釋手-重點概念講解、融合篇
- dagger2讓你愛不釋手:基礎依賴注入框架篇
- 閱讀筆記
- Glide
- Google推薦的圖片加載庫Glide:最新版使用指南(含新特性)
- rxjava
- 這可能是最好的RxJava2.x入門教程完結版
- 這可能是最好的RxJava2.x入門教程(一)
- 這可能是最好的RxJava2.x入門教程(三)
- 這可能是最好的RxJava2.x入門教程(二)
- 這可能是最好的RxJava2.x入門教程(五)
- 這可能是最好的RxJava2.x入門教程(四)
- 2-Android-3-優化
- 優化概況
- 各種優化
- Android端秒開優化
- apk大小優化
- 內存分析
- 混淆
- 2-Android-4-工具
- adb命令
- 一鍵分析Android的BugReport
- 版本控制
- git
- git章節簡述
- 2-Android-5-源碼
- HandlerThread 源碼分析
- IntentService的使用和源碼分析
- 2-Android-9-辨析
- LRU算法
- 什么是Bitmap
- 常見圖片壓縮方式
- 3-Kotlin
- Kotlin使用筆記1-草稿
- Kotlin使用筆記2
- kotlin特性草稿
- Kotlin草稿-Delegation
- Kotlin草稿-Field
- Kotlin草稿-object
- 4-JavaScript
- 5-Python
- 6-Other
- Git
- Gradle
- Android中ProGuard配置和總結
- gradle使用筆記
- Nexus私服搭建
- 編譯提速最佳實踐
- 7-設計模式與架構
- 組件化
- 組件化探索(OKR)
- 1-參考列表
- 2-1-組件化概述
- 2-2-gradle配置
- 2-3-代碼編寫
- 2-4-常見問題
- 2-9-值得一讀
- 8-數據結構與算法
- 0臨時文件
- 漢諾塔
- 8-數據-1數據結構
- HashMap
- HashMap、Hashtable、HashSet 和 ConcurrentHashMap 的比較
- 遲到一年HashMap解讀
- 8-數據-2算法
- 1個就夠了
- Java常用排序算法(必須掌握的8大排序算法)
- 常用排序算法總結(性能+代碼)
- 必須知道的八大種排序算法(java實現)
- 9-職業
- 閱讀
- 書單
- 面試
- 面試-01-java
- Java面試題全集駱昊(上)
- Java面試題全集駱昊(下)
- Java面試題全集駱昊(中)
- 面試-02-android
- 40道Android面試題
- 面試-03-開源源碼
- Android圖片加載框架最全解析(二),從源碼的角度理解Glide的執行流程
- 面試-07-設計模式
- 面試-08-算法
- 面試-09-其他
- SUMMARY
- 版權說明
- temp111