接下去继续给大家介绍这个地址薄程序我们要做的如下:
1.创建一个用户界面
2.向文件中添加一条记录
3.从文件中读取一条记录
4.编程实现按钮的功能
代码如下:
package chapter18;
import java.io.*;
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.border.*;
public class AddressBook extends JFrame {
// Specify the size of five string fields in the record
final static int NAME_SIZE = 32;
final static int STREET_SIZE = 32;
final static int CITY_SIZE = 20;
final static int STATE_SIZE = 2;
final static int ZIP_SIZE = 5;
final static int RECORD_SIZE =
(NAME_SIZE + STREET_SIZE + CITY_SIZE + STATE_SIZE + ZIP_SIZE);
// Access address.dat using RandomAccessFile
private RandomAccessFile raf;
// Text fields
private JTextField jtfName = new JTextField(NAME_SIZE);
private JTextField jtfStreet = new JTextField(STREET_SIZE);
private JTextField jtfCity = new JTextField(CITY_SIZE);
private JTextField jtfState = new JTextField(ZIP_SIZE);
private JTextField jtfZip = new JTextField(ZIP_SIZE);
// Buttons
private JButton jbtAdd = new JButton("Add");
private JButton jbtFirst = new JButton("First");
private JButton jbtNext = new JButton("Next");
private JButton jbtPrevious = new JButton("Previous");
private JButton jbtLast = new JButton("Last");
public AddressBook() {
// Open or create a random access file
try {
raf = new RandomAccessFile("address.dat", "rw");
}
catch(IOException ex) {
System.out.print("Error: " + ex);
System.exit(0);
}
// Panel p1 for holding labels Name, Street, and City
JPanel p1 = new JPanel();
p1.setLayout(new GridLayout(3, 1));
p1.add(new JLabel("Name"));
p1.add(new JLabel("Street"));
p1.add(new JLabel("City"));
// Panel jpState for holding state
JPanel jpState = new JPanel();
jpState.setLayout(new BorderLayout());
jpState.add(new JLabel("State"), BorderLayout.WEST);
jpState.add(jtfState, BorderLayout.CENTER);
// Panel jpZip for holding zip
JPanel jpZip = new JPanel();
jpZip.setLayout(new BorderLayout());
jpZip.add(new JLabel("Zip"), BorderLayout.WEST);
jpZip.add(jtfZip, BorderLayout.CENTER);
// Panel p2 for holding jpState and jpZip
JPanel p2 = new JPanel();
p2.setLayout(new BorderLayout());
p2.add(jpState, BorderLayout.WEST);
p2.add(jpZip, BorderLayout.CENTER);
// Panel p3 for holding jtfCity and p2
JPanel p3 = new JPanel();
p3.setLayout(new BorderLayout());
p3.add(jtfCity, BorderLayout.CENTER);
p3.add(p2, BorderLayout.EAST);
// Panel p4 for holding jtfName, jtfStreet, and p3
JPanel p4 = new JPanel();
p4.setLayout(new GridLayout(3, 1));
p4.add(jtfName);
p4.add(jtfStreet);
p4.add(p3);
// Place p1 and p4 into jpAddress
JPanel jpAddress = new JPanel(new BorderLayout());
jpAddress.add(p1, BorderLayout.WEST);
jpAddress.add(p4, BorderLayout.CENTER);
// Set the panel with line border
jpAddress.setBorder(new BevelBorder(BevelBorder.RAISED));
// Add buttons to a panel
JPanel jpButton = new JPanel();
jpButton.add(jbtAdd);
jpButton.add(jbtFirst);
jpButton.add(jbtNext);
jpButton.add(jbtPrevious);
jpButton.add(jbtLast);
// Add jpAddress and jpButton to the frame
add(jpAddress, BorderLayout.CENTER);
add(jpButton, BorderLayout.SOUTH);
jbtAdd.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
writeAddress();
}
});
jbtFirst.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
try {
if (raf.length() > 0) readAddress(0);
}
catch (IOException ex) {
ex.printStackTrace();
}
}
});
jbtNext.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
try {
long currentPosition = raf.getFilePointer();
if (currentPosition < raf.length())
readAddress(currentPosition);
}
catch (IOException ex) {
ex.printStackTrace();
}
}
});
jbtPrevious.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
try {
long currentPosition = raf.getFilePointer();
if (currentPosition - 2 * RECORD_SIZE > 0)
// Why 2 * 2 * RECORD_SIZE? See the follow-up remarks
readAddress(currentPosition - 2 * 2 * RECORD_SIZE);
else
readAddress(0);
}
catch (IOException ex) {
ex.printStackTrace();
}
}
});
jbtLast.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
try {
long lastPosition = raf.length();
if (lastPosition > 0)
// Why 2 * RECORD_SIZE? See the follow-up remarks
readAddress(lastPosition - 2 * RECORD_SIZE);
}
catch (IOException ex) {
ex.printStackTrace();
}
}
});
// Display the first record if exists
try {
if (raf.length() > 0) readAddress(0);
}
catch (IOException ex) {
ex.printStackTrace();
}
}
/** Write a record at the end of the file */
public void writeAddress() {
try {
raf.seek(raf.length());
FixedLengthStringIO.writeFixedLengthString(
jtfName.getText(), NAME_SIZE, raf);
FixedLengthStringIO.writeFixedLengthString(
jtfStreet.getText(), STREET_SIZE, raf);
FixedLengthStringIO.writeFixedLengthString(
jtfCity.getText(), CITY_SIZE, raf);
FixedLengthStringIO.writeFixedLengthString(
jtfState.getText(), STATE_SIZE, raf);
FixedLengthStringIO.writeFixedLengthString(
jtfZip.getText(), ZIP_SIZE, raf);
}
catch (IOException ex) {
ex.printStackTrace();
}
}
/** Read a record at the specified position */
public void readAddress(long position) throws IOException {
raf.seek(position);
String name = FixedLengthStringIO.readFixedLengthString(
NAME_SIZE, raf);
String street = FixedLengthStringIO.readFixedLengthString(
STREET_SIZE, raf);
String city = FixedLengthStringIO.readFixedLengthString(
CITY_SIZE, raf);
String state = FixedLengthStringIO.readFixedLengthString(
STATE_SIZE, raf);
String zip = FixedLengthStringIO.readFixedLengthString(
ZIP_SIZE, raf);
jtfName.setText(name);
jtfStreet.setText(street);
jtfCity.setText(city);
jtfState.setText(state);
jtfZip.setText(zip);
}
public static void main(String[] args) {
AddressBook frame = new AddressBook();
frame.pack();
frame.setTitle("AddressBook");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setVisible(true);
}
}
如果这个文件address.dat不存在,则创建名为address.dat的随机读写文件来存储地址信息。如果它已经存在,则打开它。随机文件对象raf用于实现文件的读写操作。记录中每个域的长度都是固定的
44——100行创建用户界面,第102——156行注册监听器,在第159——164行程序开始运行后,如果存在记录,则显示第一条记录。
writeAddress()方法将文件指针设置到文件的尾(170行),并向文件中写入新纪录(第171——180行)
readAddress()方法将指针设置到指定位置(189行)并从文件中读取一条记录(190——199行)
要向文件中添加记录,需要从用户界面中收集地址信息,在写到文件之中去
处理按事件的代码第102——156行实现。Firs按钮读取文件中0的位置的记录(110行)Next按钮读取当前文件指针所指的记录(122行)读取一条记录后,文件指针从上一个位置向前移动2*RECORD_SIZE个字节。对previous按钮,需要显示当前正在显示的记录的前一条,所以,必须将当前文件指针向后移动两条记录(135行)Last按钮读取raf.length()-2*RECORD_SIZE位置的记录
再给大家截图看看效果



