Java基础:一些常用代码片段

获取环境变量

1
2
System.getenv("PATH");
System.getenv("JAVA_HOME");

获取系统属性

1
2
3
4
5
System.getProperty("pencil color");  // 得到属性值
java -Dpencil color=green
System.getProperty("java.specification.version"); // 得到 Java 版本号
Properties p = System.getProperties(); // 得到所有属性值
p.list(System.out);

String Tokenizer

1
2
3
4
5
6
7
8
// 能够同时识别, 和 |
StringTokenizer st = new StringTokenizer("Hello, World|of|Java", ", |");
while (st.hasMoreElements()) {
st.nextToken();
}

// 把分隔符视为 token
StringTokenizer st = new StringTokenizer("Hello, World|of|Java", ", |", true);

StringBuffer(同步) 和 StringBuilder(非同步)

1
2
3
4
5
StringBuilder sb = new StringBuilder();
sb.append("Hello");
sb.append("World");
sb.toString();
new StringBuffer(a).reverse(); // 反转字符串

数字

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
// 数字与对象之间互相转换 - Integer 转 int
Integer.intValue();

// 浮点数的舍入
Math.round()

// 数字格式化
NumberFormat

// 整数 -> 二进制字符串
toBinaryString()或 valueOf()

// 整数 -> 八进制字符串
toOctalString()

// 整数 -> 十六进制字符串
toHexString()

// 数字格式化为罗马数字
RomanNumberFormat()

// 随机数
Random r = new Random();
r.nextDouble();
r.nextInt();

日期和时间

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
// 查看当前日期
Date today = new Date();
Calendar.getInstance().getTime();

// 格式化默认区域日期输出
DateFormat df = DateFormat.getInstance();
df.format(today);

// 格式化制定区域日期输出
DateFormat df_cn = DateFormat.getDateInstance(DateFormat.FULL, Locale.CHINA);
String now = df_cn.format(today);

// 按要求格式打印日期
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
sdf.format(today);

// 设置具体日期
GregorianCalendar d1 = new GregorianCalendar(2009, 05, 06); // 6 月 6 日
GregorianCalendar d2 = new GregorianCalendar(); // 今天
Calendar d3 = Calendar.getInstance(); // 今天
d1.getTime(); // Calendar 或 GregorianCalendar 转成 Date 格式
d3.set(Calendar.YEAR, 1999);
d3.set(Calendar.MONTH, Calendar.APRIL);
d3.set(Calendar.DAY_OF_MONTH, 12);

// 字符串转日期
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
Date now = sdf.parse(String);

// 日期加减
Date now = new Date();
long t = now.getTime();
t += 700*24*60*60*1000;
Date then = new Date(t);

Calendar now = Calendar.getInstance();
now.add(Calendar.YEAR, -2);

// 计算日期间隔 (转换成 long 来计算)
today.getTime()- old.getTime();

// 比较日期
Date 类型, 就使用 equals(), before(), after() 来计算
long 类型, 就使用 ==, <, > 来计算

// 第几日
使用 Calendar 的 get() 方法
Calendar c = Calendar.getInstance();
c.get(Calendar.YEAR);

// 记录耗时
long start = System.currentTimeMillis();
long end = System.currentTimeMillis();
long elapsed = end - start;
System.nanoTime(); // 毫秒

// 长整形转换成秒
Double.toString(t/1000D);

结构化数据

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
// 数组拷贝
System.arrayCopy(oldArray, 0, newArray, 0, oldArray.length);

// ArrayList
add(Object o) // 在末尾添加给定元素
add(int i, Object o) // 在指定位置插入给定元素
clear() // 从集合中删除全部元素
Contains(Object o) // 如果 Vector 包含给定元素, 返回真值
get(int i) // 返回指定位置的对象句柄
indexOf(Object o) // 如果找到给定对象, 则返回其索引值;否则, 返回 -1
remove(Object o) // 根据引用删除对象
remove(int i) // 根据位置删除对象
toArray() // 返回包含集合对象的数组

// Iterator
List list = new ArrayList();
Iterator it = list.iterator();
while (it.hasNext())
Object o = it.next();

// 链表
LinkedList list = new LinkedList();
ListIterator it = list.listIterator();
while (it.hasNext())
Object o = it.next();

