Thứ Năm, 26 tháng 9, 2013

Export Excel, PDF trong SpringMVC

Xuất file excel ta dùng thư viện apache poi: http://poi.apache.org/Xuất file pdf ta dùng thư viện itextpdf: http://itextpdf.com/

Ví dụ này ta sẽ xuất ra dưới dạng bảng thống kê, tức là có title, có các header cho các column và data cho bảng đó, như vậy, ta sẽ xây dựng 2 lớp builder nhận vào các đối là
Title: 1 chuỗi String
header: 1 Vector
data: 1 Vector
Đây là lớp ExcelBuilder để export ra file excel:

package vn.diepviends.util;

import java.util.Map;
import java.util.Vector;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.poi.hssf.usermodel.HSSFFont;
import org.apache.poi.hssf.usermodel.HSSFRow;
import org.apache.poi.hssf.usermodel.HSSFSheet;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.hssf.util.HSSFColor;
import org.apache.poi.ss.usermodel.CellStyle;
import org.apache.poi.ss.usermodel.Font;
import org.springframework.web.servlet.view.document.AbstractExcelView;

public class ExcelBuilder extends AbstractExcelView {
 @Override
 protected void buildExcelDocument(Map model,
   HSSFWorkbook workbook, HttpServletRequest request,
   HttpServletResponse response) throws Exception {
  // get data
  Vector vHeader = (Vector) model.get("header");
  Vector vData = (Vector) model.get("data");
  String sTitle = model.get("title").toString();
  // create a excel sheet
  HSSFSheet sheet = workbook.createSheet(sTitle);
  sheet.setDefaultColumnWidth(30);
  // create style for header cells
  CellStyle style = workbook.createCellStyle();
  Font font = workbook.createFont();
  font.setFontName("Times New Roman");
  style.setFillForegroundColor(HSSFColor.BLUE.index);
  style.setFillPattern(CellStyle.SOLID_FOREGROUND);
  font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
  font.setColor(HSSFColor.WHITE.index);
  style.setFont(font);
  // create header row
  HSSFRow title = sheet.createRow(0);
  title.createCell(0).setCellValue(sTitle);
  HSSFRow header = sheet.createRow(1);  
  for (int i = 0; i < vHeader.size(); i++) {
   header.createCell(i).setCellValue(vHeader.get(i).toString());
   header.getCell(i).setCellStyle(style);
  }
  // fill data
  int r = 2;
  for (int i = 0; i < vData.size(); i++) {
   Vector v = (Vector) vData.get(i);
   HSSFRow row = sheet.createRow(r++);
   for (int j = 0; j < v.size(); j++)
    row.createCell(j).setCellValue(v.get(j).toString());
  }
 }
}

Controller ta sẽ viết như sau:
@RequestMapping("/registerReport/export2excel")
 public String exportExcel(@RequestParam(value = "from") String from,
   @RequestParam(value = "to") String to, Model model) {
  ProvisionReportDAO dao = new ProvisionReportDAO();
  String title = "Register Report";
  Vector header = new Vector();
  Vector data = new Vector();
  ArrayList list = dao.getByDate(from, to);  
  header.add("Date");
  header.add("Quantity");
  header.add("Provision Type");
  header.add("provision Code");
  for(ProvisionReport it : list){
   Vector v = new Vector();
   v.add(it.getDate());
   v.add(it.getQuantity());
   v.add(it.getType());
   v.add(it.getCode());
   data.add(v);
  }  
  model.addAttribute("title", title);
  model.addAttribute("header", header);  
  model.addAttribute("data",data); 
  return "excelView";
 }

Với PDF thì ta phải xây dựng một lớp Abstract trung gian:
package vn.diepviends.util;

import java.io.ByteArrayOutputStream;
import java.io.OutputStream;
import java.util.Map;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.web.servlet.view.AbstractView;

import com.itextpdf.text.Document;
import com.itextpdf.text.DocumentException;
import com.itextpdf.text.PageSize;
import com.itextpdf.text.pdf.PdfWriter;

