本例的學生信息添加進入數據庫的事務(可以提交事務,事務回滾,用本地線程完善)
主頁面index.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%><%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %><html> <head> <title>學生信息管理</title> </head> <body> <a href='<c:url value="/query"/>?cmd=query'>查看學生信息</a> <br><br> <!-- <a href="<c:url value='/StudServlet?cmd=save' />">學生信息添加</a> --> <h2>學生信息添加</h2> <form action='<c:url value="/query"/>?cmd=add' method="post"> 姓名:<input type="text" name="name"/><br><br> <fieldset style="border: solid;border-color: red;width: 250px;"> <legend>圖書1</legend> 書名:<input type="text" name="book"/><br><br> 價格:<input type="text" name="price"/> </fieldset> <br> <fieldset style="border: solid;border-color:green;width: 250px;"> <legend>圖書2</legend> 書名:<input type="text" name="book"/><br><br> 價格:<input type="text" name="price"/> </fieldset> <br><br> <input type="submit" value="提交"/><br><br> </form> </body></html>
獲取數據庫連接的工具ConnUtils5.java
package cn.hncu.utils;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;import java.sql.Connection;import java.sql.DriverManager;import java.util.ArrayList;import java.util.List;import java.util.Properties;public class ConnUtils5 { //本地線程管理對象,用於實現: 同一個線程獲取的連接是同一個private static ThreadLocal< Connection> t=new ThreadLocal<Connection>(); private final static List<Connection> pool=new ArrayList<Connection>(); private static int SIZE;//由資源文件讀取private ConnUtils5(){ } static{ Properties p=new Properties(); try { //下面這種方式在純Java項目中可以讀取到classpath下的資源文件,但無法讀取JavaEE項目的。因為Tomcat把系統的默認類加載器改了//p.load( ClassLoader.getSystemClassLoader().getSystemResourceAsStream("jdbc.properties"));// p.load(ClassLoader.getSystemResourceAsStream("jdbc.properties")); //讀取Web項目的classpath下的資源文件,用這個可以p.load(ConnUtils3.class.getClassLoader().getResourceAsStream("jdbc.properties")); String driver=p.getProperty("driver"); String url=p.getProperty("url"); String name=p.getProperty("username"); String pwd=p.getProperty("password"); String ssize=p.getProperty("size"); SIZE=Integer.parseInt(ssize); Class.forName(driver); for(int i=0;i<SIZE;i++){ final Connection con=DriverManager.getConnection(url,name,pwd); System.out.println("con=="+con); //更改conn.close()方法//用代理模式生成一個增強版的conn對象,把它的close()方法攔截更改掉Object nCon=Proxy.newProxyInstance( ConnUtils3.class.getClassLoader(), // conn.getClass().getInterfaces(), //後面這種方式不行,應該是驅動中的實現類和我們當前程序不在同一空間(類加載器不同) new Class[]{Connection.class}, new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { if(method.getName().equals("close")){ System.out.println("還回一個鏈接:"+(Connection)proxy); pool.add((Connection)proxy); return null; } return method.invoke(con, args); } }); pool.add((Connection)nCon); } } catch (Exception e) { e.printStackTrace(); } } public static synchronized Connection getConnection(){ //先從t中拿,如果有就拿出去,如果沒有再到池中拿且把該對象放到t中Connection con=t.get(); if(con==null){ if(pool.size()<=0){ System.out.println("池中連接沒有了..."); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } return getConnection(); } con=pool.remove(0); t.set(con);//放到t中} return con;//拿一個移一個}}資源文件jdbc.properties
##MySQLdriver=com.mysql.jdbc.Driverurl=jdbc:mysql://127.0.0.1:3306/hncu?useUnicode=true&characterEncoding=utf-8username=rootpassword=1234size=3##Oracle#driver=oracle.jdbc.driver.OracleDriver#url=jdbc:oracle:thin:@127.0.0.1:1521:orcl#username=scott#password=tiger
值對象
Stud.java
package cn.hncu.domain;import java.util.ArrayList;import java.util.List;/* * 一對多中的“一”方值對象的建法*/public class Stud { private String id; private String name; //※專為“多”方添加一個集合---體現多表中的“一對多關係” private List<Book> books=new ArrayList<Book>();//注意,該集合要在構造時或之前就new出來。 public String getId() { return id; } public void setId(String id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public List<Book> getBooks() { return books; } public void setBooks(List<Book> books) { this.books = books; } @Override public String toString() { return "id=" + id + "," + name + "," + books; }}Book.java
package cn.hncu.domain;/* * 一對多中的“多”方值對象的建法*/public class Book { private Integer id; //基本數據類型全部用包裝類的聲明,為以後使用框架做技術準備---包裝類能夠兼容框架(因為一般框架都會使用類反射) private String name; private Double price; //※專為“一”方添加一個對像類型的變量(注意,不用studid)---體現多表中的“一對多關係” private Stud s;//設置主人//private String studid;//★★不要這樣設public Integer getId() { return id; } public void setId(Integer id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public Double getPrice() { return price; } public void setPrice(Double price) { this.price = price; } public Stud getS() { return s; } public void setS(Stud s) { this.s = s; } /* * 多表關聯時的toString()方法要注意一個陷阱,就是一方輸出另一方,同時另一方又反過來輸出前一方,形成無窮遞歸! */ @Override public String toString() { return "id=" + id + "," + name + "," + price;//這裡不能輸出Stud對象,否則無窮遞歸}}stud層的servlet層QueryServlet.java
package cn.hncu.stud.servlet;import java.io.IOException;import java.util.List;import java.util.Map;import javax.servlet.ServletException;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import cn.hncu.domain.Book;import cn.hncu.domain.Stud;import cn.hncu.stud.service.IStudService;import cn.hncu.stud.service.StudServiceImpl;public class QueryServlet extends HttpServlet { //注入IStudService service=new StudServiceImpl(); public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String cmd=request.getParameter("cmd"); System.out.println("cmd:"+cmd); if("query".equals(cmd)){ query(request, response); }else if("add".equals(cmd)){ add(request, response); } } public void query(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { List<Map<String, String>> studs=service.query(); request.setAttribute("studs", studs); request.getRequestDispatcher("/jsps/show.jsp").forward(request, response); } public void add(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //1收集參數2組織參數(id字段留到dao中去補) String name[]=request.getParameterValues("name"); System.out.println(name[0]); Stud s=new Stud(); s.setName(name[0]); //圖書信息String books[]=request.getParameterValues("book"); //防護一下---價格的防護應該也要寫,這裡我們偷懶了if(books==null||books.length<=0){ return; } String prices[]=request.getParameterValues("price"); for(int i=0;i<books.length;i++){ Book b=new Book(); b.setName(books[i]); b.setPrice(Double.parseDouble(prices[i])); //※完成兩個值對象的“一對多”關係的數據封裝s.getBooks().add(b);//一方b.setS(s);//多方} //3調用service層try { service.save(s); } catch (Exception e) { //導向失敗頁面} }}stud層的service層接口:
package cn.hncu.stud.service;import java.util.List;import java.util.Map;import cn.hncu.domain.Stud;public interface IStudService { public List<Map<String, String>> query(); public void save(Stud stud);}實現類
package cn.hncu.stud.service;import java.sql.Connection;import java.sql.SQLException;import java.util.List;import java.util.Map;import cn.hncu.domain.Stud;import cn.hncu.stud.dao.BookDAO;import cn.hncu.stud.dao.BookJdbcDao;import cn.hncu.stud.dao.StudDAO;import cn.hncu.stud.dao.StudJdbcDAO;import cn.hncu.utils.ConnUtils3;/*我們以後開發時通常都要採用一個dao獨立操作一個表,系統中有幾個實體表就寫幾個dao, * 以後框架都是這麼幹的,我們也要這樣做,因為架構好! * * 採用事務的場合: * 1、如果只有一個dao,但要執行多條sql語句且涉及增刪改,則要開啟事務* 2、如果一個service調用多個dao,通常也要開啟事務。 */public class StudServiceImpl implements IStudService { //注入StudDAO dao_stud=new StudJdbcDAO(); BookDAO dao_book=new BookJdbcDao(); @Override public List<Map<String, String>> query() { return dao_stud.query(); } @Override public void save(Stud stud) { Connection con=null; try { con=ConnUtils3.getConnection(); System.out.println("拿到一個鏈接:"+con); con.setAutoCommit(false); dao_stud.save(stud); dao_book.save(stud.getBooks()); System.out.println("提交一個事務..."); con.commit(); } catch (Exception e) { try { System.out.println("回滾一個事務..."); con.rollback(); } catch (SQLException e1) { e1.printStackTrace(); } }finally{ try { con.setAutoCommit(true); con.close(); } catch (SQLException e) { e.printStackTrace(); } } }} stud層的dao層
Stud接口
package cn.hncu.stud.dao;import java.util.List;import java.util.Map;import cn.hncu.domain.Stud;public interface StudDAO { public List<Map<String, String>> query(); public void save(Stud stud) throws Exception;}Stud實現類
package cn.hncu.stud.dao;import java.sql.Connection;import java.sql.PreparedStatement;import java.sql.ResultSet;import java.sql.SQLException;import java.sql.Statement;import java.util.ArrayList;import java.util.HashMap;import java.util.List;import java.util.Map;import java.util.UUID;import cn.hncu.domain.Book;import cn.hncu.domain.Stud;import cn.hncu.utils.ConnUtils3;public class StudJdbcDAO implements StudDAO { @Override public List<Map<String, String>> query() { List<Map<String, String>> list=new ArrayList<Map<String,String>>(); //一個map就是一行數據, List<Map>就是整個數據表Connection con=null; try { con=ConnUtils3.getConnection(); Statement st=con.createStatement(); String sql="select * from stud"; ResultSet rs=st.executeQuery(sql); while(rs.next()){ Map<String,String> m=new HashMap<String, String>(); m.put("id", (String) rs.getObject(1)); m.put("name", (String) rs.getObject(2)); list.add(m); } rs.close(); st.close(); } catch (SQLException e) { e.printStackTrace(); }finally{ try { con.close(); } catch (SQLException e) { e.printStackTrace(); } } return list; } @Override public void save(Stud stud) throws Exception { Connection con=ConnUtils3.getConnection(); System.out.println("拿到一個鏈接:"+con); String sql="insert into stud values(?,?)"; String uuid=UUID.randomUUID().toString().replace("-", ""); PreparedStatement pst=con.prepareStatement(sql); stud.setId(uuid);//為了"多方"即book能夠拿到"一方"的id,專門補的pst.setString(1, uuid); pst.setString(2, stud.getName()); System.out.println("1:"+uuid+",2:"+stud.getName()); pst.executeUpdate();// con.close();//拿到同一個con,這裡就不需要關了}}Book接口
package cn.hncu.stud.dao;import java.util.List;import cn.hncu.domain.Book;public interface BookDAO { public void save(List<Book> books) throws Exception;}Book實現類
package cn.hncu.stud.dao;import java.sql.Connection;import java.sql.PreparedStatement;import java.util.List;import cn.hncu.domain.Book;import cn.hncu.utils.ConnUtils3;public class BookJdbcDao implements BookDAO { @Override public void save(List<Book> books) throws Exception { Connection con=ConnUtils3.getConnection(); System.out.println("拿到一個鏈接:"+con); String sql="insert into book(name,price,studid) values(?,?,?)"; PreparedStatement pst=con.prepareStatement(sql); for(Book b:books){ pst.setString(1, b.getName()); pst.setDouble(2, b.getPrice()); pst.setObject(3, "12132312");//異常(故意給一個不存在的外鍵字段,以測試事務回滾)--測事務回滾// pst.setObject(3, b.getS().getId()); System.out.println("1:"+b.getName()+",2:"+b.getPrice()+",3:"+b.getS().getId()); pst.addBatch();//添加到批處理} pst.executeBatch();//執行批處理// con.close();//這裡拿到同一個con,這裡不需要關}}顯示學生信息頁面jsps/show.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%><%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %><html> <head> <title>學生信息管理</title> </head> <body> <h2>學生信息</h2> <c:forEach items="${studs}" var="x"> ${x.id},${x.name}<br/> </c:forEach> </body></html>效果圖:
關於管理系統的更多內容請點擊《管理系統專題》進行學習
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。