怎樣在java中定義一個(gè)抽象屬性
Abstract關(guān)鍵字通常被用于類(lèi)和方法,用來(lái)把某些行為的實(shí)現(xiàn)委托給子類(lèi)。由于Java不支持抽象屬性,如果你試圖將類(lèi)屬性標(biāo)記為抽象,將會(huì)得到一個(gè)編譯時(shí)錯(cuò)誤。
在本教程中,我們將介紹兩種定義抽象屬性的方法,這些抽象屬性可以由子類(lèi)進(jìn)行設(shè)置,而且不使用Abstract 關(guān)鍵字。
實(shí)用案例
假設(shè)我們想要實(shí)現(xiàn)一個(gè)記錄事務(wù)的日志模塊,用來(lái)記錄特定事務(wù)的信息。我們希望這個(gè)模塊是抽象的,這樣我們可以實(shí)現(xiàn)不同的日志記錄方式,例如:記錄到文件或數(shù)據(jù)庫(kù)中。
我們的引擎使用預(yù)定義的分隔符來(lái)連接日志中的信息,并存儲(chǔ)在一個(gè)String中。具體應(yīng)該使用哪個(gè)分隔符,這將取決于日志記錄的規(guī)則,例如可以用字符“,”對(duì)日志記錄中不同部分的信息進(jìn)行分割。
因此,分隔符看起來(lái)對(duì)我們的引擎是抽象的,需要由每個(gè)日志記錄規(guī)則明確定義。
下面我提供兩種方式,來(lái)實(shí)現(xiàn)把分隔符的定義委托給子類(lèi)。
在抽象類(lèi)中定義帶參數(shù)的構(gòu)造函數(shù)
在抽象類(lèi)中定義動(dòng)態(tài)屬性的***種方法是:定義一個(gè)參數(shù)的構(gòu)造函數(shù)。
所以我們可以這樣實(shí)現(xiàn)這個(gè)引擎:
- // TransactionManager.java
- public abstract class TransactionManager {
- private String separator;
- public TransactionManager(String separator) {
- this.separator = separator;
- }
- public abstract void writeTransaction(String result);
- public Transaction startTransaction()
- {
- Transaction transaction = new Transaction(System.currentTimeMillis());
- return transaction;
- }
- public void endTransaction(Transaction t) {
- long processingTime = System.currentTimeMillis() - t.getStartTime();
- StringBuilder logBuilder = new StringBuilder();
- logBuilder.append(t.getStartTime());
- // Notice the use of this.separator
- logBuilder.append(this.separator);
- logBuilder.append(processingTime);
- logBuilder.append(this.separator);
- logBuilder.append(t.getData());
- String result = logBuilder.toString();
- writeTransaction(result);
- }
- }
在抽象類(lèi)中定義帶參數(shù)的構(gòu)造函數(shù)時(shí),子類(lèi)將會(huì)被強(qiáng)制定義自己的構(gòu)造函數(shù)并調(diào)用super()。 這樣我們就能強(qiáng)制separator屬性依賴(lài)于已使用的日志記錄機(jī)制。
注意,我們的引擎實(shí)現(xiàn)了所有日志機(jī)制共有的靜態(tài)行為:startTransaction(), endTransaction(),同時(shí)將動(dòng)態(tài)行為writeTransaction()交給子類(lèi)去實(shí)現(xiàn)。
現(xiàn)在,如果我們想要?jiǎng)?chuàng)建一個(gè)事務(wù)管理器,用它將日志內(nèi)容記錄到一個(gè)文件中,那么可以這樣去定義:
- public class TransactionManagerFS extends TransactionManager{
- // The IDE forces you to implement constructor.
- public TransactionManagerFS(String separator) {
- super(separator);
- }
- @Override
- public void writeTransaction(String result) {
- System.out.println("The following transaction has just finished: " );
- System.out.println(result);
- }
- }
接下來(lái)做一個(gè)測(cè)試,看看代碼是怎樣工作的
- public static void main(String[] args) throws InterruptedException {
- // we pass the separator explicitly in the constructor
- TransactionManager transactionManager = new TransactionManagerFS(",");
- Transaction transaction = transactionManager.startTransaction();
- transaction.setData("This is a test transaction !!");
- Thread.sleep(1500);
- transactionManager.endTransaction(transaction);
- }
輸出:
- The following transaction has just finished:
- 1502179140689,1501,This is a test transaction !!
通過(guò)getter方法傳遞分隔符
另外一種實(shí)現(xiàn)動(dòng)態(tài)屬性的方法是:通過(guò)定義一個(gè)抽象的getter方法,該方法根據(jù)當(dāng)前的日志記錄機(jī)制來(lái)檢索所需的分隔符。在我們的引擎中,當(dāng)需要要使用分隔符時(shí),可以通過(guò)調(diào)用這個(gè)getter方法得到。
接下來(lái)我們將引擎修改成這樣:
- public abstract class TransactionManager {
- public abstract String getSeperator();
- public abstract void writeTransaction(String result);
- public Transaction startTransaction()
- {
- Transaction transaction = new Transaction(System.currentTimeMillis());
- return transaction;
- }
- public void endTransaction(Transaction t) {
- long processingTime = System.currentTimeMillis() - t.getStartTime();
- StringBuilder logBuilder = new StringBuilder();
- logBuilder.append(t.getStartTime());
- // Notice the use of getSeparator()
- logBuilder.append(getSeperator());
- logBuilder.append(processingTime);
- logBuilder.append(getSeperator());
- logBuilder.append(t.getData());
- String result = logBuilder.toString();
- writeTransaction(result);
- }
- }
另外修改TransactionManagerFS如下:
- public class TransactionManagerFS extends TransactionManager{
- @Override
- public String getSeperator() {
- return ",";
- }
- @Override
- public void writeTransaction(String result) {
- System.out.println("The following transaction has just finished: " );
- System.out.println(result);
- }
- }
然后,修改main以使用新的實(shí)現(xiàn),并確保得到正確的結(jié)果。
- public static void main(String[] args) throws InterruptedException {
- // The separator is defined implicitly using getSeparator() method of the manager
- TransactionManager transactionManager = new TransactionManagerFS();
- Transaction transaction = transactionManager.startTransaction();
- transaction.setData("This is a test transaction !!");
- Thread.sleep(1500);
- transactionManager.endTransaction(transaction);
- }
輸出:
- The following transaction has just finished:
- 1502179140689,1501,This is a test transaction !!