庄周梦蝶

生活、程序、未来
   :: 首页 ::  ::  :: 聚合  :: 管理

读R4R,才读完第2章,前两章没什么新内容,算是复习吧

rails的lifecycle:
test3.jpg

从浏览器发起一个请求的基本流程:

A.server to dispatcher

服务器通过你指定的路径查找dispatcher,比如,你在apache配置:

<VirtualHost www.r4rmusic.com>
ServerName www.r4rmusic.com
ServerAlias r4rmusic.com
DocumentRoot "/usr/local/share/railsapps/r4rmusic/public/"
</VirtualHost>

服务器将在"/usr/local/share/railsapps/r4rmusic/public/下查找分发器,在rails应用下面有这么几个文件就是:dispatch.cgi dispatch.fcgi dispatch.rb

这几个文件中的某个将被调用

B.dispatcher to controller

分发器将解析发起请求的URL,按照:

1.controller的名称

2.action名称

3.请求id值

来寻找相应的controller的action,并执行.当然,你可以通过编辑config下面的routes.rb来定义自己的路由规则

C.performance of a controller action

当相应的controller以及相应的action被找到并且执行,他们将可以访问:

1.CGI数据,form提交的数据

2.controller的session信息,内建的session方法

本质上rails是把业务逻辑放在controller的action里面,也就是没有我们在J2EE中所谓的业务逻辑层.当业务逻辑或者说action较多时,你的controller文件不可避免的变的很长...在维护性和可读性也许会带来问题,当然,细分controller是更好的方法

posted @ 2007-02-06 13:37 dennis 阅读(443) | 评论 (0)编辑 收藏

来自railscn的讨论:

http://www.railscn.com/viewtopic.php?t=2289

 

