環境的安裝和設定以及hello world
這方面的文章網絡上一搜一大堆。偶也不引用了。
偶的感覺是python的安裝和組件安裝亂七八糟。ruby的安裝和插件安裝感覺比較爽。其理念是學習linux的port和apt的包管理思路。
posted @ 2008-04-29 11:49 wanglin 閱讀(271) | 評論 (0) | 編輯 收藏
這方面的文章網絡上一搜一大堆。偶也不引用了。
偶的感覺是python的安裝和組件安裝亂七八糟。ruby的安裝和插件安裝感覺比較爽。其理念是學習linux的port和apt的包管理思路。
posted @ 2008-04-29 11:49 wanglin 閱讀(271) | 評論 (0) | 編輯 收藏
posted @ 2008-04-29 11:20 wanglin 閱讀(595) | 評論 (1) | 編輯 收藏
我曾是個技術粉絲
但是多年的開發經驗,使得我對技術的本質認識的越來越清楚。至少對企業軟件開發人員來說,純粹的技術coding是沒有多少價值的。如同建筑行業一樣,真正有價值的東西在設計階段已經完成了。
和傳統建筑行業開發不同,軟件開發行業不光是技術設計,還包括業務的設計。業務和技術摻雜在一起,構成了軟件開發的復雜性。
在業務上,在技術上,尤其是在技術和業務的鴻溝之間,存在了太多太多因素。使得我們本來對相對簡單的軟件開發不敢抱有那么大的樂觀。更何況真正一個成功的項目還需要市場,客戶等等各個方面。
作為一個軟件開發人員,真的應該放棄軟件自大的心態,客觀的去看待軟件開發技術在整個軟件開發工程中的位置和地位。以一種推動企業發展,推動項目發展和成功的心態和目的去看待整個項目。就明白了軟件開發的真正意義和任務。也就能更好的完成自己的工作,甚至可以改變項目的成敗。
所以成敗不由技術,成敗由你我的視野和努力。
posted @ 2008-04-28 15:04 wanglin 閱讀(223) | 評論 (0) | 編輯 收藏
最近公司項目經理派我研究工作流并考慮在項目中使用。很有一些心得。工作流應用我將之分為狹義工作流和廣義工作流。對狹義工作流而言,你可以將之理解為在工作流設計器里畫畫節點以及方向箭頭,設置好就節點數據,動作就差不多了。(具體可以參見jbpm的websale這個demo)。
廣義的工作流是對服務之間的整合。核心問題是業務節點和工作流節點之間的映射,以及業務數據和工作流數據之間的映射,和普通工作流一樣還有流程判斷等等服務。實現了這些,各個業務模塊之間的數據就可以通過服務,以定好的方式(進行方向控制和格式轉化)在各個節點之間流通,達到了服務整合的目的。
IBM為ESB定義了四個必備的功能:“路由器”——根據信息內容,在不同應用和服務之間進行信息傳輸和路由;“轉換器”——進行應用之間的通信協議轉換;“翻譯機”——進行應用之間的消息格式轉換;“收發室”——處理來自不同渠道的業務事件(同步傳輸,異步傳輸,發布/訂閱等方式)。
其中“路由器”和“收發室”都是針對服務的重用而設計的,而“轉換器”和“翻譯機”則專門用來解決異構的通信問題。
針對重用和異構這兩個難題,倪曉兵認為ESB提供了兩個核心的功能,服務的管理和數據的轉換。
我們DEC項目的目標就是建立一個全能服務倉庫(暫時我在DEC設計人員zy哪里得到的信息),而服務之間如何路由,如何轉換,語義的協調都沒有考慮,而后者卻是成敗的關鍵。
最關鍵的語義翻譯這一點,就現在的技術上來說還不能做到(需要很高的機器智能才能達到使得不同的系統的業務詞匯可以正確的映射,更何況是在所有的系統之間進行映射,同時應用在企業級的應用環境中)
也許真的有這樣的幻想,但是真的能夠做到這一步么?我深深的懷疑。就目前的技術手段,如果要達到數據映射的高度正確性,必須由人不同系統之間需要協調的數據進行語義確認方能進行有效的映射。
當考慮到還必須做到ESB系統對其接入的所有的服務數據的語義都這樣做時。我懷疑真的需要做到協調所有的服務么?
也許ESB的應用范圍就是在公司內部或者有限范圍內的整合目標明確的業務節點之間業務的整合。
posted @ 2008-04-11 17:11 wanglin 閱讀(685) | 評論 (1) | 編輯 收藏
ruby很火,ror很火。但凡一個東西火,我們要知道他火的原因。
因為他開發快,你看
rails project_name
#config db
rake db:create:all
rake db:mirage scoffled table_name [field_name:field_type,.....]
#編輯model
rake db:mirage
#編輯action和route
ruby script/server
然后一個應用程序就生成啦,這個過程大概就2、3分鐘;而且他熱部署,所寫即所得,語法超級強大,簡單幾句話就可以表達很復雜的邏輯,真正讓人把精力集中在業務邏輯上和頁面邏輯上(他的mirage真是太cool了,完美的體現了定義一次schame,到處使用的原則)
坦率的講,這些別的東西——包括java都可以做到~,為什么到現在java還是這么殺手呢(不是應用程序殺手,是程序員殺手,開發起來羅嗦到死。
既然ror出現了,所以我想jor也很快了,不過ruby使人愉快的是,它從不限制你,包括寫的更難懂——如果你真的覺得別人寫的你看不懂的話——幸運的是,它也沒有限制你寫的更簡單。
那就用ruby去快樂的編程吧
posted @ 2008-03-05 19:08 wanglin 閱讀(288) | 評論 (0) | 編輯 收藏
linux控制臺分辨率調節
2007年12月07日 上午 11:16 | 640x480 800x600 1024x768 1280x1024
-----+-----------------------------------------------------
256 | 257 259 261 263
32k | 272 275 278 281
64k | 273 276 279 282
16M| 274 277 280 283
VESA:
Colors (depth) 640x480 800x600 1024x768 1280x1024 1600x1200
------------------+-----------+-----------+------------+-------------+-------------
256 ( 8 bit) | 769 771 773 775 796
32,768 (15 bit)| 784 787 790 793 797
65,536 (16 bit)| 785 788 791 794 798
16.8M (24 bit) | 786 789 792 795 799
查上面的表,編輯/boot/grub/menu.lst
kernel /boot/vmlinuz-2.6.15-23-386 root=/dev/hdb10 ro quiet splash vga=791
這行最后補上vga=792
posted @ 2008-02-21 09:44 wanglin 閱讀(1620) | 評論 (3) | 編輯 收藏
posted @ 2008-01-22 22:46 wanglin 閱讀(214) | 評論 (0) | 編輯 收藏
posted @ 2007-11-16 08:51 wanglin 閱讀(304) | 評論 (1) | 編輯 收藏
posted @ 2007-10-17 13:31 wanglin 閱讀(652) | 評論 (0) | 編輯 收藏
posted @ 2007-10-06 18:28 wanglin 閱讀(292) | 評論 (0) | 編輯 收藏
posted @ 2007-09-19 14:10 wanglin 閱讀(696) | 評論 (2) | 編輯 收藏
posted @ 2007-09-10 11:05 wanglin 閱讀(119) | 評論 (0) | 編輯 收藏
posted @ 2007-08-25 11:41 wanglin 閱讀(181) | 評論 (0) | 編輯 收藏
posted @ 2007-08-19 18:16 wanglin 閱讀(151) | 評論 (0) | 編輯 收藏
public class ExcelTemplateUtil {
private static String CHARSET = "";
private static final String ROOT = "ROOT";
private static final String TITLE = "TITLE";
private static final String RECORD = "RECORD";
private static Map temp = new HashMap();
public static String generateListToTemplate(Object titleObj, List recordList, File templateFile)
{
readTemplateFile(templateFile);
ByteArrayOutputStream os = (ByteArrayOutputStream) builderExcelOutput(titleObj, recordList);
return removeXMLHeader(os);
}
public static void readTemplateFile(File file)
{
try {
Document templateDocument = new SAXReader().read(file);
Element root = templateDocument.getRootElement();
List trList = root.selectNodes("http://div/table/tr");
Element titleTemp = (Element) trList.get(0);
Element recordTemp = (Element) trList.get(1);
root.element("table").remove(titleTemp);
root.element("table").remove(recordTemp);
temp.put(TITLE, trList.get(0));
temp.put(RECORD, trList.get(1));
temp.put(ROOT, root);
} catch (DocumentException e) {
e.printStackTrace();
throw new RuntimeException("Parse xml file error, Cause:", e);
}
}
public static OutputStream builderExcelOutput(Object titleObj, List list)
{
ByteArrayOutputStream os = new ByteArrayOutputStream();
Element root = (Element) ((Element) temp.get(ROOT)).clone();
Document document = DocumentHelper.createDocument();
document.setRootElement(root);
Element tableEle = root.element("table");
tableEle.add(parseTitleElement(titleObj));
for (int i = 0; i < list.size(); i++) {
tableEle.add(parseRecordElement(list.get(i)));
}
try {
OutputFormat format = new OutputFormat("", true, "GB2312");
XMLWriter writer = new XMLWriter(os, format);
writer.write(document);
writer.flush();
writer.close();
os.close();
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
throw new RuntimeException("Parse outstream error, Cause:", e);
} catch (IOException e) {
e.printStackTrace();
throw new RuntimeException("Parse outstream error, Cause:", e);
}
return os;
}
public static Element parseTitleElement(Object titleObj)
{
Element titleEle = (Element) ((Element) temp.get(TITLE)).clone();
if (null == titleObj) return titleEle;
List tdList = titleEle.selectNodes("td");
Element td;
for (int i = 0; i < tdList.size(); i++) {
td = (Element) tdList.get(i);
fullField(td, titleObj);
}
return titleEle;
}
public static Element parseRecordElement(Object recordObj)
{
Element recordEle = (Element) ((Element) temp.get(RECORD)).clone();
List tdList = recordEle.selectNodes("td");
Element td;
for (int i = 0; i < tdList.size(); i++) {
td = (Element) tdList.get(i);
fullField(td, recordObj);
}
return recordEle;
}
public static void fullField(Element tdEle, Object obj)
{
Attribute att = tdEle.attribute("id");
if (null == att || null == att.getText() || 0 == att.getText().trim().length()) {
return;
}
String fieldName = att.getText();
if (null == fieldName || fieldName.trim().length() == 0) return;
Method[] objMethod = obj.getClass().getDeclaredMethods();
Object value;
for (int i = 0; i < objMethod.length; i++) {
if (("get" + (fieldName.trim())).equals(objMethod[i].getName())) {
try {
value = objMethod[i].invoke(obj, new Object[]{});
value = (null == value ? "" : value);
tdEle.setText(value.toString());
} catch (IllegalAccessException e) {
e.printStackTrace();
throw new RuntimeException(e);
} catch (InvocationTargetException e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
}
}
public static String removeXMLHeader(OutputStream os)
{
String xml = os.toString();
int position = xml.indexOf(">");
// xml = xml.substring(position+1,xml.length());
// position = xml.indexOf(">");
return xml.substring(position + 1, xml.length());
}
}
posted @ 2007-08-08 21:47 wanglin 閱讀(1348) | 評論 (3) | 編輯 收藏
異常爭論
異常有兩個模型:中止模型和繼續模型
中止模型認為異常不應該再回來,他做的是善后工作。而繼續模型保持異常時環境,希望再一次能運行成功。
Java采用的是前者(一般語言都是前者),而OS一般采用后者。
Java異常有三類:錯誤,運行時異常,檢查型異常。
官方的觀點是
第 39 條:最好為異常條件使用異常。也就是說,最好不為控制流使用異常。
第 40 條:為可恢復的條件使用檢查型異常,為編程錯誤使用運行時異常。
第 41 條:避免不必要的使用檢查型異常。
第 43 條:拋出與抽象相適應的異常。(使處理異常更直觀)
在異常的使用上,專家的觀點是很不一樣的
C#作者Anders根本就忽略檢查型異常。
Bruce Eckel,聲稱在使用 Java 語言多年后,他已經得出這樣的結論,認為檢查型異常是一個錯誤 —— 一個應該被聲明為失敗的試驗。
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
缺點1,代碼中包含了過多的catch,使得代碼不清晰
缺點2,有時候捕捉的異常沒有什么實際意義
缺點3,不夠清晰的錯誤指示。
缺點4,過深的異常層次。
缺點4,性能。
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
Eckel 提倡將所有的異常都作為非檢查型的,并且提供將檢查型異常轉變為非檢查型異常的一個方法,同時保留當異常從棧向上擴散時捕獲特定類型的異常的能力
Rod Johnson ,他采取一個不太激進的方法。他列舉了異常的多個類別,并且為每個類別確定一個策略。一些異常本質上是次要的返回代碼(它通常指示違反業務規則),而一些異常則是“發生某種可怕錯誤”(例如數據庫連接失?。┑淖兎N。Johnson 提倡對于第一種類別的異常(可選的返回代碼)使用檢查型異常,而對于后者使用運行時異常。在“發生某種可怕錯誤”的類別中,其動機是簡單地認識到沒有調用者能夠有效地處理該異常,因此它也可能以各種方式沿著棧向上擴散而對于中間代碼的影響保持最?。ú⑶易钚』惓Q蜎]的可能性)。
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
解決1:謹慎的拋出檢查型異常?;蛘吣阏J為,你可以處理它。否則,包裝為運行時異常。
解決2:如果遵守1,2不是問題
解決3:異常不跨層,否則必須捕捉或者包裝。
比如持久層丟出的SalException,你或者丟棄/處理/包裝(為運行時異常),或者重新包裝為業務層異常。保持JEE層的獨立和異常的清晰性。
包裝底層異常,保持異常鏈。
解決4:如果符合1,4也不是問題。再次強調,能捕捉就捕捉。
解決5:減少異常使用,減少層次。
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
在je里面,robin認為異常是流程控制的一部分——當然,考慮到性能問題,這個流程不應該是大概率流程——也就是異常流程
例如用戶登錄
Try{
用戶登錄(用戶名,密碼);
登錄成功;
}catch(沒有這個用戶異常 e){
錯誤提示界面;
}
Potian則認為,沒有用戶是正常業務邏輯的一部分
If(!用戶業務層.沒有這個用戶(用戶名))錯誤提示界面;
If(用戶業務層.檢驗密碼(用戶名,密碼))登錄成功;
else 登錄失?。?/p>
Potian認為不應該在一個業務中包含了過多的責任。
Ps:在servlet中,我喜歡僅僅簡單的在action中調用最好一個業務層方法就可以完成此action的任務。這意味著我的servlet非常瘦,可以比較容易的被替換。如果采用了potian的辦法,則意味著我要把業務層中的代碼前移到servlet中來,這模糊了業務層的責任。解決的辦法是回到老路子上來。
Ps:我還認為,沒有異常的業務方法表達能力太弱,異常給了他們更豐富的表達能力。這使得業務層可以更豐富的表達業務意義。避免將業務責任分散掉。
我認為在業務層中,恰恰要包含足夠的責任。不多也不要少(流程分支-2最好)。在別的層次中,要細致一點。
posted @ 2007-05-11 15:22 wanglin 閱讀(3658) | 評論 (1) | 編輯 收藏
在爵士主場被連扳2場,比分2:2平。
比分不重要,關鍵氣勢上被完全壓倒了。
當初頭又大中心win那兩場,也比較懸;輸掉的這2場卻比較爽快,這說明爵士是非常有韌性的球隊,恰巧mm都是比較軟的0
難道又是05?
那jeff可以考慮走人,火箭考慮重建。
可以看的出來,火箭當初請jeff很大一個原因就是在最好的教練(jeff,布朗,禪師)里面,他比較擅長執教中鋒。
可是輸掉這兩場,卻看的出來他有幾個比較致命的缺點:臨場戰術指揮能力差,戰術死板缺少變化,使用板凳或保守或激進——毫無章法
火箭輸掉的這兩場可以看的出來mm幾乎被對方研究透了,經常看到yao一轉身,人家手一伸把球給給拍掉,tm高位擋拆,人家阻夾tm(tm和yao的擋拆只擋不拆)。。。。。戰術被人家研究透了,節奏完全混亂,常??吹教嫜a們空位不進。這個時候教練應該做點什么,可是jeff做了什么??!
jeff做的非常好的地方,還是防守。我常??吹交鸺M攻的時候就tm底線卷切出來接球,這個時候yao提到45度附近一個擋拆,然后tm就得到空位突破或者跳投,如果突破得到協防的話,常常的看到tm分到空位三分。還有yao底線卡位拿球攻擊內線,迫使對方收縮防線,或投或傳??墒钱斶@兩招被對方破解以后,我看到火箭隊員在進攻的時候開始茫然無措了,不知道如何跑位,不知道如何配合了。——于此相反,一回到自己半場,就好像死魚放回水里一樣,防守起來卻是井井有條,活躍起來。
你永遠也別向光用防守來解決問題——因為籃球經常出現無法防守的局面(比如kb大嬸發飆,tm手感到來,這個時候你派誰,用什么戰術,使他陷入多么不合理的出手都不行),這個時候你要用進攻來回應他。
jeff沒有辦法。
你回家吧jeff,如果你證明自己只是這個樣子的話。不是我黑你,你太讓我們失望了,你在浪費yao和tm的生命。
posted @ 2007-04-29 13:56 wanglin 閱讀(174) | 評論 (0) | 編輯 收藏
public class FindCustomerModel extends YModel {
private String name;
private String id;
private Collection customers;
private CustomerServiceDelegate delegate = new CustomerServiceDelegate();
public void findCustomers() {
setCustomers(delegate.findCustomers(id, name));
notifyObservers("customers");
}
public void setCustomers(Collection customers) {
this.customers = customers;
}
public Collection getCustomers() {
return customers;
}
public void setId(String id) {
this.id = id;
}
public String getId() {
return id;
}
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
public class FindCustomerView extends YPanel {
private YLabel idLabel = new YLabel("Id");
private YLabel nameLabel = new YLabel ("Name");
private YTextField idField = new YTextField();
private YTextField nameField = new YTextField();
private YPanel criteriaPanel = new YPanel();
private YTable resultTable = new YTable();
private YButton findButton = new YButton("Find");
public FindCustomerView () {
addComponents();
setMVCNames();
}
private void setMVCNames() {
idField.getYProperty().put(YIComponent.MVC_NAME,"id");
nameField.getYProperty().put(YIComponent.MVC_NAME,"name");
resultTable.getYProperty().put(YIComponent.MVC_NAME,"customers");
findButton.getYProperty().put(YIComponent.MVC_NAME,"findButton");
YColumn[] columns = {
new YColumn("id"),
new YColumn("name")};
resultTable.setColumns(columns);
}
private void addComponents() {
this.setLayout(new BorderLayout());
this.add(criteriaPanel, BorderLayout.NORTH);
idField.setPreferredSize(new Dimension(100, 19));
nameField.setPreferredSize(new Dimension(100, 19));
criteriaPanel.add(idLabel);
criteriaPanel.add(idField);
criteriaPanel.add(nameLabel);
criteriaPanel.add(nameField);
criteriaPanel.add(findButton);
this.add(resultTable, BorderLayout.CENTER);
}
public YTextField getIdField() {
return idField;
}
public YLabel getIdLabel() {
return idLabel;
}
public YTextField getNameField() {
return nameField;
}
public YLabel getNameLabel() {
return nameLabel;
}
public YTable getResultTable() {
return resultTable;
}
public YButton getFindButton() {
return findButton;
}
}
public class FindCustomerController extends YController {
private FindCustomerView view = new FindCustomerView();
private FindCustomerModel model = new FindCustomerModel();
public FindCustomerController() {
super();
setUpMVC(model, view);
}
public void findButtonPressed() {
model.findCustomers();
}
}
public class YIntegerField extends JTextField implements YIModelComponent {
/** Gets value of this field for the model. */
public Object getModelValue() {
try {
return new Integer(getText());
} catch (Exception ex) {
return null;
}
}
/** Sets the model value into this field. */
public void setModelValue(Object obj) {
if (obj == null) {
setText("");
} else {
setText(obj.toString());
}
}
/** Notifies the framework when the component value might have changed. */
public void addViewListener(final YController controller) {
this.addFocusListener(new FocusAdapter() {
public void focusLost(FocusEvent ev) {
controller.updateModelAndController(YIntegerField.this);
}
});
}
// The rest is for the framework internal use,
// the implementation must be copied to each new component:
private YProperty myProperty = new YProperty();
public YProperty getYProperty() {
return myProperty;
}
}
posted @ 2007-04-27 16:14 wanglin 閱讀(278) | 評論 (0) | 編輯 收藏