// HashMap
HashMap<String, String> hm = new HashMap<String, String>();
hm.get(key); // 通过 key 得到 value
hm.put("No1", "Hexinyu");
hm.put("No2", "Sean");
// 方法 1: 获取全部键值
Iterator<String> it = hm.values().iterator();
while (it.hasNext()) {
String myKey = it.next();
String myValue = hm.get(myKey);
}
// 方法 2: 获取全部键值
for (String key : hm.keySet()) {
String myKey = key;
String myValue = hm.get(myKey);
}

// Preferences - 与系统相关的用户设置, 类似名 - 值对
Preferences prefs = Preferences.userNodeForPackage(ArrayDemo.class);
String text = prefs.get("textFontName", "lucida-bright");
String display = prefs.get("displayFontName", "lucida-balckletter");
System.out.println(text);
System.out.println(display);
// 用户设置了新值, 存储回去
prefs.put("textFontName", "new-bright");
prefs.put("displayFontName", "new-balckletter");

// Properties - 类似名 - 值对, key 和 value 之间, 可以用"=", ":"或空格分隔, 用"#"和"!"注释
InputStream in = MediationServer.class.getClassLoader().getResourceAsStream("msconfig.properties");
Properties prop = new Properties();
prop.load(in);
in.close();
prop.setProperty(key, value);
prop.getProperty(key);

// 排序
1. 数组: Arrays.sort(strings);
2. List: Collections.sort(list);
3. 自定义类: class SubComp implements Comparator
然后使用 Arrays.sort(strings, new SubComp())

// 两个接口
1. java.lang.Comparable: 提供对象的自然排序, 内置于类中
int compareTo(Object o);
boolean equals(Object o2);
2. java.util.Comparator: 提供特定的比较方法
int compare(Object o1, Object o2)

// 避免重复排序, 可以使用 TreeMap
TreeMap sorted = new TreeMap(unsortedHashMap);

// 排除重复元素
Hashset hs - new HashSet();

// 搜索对象
binarySearch(): 快速查询 - Arrays, Collections
contains(): 线型搜索 - ArrayList, HashSet, Hashtable, linkedList, Properties, Vector
containsKey(): 检查集合对象是否包含给定 - HashMap, Hashtable, Properties, TreeMap
containsValue(): 主键 ( 或给定值) - HashMap, Hashtable, Properties, TreeMap
indexOf(): 若找到给定对象, 返回其位置 - ArrayList, linkedList, List, Stack, Vector
search(): 线型搜素 - Stack

// 集合转数组
toArray();

// 集合总结
Collection: Set - HashSet, TreeSet
Collection: List - ArrayList, Vector, LinkedList
Map: HashMap, HashTable, TreeMap

泛型与 foreach

1
2
3
4
5
6
7
// 泛型
List<String> myList = new ArrayList<String>();

// foreach
for (String s : myList) {
System.out.println(s);
}

面向对象

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
// toString() 格式化
public class ToStringWith {
int x, y;
public ToStringWith(int anX, int aY) {
x = anX;
y = aY;
}
public String toString() {
return "ToStringWith[" + x + "," + y + "]";
}
public static void main(String[] args) {
System.out.println(new ToStringWith(43, 78));
}
}

// 覆盖 equals 方法
public boolean equals(Object o) {
if (o == this) // 优化
return true;
if (!(o instanceof EqualsDemo)) // 可投射到这个类
return false;
EqualsDemo other = (EqualsDemo)o; // 类型转换
if (int1 != other.int1) // 按字段比较
return false;
if (!obj1.equals(other.obj1))
return false;
return true;
}

// 覆盖 hashcode 方法
private volatile int hashCode = 0; // 延迟初始化
public int hashCode() {
if (hashCode == 0) {
int result = 17;
result = 37 * result + areaCode;
}
return hashCode;
}

// Clone 方法
要克隆对象, 必须先做两步: 1. 覆盖对象的 clone() 方法; 2. 实现空的 Cloneable 接口
public class Clone1 implements Cloneable {
public Object clone() {
return super.clone();
}
}

// Finalize 方法
Object f = new Object() {
public void finalize() {
System.out.println("Running finalize()");
}
};
Runtime.getRuntime().addShutdownHook(new Thread() {
public void run() {
System.out.println("Running Shutdown Hook");
}
});
在调用 System.exit(0); 的时候, 这两个方法将被执行

// Singleton 模式
// 实现 1
public class MySingleton() {
public static final MySingleton INSTANCE = new MySingleton();
private MySingleton(){}
}
// 实现 2
public class MySingleton() {
public static MySingleton instance = new MySingleton();
private MySingleton(){}
public static MySingleton getInstance() {
return instance;
}
}

