publicclassUserDao{privateDataSourcedataSource;publicUserDao(){}publicvoidsetDataSource(DataSourcedataSource){this.dataSource=dataSource;}publicvoidadd(Useruser)throwsSQLException{Connectionc=null;PreparedStatementps=null;try{c=dataSource.getConnection();ps=c.prepareStatement("insert into user values(?, ?, ?)");ps.setString(1,user.getId());ps.setString(2,user.getName());ps.setString(3,user.getPassword());ps.executeUpdate();}catch(SQLExceptione){throwe;}finally{if(ps!=null)try{ps.close();}catch(SQLExceptione){}if(c!=null)try{c.close();}catch(SQLExceptione){}}}publicvoiddeleteAll()throwsSQLException{Connectionc=null;PreparedStatementps=null;try{c=dataSource.getConnection();ps=c.prepareStatement("truncate table user");ps.executeUpdate();}catch(SQLExceptione){throwe;}finally{if(ps!=null)try{ps.close();}catch(SQLExceptione){}if(c!=null)try{c.close();}catch(SQLExceptione){}}}publicUserget(Stringid)throwsSQLException{Connectionc=null;PreparedStatementps=null;ResultSetrs=null;Useruser=null;try{c=dataSource.getConnection();ps=c.prepareStatement("select * from user where id = ?");ps.setString(1,id);rs=ps.executeQuery();if(rs.next()){user=newUser(rs.getString("id"),rs.getString("name"),rs.getString("password"));}returnuser;}catch(SQLExceptione){throwe;}finally{if(rs!=null)try{rs.close();}catch(SQLExceptione){}if(ps!=null)try{ps.close();}catch(SQLExceptione){}if(c!=null)try{c.close();}catch(SQLExceptione){}if(user==null)thrownewEmptyResultDataAccessException(1);}}publicintgetCount()throwsSQLException{Connectionc=null;PreparedStatementps=null;ResultSetrs=null;try{c=dataSource.getConnection();ps=c.prepareStatement("select count(id) from user");rs=ps.executeQuery();rs.next();returnrs.getInt(1);}catch(SQLExceptione){throwe;}finally{if(rs!=null)try{rs.close();}catch(SQLExceptione){}if(ps!=null)try{ps.close();}catch(SQLExceptione){}if(c!=null)try{c.close();}catch(SQLExceptione){}}}}
publicclassUserDao{privateDataSourcedataSource;privateJdbcContextjdbcContext;publicvoidsetDataSource(DataSourcedataSource){this.jdbcContext=newJdbcContext();this.jdbcContext.setDataSource(dataSource);this.dataSource=dataSource;}publicvoidadd(finalUseruser)throwsSQLException{jdbcContext.executeSql("insert into user values(?, ?, ?)",user.getId(),user.getName(),user.getPassword());}publicvoiddeleteAll()throwsSQLException{jdbcContext.executeSql("truncate table user");}...}
executeUpdate() 에서 반복되는 try/catch/finally 부분을 기본적인 탬플릿/콜백 패턴을 사용해 중복을 제거했다.
위의 코드는 일종의 전략 패턴이 적용된것 이라고 볼 수 있다. 복잡하지만 바뀌지 않는 일정한 패턴을 갖는 작업 흐름이 존재하고 그중 일부분만 자주 바꿔서 사용해야 하는 경우에 적합한 구조다. 전략 패턴의 기본 구조에 익명 내부 클래스를 활용한 방식이다. 이런 방식을 스프링에서는 탬플릿/콜백 패턴이라고 부른다.
전략패턴의 컨텍스트를 탬플릿이라 부르고,익명 내부 클래스로 만들어지는 오브젝트를 콜백이라고 부른다.
스프링의 JdbcTemplate
UserDao.java
publicclassUserDao{publicvoidsetDataSource(DataSourcedataSource){this.jdbcTemplate=newJdbcTemplate(dataSource);}privateJdbcTemplatejdbcTemplate;privateRowMapper<User>userMapper=newRowMapper<User>(){publicUsermapRow(ResultSetresultSet,inti)throwsSQLException{returnnewUser(resultSet.getString("id"),resultSet.getString("name"),resultSet.getString("password"));}};publicvoidadd(finalUseruser){jdbcTemplate.update("insert into user values(?, ?, ?)",user.getId(),user.getName(),user.getPassword());}publicvoiddeleteAll(){this.jdbcTemplate.update("truncate table user");}publicUserget(Stringid){returnthis.jdbcTemplate.queryForObject("select * from user where id = ?",newObject[]{id},this.userMapper);}publicList<User>getAll(){returnthis.jdbcTemplate.query("select * from user",this.userMapper);}publicintgetCount(){returnthis.jdbcTemplate.queryForObject("select count(id) from user",Integer.class);}}
Leave a comment