怎樣在java中定義一個抽象屬性
Abstract關鍵字通常被用于類和方法,用來把某些行為的實現委托給子類。由于Java不支持抽象屬性,如果你試圖將類屬性標記為抽象,將會得到一個編譯時錯誤。
在本教程中,我們將介紹兩種定義抽象屬性的方法,這些抽象屬性可以由子類進行設置,而且不使用Abstract 關鍵字。
實用案例
假設我們想要實現一個記錄事務的日志模塊,用來記錄特定事務的信息。我們希望這個模塊是抽象的,這樣我們可以實現不同的日志記錄方式,例如:記錄到文件或數據庫中。
我們的引擎使用預定義的分隔符來連接日志中的信息,并存儲在一個String中。具體應該使用哪個分隔符,這將取決于日志記錄的規則,例如可以用字符“,”對日志記錄中不同部分的信息進行分割。
因此,分隔符看起來對我們的引擎是抽象的,需要由每個日志記錄規則明確定義。
下面我提供兩種方式,來實現把分隔符的定義委托給子類。
在抽象類中定義帶參數的構造函數
在抽象類中定義動態屬性的***種方法是:定義一個參數的構造函數。
所以我們可以這樣實現這個引擎:
- // 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);
- }
- }
在抽象類中定義帶參數的構造函數時,子類將會被強制定義自己的構造函數并調用super()。 這樣我們就能強制separator屬性依賴于已使用的日志記錄機制。
注意,我們的引擎實現了所有日志機制共有的靜態行為:startTransaction(), endTransaction(),同時將動態行為writeTransaction()交給子類去實現。
現在,如果我們想要創建一個事務管理器,用它將日志內容記錄到一個文件中,那么可以這樣去定義:
- 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);
- }
- }
接下來做一個測試,看看代碼是怎樣工作的
- 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 !!
通過getter方法傳遞分隔符
另外一種實現動態屬性的方法是:通過定義一個抽象的getter方法,該方法根據當前的日志記錄機制來檢索所需的分隔符。在我們的引擎中,當需要要使用分隔符時,可以通過調用這個getter方法得到。
接下來我們將引擎修改成這樣:
- 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以使用新的實現,并確保得到正確的結果。
- 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 !!