《代码大全》表驱动法-Table Driven Approach-2

package com.ljn.base;

import java.io.BufferedReader;
import java.io.FileInputStream;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.apache.commons.io.IOUtils;
import org.joda.time.DateTime;
import org.joda.time.format.DateTimeFormat;
import org.joda.time.format.DateTimeFormatter;

/**
《代码大全》-表驱动法-消息打印
问题描述:
Suppose you’re writing a routine to print messages that are stored in a file. The file
usually has about 500 messages, and each file has about 20 kinds of messages. The
messages originally come from a buoy and give water temperature, the buoy’s location, and so on.
Each of the messages has several fields, and each message starts with a header that has
an ID to let you know which of the 20 or sokinds of messages you’re dealing with.

书上对于这个问题的解法,我看得不是很明白,动手写了一下,就有了以下代码,基本达到目的:
*/
public class TableDriven2 {

private static final Map<String, Message> messageMap;

/**
* 正如书上所说,“消息表”可以硬编码到程序中;也可以定义在配置文件里,程序初始化时读取
* 定义在配置文件的优点是,消息格式变动时,不需要改动java代码;缺点是要解析配置文件
* 这里简单起见,只定义三种消息,且硬编码到程序中
*/
static {
messageMap = new HashMap<String, Message>();

Message temperature = new Message("001", "Temperature Message");
temperature.addField(new FloatField("Average Temperature"));
temperature.addField(new IntegerField("Number of Samples"));
temperature.addField(new StringField("Location"));
temperature.addField(new DayField("Time of Measurement"));
messageMap.put(temperature.getId(), temperature);

Message drift = new Message("002", "Drift Message");
drift.addField(new FloatField("Change in Latitude"));
drift.addField(new FloatField("Change in Longtitude"));
drift.addField(new DayField("Time of Measurement"));
messageMap.put("002", drift);

Message location = new Message("003", "Location Message");
location.addField(new FloatField("Latitude"));
location.addField(new FloatField("Longtitude"));
location.addField(new IntegerField("Depth"));
location.addField(new DayField("Time of Measurement"));
messageMap.put(location.getId(), location);
}

//示例代码。实际开发中,还应该考虑很多细节,例如每行前后的空格要去掉,要检查每行的数据是否合法,等等
public static void main(String[] args){
FileInputStream fs = null;
try {
fs = new FileInputStream("C:/Users/lijinnan/Desktop/message.txt");
BufferedReader br = new BufferedReader(new InputStreamReader(fs));
String line = null;
boolean messageBegin = true;
Message message = null;
int fieldIndex = 0;
while ((line = br.readLine()) != null) {
if (messageBegin) {
String messageId = line;
message = messageMap.get(messageId);
System.out.println("#" + message.getName() + "#");
messageBegin = false;
continue;
}
if (message != null && message.getFields() != null) {
if (fieldIndex < message.getFields().size()) {
Field curField = message.getFields().get(fieldIndex++);
curField.readAndPrint(line);
}
if (fieldIndex == message.getFields().size()) {
messageBegin = true;
fieldIndex = 0;
System.out.println();
}
}
}
} catch (Exception e) {
//ignore for test
} finally {
IOUtils.closeQuietly(fs);
}
}

}


/*
定义消息字段的类型
只定义四种:FloatField IntegerField StringField DayField
当然这四种类型也可以定义为enum
*/

abstract class Field {
private String label;
public abstract void readAndPrint(String value);
public Field(String label) {
this.label = label;
}
public String getLabel() {
return label;
}
public void setLabel(String label) {
this.label = label;
}
}

class FloatField extends Field {

public FloatField(String label) {
super(label);
}

/**
* 假设浮点数输出的格式是保留两位小数点
*/
@Override
public void readAndPrint(String value) {
float val = Float.parseFloat(value);
String formattedValue = String.format("%.2f", val);
System.out.println(getLabel() + ":" + formattedValue);
}

}

class IntegerField extends Field {

public IntegerField(String label) {
super(label);
}

/**
* 假设整数的输出格式为原样输出
*/
@Override
public void readAndPrint(String value) {
System.out.println(getLabel() + ":" + value);
}

}

class StringField extends Field {

public StringField(String label) {
super(label);
}

/**
* 假设字符串的输出格式为原样输出
*/
@Override
public void readAndPrint(String value) {
System.out.println(getLabel() + ":" + value);
}

}

class DayField extends Field {

public DayField(String label) {
super(label);
}

/**
* 假设时间的输入格式是dd/MM/yyyy HH:mm:ss
* 输出格式是yyyy-MM-dd
* 时间的处理用到了joda-time
*/
@Override
public void readAndPrint(String value) {
DateTimeFormatter formatter = DateTimeFormat.forPattern("dd/MM/yyyy HH:mm:ss");
DateTime dt = formatter.parseDateTime(value);
System.out.println(getLabel() + ":" + dt.toString("yyyy-MM-dd"));
}

}


class Message {
private String id; //消息ID
private String name; //消息名
private List<Field> fields; //消息字段

public Message(String id, String name) {
this.id = id;
this.name = name;
}

public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public List<Field> getFields() {
return Collections.unmodifiableList(fields);
}
public void addField(Field field) {
if (fields == null) {
fields = new ArrayList<Field>();
}
fields.add(field);
}
}

/*
测试数据,message.txt的内容:
001
1.111
11
East 111
01/01/2011 11:11:11
001
1.010
10
West 001
11/11/2011 11:11:11
002
2.0
2.1
02/02/2012 22:22:22
003
3.0
3.123
3000
03/03/2013 23:33:33
002
22.0
22.1
22/02/2012 22:22:22
003
30.0
30.123
3000
13/03/2013 23:33:33

程序输出:

#Temperature Message#
Average Temperature:1.11
Number of Samples:11
Location:East 111
Time of Measurement:2011-01-01

#Temperature Message#
Average Temperature:1.01
Number of Samples:10
Location:West 001
Time of Measurement:2011-11-11

#Drift Message#
Change in Latitude:2.00
Change in Longtitude:2.10
Time of Measurement:2012-02-02

#Location Message#
Latitude:3.00
Longtitude:3.12
Depth:3000
Time of Measurement:2013-03-03

#Drift Message#
Change in Latitude:22.00
Change in Longtitude:22.10
Time of Measurement:2012-02-22

#Location Message#
Latitude:30.00
Longtitude:30.12
Depth:3000
Time of Measurement:2013-03-13


*/
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值