public abstract class AbstractITextPdfView extends AbstractView {

 public AbstractITextPdfView() {
  setContentType("application/pdf");
 }

 @Override
 protected boolean generatesDownloadContent() {
  return true;
 }
  
 @Override
 protected void renderMergedOutputModel(Map model,
   HttpServletRequest request, HttpServletResponse response) throws Exception {
  // IE workaround: write into byte array first.
  ByteArrayOutputStream baos = createTemporaryOutputStream();

  // Apply preferences and build metadata.
  Document document = newDocument();
  PdfWriter writer = newWriter(document, baos);
  prepareWriter(model, writer, request);
  buildPdfMetadata(model, document, request);

  // Build PDF document.
  document.open();
  buildPdfDocument(model, document, writer, request, response);
  document.close();

  // Flush to HTTP response.
  writeToResponse(response, baos);
 }

 protected Document newDocument() {
  return new Document(PageSize.A4);
 }
 
 protected PdfWriter newWriter(Document document, OutputStream os) throws DocumentException {
  return PdfWriter.getInstance(document, os);
 }
 
 protected void prepareWriter(Map model, PdfWriter writer, HttpServletRequest request)
   throws DocumentException {

  writer.setViewerPreferences(getViewerPreferences());
 }
 
 protected int getViewerPreferences() {
  return PdfWriter.ALLOW_PRINTING | PdfWriter.PageLayoutSinglePage;
 }
 
 protected void buildPdfMetadata(Map model, Document document, HttpServletRequest request) {
 }
 
 protected abstract void buildPdfDocument(Map model, Document document, PdfWriter writer,
   HttpServletRequest request, HttpServletResponse response) throws Exception; 
}

Lớp PDFBuilder extends lớp Abstract trên:
package vn.diepviends.util;

import java.util.Map;
import java.util.Vector;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import com.itextpdf.text.BaseColor;
import com.itextpdf.text.Document;
import com.itextpdf.text.Font;
import com.itextpdf.text.FontFactory;
import com.itextpdf.text.Paragraph;
import com.itextpdf.text.Phrase;
import com.itextpdf.text.pdf.PdfPCell;
import com.itextpdf.text.pdf.PdfPTable;
import com.itextpdf.text.pdf.PdfWriter;

public class PDFBuilder extends AbstractITextPdfView {

