Extract Class(提炼类)

某个class做了应该由两个class做的事.

 

建立一个新class,将相关的值域和函数从旧class搬移到新class。

 

class Person..

    private String _name;
    private String _officeAreaCode;
    private String _officeNumber;

    public String getTelephoneNumber()

 

==>

class Person..

    private String _name;

    private TelephoneNumber _officeTelephone;

    public String getTelephoneNumber()

 

class TelephoneNumber..

    private String _areaCode;
    private String _number;

    public String getTelephoneNumber()

    

动机

如果某些数据和某些函数总是一起出现,如果某些数据经常同时变化甚至彼此相依,这就表示你应该将它们分离出去。

另一个往往在开发后期出现的信号是class的[subtyped方式]。如果你发现subtyping只影响class的部分特性,或如果你发现某些特性[需要以此方式subtyped],某些特性[需要以彼此方式subtyped],这就是意味你需要分解原来的class。

作法

1. 决定如何分解class所负责任。

2. 建立一个新class,用以表现从旧class中分离出来的责任。 

如果旧class剩下的责任与旧class名称不符,为旧class易名。

3. 建立[从旧class访问新class]的连接关系(link)。

也许你有可能需要一个双向连接。但是在真正需要它之前,不要建立[从新class同往旧class]的连接。

4. 对于你想搬移的每一个值域,运用Move Field(146)搬移之。

5. 每次搬移后,编译、测试。

6. 使用Move Method(142)将必要函数搬移到新class。先搬移较低层函数(也就是[就其他函数调用]多于[调用其他函数]者),再搬移较高层函数。

7. 每次搬移之后,编译、测试。

8. 检查,精简每个class的接口。

如果你建立起双向连接,检查是否可以将它改为单向连接。

9. 决定是否让新class暴光。如果你的的确需要暴光它,决定让它成为reference object(引用型对象)或immutable value object(不可变之[实值型对象])。

 

让我们从一个简单的Person class开始:
class Person...
    public String getName() {
       return _name;
    }
    public String getTelephoneNumber() {
       return ("(" + _officeAreaCode + ")" + _officeNumber);
    }
    String getOfficeAreaCode() {
       return _officeAreaCode;
    }
    void setOfficeAreaCode(String arg) {
       _officeAreaCode = arg;
    }
    String getOfficeNumber() {
       return _officeNumber;
    }
    void setOfficeNumber(String arg) {
       _officeNumber = arg;
    }

    private String _name;
    private String _officeAreaCode;
    private String _officeNumber;

 

在这个例子,我可以将[与电话号码相关]的行为分离到一个独立class中。首先我要定义一个TelephoneNumber class来表示[电话号码]这个概念:


class TelephoneNumber {
}

 

易如反掌!然后,我要建立从Person到TelephoneNumber的连接:

 

class Person...
    private TelephoneNumber _officeTelephone = new TelephoneNumber();

 

现在,我运用Move Field(146)移动一个值域:

 

class TelephoneNumber {
    String getAreaCode() {
       return _areaCode;
    }
    void setAreaCode(String arg) {
       _areaCode = arg;
    }
    private String _areaCode;
}
class Person...
    public String getTelephoneNumber() {
       return ("(" + getOfficeAreaCode() + ")" + _officeNumber);
    }
    String getOfficeAreaCode() {
       return _officeTelephone.getAreaCode();
    }
    void setOfficeAreaCode(String arg) {
       _officeTelephone.setAreaCode(arg);
    }

 

然后我可以移动其他值域,并运用Move Method(142)将相关函数移动到TelephoneNumber class中:

 

class Person...
    public String getName() {
       return _name;
    }
    public String getTelephoneNumber() {
       return _officeTelephone.getTelephoneNumber();
    }
    TelephoneNumber getOfficeTelephone() {
       return _officeTelephone;
    }

    private String _name;
    private TelephoneNumber _officeTelephone = new TelephoneNumber();