// 自定义异常
Exception: 编译时检查
RuntimeException: 运行时检查
public class MyException extends RuntimeException {
public MyException() {
super();
}
public MyException(String msg) {
super(msg);
}
}

输入和输出

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
// Stream, Reader, Writer
Stream: 处理字节流
Reader/Writer: 处理字符, 通用 Unicode

// 从标准输入设备读数据
1. 用 System.in 的 BufferedInputStream() 读取字节
int b = System.in.read();
System.out.println("Read data: " + (char)b); // 强制转换为字符
2. BufferedReader 读取文本
如果从 Stream 转成 Reader, 使用 InputStreamReader 类
BufferedReader is = new BufferedReader(new
InputStreamReader(System.in));
String inputLine;
while ((inputLine = is.readLine()) != null) {
System.out.println(inputLine);
int val = Integer.parseInt(inputLine); // 如果 inputLine 为整数
}
is.close();

// 向标准输出设备写数据
1. 用 System.out 的 println() 打印数据
2. 用 PrintWriter 打印
PrintWriter pw = new PrintWriter(System.out);
pw.println("The answer is " + myAnswer + " at this time.");

// Formatter 类
格式化打印内容
Formatter fmtr = new Formatter();
fmtr.format("%1$04d - the year of %2$f", 1951, Math.PI);
或者 System.out.printf(); 或者 System.out.format();

// 原始扫描
void doFile(Reader is) {
int c;
while ((c = is.read()) != -1) {
System.out.println((char)c);
}
}

// Scanner 扫描
Scanner 可以读取 File, InputStream, String, Readable
try {
Scanner scan = new Scanner(new File("a.txt"));
while (scan.hasNext()) {
String s = scan.next();
}
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}

// 读取文件
BufferedReader is = new BufferedReader(new FileReader("myFile.txt"));
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("bytes.bat"));
is.close();
bos.close();

// 复制文件
BufferedIutputStream is = new BufferedIutputStream(new FileIutputStream("oldFile.txt"));
BufferedOutputStream os = new BufferedOutputStream(new FileOutputStream("newFile.txt"));
int b;
while ((b = is.read()) != -1) {
os.write(b);
}
is.close();
os.close();

// 文件读入字符串
StringBuffer sb = new StringBuffer();
char[] b = new char[8192];
int n;
// 读一个块, 如果有字符, 加入缓冲区
while ((n = is.read(b)) > 0) {
sb.append(b, 0, n);
}
return sb.toString();

// 重定向标准流
String logfile = "error.log";
System.setErr(new PrintStream(new FileOutputStream(logfile)));

// 读写不同字符集文本
BufferedReader chinese = new BufferedReader(new InputStreamReader(new FileInputStream("chinese.txt"), "ISO8859_1"));
PrintWriter standard = new PrintWriter(new OutputStreamWriter(new FileOutputStream("standard.txt"), "UTF-8"));

// 读取二进制数据
DataOutputStream os = new DataOutputStream(new FileOutputStream("a.txt"));
os.writeInt(i);
os.writeDouble(d);
os.close();

// 从指定位置读数据
RandomAccessFile raf = new RandomAccessFile(fileName, "r"); // r 表示已只读打开
raf.seek(15); // 从 15 开始读
raf.readInt();
raf.radLine();

// 串行化对象
对象串行化, 必须实现 Serializable 接口
// 保存数据到磁盘
ObjectOutputStream os = new ObjectOutputStream(new BufferedOutputStream(new FileOutputStream(FILENAME)));
os.writeObject(serialObject);
os.close();
// 读出数据
ObjectInputStream is = new ObjectInputStream(new FileInputStream(FILENAME));
is.readObject();
is.close();

// 读写 Jar 或 Zip 文档
ZipFile zippy = new ZipFile("a.jar");
Enumeration all = zippy.entries(); // 枚举值列出所有文件清单
while (all.hasMoreElements()) {
ZipEntry entry = (ZipEntry)all.nextElement();
if (entry.isFile())
println("Directory: " + entry.getName());

// 读写文件
FileOutputStream os = new FileOutputStream(entry.getName());
InputStream is = zippy.getInputStream(entry);
int n = 0;
byte[] b = new byte[8092];
while ((n = is.read(b)) > 0) {
os.write(b, 0, n);
is.close();
os.close();
}
}