 @Override
 protected void buildPdfDocument(Map model, Document doc,
   PdfWriter writer, HttpServletRequest request,
   HttpServletResponse response) throws Exception {
  // get data
  Vector vHeader = (Vector) model.get("header");
  Vector vData = (Vector) model.get("data");
  String sTitle = model.get("title").toString();
  
  doc.add(new Paragraph(sTitle));
  PdfPTable table = new PdfPTable(vHeader.size());
  table.setWidthPercentage(100.0f); 
  table.setSpacingBefore(10);
  // define font for table header row
  Font font = FontFactory.getFont(FontFactory.HELVETICA);
  font.setColor(BaseColor.WHITE);

  // define table header cell
  PdfPCell cell = new PdfPCell();
  cell.setBackgroundColor(BaseColor.BLUE);
  cell.setPadding(5);
  for (int i = 0; i < vHeader.size(); i++) {
   // write table header
   cell.setPhrase(new Phrase(vHeader.get(i).toString(), font));
   table.addCell(cell);
  }  
  // write table row data
  int x = vData.size();
  int y = vHeader.size();
  for(int i = 0; i


Controller ta sẽ viết như sau:
@RequestMapping("/registerReport/export2pdf")
 public String exportPdf(@RequestParam(value = "from") String from,
   @RequestParam(value = "to") String to, Model model) {
  ProvisionReportDAO dao = new ProvisionReportDAO();
  String title = "Register Report";
  Vector header = new Vector();
  Vector data = new Vector();
  ArrayList list = dao.getByDate(from, to);  
  header.add("Date");
  header.add("Quantity");
  header.add("Provision Type");
  header.add("provision Code");
  for(ProvisionReport it : list){
   Vector v = new Vector();
   v.add(it.getDate());
   v.add(it.getQuantity());
   v.add(it.getType());
   v.add(it.getCode());
   data.add(v);
  }  
  model.addAttribute("title", title);
  model.addAttribute("header", header);  
  model.addAttribute("data",data); 
  return "pdfView";
 }

Định nghĩa các view để hiển thị ra excel và pdf

     
    
     


Trong dispatcher của spring ta gọi view.xml vào:

        
 
 
  
  
         
        
   
   
    



và đây là kết quả:



Demo Job Scheduler sử dụng Quartz

maven dependences

    log4j
    log4j
    1.2.17
   
   
    org.quartz-scheduler
    quartz
    2.0.2
   

Xây dựng lớp TestJob implement Job của Quartz. lớp này để thực hiện một công việc mà ta cần làm sau một khoảng thời gian xác định. Cụ thể công việc ở đây là in ra dòng thông báo.
package vn.diepviends.job;

import org.apache.log4j.Logger;
import org.quartz.Job;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;

public class TestJob implements Job {
 private Logger log = Logger.getLogger(TestJob.class);
 public void execute(JobExecutionContext arg0) throws JobExecutionException {
  log.debug("Test job run successfully after 2s");    
 }

}
Trong lớp chính cần gọi ta cài đặt khoảng thời gian lặp lại cho job và gọi TestJob class để thực hiện
package vn.diepviends.job;

import org.quartz.JobBuilder;
import org.quartz.JobDetail;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.quartz.SchedulerFactory;
import org.quartz.SimpleScheduleBuilder;
import org.quartz.Trigger;
import org.quartz.TriggerBuilder;
import org.quartz.impl.StdSchedulerFactory;

public class JobScheduler {
 public static void main(String[] args) {
  try {

   // chi ra job can thuc hien
   JobDetail job = JobBuilder.newJob(TestJob.class)
     .withIdentity("testJob").build();

   // qui dinh thoi gian la 2s, lap lai mai mai
   Trigger trigger = TriggerBuilder
     .newTrigger()
     .withSchedule(
       SimpleScheduleBuilder.simpleSchedule()
         .withIntervalInSeconds(2).repeatForever())
     .build();

   // start
   SchedulerFactory schFactory = new StdSchedulerFactory();
   Scheduler sch = schFactory.getScheduler();
   sch.start();
   sch.scheduleJob(job, trigger);   

  } catch (SchedulerException e) {
   e.printStackTrace();
  }
 }
}
Và đây là kết quả khi run lớp JobScheduler


Cấu trúc chương trình:

Thứ Ba, 6 tháng 8, 2013

Lấy URL bằng đối tượng Request

Để lấy URL của website ví dụ: http://www.google.com.vn/ ta dùng cú pháp:

String  url = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + request.getContextPath(); 
Để lấy đường dẫn thực trên server ví dụ "E:\tomcat\demo\" ta dùng:

String path  = request.getSession().getServletContext().getRealPath("/")

Thứ Hai, 5 tháng 8, 2013

Coppy and Remove file trong Java

Trong lập trình nhiều khi chúng ta phải thao tác với file, ngoài việc đọc và ghi, thì còn phải coppy và delete file, delete file theo .extend
Class util  hỗ trợ mục đích của ta:
FileUtil.java
package vn.ds.util;

import java.io.*;

public class FileUtil {

 /**
  * Coppy data từ một file nguồn vào file đích
  */
 public static void copyFile(File source, File dest) throws IOException {

  if (!dest.exists()) {
   dest.createNewFile();
  }
  InputStream in = null;
  OutputStream out = null;
  try {
   in = new FileInputStream(source);
   out = new FileOutputStream(dest);
   // Transfer bytes from in to out
   byte[] buf = new byte[1024];
   int len;
   while ((len = in.read(buf)) > 0) {
    out.write(buf, 0, len);
   }

  } catch (Exception ex) {
   System.out.println("error");
  } finally {
   in.close();
   out.close();
  }
 }

 /**
  * Coppy tất cả các file trong thư mục nguồn đến thư mục đích
  */
 public static void copyDirectory(File sourceDir, File destDir)
   throws IOException {
  if (!destDir.exists()) {
   destDir.setWritable(true);
   destDir.mkdirs();
  } else {
   destDir.setWritable(true);
  }
  File[] children = sourceDir.listFiles();
  for (File sourceChild : children) {
   String name = sourceChild.getName();
   File destChild = new File(destDir, name);
   if (sourceChild.isDirectory()) {
    copyDirectory(sourceChild, destChild);
   } else {
    copyFile(sourceChild, destChild);
   }
  }
 }

 /**
  * xoá một thư mục (bao gồm các file trong thư mục đó)
  */
 public static boolean delete(File resource) throws IOException {
  if (resource.isDirectory()) {
   File[] childFiles = resource.listFiles();
   for (File child : childFiles) {
    delete(child);
   }
  }
  return resource.delete();
 }

 /**
  * xoá các file trong một thư mục có đuôi mở rộng là một parameter
  * 
  * @param String
  *            ext
  */
 public static void deleteFile(String folder, String ext) {

  GenericExtFilter filter = new GenericExtFilter(ext);
  File dir = new File(folder);

  // list out all the file name with .txt extension
  String[] list = dir.list(filter);

  if (list.length == 0)
   return;

  File fileDelete;

  for (String file : list) {
   String temp = new StringBuffer(folder).append(File.separator)
     .append(file).toString();
   fileDelete = new File(temp);
   boolean isdeleted = fileDelete.delete();
  }
 }

 /**
  * lớp filter để để lọc các file có đuôi mở rộng là một parameter được
  * truyền vào trong constructor method
  */
 public static class GenericExtFilter implements FilenameFilter {

  private String ext;

  public GenericExtFilter(String ext) {
   this.ext = ext;
  }

  public boolean accept(File dir, String name) {
   return (name.endsWith(ext));
  }
 }

}



Thks & Rgds!!!

Thứ Tư, 31 tháng 7, 2013

Tạo connection pool MySQL trong java

Trước tiên ta xây dựng lớp Logger để ghi lại những thông báo hoặc ném ra lỗi, điều này rất cần thiết đối với dự án lớn, đặc biết là có nhiều lớp, nhiều hàm, khi debug, fix error. có thể sử dụng lớp Logger của gói java.util hoặc org.apache.log4j, để đơn giản ta xây dựng lớp Logger gồm 2 method là log method để in thông báo và error method để thông báo lỗi.
public class Logger {
 private String classname;

 public Logger(String classname) {
  this.classname = classname;
 }

 public void log(String msg) {
  System.out.println("[" + this.classname + "] " + msg);
 }

 public void error(String msg) {
  System.out.println("[ERROR] [" + this.classname + "] " + msg);
 }
}

Bây giờ ta xây dựng lớp để tạo connections pool đến MySQL trong Java, sự quan trọng và cần thiết cũng như khái niệm về connection pool trong lập trình thì có thể search trên google.com. ở đây ta không đề cập đến. Lớp kết nối sẽ có tên là DBPool:
  public class DBPool {
    private static Logger logger = new Logger("DBPool"); // biến logger của lớp Logger trên
    private static LinkedList pool = new LinkedList(); // pool để chứa các connections
    public final static int MAX_CONNECTIONS = 10;  // số connection tối đa trong pool là 10, tuỳ ý!!
    public final static int INI_CONNECTIONS = 2; // số connection khi bắt đầu khởi tạo là 2
}
Các phương thức quan trọng trong lớp:
Phương thức tạo một kết nối:
public static Connection makeDBConnection() throws SQLException {
        Connection conn = null;
        try {
            Class.forName(DBConfig.getProperties("_mf_driver", ""));
            logger.log("URL:" + DBConfig.getProperties("_mf_url", ""));
            logger.log("User:" + DBConfig.getProperties("_mf_user", ""));
            conn = DriverManager.getConnection(
             DBConfig.getProperties("_mf_url", ""),
             DBConfig.getProperties("_mf_user", ""),
                DBConfig.getProperties("_mf_passwd", ""));
            
        } catch (ClassNotFoundException ex) {
            throw new SQLException(ex.getMessage());
        }catch (Exception e) {
   // TODO: handle exception
  }
        return conn;
    }
Phương thức khởi tạo số connections theo biến INI_CONNECTIONS:
public static void build(int number) {
        logger.log("Establishing " + number + " connections...");
        Connection conn = null;
        release();
        for (int i = 0; i < number; i++) {
            try {
                conn = makeDBConnection();
            } catch (SQLException ex) {
                logger.log("Error: " + ex.getMessage());                
            }
            if (conn != null) {
                pool.addLast(conn);
            }
        }
        logger.log("Number of connection: " + size());
    }
Phương thức lấy về connection, khi các class khác cần connect DB, thì chỉ cần gọi phương thức này là được:
public static Connection getConnection() {
        Connection conn = null;
        try{
         synchronized (pool) {
             conn = (java.sql.Connection) pool.removeFirst();
         }
         if (conn == null) {
          conn = makeDBConnection();
         }
         try {
             conn.setAutoCommit(true);
         } catch (Exception ex) {}
         
        } catch(Exception e){            
            logger.error("Method getConnection(): Error executing >>>" + e.toString());
            try {
             logger.log("Make connection again.");
    conn = makeDBConnection();
    conn.setAutoCommit(true);
   } catch (SQLException e1) {
   }
   logger.log(""+conn);
        }
        return conn;
    }

Phương thức đóng một kết nối có trong pool:
 public static void putConnection(java.sql.Connection conn) {
        try { // Ignore closed connection
            if (conn == null || conn.isClosed()) {
                logger.log("putConnection: conn is null or closed: " + conn);
                return;
            }
            if (pool.size() >= MAX_CONNECTIONS) {
                conn.close();
                return;
            }
        } catch (SQLException ex) {}

        synchronized (pool) {
            pool.addLast(conn);
            pool.notify();
        }
    }
Phương thức đóng và xoá tất cả connection trong pool:
public static void release() {
        logger.log("Closing connections in pool...");
        synchronized (pool) {
            for (Iterator it = pool.iterator(); it.hasNext(); ) {
                Connection conn = (Connection) it.next();
                try {
                    conn.close();
                }
                catch (SQLException e) {
                    logger.error(
                        "release: Cannot close connection! (maybe closed?)");
                }
            }
            pool.clear();
        }
        logger.log("Release connection OK");
    }
Ngoài ra còn có các phương thức khác trong lớp DBPool như sau:
//  phương thức close một connection và preparedStatement
public static void releaseConnection(Connection conn, PreparedStatement preStmt) {       
        putConnection(conn);
        try {
            if (preStmt != null) {
                preStmt.close();
            }
        } catch (SQLException e) {}
    }

    public static void releaseConnection(Connection conn, PreparedStatement preStmt, ResultSet rs) {
        releaseConnection(conn, preStmt);
        try {
            if (rs != null) {
                rs.close();
            }
        } catch (SQLException e) {}
    }

    public static void releaseConnection(Connection conn, PreparedStatement preStmt, Statement stmt, ResultSet rs) {
        releaseConnection(conn, preStmt, rs);
        try {
            if (stmt != null) {
                stmt.close();
            }
        } catch (SQLException e) {}
    }
    public static void releaseConnection(Connection conn, CallableStatement cs, ResultSet rs) {
     putConnection(conn);
        try {
            if (cs != null) {
             cs.close();
            }
            if(rs!=null){
             rs.close();
            }
        } catch (SQLException e) {}
    }
Để sử dụng lớp DBPool này ta thử demo với lớp EmployeeDAO dùng để lấy tất cả các nhân viên và để thêm một nhân viên.
public ArrayList getall() {
  String sql = "SELECT * FROM employee";
  ArrayList lst = new ArrayList();
  conn = DBPool.getConnection();
  try {
   psmt = conn.prepareStatement(sql);
   rs = psmt.executeQuery();
   while (rs.next()) {
    Employee employee = new Employee();
    employee.setId(rs.getInt("EmployeeId"));
    employee.setName(rs.getString("EmployeeName"));
    lst.add(employee);
   }
  } catch (SQLException e) {
   logger.error("Error: Not get all employee " + e.getMessage());
   e.printStackTrace();
  } finally {
   DBPool.releaseConnection(conn, psmt, rs);
  }
  return lst;
 }

  public void insert(Employee employee) {
  String sql = "INSERT INTO employee(name) VALUES(?)";
  conn = DBPool.getConnection();
  try {
   psmt = conn.prepareStatement(sql);
   psmt.setString(1, employee.getName());
   psmt.executeUpdate();
  } catch (SQLException e) {
   logger.error("Error: Not insert employee! " + e.getMessage());
   e.printStackTrace();
  } finally {
   DBPool.releaseConnection(conn, psmt);
  }
 }
Cuối cùng cần import các thư viện sau vào:
import java.sql.CallableStatement;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Iterator;
import java.util.LinkedList;
Để connect tới được CSDL MySQL ta dùng gói mysql-connector-java_5.1.14.jar. còn nếu dùng maven thì có thể thêm doạn dependency sau:

 mysql
 mysql-connector-java
 5.1.14

Thanks & Rgds!

Thứ Hai, 29 tháng 7, 2013

Một số kỹ thuật thông dụng trong MySQL

1. Sử dụng Regx trong truy vấn
Tìm ra tất cả các nhân viên có lastName bắt đầu bằng chữ M, B hoặc T
SELECT lastname,firstname
FROM employees
WHERE lastname REGEXP  '^(M|B|T)'
2. Sử dụng LIMIT
Để lấy ngẫu nhiên một dòng:
SELECT * FROM table  ORDER BY RAND()  LIMIT 1
lấy ngẫu nhiên n dòng:
SELECT * FROM table  ORDER BY RAND()  LIMIT n
Cách trên giải quyết tốt trên những bảng dữ liệu có kích cở nhỏ. Với những bảng lớn thì nó sẽ cho kết quả chậm do phải sắp xếp lại toàn bộ bảng rồi sau đó mới tiến hành lấy ngẫu nhiên. Để giải quyết vấn đề này, chúng ta giải quyết như sau:
·  Trước tiên, lấy ngẫu nhiên ID của một cột. Cột này nên là khóa chính và giá trị tăng tuần tự.
·   Sau đó, lấy ra các dòng dữ liệu dựa trên ID mà chúng ta đã chọn
SET @ID = FLOOR(RAND( )* N) + 1;
 SELECT *  FROM table  WHERE ID >= @ID LIMIT 1
Phân trang
SELECT *
FROM table_name
LIMIT page, limit
page: trang hiện tại
limit: số dòng hiển thị trên một trang

Regular Expressions in Javascript

Tham khảo tại:
http://www.w3schools.com/jsref/jsref_obj_regexp.asp
http://net.tutsplus.com/tutorials/other/8-regular-expressions-you-should-know/
ví dụ:
yyyy-mm-dd:
/^(\d{4})-(\d{1,2})-(\d{1,2})$/
file đuôi .wav :
/(\.wav)$/
áp dụng:
re = /^(\d{4})-(\d{1,2})-(\d{1,2})$/;
if (frm.date.value != '' && !frm.date.value.match(re)) {
alert("Invalid date format: " + frm.date.value);
form.date.focus();
return false;
}

Thks & rgds!