class TelephoneNumber...
    public String getTelephoneNumber() {
       return ("(" + _areaCode + ")" + _number);
    String getAreaCode() {
       return _areaCode;
    }
    void setAreaCode(String arg) {
       _areaCode = arg;
    }
    String getNumber() {
       return _number;
    }
    void setNumber(String arg) {
       _number = arg;
    }
    private String _number;
    private String _areaCode;

下一步要做的决定是:要不要对客户揭示这个新class?我可以将Person中[与电话号码相关]的函数委托(delegating)至 TelephoneNumber,从而完全隐藏这个新class;也可以直接将对用户曝光。我还可以将它暴露给部分用户(位于同一个package中的用户),而不暴露给其他用户。

如果我选择暴露新class吗,我就需要考虑别名(aliasing)带来的危险。如果我暴露了TelephoneNumber,而有个用户修改了对象中的_areaCode值域值,我又怎么能知道呢?而且,做出修改的可能不是直接用户,而是用户的用户的用户。


面对这个问题,我有下列数种选择:
1.允许任何对象修改TelephoneNumber对象的任何部分。这就使得TelephoneNumber对象成为引用对象(reference object),对于我应该考虑使用Change Value to Reference(179)。这种情况下,Person应该是TelephoneNumber的访问点。
2.不许任何人[不通过Person对象就修改TelephoneNumber对象]。为了达到目的,我可以将TelephoneNumber设为不可修改的(immutable),或为它提供一个不可修改的接口(immutable interface)。
3.另一个办法是:先复制一个TelephoneNumber对象,然后将复制得到的新对象传递给用户。但这可能会造成一定程度的迷惑,因为人们会认为他们可以修改TelephoneNumber对象值。此外,如果同一个TelephoneNumber对象被传递给多个用户,也可能在用户之间造成别名(aliasing)问题。

Extract Class(149)是改善并发(concurrent)程序的一种常用技术,因为它使你可以提炼后的两个classes分别加锁(locks)。


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
可以通过 JDBC API 来实现。具体来说,需要进行以下步骤: 1. 导入 JDBC 驱动程序。 2. 载入驱动程序并建立数据库连接。 3. 创建用于执行 SQL 语句的 Statement 对象。 4. 执行 SQL 语句并处理结果。 5. 关闭数据库连接。 以下是一个示例代码: ``` import java.sql.*; public class MyClass { public static void main(String[] args) { // JDBC driver name and database URL String JDBC_DRIVER = "com.mysql.jdbc.Driver";//这里以使用 MySQL 数据库为例 String DB_URL = "jdbc:mysql://localhost/EMP"; // Database credentials String USER = "username"; String PASS = "password"; Connection conn = null; Statement stmt = null; try { // Register JDBC driver Class.forName("com.mysql.jdbc.Driver"); // Open a connection conn = DriverManager.getConnection(DB_URL, USER, PASS); // Execute a query stmt = conn.createStatement(); String sql = "SELECT id, name, age FROM Employees"; ResultSet rs = stmt.executeQuery(sql); // Extract data from result set while (rs.next()) { // Retrieve by column name int id = rs.getInt("id"); String name = rs.getString("name"); int age = rs.getInt("age"); // Display values System.out.print("ID: " + id); System.out.print(", Name: " + name); System.out.println(", Age: " + age); } // Clean-up environment rs.close(); stmt.close(); conn.close(); } catch (SQLException se) { // Handle errors for JDBC se.printStackTrace(); } catch (Exception e) { // Handle errors for Class.forName e.printStackTrace(); } finally { // 释放资源 try { if (stmt!=null) stmt.close(); } catch (SQLException se2) {} try { if (conn!=null) conn.close(); } catch (SQLException se) { se.printStackTrace(); } } } } ``` 在这个示例中,首先定义了一个字符串变量 JDBC_DRIVER,它包含了 JDBC 驱动程序的名称;另外一个字符串变量 DB_URL 描述了数据库的 URL 地址。然后定义了用户名和密码,这是为了访问数据库而设置的身份验证信息。 接下来,代码通过 Class.forName() 来加载 JDBC 驱动程序,这样就能够开始使用 JDBC API。然后,代码通过 DriverManager.getConnection() 来获得数据库连接。接着,代码创建了一个 Statement 对象,它用于执行 SQL 语句。使用 stmt.executeQuery() 方法来执行查询操作。在 while 循环中,使用 ResultSet 对象来处理查询结果,并显示到控制台。 最后,代码释放了资源,关闭了数据库连接和 Statement 对象。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值