• Dspace架设经验

    1.Dspace是什么?
          dspace.org
          The DSpace digital repository system captures, stores, indexes, preserves, and distributes digital research material.
          简单一点,Dspace是一个数字图书馆系统,一般用于研究机构储存和发布研究资源.

    2.Dspace的安装:
       windows
        wiki.dspace.org/index.php/DSpaceOnWindows
        linux:
        wiki.dspace.org/index.php/Installing_DSpace_on_Ubuntu_6.06_(LTS)

    3.Dspace架设的一些经验
         我当时使用的版本是dspace3.2
        中文问题
         Dspace会把每个item的标题和作者的首字母截取下来,做为按字母索引.但这对中文就不起作用了.所以我用pinyin4j这个开源项目,把标题和作者转换为拼音,然后截取首字母.
         邮件验证问题
         Dspace中只让用户设置邮件服务器,但并没让设置用户和密码,所以你必须用自己的邮件服务器.这个可以修改一下org.dspace.core.Email类的源码,加入验证.在代码Transport.send(message)处,加入以下代码:

    java 代码

    //jolestar add for mail auth
             String user = ConfigurationManager.getProperty("mail.user");
             String password = ConfigurationManager.getProperty("mail.password");
             Transport transport=session.getTransport("smtp");
             if(user==null||password==null)
             {
                 user = "";
                 password="";
             }
             transport.connect(server,user,password);
             transport.sendMessage(message,message.getAllRecipients());
             transport.close();
    
             //Transport.send(message); 
    

     并将上面那行代码注释了.在配置文件中就可以用mail.user和mail.password配置邮件用户了.不过这个问题在最新版的dspace中已经不存在了.

  • struts2 checkboxlist 的一个问题

    下面是java对象。

    package cn.jolestar.struts;
    
    /**
     * @author jolestar
     *
     */
    public class Language {
    
    	private Long id;
    	private String name;
    
    	/**
    	 *
    	 */
    	public Language() {
    	}
    
    	public Long getId() {
    		return id;
    	}
    
    	public void setId(Long id) {
    		this.id = id;
    	}
    
    	public String getName() {
    		return name;
    	}
    
    	public void setName(String name) {
    		this.name = name;
    	}
    
    	@Override
    	public int hashCode() {
    		final int prime = 31;
    		int result = 1;
    		result = prime * result   ((id == null) ? 0 : id.hashCode());
    		return result;
    	}
    
    	@Override
    	public boolean equals(Object obj) {
    		if (this == obj)
    			return true;
    		if (obj == null)
    			return false;
    		if (getClass() != obj.getClass())
    			return false;
    		final Language other = (Language) obj;
    		if (id == null) {
    			if (other.id != null)
    				return false;
    		} else if (!id.equals(other.id))
    			return false;
    		return true;
    	}
    
    }
    

     

     

     

    package cn.jolestar.struts;
    
    import java.util.List;
    
    /**
     * @author jolestar
     *
     */
    public class Album {
    
    	private Long id;
    	private String name;
    	private List languages;
    	/**
    	 *
    	 */
    	public Album() {
    	}
    	public List getLanguages() {
    		return languages;
    	}
    	public void setLanguages(List languages) {
    		this.languages = languages;
    	}
    
    }
    

     

     下面是jsp标签:

     

     

     

     

    action中有个getLanguages()方法,返回数据库中的所有语言。按道理我编辑Album的时候,checkboxlist中Album的已经拥有的语言会自动选中。但怎么弄,它就是不选中。没办法,看了一下struts2的相关源码,才明白了。

     

    struts2 checkboxlist 的freemarker模板:

     

    ${itemValue?html}
        
    
       
    

     

     if tag.contains(parameters.nameValue,
    itemKey)这个if语句决定该checkbox是否选中。tag.contains()最终调用的是
    org.apache.struts2.util.ContainUtil中的这个方法:

     

    public static boolean contains(Object obj1, Object obj2) {
            if ((obj1 == null) || (obj2 == null)) {
                //log.debug("obj1 or obj2 are null.");
                return false;
            }
    
            if (obj1 instanceof Map) {
                if (((Map) obj1).containsKey(obj2)) {
                    //log.debug("obj1 is a map and contains obj2");
                    return true;
                }
            } else if (obj1 instanceof Collection) {
                if (((Collection) obj1).contains(obj2) || ((Collection) obj1).contains(obj2.toString())) {
                    //log.debug("obj1 is a collection and contains obj2");
                    return true;
                }
            } else if (obj1.getClass().isArray()) {
                for (int i = 0; i < Array.getLength(obj1); i  ) {
                    Object value = null;
                    value = Array.get(obj1, i);
    
                    if (value.equals(obj2)) {
                        //log.debug("obj1 is an array and contains obj2");
                        return true;
                    }
                }
            } else if (obj1.toString().equals(obj2.toString())) {
                //log.debug("obj1 is an object and it's String representation equals obj2's String representation.");
                return true;
            } else if (obj1.equals(obj2)) {
                //log.debug("obj1 is an object and equals obj2");
                return true;
            }
    
            //log.debug("obj1 does not contain obj2: "   obj1   ", "   obj2);
            return false;
        }
    

     

     

    tag.contains(parameters.nameValue,
    itemKey)的参数parameters.nameValue是該标签的值,在这里就是album.languages,
    parameters.list
    是languages,itemKey是循环中的当前language的id。album.languages包含的是Language对象,而
    struts2却在其中查找itemKey–就是Language的id,当然就无法查找到,所以不能自动选中。

     

    我把checkboxlist的模板改了一下。

     

    ${itemValue?html}
    	
    
      &nbsp;
    

     就是将tag.contains(parameters.nameValue,
    itemKey)改成了tag.contains(parameters.nameValue,it),it就是当前循环的Language对象,在
    struts2的iterator 标签里设置

     

    .(注意:如果是最新版的struts2,应该用var 而不是id).

     

    这样就好了。需要注意的是language必须正确overwrite  equals和hashCode方法。

     

  • SUSE10.3的桌面

     

       虽然认识linux也有三年多了,但这三年多里,从windows到linux,不习惯,再换到windows,兴趣来了,再换过来。翻来覆去,折
    腾了好多遍。终于在半年前完全抛弃了windows,转移到linux下来。linux和windows是两种不同的思维方式,思维方式换了,别的就都习
    惯了。最近把我的笔记本的系统升级到suse10.3,贴几张桌面截图。

       我的笔记本是联想F40A,装上suse之后,除了声卡需要特别配置,别的都完全正常使用,包括无线网卡以及蓝牙。因为声卡用了一个微软的啥UAA技术,windows下安装声卡驱动都要先装那个UAA驱动。
    当然,那些鸡肋似的按钮是没办法用了。所以建议打算买本装linux的朋友们优先考虑预装linux的笔记本。

     

     

  • javascript获取网页元素的最终样式

        今天遇到个问题,用js获取网页元素的样式,如果直接用 document.getElementById(“idname”).style
    获取的话,只能获取该元素的专有样式,即在该元素上直接用style=“”指定的样式,而无法获取由class设置的和从上级元素继承来的样式,也就是浏
    览器生成dom后最终的样式,所以很头痛。网上搜索了一下,找到一个办法。

         
         
         div.special{
          background-color:red;
          height:50px;
          width:50px;
          margin:10px;
         }
        
        
        function getBackgroundColor()
        {
         var oDiv=document.getElementById("div1");
         alert(oDiv.currentStyle.backgroundColor||document.defaultView.getComputedStyle(oDiv,null).backgroundColor);
        }
        
        
        
        
        
          
        
          
    

     

    注意:oDiv.currentStyle.backgroundColor是IE使用的方法,document.defaultView.getComputedStyle(oDiv,null).backgroundColor是DOM使用的方法。

           内容来源:http://blog.tianya.cn/blogger/post_show.asp?BlogID=666817&PostID=7224874&idWriter=0&Key=0

  • 读巴别塔之犬

    去书店逛,本来是要找本计算机方面的书的,但出来后瞄了一眼热门书榜,一本薄薄的书吸引了我,主要是封面上那短短的几十字介绍:

    写道

    一个女人从苹果树上坠地身亡。
    是意外还是自杀?无人知晓。
    唯一的目击者,就是她心爱的狗。
    女人的丈夫是一位语言学家,
    哀伤又困惑的他,
    思念妻子却无从得知她真正的死因。
    他决定以自己毕生的研究,
    教这只狗开口说话,
    让它道出事情的真相…..

     

    于是买了本回来。

            基本是一口气读完的,开始的时候还想着女人的死因和这只狗到底会说话了么?但到后来这些都不重要了。

            作者并没有直接描写那个女人—露西,而是通过女人的丈夫,那个语言学家,也就是保罗的回忆,慢慢一点一点在大家的想象中勾勒出了这个女人,一个精灵般的女人,一个很难用语言形容的女人,一个不属于这个现实世界的女人。

            有人说这本小说是本爱情小说,我不同意,你不能因为小说描写了爱情就是爱情小说,这都是人们的帖标签习惯带来的。好的小说或者作品就是一个鲜活的生命,有很多种可能,有很多种表现,能给你带来无限的思考空间,很难用一个简单的标签来区分的。

          
    “其实书里想要表达的东西并不是这个,巴别塔象征了人和人之间难以融通的隔阂,主人公保罗所要面对的不是如何让他家的狗开口说话的难题,而是他和妻子间幸福生活下潜藏着的观念鸿沟。他们彼此相爱,但并不能彼此理解,对于生活习惯和爱好的争端从未停止。保罗深爱妻子露西,因为她带给他惊喜和快乐,但他忽略了在快乐表面下露西那颗伤痕累累的心。在断断续续讲述自己过去的时候,露西渴望抚慰,然而保罗没有给。在冲着像自己孩子一样的面具发火时,露西渴望理解,但保罗没有给。 ”——-摘自豆瓣()

           但我觉得并非保罗不理解露西,而是保罗不善于表达,虽然他是个语言学家,甚至想让狗说话。当保罗看到露西做的第一个面具时,那个面具带给保罗内心的震动,我想他理解了露西在用面具阐释的生命和死亡的关系,但当表述为语言时,两个人的理解就发生歧义了。当露西认为保罗不理解她,冲着面具发火并把面具扎碎时,我竟然能体会到保罗的那种心碎的感觉。我想他当时一定非常恨上帝,恨上帝造人的时候为什么只能让人用语言交流?而相对于人的内心,语言本身又是如此的苍白无力。

          每个人都是孤独的。

  • JavaFX尝鲜

         java6出来以后,其一大特色就是对脚本语言的支持。以前只是粗略看了一下,没感觉到有什么具体的用处。最近又对java的特有脚本语言javaFX了解一下,感觉挺有意思。

    **一.以下是几个相关网址: **
       1.sun javafx 官方网站:.www.sun.com/software/javafx/script/
       2.openjfx 网站:  openjfx.dev.java.net/
       3.JavaFX: First Steps – “Hello OnJava” App  www.oreillynet.com/onjava/blog/2007/05/javafx_first_steps_hello_onjav_1.html
       4.The JavaFX Script Programming Language Reference (Chinese version)

    hello world 式的程序,很简单,稍看一下文档就明白了。
    二.我比较喜欢这个语言的下面几个特征:
     
         这个语言融合了javascript,php,actionscript,sql 等许多脚本语言的特征。

        1.数组操作很有特色
             var nums = [0..3];
             ..  是范围表达式,上面的语句相当于  var nums == [0,1,2,3];
             数组支持类似于sql语句一样的查询表达式。
            var nums = select n*n from n in [1..100];
            结果便是从1到100每个数字的平方组成的数组。

         2.字符串
            javaFX中的字符串支持嵌入表达式

     

     

    var name = 'Joe';
     var s = "Hello {name}";
    System.out.println(s);
    

     

          程序在输出结果的时候会自动把字符串中的表达式 {} 算出来,替换成结果 。这段程序的输出结果是 : Hello Joe
           这个特点和php很相似。用在来做结果输出模板时很有用,不必用自己创造标签语法,输出的时候又得用正则分析,替换。

        3.类与对象
           对象声明可以用一种类似于json的语法进行。

     

    class Person {
     attribute name: String;
     attribute parent: Person inverse Person.children;
     attribute children: Person* inverse Person.parent;
     function getFamilyIncome(): Number;
     function getNumberOfChildren(): Number;
     operation marry(spouse: Person);
     }
    
    var chris = Person {
    name: "Chris"
    children:
    [Person {
    name: "Dee"
    },
    Person {
    name: "Candice"
    }]
    };
    

     

     

       4.很有特色的反射操作
          javaFX的反射很奇妙,我都在想,为什么java的反射也不做成这样。

      import java.lang.System;

         System.out.println(1.class.Name) // prints "Number"
         System.out.println("Hello".class.Name); // prints "String" 
    

     

    方便吧?
    **
    三.在web中的尝试。**
          看了说明,好像javaFX出现的目标是简化Swing的界面开发编程。一方面javaFX还没有一个好的所见即所得的ide支持,另外本人对web方面的编程比较感兴趣。所以进行了一个小实验。
         我的思路是用一个servlet拦截所有对fx的请求,根据requestURI,获得javafx文件的绝对路径,然后在servelt中调用java6的脚本引擎,解释执行。
     FXServlet.java

    import java.io.File;
     import java.io.FileInputStream;
     import java.io.IOException;
     import java.io.InputStreamReader;
     import javax.servlet.ServletException;
     import javax.servlet.http.HttpServlet;
     import javax.servlet.http.HttpServletRequest;
     import javax.servlet.http.HttpServletResponse;
    
    import javax.script.Bindings;
    import javax.script.ScriptContext;
    import javax.script.ScriptEngine;
    import javax.script.ScriptEngineManager;
    import javax.script.SimpleScriptContext;
    
    public class FXServlet extends HttpServlet {
    
    private String realPath;
    
    public void doGet(HttpServletRequest request, HttpServletResponse response)
    throws ServletException, IOException {
    
    this.doWork(request, response);
    }
    
    public void doPost(HttpServletRequest request, HttpServletResponse response)
    throws ServletException, IOException {
    
    this.doWork(request, response);
    }
    
    public void doWork(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException{
    System.out.println("begin ....");
     try {
    ClassLoader loader = Thread.currentThread().getContextClassLoader();
    ScriptEngineManager manager = new ScriptEngineManager(loader);
    ScriptEngine engine = manager.getEngineByExtension("fx");
    
    Bindings bindings = engine.createBindings();
    //将request和response两个对象传递给脚本引擎。
    bindings.put("REQUEST:javax.servlet.http.HttpServletRequest",request);
    bindings.put("RESPONSE:javax.servlet.http.HttpServletResponse",response);
    
    ScriptContext context = new SimpleScriptContext();
    context.setBindings(bindings, ScriptContext.GLOBAL_SCOPE);
    context.setBindings(bindings, ScriptContext.ENGINE_SCOPE);
    engine.setContext(context);
    String uri = request.getRequestURI();
    String contextPath = request.getContextPath();
    //将requestURI中的contextPath替换掉
    if(contextPath!=null&&contextPath.trim().length()>0){
    uri = uri.replaceAll(contextPath,"");
    }
    //根据requestURI取得fx文件的绝对地址我们把fx文件放在站点的 WEB-INF下的 jfx目录下。
    String filePath = this.realPath   "WEB-INF"  File.separator  "jfx"   uri;
    File jfxFile =new File(filePath);
    
    if(!jfxFile.exists()){
    response.sendError(404);
    }else{
    InputStreamReader reader = new InputStreamReader(new FileInputStream(jfxFile));
    engine.eval(reader);
    reader.close();
    }
    } catch (Exception e) {
    e.printStackTrace();
    }
    System.out.println("... end");
    }
    
    public void init() throws ServletException {
    this.realPath = this.getServletContext().getRealPath("/");
    }
    
    }
    

    web.xml

           FXServlet
           net.teamhot.jfxweb.FXServlet
        
    
        
          FXServlet
          *.fx
    

     

    Hello.fx

    import java.lang.System;
     import javax.servlet.http.HttpServlet;
     import javax.servlet.http.HttpServletRequest;
     import javax.servlet.http.HttpServletResponse;
     import java.io.PrintWriter;
    
     //接受从java中传递过来的参数
     var request:HttpServletRequest = REQUEST;
     var response:HttpServletResponse = RESPONSE;
    
    var name = request.getParameter("name");
    var out:PrintWriter = response.getWriter();
    
    var template = "Hello World!{if name ==null then 'guest' else name}";
    
    out.print(template);
    

     

    运行:http://localhost:8080/jfx/Hello.fx
    结果:Hello World!guest 运行:http://localhost:8080/jfx/Hello.fx?name=jolestar
    结果:Hello World!jolestar

    注意:本程序需要tomcat在jdk6环境下运行,如果还是提示找不到javax.script.ScriptEngine,请将jdk6 lib 下面的rt.jar中的javax.script包抽取出来,重新打包后放到 tomcat  common lib 下面。 这个程序没有考虑效率问题。其实这样的事情应该由容器完成,本人只是做了个小小的实验。

subscribe via RSS