// 读写 gzip 文档
FileInputStream fin = new FileInputStream(FILENAME);
GZIPInputStream gzis = new GZIPInputStream(fin);
InputStreamReader xover = new InputStreamReader(gzis);
BufferedReader is = new BufferedReader(xover);
String line;
while ((line = is.readLine()) != null)
System.out.println("Read: " + line);

目录和文件操作

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
// 获取文件信息
exists(): 如果文件存在, 返回 true
getCanonicalPath(): 获取全名
getName(): 文件名
getParent(): 父目录
canRead(): 如果文件可读, 返回 true
canWrite(): 如果文件可写, 返回 true
lastModified(): 文件更新时间
length(): 文件大小
isFile(): 如果是文件, 返回 true
ifDirectory(): 如果是目录, 返回 true
要调用文件的这些方法, 必须
File f = new File(fileName);

// 创建文件
File f = new File("c:\\test\\mytest.txt");
f.createNewFile(); // 创建 mytest.txt 文件到 test 目录下

// 修改文件名
File f = new File("c:\\test\\mytest.txt");
f.renameTo(new File("c:\\test\\google.txt"));
把 mytest.txt 修改成 google.txt

// 删除文件
File f = new File("c:\\test\\mytest.txt");
f.delete();

// 临时文件
File f = new File("C:\\test"); // 指定一个文件夹
// 在 test 文件夹中创建 foo 前缀, tmp 后缀的临时文件
File tmp = File.createTempFile("foo", "tmp", f);
tmp.deleteOnExit(); // 在程序结束时删除该临时文件

// 更改文件属性
setReadOnly(): 设置为只读
setlastModified(): 设置最后更改时间

// 列出当前文件夹的文件列表
String[] dir = new java.io.File(".").list();
java.util.Arrays.sort(dir);
for (int i = 0; i < dir.length; i++) {
System.out.println(dir[i]);
}

// 过滤文件列表
class OnlyJava implements FilenameFilter {
public boolean accept(File dir, String s) {
if (s.endsWith(".java") || s.endsWith(".class") || s.endsWith(".jar"))
return true;
}
}

// 获取根目录
File[] rootDir = File.listRoots();
for (int i = 0; i < rootDir.length; i++) {
System.out.println(rootDir[i]);
}

// 创建新目录
new File("/home/ian/bin").mkdir(); // 如果"/home/ian"存在, 则可以创建 bin 目录
new File("/home/ian/bin").mkdirs(); // 如果"/home/ian"不存在, 会创建所有的目录

国际化和本地化

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
// I18N 资源
ResourceBundle rb = ResourceBundle.getBundle("Menus");
String label = rb.getString("exit.label");
// ResourceBundle 相当于名值对, 获取 Menus 按钮的区域属性
Menus_cn.properties: 不同区域的属性文件

// 列出有效区域
Locale[] list = Locale.getAvailableLocales();

// 指定区域
Locale cnLocale = Locale.CHINA;

// 设置默认区域
Locale.setDefault(Locale.CHINA);

// 格式化消息
public class MessageFormatDemo {
static Object[] data = {
new java.util.Date(),
"myfile.txt",
"could nto be opened"
};
public static void main(String[] args) {
String result = MessageFormat.format("At {0,time} on {0,date}, {1} {2}.", data);
System.out.println(result);
}
}
输出: At 10:10:08 on 2009-6-18, myfile.txt could nto be opened.

// 从资源文件中读消息
Widgets.properties 在 com.sean.cook.chap11 下
ResourceBundle rb = ResourceBundle.getBundle("com.sean.cook.chap11.Widgets");
String propt = rb.getString("filedialogs.cantopen.string");
String result = MessageFormat.format(rb.getString("filedialogs.cantopen.format"), data);

网络客户端

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
// 访问服务器
Socket socket = new Socket("127.0.0.1", 8080);
// todo something
socket.close();

// 查找网络地址
InetAddress.getByName(hostName).getHostAddress()); // 根据主机名得到 IP 地址
InetAddress.getByName(ipAddr).getHostName()); // 根据 IP 地址得到主机名

// 连接具体异常
UnknownHostException
NoRouteToHostException
ConnectException

// Socket 读写文本数据
BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
String remoteTime = in.readline();
PrintWriter out = new PrintWriter(socket.getOutputStream(), true);
out.print("send message to client \r\n");
out.flush();

