上一節我們做完了首頁UI界面,但是有個問題:如果我在後台添加了一個商品,那麼我必須重啟一下服務器才能重新同步後台數據,然後刷新首頁才能同步數據。這明顯不是我們想要的效果,一般這種網上商城首頁肯定不是人為手動同步數據的,那麼如何解決呢?我們需要用到線程和定時器來定時自動同步首頁數據。
1. Timer和TimerTask
我們需要用到Timer和TimerTask兩個類。先來介紹下這兩個類。
Timer是一種工具類,在java.util包中,線程用其安排以後在後台線程中執行的任務。可安排任務執行一次,或者定期重複執行。它有個構造函數:
Timer(boolean isDaemon) //創建一個新計時器,可以指定其相關的線程作為守護程序運行。
守護線程即主線程結束後,該線程也結束,非守護線程即主線程結束後,該線程仍然繼續執行。 isDaemon為true時為守護線程。 Timer類有個schedule方法可以創建一個任務,如下:
void schedule(TimerTask task, Date firstTime, long period) //安排指定的任務在指定的時間開始進行重複的固定延遲執行。 //第一個參數是指定任務,即TimerTask對象;第二個參數為第一次開啟任務時間;第三個參數為時間間隔,即每隔多長時間執行一次
我們再來看看TimerTask,TimerTask是用來創建一個新的線程任務的,它實現了Runnable接口,如果我們要創建一個新的線程任務,只需要繼承TimerTask,並重寫run方法即可。
2. 創建一個新的線程任務
下面我們來創建一個新的線程任務,用來更新後台數據:
@Component //把該對象交給Spring管理public class ProductTimerTask extends TimerTask { @Resource private ProductService productService = null; //注入productService @Resource private CategoryService categoryService = null; //注入categoryService private ServletContext application = null; //定義一個ServletContext對象,因為我們更新了後台數據後,需要存入application域裡面public void setApplication(ServletContext application) { this.application = application; //通過監聽器將這個application對象set進來,因為這裡是無法拿application對象的} @Override //和監聽器在項目啟動的時候數據初始化的邏輯一樣public void run() { System.out.println("----run----"); List<List<Product>> bigList = new ArrayList<List<Product>>(); //bigList中存放一個裝有Category類的list // 1. 查詢出熱點類別for(Category category : categoryService.queryByHot(true)) { //根據熱點類別id獲取推薦商品信息List<Product> lst = productService.querByCategoryId(category.getId()); bigList.add(lst); //將裝有category的list放到bigList中} // 2. 把查詢的bigList交給application內置對象application.setAttribute("bigList", bigList); //假設我們已經拿到了application對象} }接下來,我們修改項目啟動時監聽器裡面的內容,原本上面的這個查詢操作是放在監聽器中,當項目啟動時,監聽器開始執行,獲取後台數據,存到application域中,然後前台通過jstl標籤從application域中拿到數據。現在我們把這些事情交給我們定義的ProductTimerTask去做,那麼監聽器中只要設置一下定時器,讓ProductTimerTask定時去更新一下後台數據即可。看看監聽器中修改後的代碼:
3. 在監聽器中啟動定時器
//@Component //監聽器是web層的組件,它是tomcat實例化的,不是Spring實例化的。不能放到Spring中public class InitDataListener implements ServletContextListener { private ProductTimerTask productTimerTask = null; //定義一個ProductTimerTask對象private ApplicationContext context = null; @Override public void contextDestroyed(ServletContextEvent event) { // TODO Auto-generated method stub } @Override public void contextInitialized(ServletContextEvent event) { context = WebApplicationContextUtils.getWebApplicationContext(event.getServletContext()); productTimerTask = (ProductTimerTask) context.getBean("productTimerTask");//從配置文件中獲取ProductTimerTask對象//把內置對象交給productTimerTask,因為productTimerTask裡面是拿不到application的,只能通過監聽器set給它productTimerTask.setApplication(event.getServletContext()); //通過設置定時器,讓首頁的數據每個一小時同步一次(配置為守護線程) new Timer(true).schedule(productTimerTask, 0, 1000*60*60);//每個一小時執行一次productTimerTask任務,即更新一下後台數據} }關於InitDataListener監聽器中原來的操作代碼,可以對比上一節中的內容,其實就是ProductTimerTask中的更新後台數據,只不過現在放到TimerTask中去做了而已。這樣我們就完成了使用線程和定時器定期同步首頁數據,這個時間間隔可以自己設定。
其實CSDN博客裡的部分首頁數據也不是實時更新的,每天晚上會有個時間更新一次,例如左側欄目中的博客排名,閱讀排行後的顯示的閱讀量等,這些都是每天晚上更新一次,應該就是在後台設置了每天更新一次,原理跟這裡應該是一樣的。這樣也減輕了服務器的壓力。
本文鏈接:http://blog.csdn.net/eson_15/article/details/51387378
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。