2007年4月15日
从上班开始没多久,就开始用Evernote,是portable 2.2版本。
好记性不如烂笔头。真的是做到了随手记,尤其是搜索功能简单易用,用来挖很久以前记下的tips,界面也很简洁,明了、够用。
备份也方便,反正换了电脑,文件夹直接拷贝,数据没丢。
就这样,一用多少年过去了。
如今,“印象笔记”在iphone应用推荐榜的笔记类里一直居高位,图标也早就不是我v2.2的老土模样,如果不是英文名字就叫evernote,我完全对不上号。
人家其实从v3开始就不是老土的本地应用了,都讲究“云”了。
虽然手头的v2.2完全满足我随手记一记的需求,但是,本着与时俱进和稍微保证一下数据的安全性,我还是下了个最近的v4.x的版本,结果发现还是要从v3开始才能导入v2的数据。嗯,看来是落后太久了。
有一点夸奖一下:免费账户的每月上传容量是有限的,但是对于我这种文本笔记,够传3w条的。
——嗯,其实,本来只是个小软件的更新而已,竟然跑到blogjava写上一篇,可能是因为伴随感吧,再有点儿时过境迁的唏嘘。
春节过后这几天干的活儿就没一件痛痛快快的?
第一天是发现编译环境有异动,我以为是代码有问题,折腾了1天多。。。
第二天是自己机器上的IE8抽疯弄得我以为是build有问题。。。
第三天从TimesTen转到Psql的各种毛病。。。
好吧。言归正传:
1)找一个跟表无关的通用的单纯的select语句作为测试语句不容易啊,
Dual是Oracle自带的表,其他DBS木有的。。不用Dual的话Oracle又不认。
2)Cause: org.postgresql.util.PSQLException: Method org.postgresql.jdbc2.Jdbc2PreparedStatement.setQueryTimeout(int) is not yet implemented.
这个错误,把sqlMapConfig.xml中在settings里配置的defaultStatementTimeout去掉就可以避免了。但是,这个配置对PSQL应该OK啊。。。待查
【续】——
defaultStatementTimeout 确实不是所有的driver都支持的。
但去掉的话,TimesTen又不干了:太囧了。。。
[TimesTen][TimesTen 7.0.5.9.0 CLIENT]Query Timeout must be numeric; if the Network Timeout
is non-zero, the Query Timeout must be greater than zero and less than the Network Timeout, else if the Network Timeout is zero, the Query Timeout must be greater than or equal to zero.
兜兜转转又遇到了这个之前遇到过的问题。这些driver就不能统一统一咩!
parseInt(String s, int radix)
Parses the string argument as a signed integer in the radix specified by the second argument.
今儿头回用,真是……反省ing
文档写到现在,猛然说不上
abbreviation 和 acronym 的区别。。。真是惭愧。。。
abbreviation:单词、语句的缩略形式,缩到短一点儿。
acronym:“首字母”缩写
之前都没注意过有这么个方法,否则判断日期格式就太不靠谱了。
又是晦涩的单词:setLenient(false),严格匹配日期格式
SimpleDateFormat df = new SimpleDateFormat("MMddyyyy");
/*
* With lenient parsing, the parser may use heuristics to interpret inputs that
* do not precisely match this object's format.
* With strict parsing, inputs must match this object's format.
*/
df.setLenient(false);
try {
Date a = df.parse("23232011");
System.out.print("date: " + a);
} catch (java.text.ParseException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
i18n,之前在J2EE里,自然用struts来做。但是Web Service里貌似应该不需要这部分,但是实际上,有些操作还是避免不了要支持多国语言。
查了一圈儿,好像WS-i18n还在working draft的阶段,而且好像已经好几年了……也没找到什么开源中间件的实现。
大意是在SOAP header中加上<i18n:international>相关的标签。
因为毕竟是Web Service,利用http header还是不太妥,还是要放在SOAP里面。
记录一下查到的链接,备用。
For a SOAP web service the language/country code should be part of the SOAP request and possibly the SOAP response. You can always base your solution on the above WS-I18N working draft.
Internationalization isn't as big a priority with SOAP web services as it is with web pages as web services are consumed by other applications - not humans. In most cases normalized "generic data" is exchanged which is displayed on the client using the client's locale. SOAP web services exchanging localized text are pretty much in the minority.
这回是更加深深地体会到IDE的重要性了!
想当初要不是eclipse,估计上手过程会很痛苦;
后来myeclipse的官方网站竟然封掉了;
现在,眼巴巴瞅着NetBeans美丽的GUI Builder,假如当年Swing那部分是用NetBeans开发的,或者NetBeans可以不要那个.form的文件直接解析代码,就好了啊!!!
唉……
繁琐的登陆过程让人抓狂,telnet到第一台机器输入用户名之前,点击SecureCRT里menu->Script->Start recording script,录下接下来的步骤,然后点击Stop recording script,就会提示保存脚本。以后登陆时,选择Run脚本就可以了。
引子只是个小问题,JTextArea多行文本,想设置文字的对齐方式为居中对齐,却发现死活没有合适有效的方法,又试了JTextField和JLabel,都不理想。搜了一溜够看到JTextPane的解决方法,其实真是有点儿大材小用了。JTextPane里可以设置style,对齐方式、字体字号、颜色等等,并且还可以设置style的作用范围,挺方便的。另外,想把文字区域背景设为透明的也是在API中找了一通,才发现敢情就是setOpacity,呃,对这个单词实在是太不敏感了。。。。
把在coderanch上搜的例子贴在这里吧,以防以后忘了:
import java.awt.*;
import javax.swing.*;
import javax.swing.text.*;
public class Styling
{
public Styling()
{
String text = "To refer to locations within the sequence, the " +
"coordinates used are the location between two " +
"characters.\nAs the diagram below shows, a location " +
"in a text document can be referred to as a position, " +
"or an offset. This position is zero-based.";
SimpleAttributeSet aSet = new SimpleAttributeSet();
StyleConstants.setForeground(aSet, Color.blue);
StyleConstants.setBackground(aSet, Color.orange);
StyleConstants.setFontFamily(aSet, "lucida bright italic");
StyleConstants.setFontSize(aSet, 18);
SimpleAttributeSet bSet = new SimpleAttributeSet();
StyleConstants.setAlignment(bSet, StyleConstants.ALIGN_CENTER);
StyleConstants.setUnderline(bSet, true);
StyleConstants.setFontFamily(bSet, "lucida typewriter bold");
StyleConstants.setFontSize(bSet, 24);
JTextPane textPane = new JTextPane();
textPane.setText(text);
StyledDocument doc = textPane.getStyledDocument();
doc.setCharacterAttributes(105, doc.getLength()-105, aSet, false);
doc.setParagraphAttributes(0, 104, bSet, false);
JFrame f = new JFrame();
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f.add(new JScrollPane(textPane));
f.setSize(400,400);
f.setLocation(200,200);
f.setVisible(true);
}
public static void main(String[] args)
{
new Styling();
}
}
之前用axis2生成代码的时候都没有在意过,一路用default configuration下来。
前一阵子发现以前的代码在生成时选择的data binding不是用默认的adb方式,而是用xmlbeans。
google了一番不同数据绑定的区别,好像也没什么结果,大意就是adb的最简单,但是有局限性;xmlbeans支持的比较全面,但是用起来有点儿复杂。不过,我也没有体会出来xmlbeans强大在哪里……只是知道生成的代码更多了,调用起来更绕弯子而已。。。
server端用xmlbeans生成,client端用adb生成,互通是没有问题的。
client端用xmlbeans生成时,需要把产生于resources里面的所有.class文件打成jar包加到client端的build path里,否则运行client加载类的时候就会报错了,好诡异,太不友好了。
类似这种错误:
ClassNotFoundException : Cannot load SchemaTypeSystem. Unable to load class with name schemaorg_apache_xmlbeans.system.s68C41DB812F52C975439BA10FE4FEE54.TypeSystemHolder. Make sure the generated binary files are on the classpath.
所幸是在官方网站上有说明:http://ws.apache.org/axis2/tools/1_2/CodegenToolReference.html
一个JNLP文件中的资源必须由同一个证书签名
当需要不同证书时,可以创建多个JNLP文件,并在主JNLP文件中引用之。
遇到过的问题:找不到子JNLP文件中的资源
解决方法:竟然把子JNLP文件放到比主JNLP文件所在目录低一级就可以了?!
JLabel.setText("<html>auto new line</html>");
JLabel.setText("<html>line<br>force new line</html>");
费了半天劲,敢情Firefox在3.0版本后又支持弹出modal window啦
window.showModalDialog
HttpClient.getHttpConnectionManager().getParams().setSoTimeout()
HttpClient.getHttpConnectionManager().getParams().setConnectionTimeout()
第一个针对连接建立后,但是没有收到response的超时时间,测试时可将server simulator收到request后等一段时间后再回response。
出错信息:
java.net.SocketTimeoutException: Read timed out
第二个针对连接建立的超时时间,测试时可将目的IP地址设为不存在的IP地址。
出错信息:
org.apache.commons.httpclient.ConnectTimeoutException: The host did not accept the connection within timeout of 8000 ms
at org.apache.commons.httpclient.protocol.ReflectionSocketFactory.createSocket(ReflectionSocketFactory.java:154)
默认值为0,如果不设置的话大概2分多钟才会得到异常
导入了某证书至truststore,建立SSL连接时遇到错误
javax.net.ssl.SSLKeyException: RSA premaster secret error
Caused by: java.security.InvalidKeyException: Illegal key size or default parameters
在windows上双击查看该证书时发现Public Key是4096bit,查了一番后发现Java默认好像不能处理这么长的key,必须要按照JCE的一个Unlimited Strength Jurisdiction Policy
http://java.sun.com/javase/downloads/index_jdk5.jsp里面就有下载,这样貌似就可以处理这种强度比较高的了。
打印数组的方法,小地方,却老忘记,来自FindBugs的提示:
The code invokes toString on an array, which will generate a fairly useless
result such as [C@16f0472. Consider using Arrays.toString to convert the array
into a readable String that gives the contents of the array.
Arrays提供一系列接收不同类型数组作为参数的toString方法
String[] array = new String[]{"a","b","c"};
System.out.println(Arrays.toString(array));
FindBugs,
http://findbugs.sourceforge.net/
发现代码中潜在bug的工具,有eclipse的插件,安装后右键单击java project name,点击Find Bugs,切换到FindBugs得perspective可以看到结果,速度还比较快,比之前用过的一个(虽然已经记不得名字了)快一些~具体的效果还是要逐个分析。
Bug categories:
- Correctness bug
- Probable bug - an apparent coding mistake
resulting in code that was probably not what the
developer intended. We strive for a low false positive rate.
- Bad Practice
-
Violations of recommended and essential
coding practice. Examples include hash code and equals
problems, cloneable idiom, dropped exceptions,
serializable problems, and misuse of finalize.
We strive to make this analysis accurate,
although some groups may
not care about some of the bad practices.
- Dodgy
-
Code that is confusing, anomalous, or
written in a way that leads itself to errors.
Examples include dead local stores, switch fall through,
unconfirmed casts, and redundant null check of value
known to be null.
More false positives accepted.
In previous versions of FindBugs, this category was known as Style.
Multithreaded correctness
Incorrect lazy initialization and update of static field
Calls Thread.sleep() with a lock held: better to use wait(lock)
Synchronization on interned String could deadlock: 最好不用字符串,以免重复
Performance
invokes inefficient new String() constructor
concatenates strings using + in a loop: better to use append StringBuffer
inner class usage
Server:
创建本地证书:
keytool -genkey -alias testserver-keyalg RSA -keystore keystore
其中alias自己起一个别名,keystore为证书库的文件路径
还可以加上-keysize 1024、2048、4096等来指定公钥的大小,由此导出的证书查看时可以看到公钥的大小是与之一致的。keysize越大genkey时耗费时间越长。
会要求键入一个密码,为这个证书库的访问密码
会要求填写一些信息,姓名、单位、地区之类
最后要求键入一个密码,为这条证书别名的密码
导出cert:
keytool -export -alias testserver-file testcert.cer -keystore keystore
这里的别名和文件名同上一步
Client:
直接导入cert:
Truststore文件中存储的是作为client,信任那些server的证书。所以需要将server提供的证书导入进来(当然可以导入n个),client才能信任。
keytool -import -alias testserver-file testcert.cer -keystore truststore
导入时需要输入密码,该密码应该是truststore文件的访问密码,密码正确才能修改其信息。
提示是否信任该证书信息,确认。
查看证书信息:
keytool –list –v –keystore truststore
如果导入过多个,则可以看到多条entry
删除某一个证书entry,通过指定别名来删:
keytool -delete -alias testserver-keystore truststore
上述是自签名的证书,证书链的长度只为1
真正商用时,需要找相关机构(例如verisign)认证通过才能成为有效的证书:
生成证书签名请求:
keytool -certreq -keyalg RSA -alias testserver -file certreq.csr -keystore keystore
之后能够收到一个证书文件,证书链信息包含了该机构的一些信息,然后再导入。
有两种方法建立HTTPS连接,
一种是,在java中可以设置相关的4个系统参数,指向相关的keystore,truststore,一旦设置,在运行时就不可改变了。除非一定要使用多个不同的证书库文件,否则可以把多个不同证书都导入到同一个证书库里,这样,设置系统参数为唯一的值也够用,而且也比较简单一些。
另一种是,自己继承相关接口实现自己的证书管理器,这样可以自定义相关行为,也可以load不同的证书库。
如果不设置password,就认为不检查文件完整性,也能通过。如果设置了password,但是不正确,则会遇到密码错误的异常;如果server证书没有导入到client得信任列表里,则会遇到找不到可信证书的异常。
一个反编译的好工具,600KB,简单好使又免费,比之前用过的好~
http://java.decompiler.free.fr/?q=jdgui
MySQL Cluster 不支持windows
之前的说法是max版支持集群,standard版不支持,但是官方下载点里已经看不出任何关于max,standard的标志。官方网站上又说5.0版本已经包含了MySQL Cluster,所以还是下了个5.0的试试,结果是可以的。
基本的安装:
linux下官方网站的安装说明还是挺明确的,一步一步做就是了。
遇到了一个问题与这篇文章一模一样:http://www.91linux.com/html/linux_pub/fedora/20071114/8195.html
按它的方法修改之后再次
cd /usr/local/mysql
bin/mysqld_safe --user=mysql &
Starting mysqld daemon with databases from /var/lib/mysql
看到上面这一行说明启动成功。
修改密码:
bin/mysqladmin -u root password yourpassword
允许远程访问:
bin/mysql -u root -p
grant all on *.* to root@"202.11.10.253" Identified by "dboomysql"
允许地址202.11.10.253上用root用户,密码dboomysql来连接mysql的所有数据库,付给所有权限。
实质上是修改了自带的database:mysql中user这一张表,与PostgreSQL需要修改配置文件相比还是方便一些。
另外就是MySQL的GUI客户端(MySQL Administrator)也比PostgreSQL的漂亮很多。
停止MySQL:
bin/mysqladmin -u root shutdown -p
安装完成后,按照这篇文章配置双机互备:
http://hi.baidu.com/%CA%AB%D5%B9/blog/item/3b99dbc48140f7ca38db49f7.html
但是这篇文章有笔误:
四、初始化集群
在Server1中
# ndbd --nodeid=3 --initial
在Server2中
# ndbd --nodeid=4 --iniitial
应该改为——
四、初始化集群
在Server1中
# ndbd --ndb_nodeid=3 --initial
在Server2中
# ndbd --ndb_nodeid=4 --initial
这样的集群配置两台机器都是一样的,所以有2个管理节点,2个SQL节点,2个数据节点。集群的验证:从任何一台机器登上去操作数据都能双写。
ndb_mgm -e show
可以看到集群节点的信息。
从官方网站上看到的例子都是用了多于3台的机器来做的集群,仅用两台来做的方法好像是大家特意这么做的。
尽管集群配置完毕,但是应用程序与数据库建立连接不能使用其中任何一台机器的真实IP,否则一旦该机器网络连接出问题还是不行,所以还是要给DB集群的机器配置虚拟IP以供外部访问。
遇到过的问题:
无法mysqld_safe启动,也找不到日志
解决方法:加启动参数以指定日志
mysqld_safe --log-error=/var/log/mysqld.log
可以查看到错误日志了。
[ERROR] Fatal error: Can't open and lock privilege tables: Table 'mysql.host' doesn't exist
没辙,google呗,好像是系统数据库mysql里host表的权限有问题什么的,而mysql这个库是之前安装时初始化过的。
解决方法:删除data,重新install_db
rm -rf /usr/local/mysql/data
cd /usr/local/mysql
scripts/mysql_install_db --user=mysql
chown -R root .
chown -R mysql data
chgrp -R mysql .
bin/mysqld_safe --log-error=/var/log/mysqld.log --ndb_nodeid=5 --user=mysql &
启动成功了。
btw, mysql官方网的forum好像不能“平板显示”,好不方便。
关闭Cluster,在管理节点所在的机器上:
/usr/local/mysql/ndb_mgm -e shutdown
关闭SQL节点的mysqld服务:
/usr/local/mysql/bin/mysqladmin -uroot shutdown
java默认的JVM内存占用:初始1MB,最大64MB
可以通过java -Xms256m -Xmx256m ClassName来调整。
xms和xmx的值最好相同,否则调整的效率可能不高。
如果程序运行中,发生内存溢出,那么不会以Exception的形式抛出,而是"Error"
如果没有catch(Error error),那么控制台将显示错误:Exception in thread "main" java.lang.OutOfMemoryError: Java heap space,而且后面的程序也不会执行。
虽然RMI已经不大推荐使用了,但是方法还是要记住的。
在产生.class文件的根目录,例如/bin
cd /bin
rmic com.test.RMIImpl
将在/bin/com/test目录下产生RMIImpl_Stub.class文件
cd /bin
jar cvf test.jar com/test/RMIIntf.class com/test/RMIImpl_Stub.class
将在/bin目录下产生test.jar文件,供RMI客户端调用
遇到经典的getParameter中文乱码的问题,网上已经有很多总结和讨论了。
在jsp里调用了*.do?param=name,当name本身的值为中文时,在action里获取时有乱码问题。
调用.do的jsp页面头已经设置过
<%@ page language="java" import="java.util.*" pageEncoding="gb2312"%>
即便在action里设置了
request.setCharacterEncoding
也没有效果。
最后的解决方法:
String str = request.getParameter("param");
改为
String str = new String(request.getParameter("param").getBytes("ISO-8859-1"),"GB2312");
windows下.exe安装版中,TOMCAT_HOME/conf下,是没有logging.properties这个文件的(在解压版中有)。安装版的logging.properties文件位于TOMCAT_HOME/common/classes下。
Windows->Preferences->General->Compare/Patch->Ignore white space
Team->Branch/Tag时,必须输入注释,否则即便看起来好像执行了操作,但是却没有真正创建分支在SVN server上。
其实还是挺糊涂的,org.apache.struts.action.ActionForm的Reset()方法,不过知道可以在该方法中set该Form的一些属性值,就能够在页面上显示为该对应控件的默认值,比较方便。
public void reset(ActionMapping mapping, HttpServletRequest request) {
//set default value for the first display
this.setType("0");
}
在eclipse中已经集成了ant,如果创建名为build.xml的文件,就会被识别为ant的文件。ShowView->Outline会有一些提示。
在eclipse中,右键build.xml,
选择:Run As Ant Build,会执行默认的任务
选择:Run As Ant Build...,在弹出框中,可以选择执行哪些任务.
当直接使用ant命令时,可以用-D来输入参数,
例如ant -buildfile package.xml -Dversion=1_0,指定了version的值。
more *.log | grep keywords
vi一个写有中文内容的文件,只能看到其unicode。more该文件就可以看到中文内容。
当keywords中包含空格时,需要用""将keywords扩起来。
Linux和windows下,文件中的换行符好像是不同的,当在windows下用UE打开一来自linux的文件时,好像会问你是不是要转换到DOS格式什么的。在eclipse中打开代码时,有时也会遇到类似的问题,尤其是.xml文件。
今天在看weka的网站时,无意看到在eclipse中有一个设置貌似是专门用来做这个的:
eclipse->Window -> Preferences -> General -> Editors 里面有一个"New text file line delimiter",可以在other中选择Unix。
7种武器
array/String/HashMap
regular expressions
debug
two programming languages
one IDE
SQL
software idea
小心仔细的使用静态变量,因为有的时候它表现的过于“静态”,而不符合习惯性的想法,容易引入不易发现的bug。
因为经常是在单次运行程序时还不会反映出来,只有在多次运行时才能显露效果。
windows命令行支持的ftp命令,下载东西时默认是ASCII编码,于是一个.war文件下载后的文件大小就不大对,部署时tomcat也不认识,报出invalid header的错误。
因此用这个命令下载时需要转换成二进制。而如果用现成的FTP工具下载就不会有问题。
以前天真的认为只要把两个tomcat的端口号修改成不同的就哦了呢,结果不行。
实际情况是(转自CSDN):
1.特别要注意:不要设置CATALINA_HOME
2.分别修改安装目录下的conf子目录中的server.xml文件:
a.修改http访问端口(默认为8080端口),将8080修改为tomcat不在使用的端口号。此处所设的端口号即是以后访问web时所用的端口号。
b.修改Shutdown端口(默认为8005端口),将8005修改为没有在使用的端口号,例如8055。
c.修改8009端口,将8009修改为没有在使用的端口号,例如8099
(注意:两个文件中对应的端口号要不一样)
3.依次启动两个tomcat。
另外,因为打开了axis2的SOAPMonitor,今天不知道怎么回事儿monitor页面无法正常显示了,tomcat启动时也报出了address in use:5001的错误。一开始以为是两个tomcat的问题,后来才意识到是SOAPMonitor的端口冲突了(也不知道是和谁冲突了。。。)反正改吧:
webapps/axis2/WEB-INF/web.xml
<init-param>
<param-name>SOAPMonitorPort</param-name>
<param-value>5011</param-value>
</init-param>
把原来的5001改成比如5011就哦了。
在web.xml中,会将HTTP的404错误导向一个错误页面:
<error-page>
<error-code>404</error-code>
<location>/errors.jsp</location>
</error-page>
但是今天发现随便访问一个不存在页面时,并没有如预期的一样出现自定义的错误页面。在网上查了一下,原来是IE对于内容非常少的自定义错误页面是不会显示出来的,而firefox就没有问题。
于是,把错误页面的内容多加了几行字,就可以了。