怎样在java中定义一个抽象属性

334 阅读4分钟
原文链接: mp.weixin.qq.com

点击"充实的脑洞"关注我! 

Abstract关键字通常被用于类和方法,用来把某些行为的实现委托给子类。由于Java不支持抽象属性,如果你试图将类属性标记为抽象,将会得到一个编译时错误。

在本教程中,我们将介绍两种定义抽象属性的方法,这些抽象属性可以由子类进行设置,而且不使用 Abstract关键字。

实用案例

假设我们想要实现一个记录事务的日志模块,用来记录特定事务的信息。我们希望这个模块是抽象的,这样我们可以实现不同的日志记录方式,例如:记录到文件或数据库中。

我们的引擎使用预定义的分隔符来连接日志中的信息,并存储在一个 String中。具体应该使用哪个分隔符,这将取决于日志记录的规则,例如可以用字符“,”对日志记录中不同部分的信息进行分割。

因此,分隔符看起来对我们的引擎是抽象的,需要由每个日志记录规则明确定义。

下面我提供两种方式,来实现把分隔符的定义委托给子类。

在抽象类中定义带参数的构造函数

在抽象类中定义动态属性的第一种方法是:定义一个参数的构造函数。

所以我们可以这样实现这个引擎:

  1. // TransactionManager.java

  2. public abstract class TransactionManager {

  3.    private String separator;

  4.    public TransactionManager(String separator) {

  5.        this.separator = separator;

  6.    }

  7.    public abstract void writeTransaction(String result);

  8.    public Transaction startTransaction()

  9.    {

  10.        Transaction transaction = new Transaction(System.currentTimeMillis());

  11.        return transaction;

  12.    }

  13.    public void endTransaction(Transaction t) {

  14.        long processingTime = System.currentTimeMillis() - t.getStartTime();

  15.        StringBuilder logBuilder = new StringBuilder();

  16.        logBuilder.append(t.getStartTime());

  17.        // Notice the use of this.separator

  18.        logBuilder.append(this.separator);

  19.        logBuilder.append(processingTime);

  20.        logBuilder.append(this.separator);

  21.        logBuilder.append(t.getData());

  22.        String result = logBuilder.toString();

  23.        writeTransaction(result);

  24.    }

  25. }

在抽象类中定义带参数的构造函数时,子类将会被强制定义自己的构造函数并调用 super()。 这样我们就能强制 separator属性依赖于已使用的日志记录机制。

注意,我们的引擎实现了所有日志机制共有的静态行为: startTransaction(), endTransaction(),同时将动态行为 writeTransaction()交给子类去实现。

现在,如果我们想要创建一个事务管理器,用它将日志内容记录到一个文件中,那么可以这样去定义:

  1. public class TransactionManagerFS extends TransactionManager{

  2.    // The IDE forces you to implement constructor.

  3.    public TransactionManagerFS(String separator) {

  4.        super(separator);

  5.    }

  6.    @Override

  7.    public void writeTransaction(String result) {

  8.        System.out.println("The following transaction has just finished: " );

  9.        System.out.println(result);

  10.    }

  11. }

接下来做一个测试,看看代码是怎样工作的

  1. public static void main(String[] args) throws InterruptedException {

  2.        // we pass the separator explicitly in the constructor

  3.        TransactionManager transactionManager = new TransactionManagerFS(",");

  4.        Transaction transaction = transactionManager.startTransaction();

  5.        transaction.setData("This is a test transaction !!");

  6.        Thread.sleep(1500);

  7.        transactionManager.endTransaction(transaction);

  8.    }

输出:

  1. The following transaction has just finished:

  2. 1502179140689,1501,This is a test transaction !!

通过getter方法传递分隔符

另外一种实现动态属性的方法是:通过定义一个抽象的 getter方法,该方法根据当前的日志记录机制来检索所需的分隔符。在我们的引擎中,当需要要使用分隔符时,可以通过调用这个 getter方法得到。

接下来我们将引擎修改成这样:

  1. public abstract class TransactionManager {

  2.    public abstract String getSeperator();

  3.    public abstract void writeTransaction(String result);

  4.    public Transaction startTransaction()

  5.    {

  6.        Transaction transaction = new Transaction(System.currentTimeMillis());

  7.        return transaction;

  8.    }

  9.    public void endTransaction(Transaction t) {

  10.        long processingTime = System.currentTimeMillis() - t.getStartTime();

  11.        StringBuilder logBuilder = new StringBuilder();

  12.        logBuilder.append(t.getStartTime());

  13.        // Notice the use of getSeparator()

  14.        logBuilder.append(getSeperator());

  15.        logBuilder.append(processingTime);

  16.        logBuilder.append(getSeperator());

  17.        logBuilder.append(t.getData());

  18.        String result = logBuilder.toString();

  19.        writeTransaction(result);

  20.    }

  21. }

另外修改 TransactionManagerFS如下:

                                                
  1. public class TransactionManagerFS extends TransactionManager{

  2.     @Override

  3.     public String getSeperator() {

  4.         return "," ;

  5.     }

  6.     @Override

  7.     public void writeTransaction( String result ) {

  8.         System.out .println("The following transaction has just finished: " );

  9.         System.out .println(result );

  10.     }

  11. }

然后,修改 main以使用新的实现,并确保得到正确的结果。

                                                    
  1. public static void main(String[] args) throws InterruptedException {

  2.         // The separator is defined implicitly using getSeparator() method of the manager

  3.         TransactionManager transactionManager = new TransactionManagerFS ();

  4.         Transaction transaction = transactionManager .startTransaction ();

  5.        transaction .setData("This is a test transaction !!" );

  6.         Thread.sleep( 1500);

  7.        transactionManager .endTransaction(transaction );

  8.     }

输出:cnc

  1. The following transaction has just finished:

  2. 1502179140689,1501,This is a test transaction !!

----------充实的脑洞----------

长按二维码关注充实的脑洞,一个技术宅的自留地。