// Socket 读写二进制数据
DataInputStream in = new DataInputStream(new BufferedInputStream(socket.getInputStream()));
long remoteTime = (long)(in.readUnsignedByte() << 24);
DataOutputStream out = new DataOutputStream(socket.getOutputStream(), true);

// Socket 读写串行化数据
ObjectInputStream in = new ObjectInputStream(new BufferedInputStream(socket.getInputStream()));
Object o = in.readObject();
if (o instanceof Date) // 验证对象类型
ObjectOutputStream out = new ObjectOutputStream(socket.getOutputStream(), true);

// UDP 数据报
private final static int PACKET_SIZE = 1024;

String host = "EV001B389673DE";
InetAddress serverAddr = InetAddress.getByName(host);
DatagramSocket socket = new DatagramSocket();
byte[] buffer = new byte[PACKET_SIZE]; // 分配数据缓冲空间
DatagramPacket packet = new DatagramPacket(buffer, PACKET_SIZE, serverAddr, 8080);
packet.setLength(PACKET_SIZE-1); // 设置数据长度
socket.send(packet);
socket.receive(packet); // 接收数据

服务器端: Socket

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
// 创建 ServerSocket
ServerSocket serverSocket;
Socket clientSocket;

serverSocket = new ServerSocket(9999);
while ((clientSocket = serverSocket.accept()) != null) {
System.out.println("Accept from client " + s.getInetAddress());
s.close();
}

// 监听内部网
public static final short PORT = 9999;
public static final String INSIDE_HOST = "acmewidgets-inside"; // 网络接口名
public static final int BACKLOG = 10; // 待发数
serverSocket = new ServerSocket(PORT, BACKLOG, InetAddress.getByName(INSIDE_HOST));

// 返回相应对象
ServerSocket serverSocket = new ServerSocket(9999);;
Socket clientSocket;
BufferedReader in = null;
PrintWriter out = null;
while (true) {
clientSocket = serverSocket.accept();
in = new BufferedReader(new InputStreamReader(clientSocket.getInputStream(), "8859_1"));
out = new PrintWriter(new OutputStreamWriter(clientSocket.getOutputStream(), "8859_1"), true);
String echoLine;
while ((echoLine = in.readLine()) != null) {
System.out.println("Read " + echoLine);
out.print(echoLine + "\r\n");
}
}
以上例子返回字符串, 如果返回二进制, 则使用 DataOutputStream;返回对象, 使用 ObjectOutputStream

// 处理多客户端
需要把接收数据的处理放入多线程中
public class EchoServerThreaded {
public static final int ECHOPORT = 7;
public static final int NUM_THREADS = 4;

public static void main(String[] av) {
new EchoServerThreaded(ECHOPORT, NUM_THREADS);
}

public EchoServerThreaded2(int port, int numThreads) {
ServerSocket servSock;
Socket clientSocket;
try {
servSock = new ServerSocket(ECHOPORT);
} catch(IOException e) {
throw new RuntimeException("Could not create ServerSocket " + e);
}
for (int i = 0; i < numThreads; i++) {
new Handler(servSock, i).start();
}
}
}
class Handler extends Thread {
ServerSocket servSock;
int threadNumber;

Handler(ServerSocket s, int i) {
super();
servSock = s;
threadNumber = i;
setName("Thread " + threadNumber);
}

public void run() {
while (true) {
try {
System.out.println(getName() + " waiting");
Socket clientSocket;
synchronized (servSock) {
clientSocket = servSock.accept();
}
System.out.println(getName() + " starting, IP=" + clientSocket.getInetAddress());
BufferedReader is = new BufferedReader(new InputStreamReader(
clientSocket.getInputStream()));
PrintStream os = new PrintStream(clientSocket.getOutputStream(), true);
String line;
while ((line = is.readLine()) != null) {
os.print(line + "\r\n");
os.flush();
}
System.out.println(getName() + " ENDED ");
clientSocket.close();
} catch (IOException ex) {
System.out.println(getName() + ": IO Error on socket " + ex);
return;
}
}
}
}

// 使用 SSL 和 JSSE 保护 Web 服务器
SSLServerSocketFactory ssf = (SSLServerSocketFactory)SSLServerSocketFactory.getDefault();
ServerSocket serverSocket = ssf.createServerSocket(8080);

// Log4j
Level 级别: DEBUG < INFO < WARN < ERROR < FATAL < OFF
Appender: 输出信息
ConsoleAppender: 输出控制台 System.out