1 . eval   " obj = #{str}.new "     # 只能在irb中使用
2 . obj  =   eval   " obj = #{str}.new "  
3 . obj  =   eval (str) . new
4 . =  Object . const_get( " Array "
  s 
=  c . new

 

相比于java的class.forName,与javascript等脚本语言一致,更为简单和轻量级 

posted @ 2007-02-06 13:35 dennis 阅读(261) | 评论 (0)编辑 收藏

刚学ruby时写的工具,我自己一直在用,放在桌面上,遇到不懂单词就得问它。发在railscn上,ranchgirl修改的更为ruby way。

 受blogjava上的帖子启发,实在是个有趣的主意,写了个ruby版本的,对俺这等英语一般的同学来说,google在线翻译经常使用呐,原贴之java版
http://www.blogjava.net/ekinglong/archive/2006/11/12/80704.html?Pending=true#Post

require 'net/http'
def translate
  txt
=STDIN.gets
  break 
if txt.strip=='e' or txt.strip=='exit'
  temp
=txt.split(' ')
  
if temp[1]=='1' or temp.size==1
    langpair
='en|zh-CN'
  
else
    langpair
='zh-CN|en'
  end
 
#使用代理  
  $proxy_addr = '127.0.0.1'
  
$proxy_port = 80

  response 
= Net::HTTP.Proxy($proxy_addr, $proxy_port).post_form(URI.parse("http://translate.google.com/translate_t"),{'text'=>temp[0],'langpair'=>langpair})

  response
.body =~ /<div id=result_box dir=ltr>(.*)<\/div>/
  result 
= $1 
  puts '翻译内容:'+temp[0]
  puts 
'google返回:'+result
  puts 
'-------------------退出请打e或者exit---------------'
  translate
end
translate


ranchgirl修改的版本,没有使用递归:
require 'net/http'
def usage
  
"usage: word [lang2 [lang1]]\n" +
  
"Translate word from lang1 (default en, English) to lang2 (default es, Spanish)\n" +
  
"ISO language code: http://www.unicode.org/unicode/onlinedat/languages.html" 
end
def translate
  arr 
= ARGV
  
if !arr[0] then puts usage; return end
  arr[
1= "es" unless arr[1]
  arr[
2= "en" unless arr[2
  langpair 
= "#{arr[2]}|#{arr[1]}"   
  response 
= Net::HTTP.post_form(URI.parse("http://translate.google.com/translate_t"),
                                 {
:text => arr[0], :langpair => langpair})
  response
.body =~ /<div id=result_box dir=ltr>(.*)<\/div>/
  result 
= $1 
  result 
= "No #{langpair} translation available for #{arr[0]}" if result.size == 0
  puts result
end
translate 

posted @ 2007-02-06 13:33 dennis 阅读(1637) | 评论 (3)编辑 收藏

 项目算完成了,等测试报告出来的时间里玩玩groovy,不玩不知道,一玩吓一跳,groovy与ruby好象啊。
看看下面的代码:
ruby:
[
1,2,3,4,5,6,7].inject(0){|sum,i| sum+=i}

groovy:

[
1,2,3,4,5,6,7].inject(0){sum,-> sum+=i}


ruby的closures与groovy的closures机制几乎完全一样,除了符号改变之外。有兴趣的话看看这个:
http://groovy.codehaus.org/Tutorial+2+-+Code+as+data%2C+or+closures

如果说groovy有什么优点的话,就是它可以直接利用java丰富的类库,实在爽:
//readFile.groovy
myFileName 
= "E://groovy//readFile.groovy"
myFile 
= new File( myFileName)
printFileLine 
= { println "File line: " + it }
myFile
.eachLine( printFileLine )


语法层面没什么好谈的,我更希望了解下groovy在实际应用中的表现,IBM的develper works上面有个实战groovy系列,值的好好读读

补充下,在spring中调用groovy beans的方法:
首先需要一个java接口,比如:

package com.sohu.blog.dennis_zane.springdemo.scripting;

public interface Calculator {
   
int add(int x,int y);
}

然后是Groovy bean
:
//GroovyCalculator.groovy
package com.sohu.blog.dennis_zane.springdemo.scripting;
class GroovyCalculator implements Calculator{
  
int add(int x, int y){
      x
+y    
  }
}

在spring配置文件中配置:

<lang:groovy id="calculator" script-source="classpath:/com/sohu/blog/dennis_zane/springdemo/scripting/GroovyCalculator.groovy"/>


调用的话跟使用一般的bean没什么不同:

Calculator calculator=(Calculator)ctx.getBean("calculator");
System.out.println(calculator.add(100, 20));

也许随着groovy性能的提升,将业务层完全由动态语言来实现不是没有可能。

 

posted @ 2007-02-06 13:11 dennis 阅读(481) | 评论 (0)编辑 收藏

ruby语言本身并没有提供abstract class和abstract method机制。这是ruby的spirit所决定的。但如果我们真的需要定义一个公共抽象类(或者抽象方法)来让子类来实现,又该如何做呢?
    我们可以通过在调用方法时抛出NotImplementedError来防止方法被调用。如(来自《ruby cookbook》的例子):

 class Shape2D
  def area
    raise NotImplementedError
.new("#{self.class.name}#area是抽象方法")
  end
end 
class Square 
< Shape2D
  def initialize(
length)
    
@length = length
  end
  
  def area
    
@length ** 2
  end
end


   父类
Shape2D的方法area就是我们所需要的“抽象方法”了。你不能直接调用:

s1=Shape2D.new
s1
.area

这样调用将抛出错误:Shape2D#area是抽象方法 (NotImplementedError)
Shape2D的子类Square覆写了此方法。由此我们模拟实现了抽象方法。那么抽象类该如何实现呢?自然而然,我们想到如果把类的initialize方法这样处理,那么这样的类将没办法被new生成,不正是我们所需要的抽象类?说干就干:

 class Shape2D
  def initialize
    raise NotImplementedError
.new("#{self.class.name}#area是抽象类")
  end
  def area
    raise NotImplementedError
.new("#{self.class.name}#area是抽象方法")
  end
end 


当你调用Shape2D.new时,解释器将提示你:Shape2D是抽象类(NotImplementedError)

我们已经实现了抽象方法和抽象类,感觉还是不够好,对每一个需要实现抽象类的类来说,我们都需要去写一句:raise NotImplementedError.new...实在不够爽。ruby鼓励我们去修改类的行为,甚至是标准库,那好,我们修改Class类吧,提供类似attr_reader的声明式服务:
class Class
  def  
    abstract(
*args)
    args
.each do |method_name|
      
      define_method(method_name) 
do |*args|
        
if method_name == :initialize   
          msg 
= "#{self.class.name}是抽象类"
        
else
          msg 
= "#{self.class.name}##{method_name}是抽象方法"
        end
        raise NotImplementedError
.new(msg)
        
      end
    end
  end
end


OK,如此一来,我们的Shape2D可以写成:
 class Shape2D
    abstract
:initialize,:area  #initialize和area是抽象方法
 end


尽管在ruby中,抽象类和抽象方法是否有用存在怀疑,不过知道这样的办法总是不错的主意

posted @ 2007-02-06 13:07 dennis 阅读(1447) | 评论 (1)编辑 收藏

rails1.1已经加入对json的全面支持,现在的Array,Hash,String,Object...等等都有一个to_json方法,生成 json字符串。反过来,我们该如何解析json呢?查了下http://www.json.org/上面ruby语言的链接,在rubyforge上找到了一个项目。解析json对ruby来说非常简单,只要一行代码:



例子:
json = '["a", "B", "C"]'
 
puts "Unsafe #{unsafe_json(json).inspect}"   #输出Unsafe ["a", "B", "C"]






把上面的json字符串解析成Array。这样的方法并不安全,比如:
json = 'puts "Danger Will Robinson"'
  puts "Unsafe #{unsafe_json(json).inspect}" 







又该输出什么呢?很遗憾,解析不出什么东西,跳出一个警告:
warning: character class has `[' without escape


安全的方法如下:
module SafeJSON
  require 
'monitor'
  def SafeJSON.build_safe_json
    ret 
= nil
    waiter 
= ''
    waiter.extend(MonitorMixin)
    wait_cond 
= waiter.new_cond
    
    Thread.start 
do
      $SAFE 
= 4
      ret 
= Proc.new {|json|
        eval(json.gsub(
/(["'])/s*:/s*(['"0-9tfn/[{])/){"#{$1}=>#{$2}"})
      }

      waiter.synchronize 
do
        wait_cond.signal
      end
    end
    waiter.synchronize 
do
      wait_cond.wait_while 
{ ret.nil? }
    end
    
return ret
  end
  @@parser 
= SafeJSON.build_safe_json
  
  # Safely parse the JSON input
  def SafeJSON.parse(input)
    @@parser.call(input)
  rescue SecurityError
    
return nil
  end
end






包含这个Module,你就可以这样使用:
peoples=SafeJSON.parse('{"peoples":[{"name":"site120","email":"site120@163.com","sex":"男"},{"name":"site120_2","email":"site120@163.com_2","sex":"男_2"}]}')

puts peoples[
"peoples"][1]["name"]  #输出site120_2







rails通过RJS内置了对AJAX的支持,也许用到json的机会并不多,不过作为一种数据交换的方便格式,还是值的注意。

posted @ 2007-02-06 13:04 dennis 阅读(4676) | 评论 (1)编辑 收藏

 虽然sohu不大可能倒闭,但是我也想导出我所有的文章,备份在自己的数据库中。javaeye的blog系统提供了一个blog导入导出的功能,用起来很酷。其实你自己用ruby写一下也很简单,比如我要读取我的blog的RSS源:

require 
'rss/2.0'
require 
'open-uri'
url 
= 'http://dennnis-zane.blog.sohu.com/rss'
feed 
= RSS::Parser.parse(open(url).read, false
puts 
"=== blog名称: #{feed.channel.title} ===" 
feed.items.each 
do |item| 
    puts item.title 
    puts 
" (#{item.link})"
    puts puts item.description 
end 

feed.items对应rss2.0的item元素,每个item元素描述了一篇文章的标题,链接,内容等信息,你可以通过item.title,item.link等来读取,并存入你自己的数据库。

再给一个使用代理的例子,因为我的机子是使用代理上网的:

require 'rss/2.0'
require 
'net/http'
url 
= 'http://dennnis-zane.blog.sohu.com/rss'
$proxy_addr 
= '172.16.51.10'
$proxy_port 
= 807

response 
= Net::HTTP.Proxy($proxy_addr, $proxy_port).get_response(URI.parse("http://dennnis-zane.blog.sohu.com/rss"))
feed 
= RSS::Parser.parse(response.body, false)
puts 
"=== Channel: #{Iconv.conv('GBK','UTF-8',feed.channel.title)} ==="
feed.items.each 
do |item|
  puts Iconv.conv(
'GBK','UTF-8',item.title)
  puts 
" (#{item.link})"
  puts
 # puts Iconv.conv(
'GBK','UTF-8',item.description)
end


打印:

=== Channel: 花非花 ===
websphere错误备忘录
 (http://dennnis-zane.blog.sohu.com/29898836.html)

感冒,寒冷的夜
 (http://dennnis-zane.blog.sohu.com/29859082.html)

ECMAScript对象基础
 (http://dennnis-zane.blog.sohu.com/29499101.html)

用ruby创建领域特定语言(DSL)——转载
 (http://dennnis-zane.blog.sohu.com/29350052.html)

使用ruby解析json
 (http://dennnis-zane.blog.sohu.com/29200192.html)

ruby实现抽象类和抽象方法
 (http://dennnis-zane.blog.sohu.com/29145303.html)

看了《父子》,看了《绿帽子》
 (http://dennnis-zane.blog.sohu.com/29047444.html)

ECMAScript基础
 (http://dennnis-zane.blog.sohu.com/28876856.html)

假期结束,专心工作
 (http://dennnis-zane.blog.sohu.com/28604494.html)

不解
 (http://dennnis-zane.blog.sohu.com/27564529.html)

posted @ 2007-02-06 12:57 dennis 阅读(299) | 评论 (0)编辑 收藏

呵呵,在《develope J2EE without EJB》中,DTO被狠很地批判了一把,rod说这完全是反模式。可实际项目当中,我们还是不得不在使用。VO,PO,一牵扯到概念总是多么复杂。。。把一个PO从头传到尾??从页面到数据库,一捅到底?NO,NO,万一你要显示给用户的是几个PO的结合怎么办?万一我们只是需要某几个属性组合在一起显示怎么办?一捅到底的策略是多么丑陋,而且你完全把你的数据库设计模型暴露给用户。所以我们需要一些map工具来转换,在这方面,过去我只知道有个 BeanUtils,不够灵活,而今天,接触了下dozer,啊,跟spring一样的理念!灵活多变,你想怎么映射,想怎么换都可以。看看它支持的转换类型:

• Primitive to Primitive Wrapper
• Primitive to Custom Wrapper
• Primitive Wrapper to Primitive Wrapper

• Primitive to Primitive
• Complex Type to Complex Type
• String to Primitive
• String to Primitive Wrapper
• String to Complex Type if the Complex Type contains a String constructor
• Each of these can be mapped to one another: java.util.Date, java.sql.Date, java.sql.Time,
java.sql.Timestamp, java.util.Calendar, java.util.GregorianCalendar
• String to any of the supported Date/Calendar Objects if an explicit date format mapping attribute is
specified.
• Objects containing a toString() method that produces a long representing time in (ms) to any
supported Date/Calendar object.

几乎我们能想到的,它都提供了方法来做到。而且dozer可以很容易地跟spring集成。下面举个简单例子:

定义一个Book对象:

package com.denny_blue.dozerdemo;
public class Book {
 private String name;
 private String author;
 
 
 public Book(){
  
 } 
 public void setAuthor(String author) {
  this.author = author;
 }

 public String getAuthor() {
  return (this.author);
 }
 public void setName(String name){
  this.name=name;
 } 
 public String getName(){
  return this.name;
 }
}
简单的,我们要实例化一个对象,然后clone此对象,注意,是clone!

package com.denny_blue.dozerdemo;
import net.sf.dozer.util.mapping.DozerBeanMapper;
import java.util.List;
import java.util.ArrayList;


public class MyFirstDozerDemo {
 public static void main(String args[]){
  Book book1=new Book();
  book1.setAuthor("dennis");
  book1.setName("dozer demo");
  DozerBeanMapper mapper=new DozerBeanMapper();
  Book book2=new Book();
  mapper.map(book1,book2);
  book2=(Book)mapper.map(book1,com.denny_blue.dozerdemo.Book.class);
  System.out.println("book2's name:"+book2.getName());
     }

OK,如此简单,我们把book1的属性完全复制给了book2,两者现在是完全独立的对象。可如果仅仅是这样,我们用BeanUtils不是也很容易办到? book2=(Book)BeanUtils.cloneBean(book1);可如果我要把book1映射给一个完全不同的类的对象怎么办?而且他们的属性名也不相同,怎么办?比如,一个CookBook类:

package com.denny_blue.dozerdemo;
public class CookBook {
 private String bookName;
 private String author;


 public CookBook(){}
 public String getBookName() {
  return (this.bookName);
 }

 public void setBookName(String bookName) {
  this.bookName = bookName;
 }

 public String getAuthor() {
  return (this.author);
 }

 public void setAuthor(String author) {
  this.author = author;
 }


}
它的bookName属性与Book的name属性名不一样,我们该如何复制?dozer通过xml文件的配置来灵活地达到这个目的。我们配置一个dozerBeanMapping.xml:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mappings PUBLIC "-//DOZER//DTD MAPPINGS//EN"
"http://dozer.sourceforge.net/dtd/dozerbeanmapping.dtd">
<mappings>
<configuration>
<stop-on-errors>false</stop-on-errors>
<date-format>MM/dd/yyyy HH:mm</date-format>
<wildcard>true</wildcard>
</configuration>
<mapping>
<class-a>com.denny_blue.dozerdemo.Book</class-a>
<class-b>com.denny_blue.dozerdemo.CookBook</class-b>
<field>
<a>name</a>
<b>bookName</b>
</field>
<field>
<a>author</a>
<b>author</b>
</field>
</mapping>
</mappings>

如上所示,<class-a>指定所要复制的源对象,<class-b>复制的目标对象,<a>源对象的属性名, <b>目标对象的属性名。wildcard默认为true,在此时默认对所有属性进行map,如果为false,则只对在xml文件中配置的属性进行map。此时的demo 看起来像这样:

package com.denny_blue.dozerdemo;
import net.sf.dozer.util.mapping.DozerBeanMapper;
import java.util.List;
import java.util.ArrayList;


public class MyFirstDozerDemo {
 public static void main(String args[]){
  Book book1=new Book();
  book1.setAuthor("dennis");
  book1.setName("dozer demo");
  DozerBeanMapper mapper=new DozerBeanMapper();
  book2=(Book)mapper.map(book1,com.denny_blue.dozerdemo.Book.class);
  CookBook cookBook=new CookBook();
  List myMappingFiles = new ArrayList();
  myMappingFiles.add("dozerBeanMapping.xml");
  mapper.setMappingFiles(myMappingFiles);
  cookBook=(CookBook)mapper.map(book1,CookBook.class);
  System.out.println("cookBook's name:"+   cookBook.getBookName()+"     cookBook's author:"+
                      cookBook.getAuthor());
 }
}

通过mapper.setMappingFiles()设置映射文件,可以添加多个配置文件,也可以把所有的映射写在一个配置文件里面。  更多复杂例子请见它自带的doc。

posted @ 2007-02-06 12:45 dennis 阅读(2508) | 评论 (0)编辑 收藏

最近重新再看<Inside JVM>,对JAVA编译成的字节码结构很感兴趣,希望找个工具能够对.class文件进行的解析和查看。没找到,倒发现javaassist可以对字节码进行操作和修改。此工具是JBOSS项目的一部分,JBOSS实现AOP的基础。呵呵,开眼界了,原来我们可以直接对字节码文件进行修改,哪怕不知道源文件(跟反编译完全不同)。一个简单例子:

import javassist.*;
class Hello {
    public void say() {
        System.out.println("Hello");
    }
}

public class Test {
    public static void main(String[] args) throws Exception {
        ClassPool cp = ClassPool.getDefault();
        CtClass cc = cp.get("Hello");
        CtMethod m = cc.getDeclaredMethod("say");
        m.setBody("{System.out.println(/"shit/");}");
        m.insertBefore("System.out.println(/"fuck/");");
        Class c = cc.toClass();
        Hello h = (Hello)c.newInstance();
        h.say();
    }
}

编译运行此文件,输出:

fuck

shit

我们在

 CtMethod m = cc.getDeclaredMethod("say");
  m.setBody("{System.out.println(/"shit/");}");

  m.insertBefore("System.out.println(/"fuck/");");

修改了say()方法,改成了

System.out.println("fuck");

System.out.println("shit");

这里的ClassPool是CtClass的容器,它读取class文件,并根据要求保存CtClass的结构以便日后使用,默认状态下是从当前的类装载器获得,当然你可以指定:

pool.insertClassPath("/usr/local/javalib");

当然,不仅仅是修改方法,你还可以新建一个class,利用makeClass()方法,如:

ClassPool pool = ClassPool.getDefault();
CtClass cc = pool.makeClass("Point");

还可以新增方法,下面是sample里的一个例子,同样的:

package sample;

import javassist.*;
import java.lang.reflect.*;

/*
   A very simple sample program

   This program overwrites sample/Test.class (the class file of this
   class itself) for adding a method g().  If the method g() is not
   defined in class Test, then this program adds a copy of
   f() to the class Test with name g().  Otherwise, this program does
   not modify sample/Test.class at all.

   To see the modified class definition, execute:

   % javap sample.Test

   after running this program.
*/
public class Test {
    public int f(int i) {
     i++;
    return i;
    }

    public static void main(String[] args) throws Exception {
 ClassPool pool = ClassPool.getDefault();

 CtClass cc = pool.get("sample.Test");
 Test test=new Test();
 Class c=test.getClass();
 Method []method=c.getDeclaredMethods();
 for(int i=0;i<method.length;i++){
  System.out.println(method[i]);
 }
 try {
     cc.getDeclaredMethod("g");
     System.out.println("g() is already defined in sample.Test.");
 }
 catch (NotFoundException e) {
     /* getDeclaredMethod() throws an exception if g()
      * is not defined in sample.Test.
      */
     CtMethod fMethod = cc.getDeclaredMethod("f");
     CtMethod gMethod = CtNewMethod.copy(fMethod, "g", cc, null);
     cc.addMethod(gMethod);
     cc.writeFile(); // update the class file
     System.out.println("g() was added.");
 }
    }
}
第一次运行时,因为Test里并没有g()方法,所以执行

 CtMethod fMethod = cc.getDeclaredMethod("f");
     CtMethod gMethod = CtNewMethod.copy(fMethod, "g", cc, null);  //把f方法复制给g
     cc.addMethod(gMethod);
     cc.writeFile(); //更新class文件

     System.out.println("g() was added.");
打印:g() was added

第2次运行时,因为以上步骤已经在class文件中增加了一个g方法,所以

 System.out.println("g() is already defined in sample.Test.");
打印:g() is already defined in sample.Test

 

Javassist不仅能修改你自己的class文件,而且可以同样修改JDK自带的类库(废话,类库也是人写的^_^)具体请看它的tutorial。

posted @ 2007-02-06 12:44 dennis 阅读(802) | 评论 (0)编辑 收藏

一般网站在处理用户上传图片时通常采用两种策略:一是直接把图片存入数据库中的Blob字段;二是数据库中只存储图片的在服务器上的路径信息 ,图片存放在分门别类的文件中,使用的时候从数据库读取路径信息到页面img元素即可.在此不讨论两种方案的优劣,我只是写了个hibernate的例子来实现第一种策略.例子很简单,t_user表主要两个字段,name和photo,其中photo字段类型为Blob.在此例中数据库我采用mysql, oracle的Blob字段比较特殊,你必须自定义类型,具体的请自行搜索,这方面的资料很多.

//User.java  

package com.denny_blue.hibernate;

import java.io.Serializable;
import java.sql.Blob;

public class User implements Serializable{
 private Integer id;
 private String name;
 private Blob photo;
 /**
  * @return the id
  */
 public User(){
 }
 public Integer getId() {
  return id;
 }
 /**
  * @param id the id to set
  */
 public void setId(Integer id) {
  this.id = id;
 }
 /**
  * @return the name
  */
 public String getName() {
  return name;
 }
 /**
  * @param name the name to set
  */
 public void setName(String name) {
  this.name = name;
 }
 /**
  * @return the photo
  */
 public Blob getPhoto() {
  return photo;
 }
 /**
  * @param photo the photo to set
  */
 public void setPhoto(Blob photo) {
  this.photo = photo;
 }
 
}

类User有3个属性,id,name,photo,相应的getter和setter方法以及一个无参构造函数.应该注意的是photo的类型java.sql.Blob

相应的user.hbm.xml应该如下:

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
 "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
 "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping
 package="com.denny_blue.hibernate">

 <class name="com.denny_blue.hibernate.User"
        table="t_user"
        dynamic-update="true"
        dynamic-insert="true"
        batch-size="3">
  <id name="id"
      column="id"
      type="java.lang.Integer">
   <generator class="native"/>
  </id>
  <property name="name" column="name" type="java.lang.String" lazy="true"/>
  <property name="photo" column="photo" type="java.sql.Blob"/>

 </class>

</hibernate-mapping>

对应的hibernate.cfg.xml配置文件,不再列出,请参照hibernate文档自行设定.

OK,做了这一步,我们写个测试类来进行单元测试:

package com.denny_blue.test;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.sql.Blob;

import org.hibernate.Hibernate;
import org.hibernate.HibernateException;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.hibernate.cfg.Configuration;

import com.denny_blue.hibernate.User;

import junit.framework.TestCase;

public class HibernateTest extends TestCase {
        private Session session;
 protected void setUp() throws Exception {
  try{
   Configuration config=new Configuration().configure();
   SessionFactory sf=config.buildSessionFactory();
   session=sf.openSession();
  }catch(HibernateException e){
   e.printStackTrace();
  }
 }

 protected void tearDown() throws Exception {
  try{
   session.close();
  }catch(HibernateException e){
   e.printStackTrace();
  }
 }
 
 public void testSave()throws FileNotFoundException,IOException{
  User user=new User();
  user.setName("jordan");
  FileInputStream in=new FileInputStream("C://test.gif");
  Blob photo=Hibernate.createBlob(in);
  user.setPhoto(photo);
  Transaction tx=null;
  try{
  tx=session.beginTransaction();
  session.saveOrUpdate(user);
  tx.commit();
  }catch(HibernateException e){
   if(tx!=null)
    tx.rollback();
   e.printStackTrace();
  }finally{
   in.close();
  }
 }
 public void testLoad()throws Exception{
  try{
   User user=(User)session.load(User.class, new Integer(1));
   Blob photo=user.getPhoto();
   InputStream in=photo.getBinaryStream();
   FileOutputStream out=new FileOutputStream("C://out//test2.gif");
   byte [] buf=new byte[1024];
   int len;
   while((len=in.read(buf))!=-1){
    out.write(buf, 0, len);
   }
   in.close();
   out.close();
  }catch(HibernateException e){
   e.printStackTrace();
  }
 }

}
我们读取C盘目录下的test.gif并存储到数据库中,然后再取出来写入C:/out目录,此时你可以查看下数据表中photo显示为blob,表示已经成功存入.值的注意的代码片段就是:

FileInputStream in=new FileInputStream("C://test.gif");
  Blob photo=Hibernate.createBlob(in);
我们这里是从磁盘中读取图片,实际应用中你可以利用上传组件得到图片的2进制数据流,并利用Hibernate.createBlob方法来构造相应的Blob对象.而取图片则使用

InputStream in=photo.getBinaryStream();

这只是个简单的测试类,如果我想从数据库中取出图片并现实在页面上该如何做呢?其实也很简单,我们先要写一个servlet,在它的service方法中取出图片,并"画"到指定页面上.

package com.easyjf.asp.action;

import java.io.InputStream;
import java.io.OutputStream;
import java.sql.Blob;

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

import org.hibernate.HibernateException;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;
import com.denny)blue.hibernate.User;


public class Test extends HttpServlet {

 /**
  * Destruction of the servlet. <br>
  */
 private Session session;
 public void destroy() {
  try{
   session.close();
  }catch(HibernateException e){
   e.printStackTrace();
  }
 }

 /**
  * Initialization of the servlet. <br>
  *
  * @throws ServletException if an error occure
  */
 public void init() throws ServletException {
  try{
   Configuration config=new Configuration().configure();
   SessionFactory sf=config.buildSessionFactory();
   session=sf.openSession();
  }catch(HibernateException e){
   e.printStackTrace();
  }
 }
    public void doGet(HttpServletRequest request,HttpServletResponse response)
    {
     try{
   User user=(User)session.load(User.class, new Integer(1));
   Blob photo=user.getPhoto();
   InputStream in=photo.getBinaryStream();
   OutputStream out=response.getOutputStream();
   byte [] buf=new byte[1024];
   int len;
   while((len=in.read(buf))!=-1){
    out.write(buf, 0, len);
   }
   in.close();
   out.close();
  }catch(Exception e){
   e.printStackTrace();
  }
    }

}

通过response.getOutputStream取得输出流,其他就与上段代码一致.servlet写好了,怎么在页面调用呢?那就更简单啦,直接在页面的img标签的src属性上调用该servlet即可,如:

<img id="test" src="/servlet/Test"/>

 

简单的例子,希望对初学者有帮助.

附记:如果不希望在servlet(或者action之类)中进行保存操作,希望在DAO进行此操作,那么InputStream的关闭问题可以通过hibernate的interceptor机制解决

posted @ 2007-02-06 12:43 dennis 阅读(482) | 评论 (0)编辑 收藏

仅列出标题
共56页: First 上一页 48 49 50 51 52 53 54 55 56 下一页