- 大小: 5.6 KB

- 大小: 6.6 KB
分享到:
相关推荐
在Java编程语言中,`transient`关键字是一个非常重要的概念,它与对象的序列化过程紧密相关。序列化是将一个对象的状态转换为字节流,以便存储或在网络中传输。当一个类实现了`Serializable`接口,该类的对象就可以...
Java 调用C或C++动态链接库(DLL)主要依赖于Java Native Interface (JNI) 技术。JNI 是Java平台的一部分,它允许Java代码和其他语言写的代码进行交互。以下是一个详细的步骤来说明如何使用JNI调用C或C++的动态链接...
《Xstream使用小记》 Xstream是一款Java库,它提供了简单且强大的方式来序列化和反序列化Java对象到XML,以及从XML回转换为Java对象。在本文中,我们将深入探讨Xstream的核心概念、使用场景以及如何通过实例进行...
EventBus.getDefault().post(new MessageEvent()); ``` 发布事件时,EventBus 会自动找到所有订阅了相应事件类型的订阅者,并调用它们的订阅方法。 除了基本用法,EventBus 还支持一些高级特性,比如: - **事件...
8. **IO和NIO**:JAVA的IO库提供了处理输入输出的方法,而NIO(New IO)引入了非阻塞I/O,提升了效率,适用于高并发场景。 9. **反射机制**:JAVA反射允许在运行时检查类的信息,创建和操作对象,对于理解和使用第...