// 找到网络接口
Enumeration list = NetworkInterface.getNetworkInterfaces();
while (list.hasMoreElements()) {
NetworkInterface iface = (NetworkInterface)list.nextElement();
System.out.println(iface.getDisplayName());
Enumeration addrs = iface.getInetAddresses();
while (addrs.hasMoreElements()) {
InetAddress addr = (InetAddress)addrs.nextElement();
System.out.println(addr);
}
}

Java Mail

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
// 发送 Mail
protected String msgRecIp = "hxydream@163.com";
protected String msgSubject = "babytree";
protected String msgCc = "nobody@erewhon.com";
protected String msgBody = "test body";
protected Session session;
protected Message msg;

public void doSend() {
// 创建属性文件
Properties props = new Properties();
props.put("mail.smtp.host", "mailhost");
// 创建 Session 对象
session = Session.getDefaultInstance(props, null);
session.setDebug(true);
msg = new MimeMessage(session); // 创建邮件
msg. setFrom(new InternetAddress("nobody@host.domain"));
InternetAddress toAddr = new InternetAddress(msgRecIp);
msg.addRecipient(Message.RecipientType.TO, toAddr);
InternetAddress ccAddr = new InternetAddress(msgCc);
msg.addRecipient(Message.RecipientType.CC, ccAddr);
msg.setSubject(msgSubject);
msg.setText(msgBody);
Transport.send(msg);
}

// 发送 MIME 邮件
Multipart mp = new MimeMultipart();
BodyPart textPart = new MimeBodyPart();
textPart.setText(message_body); // 设置类型"text/plain"
BodyPart pixPart = new MimeBodyPart();
pixPart.setContent(html_data, "text/html");
mp.addBodyPart(textPart);
mp.addBodyPart(pixPart);
mesg.setContent(mp);
Transport.send(mesg);

// 读 Mail
Store store = session.getStore(protocol);
store.connect(host, user, password);
Folder rf;
rf = store.getFolder(root);
rf = store.getDefaultFolder();
rf.open(Folder.READ_WRITE);

数据库访问

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
// JDO
Properties p = new Properties();
p.load(new FileInputStream("jdo.properties"));
PersistenceManagerFactory pmf = JDOHelper.getPersistenceManagerFactory(p);
PersistenceManager pm = pmf.getPersistenceManager();
// 提交数据
pm.currentTransaction().begin();
if (o instanceof Collection) {
pm.makePersistentAll((Collection) o);
} else {
pm.makePersistent(o);
}
pm.currentTransaction().commit();
pm.close();
// 取出数据
Object[] data = new Object[3];
pm.retrieveAll(data);
for (int i = 0; i < data.length; i++) {
System.out.println(data[i]);
}
pm.close();

// 数据操作
Class clz = Class.forName("oracle.jdbc.driver.OracleDriver");
String dbUrl = "jdbc:oracle:thin:@192.168.0.23:1521#:nms";
Connection conn = DriverManager.getConnection(dbUrl, "su", "1234");
Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("select * from pmtable");
while (rs.next()) {
String name = rs.getString(1);
String otherName = rs.getString("name");
}

// 使用 PreparedStatement 提高性能, 除了查询, 都使用 executeUpdate 执行操作
PreparedStatement pstmt = conn.prepareStatement("select * from pmtable where name = ?");
pstmt.setString(1, "sean");
ResultSet rs = pstmt.executeQuery();

// 调用存储过程
CallableStatement cs = conn.prepareCall("{ call ListDefunctUsers }");
ResultSet rs = cs.executeQuery();

// 显示数据库表信息
DatabaseMetaData meta = conn.getMetaData();
meta.getDatabaseProductName();
meta.getDatabaseProductVersion();
meta.getDefaultTransactionIsolation();

