在java中有一類線程,專門在後臺提供服務,此類線程無需顯式關閉,當程序結束了,它也就結束了,這就是守護線程 daemon thread。若是還有非守護線程的線程在執行,它就不會結束。 守護線程有何用處呢?讓咱們來看個實踐中的例子。java
在咱們的系統中常常應用各類配置文件(黑名單,禁用詞彙),當修改配置文件後,通常要重啓服務,系統纔可以加載;當重啓服務的代價比較高的狀況下,這種加載方式不能知足咱們的要求,這個時候守護線程該發揮它的做用了,它能夠實時加載你的配置文件,無需重啓。(固然,至關重要的配置文件,不推薦實時加載)
ide
package
com.ikon.thread.daemon;

import
java.io.File;
import
java.util.
*
;

/**
* 文件監測
* @author ikon99999
*
*/
public
abstract
class
FileWatchdog
extends
Thread
{

static final public long DEFAULT_DELAY = 20*1000;
protected HashMap fileList;
protected long delay = DEFAULT_DELAY;
boolean warnedAlready = false;
boolean interrupted = false;

public static class Entity
{
File file;
long lastModify;
Entity(File file,long lastModify)
{
this.file = file;
this.lastModify = lastModify;
}
}
protected FileWatchdog() {
fileList = new HashMap ();
setDaemon(true);
}

public void setDelay(long delay) {
this.delay = delay;
}

public void addFile(File file)
{
fileList.put(file.getAbsolutePath(),new Entity(file,file.lastModified()));
}
public boolean contains(File file)
{
if( fileList.get(file.getAbsolutePath()) != null) return true;
else return false;
}
abstract protected void doOnChange(File file);

protected void checkAndConfigure() {
HashMap map = (HashMap)fileList.clone();
Iterator it = map.values().iterator();
while( it.hasNext())
{
Entity entity = (Entity)it.next();
boolean fileExists;
try {
fileExists = entity.file.exists();
} catch(SecurityException e)
{
System.err.println ("Was not allowed to read check file existance, file:["+ entity.file .getAbsolutePath() +"].");
interrupted = true;
return;
}

if(fileExists)
{
long l = entity.file.lastModified(); // this can also throw a SecurityException
if(l > entity.lastModify) { // however, if we reached this point this
entity.lastModify = l; // is very unlikely.
newThread(entity.file);
}
}
else
{
System.err.println ("["+entity.file .getAbsolutePath()+"] does not exist.");
}
}
}
private void newThread(File file)
{
class MyThread extends Thread
{
File f;
public MyThread(File f)
{
this.f = f;
}
public void run()
{
doOnChange(f);
}
}
MyThread mt = new MyThread(file);
mt.start();
}

public void run()
{
while(!interrupted) {
try {
Thread.currentThread().sleep(delay);
} catch(InterruptedException e) {
// no interruption expected
}
checkAndConfigure();
}
}
}
FileWatchdog是個抽象類,自己是線程的子類;在構造函數中設置爲守護線程;
此類用hashmap維護着一個文件和最新修改時間值對,checkAndConfigure()方法用來檢測哪些文件的修改時間更新了,若是發現文件更新了則調用doOnChange方法來完成監測邏輯;doOnChange方法是咱們須要實現的;看下面關於一個黑名單服務的監測服務:
函數
1
package
com.ikon.thread.daemon;
2

3
import
java.io.File;
4

5
/**
6
* 黑名單服務
7
* @author ikon99999
8
* 2011-3-21
9
*/
10
public
class
BlacklistService
{
11
private File configFile = new File("c:/blacklist.txt");
12
13
public void init() throws Exception{
14
loadConfig();
15
ConfigWatchDog dog = new ConfigWatchDog();
16
dog.setName("daemon_demo_config_watchdog");//a
17
dog.addFile(configFile);//b
18
dog.start();//c
19
}
20
21
public void loadConfig(){
22
try{
23
Thread.sleep(1*1000);//d
24
25
System.out.println("加載黑名單");
26
}catch(InterruptedException ex){
27
System.out.println("加載配置文件失敗!");
28
}
29
}
30
31
public File getConfigFile() {
32
return configFile;
33
}
34
35
public void setConfigFile(File configFile) {
36
this.configFile = configFile;
37
}
38
39
40
private class ConfigWatchDog extends FileWatchdog{
41
42
@Override
43
protected void doOnChange(File file) {
44
System.out.println("文件"+file.getName()+"發生改變,從新加載");
45
loadConfig();
46
}
47
48
}
49
50
public static void main(String[] args) throws Exception {
51
BlacklistService service = new BlacklistService();
52
service.init();
53
54
Thread.sleep(60*60*1000);//e
55
}
56
}
57
ConfigWatchDog內部類實現了doOnChange(File file)方法,當文件被修改後,watchdog調用doOnChange方法完成從新加載操做; 在blackservice的init方法中初始化watchdog線程; d:模擬文件加載耗時 e:主要是防止主線程退出; 其實上面的FileWatchdog就是取自log4j;