標籤:
由於最近工作遇到效能問題,嘗試研究用多線程來實現,結果速度快了好幾倍
下面是多線程查詢的部分代碼,提供給大家參考下:
線程類:
帶返回值的類要實現Callable介面,具體商務邏輯沒有實現,只是寫了個空方法在裡面
package com.sanfy.demo.thread;
import java.util.List;
import java.util.Map;
import java.util.concurrent.Callable;
import com.sanfy.demo.model.WorkLoad;
public class QueryThread implements Callable<List<WorkLoad>> {
private List<WorkLoad> workLoads;
private WorkLoadService workLoadService;
private Map<String, Object> parameters;
private List<String> dates;
private int begin;
private int end;
public QueryThread(Map<String, Object> parameters,List<String> dates, int begin, int end,WorkLoadService workLoadService) {
super();
this.parameters = parameters;
this.dates = dates;
this.begin = begin;
this.end = end;
this.workLoadService=workLoadService;
}
/**
* @return Map<Integer, EchartsTenInDatasDTO>
* @see java.lang.Callable#call()
*/
@Override
public Map<Integer, WorkLoad> call() {
for (int i = begin; i < end; i++) {
queryListByMap(dates.get(i));
}
return workLoads;
}
/**
* 查詢方法
*
* @param date 日期
* @param i i
*/
private void queryListByMap(String date) {
//具體的查詢
workLoads=workLoadService.queryList(parameters,date);
}
public List<WorkLoad> getWorkLoads() {
return workLoads;
}
public void setWorkLoads(List<WorkLoad> workLoads) {
this.workLoads = workLoads;
}
public WorkLoadService getWorkLoadService() {
return workLoadService;
}
public void setWorkLoadService(WorkLoadService workLoadService) {
this.workLoadService = workLoadService;
}
public Map<String, Object> getParameters() {
return parameters;
}
public void setParameters(Map<String, Object> parameters) {
this.parameters = parameters;
}
public List<String> getDates() {
return dates;
}
public void setDates(List<String> dates) {
this.dates = dates;
}
public int getBegin() {
return begin;
}
public void setBegin(int begin) {
this.begin = begin;
}
public int getEnd() {
return end;
}
public void setEnd(int end) {
this.end = end;
}
}
serive調用線程類來實現多線程查詢:
package com.sanfy.demo.thread;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import com.sanfy.demo.model.WorkLoad;
public class QuerySerivce {
private WorkLoadService workLoadService;
public List<WorkLoad> queryListsByMap(String sqlCode, Map<String, Object> parameters) throws Exception {
List<WorkLoad> workLoads=new ArrayList<WorkLoad>();
long startTime = System.currentTimeMillis();
List<String> dates = DateUtil.getCurrentInYear(parameters.get("startDate").toString(), 12);
if (dates != null && dates.size() > 0) {
int len = dates.size();
// 建立線程的個數
int count = len % 2 == 0 ? len / 2: (len / 2) + 1;
// 建立線程池數量
ExecutorService pool = Executors.newFixedThreadPool(count);
List<Callable<List<WorkLoad>>> tasks = new ArrayList<Callable<List<WorkLoad>>>();
for (int i = 0; i < count; i++) {
// 每個線程負責插入資料的開始位置
int start = i * 2;
// 每個線程負責插入資料的結束位置
int end = i == count - 1 ? i * 2 + (len - i * 2) : i* 2+ 2;
Map<String, Object> parametersClone = new HashMap<String, Object>();
parametersClone.putAll(parameters);
// 線程返回值
Callable<List<WorkLoad>> task = new QueryThread(parametersClone, sqlCode, dates, start, end,workLoadService);
tasks.add(task);
}
// 執行線程
List<Future<List<WorkLoad>>> futures = pool.invokeAll(tasks);
// 處理線程返回結果
if (futures != null && futures.size() > 0) {
for (Future<List<WorkLoad>> future : futures) {
workLoads.addAll(future.get());
}
}
// 關閉線程池
pool.shutdown();
}
long endTime = System.currentTimeMillis();
logger.info("多個情況報表查詢總共花了:" + (endTime - startTime) / 1000 + "秒時間!");
return workLoads;
}
}
後續會補上多線程批量儲存與android多線程支援斷點續傳下載的內容
由於是實際項目,所以把具體的內容刪除了,直接修改的,供大家參考下
java多線程查詢