XML

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/*
SAX: 在读取文档提取相应的标记事件 (元素起始、元素结束、文档起始)
DOM: 在内存中构造与文档中元素相应的树, 可以遍历、搜索、修改
DTD: 验证文档是否正确
JAXP: 用于 XML 处理的 Java API
Castor: 开源项目, 用于 Java 对象与 XML 映射
*/
// 从对象中生成 XML
private final static String FILENAME = "serial.xml";
public static void main(String[] args) throws IOException {
String a = "hard work and best callback";
new SerialDemoXML().write(a);
new SerialDemoXML().dump();
}
public void write(Object obj) throws IOException {
XMLEncoder os = new XMLEncoder(new BufferedOutputStream(new FileOutputStream(FILENAME)));
os.writeObject(obj);
os.close();
}
public void dump() throws IOException {
XMLDecoder out = new XMLDecoder(new BufferedInputStream(new FileInputStream(FILENAME)));
System.out.println(out.readObject());
out.close();
}
serial.xml 格式内容如下:
<?xml version="1.0" encoding="UTF-8"?>
<java version="1.6.0_02" class="java.beans.XMLDecoder">
<string>hard work and best callback</string>
</java>
控制台输出
hard work and best callback

// XSLT 转换 XML
XSLT 可以用来对输出格式进行各种控制
Transformer tx = TransformerFactory.newInstance().newTransformer(new StreamSource("people.xml"));
tx.transform(new StreamSource("people.xml"), new StreamResult("people.html"));

// 用 SAX 解析 XML - 主要用于查找关键元素, 不用全文遍历
public SaxLister() throws SAXException, IOException {
XMLReader parser = XMLReaderFactory.createXMLReader("org.apache.xerces.parsers.SAXParser");
parser.setContentHandler(new PeopleHandler());
parser.parse("C:\\StudySource\\javacooksrc2\\xml\\people.xml");
}
class PeopleHandler extends DefaultHandler {
boolean parent = false;
boolean kids = false;
public void startElement(String nsURI, String localName, String rawName, Attributes attr) throws SAXException {
System.out.println("startElement: " + localName + "," + rawName);
if (rawName.equalsIgnoreCase("name"))
parent = true;
if (rawName.equalsIgnoreCase("children"))
kids = true;
}
public void characters(char[] ch, int start, int length) {
if (parent) {
System.out.println("Parent: " + new String(ch, start, length));
parent = false;
} else if (kids) {
System.out.println("Children: " + new String(ch, start, length));
kids = false;
}
}
public PeopleHandler() throws SAXException {
super();
}
}

// DOM 解析 XML - 遍历整个树
String uri = "file:" + new File("C:\\StudySource\\javacooksrc2\\xml\\people.xml").getAbsolutePath();
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
DocumentBuilder builder = factory.newDocumentBuilder();
Document doc = builder.parse(uri);
NodeList nodes = doc.getChildNodes();
for (int i = 0; i < nodes.getLength(); i++) {
Node n = nodes.item(i);
switch (n.getNodeType()) {
case Node.ELEMENT_NODE:
// todo
break;
case Node.TEXT_NODE:
// todo
break;
}
}

// 使用 DTD 或者 XSD 验证
定义好 DTD 或 XSD 文件
XmlDocument doc = XmlDocument.createXmlDocument(uri, true);

// 用 DOM 生成 XML
DocumentBuilderFactory fact = DocumentBuilderFactory.newInstance();
DocumentBuilder parser = fact.newDocumentBuilder();
Document doc = parser.newDocument();
Node root = doc.createElement("Poem");
doc.appendChild(root);
Node stanza = doc.createElement("Stanza");
root.appendChild(stanza);
Node line = doc.createElement("Line");
stanza.appendChild(line);
line.appendChild(doc.createTextNode("Once, upon a midnight dreary"));
line = doc.createElement("Line");
stanza.appendChild(line);
line.appendChild(doc.createTextNode("While I pondered, weak and weary"));

RMI

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
a. 定义客户端与服务器之间的通信接口
public interface RemoteDate extends Remote {
public Date getRemoteDate() throws RemoteException;
public final static String LOOKUPNAME = "RemoteDate";
}

b. 编写 RMI 服务器
public class RemoteDateImpl extends UnicastRemoteObject implements RemoteDate {
public RemoteDateImpl() throws RemoteException {
super();
}
public Date getRemoteDate() throws RemoteException {
return new Date();
}
}
RemoteDateImpl im = new RemoteDateImpl();
System.out.println("DateServer starting...");
Naming.rebind(RemoteDate.LOOKUPNAME, im);
System.out.println("DateServer ready.");

c. 运行 rmic 生成 stub
javac RemoteDateImpl.java
rmic RemoteDateImpl

d. 编写客户端
netConn = (RemoteDate)Naming.lookup(RemoteDate.LOOKUPNAME);
Date today = netConn.getRemoteDate();
System.out.println(today.toString());

e. 确保 RMI 注册表运行
rmiregistry

f. 启动服务器
java RemoteDateImpl

g. 运行客户端
java DateClient


// 19. 包和包装机制
jar cvf /tmp/test.jar . // 当前目录压缩到 test.jar 中
jar xvf /tmp/test.jar // 把 test.jar 解压到当前目录
从指定 class 运行 jar 文件
a. Main-Class: HelloWord // 注意中间有一个空格
b. jar cvmf manifest.mf hello.jar HelloWorld.class
c. java -jar hello.jar
*/
// 停止线程 - 不要使用 stop() 方法
private boolean done = false;
public void run() {
while (!done) {
// todo
}
}
public void shutDown() {
done = true;
}
可以调用 shutDown() 方法来结束线程

// 如果读取 IO 的时候出现堵塞, 那么可以使用下面方法
public void shutDown() throws IOException {
if (io != null)
io.close();
}

// 启动一线程, 等待控制台输入, 使用 join() 方法来暂停当前线程, 直到其他线程调用
Thread t = new Thread() {
public void run() {
System.out.println("Reading");
try {
System.in.read();
} catch (IOException e) {
System.err.println(e);
}
System.out.println("Thread finished.");
}
};
System.out.println("Starting");
t.start();
System.out.println("Joining");
try {
t.join();
} catch (InterruptedException e) {
System.out.println("Who dares imterrupt my sleep?");
}
System.out.println("Main finished.");

// 加锁保证同步
Lock lock = new ReentrantLock();
try {
lock.lock();
// todo
} finally {
lock.unlock();
}

线程通信 wait(), notify(), notifyAll()
生产者 - 消费者模式
Executors

Java 线程

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
// 停止线程 - 不要使用 stop() 方法
private boolean done = false;
public void run() {
while (!done) {
// todo
}
}
public void shutDown() {
done = true;
}
可以调用 shutDown() 方法来结束线程

// 如果读取 IO 的时候出现堵塞, 那么可以使用下面方法
public void shutDown() throws IOException {
if (io != null)
io.close();
}

// 启动一线程, 等待控制台输入, 使用 join() 方法来暂停当前线程, 直到其他线程调用
Thread t = new Thread() {
public void run() {
System.out.println("Reading");
try {
System.in.read();
} catch (IOException e) {
System.err.println(e);
}
System.out.println("Thread finished.");
}
};
System.out.println("Starting");
t.start();
System.out.println("Joining");
try {
t.join();
} catch (InterruptedException e) {
System.out.println("Who dares imterrupt my sleep?");
}
System.out.println("Main finished.");

// 加锁保证同步
Lock lock = new ReentrantLock();
try {
lock.lock();
// todo
} finally {
lock.unlock();
}

线程通信 wait(), notify(), notifyAll()
生产者 - 消费者模式
Executors

内省或“命令类的类”

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
// 反射
Class c = Class.forName("java.lang.String");
Constructor[] cons = c.getConstructors();
for (int i = 0; i < cons.length; i++) {
System.out.println(cons[i].toString());
}
Method[] meths = c.getMethods();
for (int i = 0; i < meths.length; i++) {
System.out.println(meths[i].toString());
}

// 动态装载类
Class c = Class.forName("java.lang.String");
Object obj = c.newInstance();

// 通过反射调用类的方法
class X {
public void master(String s) {
System.out.println("Working on \"" + s + "\"");
}
}
Class clx = X.class;
Class[] argTypes = {String.class};
Method worker = clx.getMethod("master", argTypes);
Object[] theData = {"Chocolate chips"};
worker.invoke(new X(), theData);
输出: Working on "Chocolate chips"

Java 与其他语言的结合

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// 执行 CMD 命令, 在 Eclipse 控制台输出
Process p = Runtime.getRuntime().exec("C:/StudySource/ver.cmd");
p.waitFor(); // 等待命令执行完
BufferedReader br = new BufferedReader(new InputStreamReader(p.getInputStream()));
String s;
while ((s = br.readLine()) != null)
System.out.println(s);

// 调用 Jython - 计算 22.0/7
BSFManager manager = new BSFManager();
String[] fntypes = {".py"};
manager.registerScriptingEngine("jython", "org.apache.bsf.engines.jython.JythonEngine", fntypes);
Object r = manager.eval("jython", "testString", 0, 0, "22.0/7");
System.out.println("Result type is " + r.getClass().getName());
System.out.println("Result value is " + r);

// 调用 C++, 使用 JNI