亚洲色无码专区在线观看,性色av极品无码专区亚洲,亚洲一区综合在线播放http://www.tkk7.com/xiaosao/不懂我的人 , 離不了我 , 該了解了解我 !而懂我的人 , 更離不了我 , 因為他們愛我 。zh-cnSat, 10 May 2025 19:04:56 GMTSat, 10 May 2025 19:04:56 GMT60<轉>標題欄新消息提示效果http://www.tkk7.com/xiaosao/archive/2011/06/15/352379.htmlcAng^ErcAng^ErWed, 15 Jun 2011 10:24:00 GMThttp://www.tkk7.com/xiaosao/archive/2011/06/15/352379.htmlhttp://www.tkk7.com/xiaosao/comments/352379.htmlhttp://www.tkk7.com/xiaosao/archive/2011/06/15/352379.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/352379.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/352379.html標題欄新消息提示如圖:

公司的項目中用到了這個新消息提示的效果,主要用于提示用戶有新消息。具體實現代碼如下:

01 var newMessageRemind={
02 _step: 0,
03 _title: document.title,
04 _timer: null,
05 //顯示新消息提示
06 show:function(){
07 var temps = newMessageRemind._title.replace("【   】", "").replace("【新消息】", "");
08 newMessageRemind._timer = setTimeout(function() {
09 newMessageRemind.show();
10 //這里寫Cookie操作
11 newMessageRemind._step++;
12 if (newMessageRemind._step == 3) { newMessageRemind._step = 1 };
13 if (newMessageRemind._step == 1) { document.title = "【   】" + temps };
14 if (newMessageRemind._step == 2) { document.title = "【新消息】" + temps };
15 }, 800);
16 return [newMessageRemind._timer, newMessageRemind._title];
17 },
18 //取消新消息提示
19 clear: function(){
20 clearTimeout(newMessageRemind._timer );
21 document.title = newMessageRemind._title;
22 //這里寫Cookie操作
23 }
24   
25 };

調用顯示新消息提示:newMessageRemind.show();

調用取消新消息提示:newMessageRemind.clear();

查看demo:http://www.css88.com/demo/newMessageRemind/

另:單純的這個代碼會出現這么一個問題:
就是當你打開一個站點很多張頁面的時候,如過有新消息,那么所有頁面都會不停的閃,當你查看消息后其他頁面仍會提示。

我們公司是通過使用Cookie的方式解決的,當查看新消息后所有標題閃動的頁面將全部取消提示。

聲明: 本文采用 BY-NC-SA 協議進行授權 | WEB前端開發
轉載請注明轉自《標題欄新消息提示效果



cAng^Er 2011-06-15 18:24 發表評論
]]>
<轉>return閉包函數http://www.tkk7.com/xiaosao/archive/2011/06/15/352374.htmlcAng^ErcAng^ErWed, 15 Jun 2011 09:56:00 GMThttp://www.tkk7.com/xiaosao/archive/2011/06/15/352374.htmlhttp://www.tkk7.com/xiaosao/comments/352374.htmlhttp://www.tkk7.com/xiaosao/archive/2011/06/15/352374.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/352374.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/352374.html高手繞道!這跟閉包本身沒什么大的關系,也不知道怎么取標題,隨便湊了個數,望見諒!

今天一個剛學js的朋友給了我一段代碼問為什么方法不執行,代碼如下:

1 function  makefunc(x)  {
2         return function (){
3             return  x;
4         }
5     }
6 alert(makefunc(0));

其實不是不執行,只是朋友的意思這里alert出來的應該是“0”,而不是function (){return x;}。
不是腳本寫錯了,只是沒搞懂return,從當前函數退出,并從那個函數返回一個值。如果返回的是一個函數,那么返回的也是函數本身。
可以這樣修改上面的代碼,就是alert(makefunc(0)()):

1 function  makefunc(x)  {
2         return (function (){
3             return  x;
4         })();
5     }
6 alert(makefunc(0)());

如果要返回函數執行的結果那么首先要讓這個函數執行,例如:

1 function  makefunc(x)  {
2         return (function (){
3             return  x;
4         })();
5     }
6 alert(makefunc(0));

這里有一個匿名函數,

1 (function (){
2            return  x;
3        })();

在第一個括號內是匿名函數,第二個括號用于調用該匿名函數,您可以在第二個括號中傳入所需的參數。例如:

1 (function( x , y){
2     alert( x + y);
3 })(2 ,3 );

聲明: 本文采用 BY-NC-SA 協議進行授權 | WEB前端開發
轉載請注明轉自《return閉包函數



cAng^Er 2011-06-15 17:56 發表評論
]]>
<轉>頁面翻轉,讓你的頁面旋轉一下http://www.tkk7.com/xiaosao/archive/2011/06/10/352061.htmlcAng^ErcAng^ErFri, 10 Jun 2011 10:08:00 GMThttp://www.tkk7.com/xiaosao/archive/2011/06/10/352061.htmlhttp://www.tkk7.com/xiaosao/comments/352061.htmlhttp://www.tkk7.com/xiaosao/archive/2011/06/10/352061.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/352061.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/352061.htmln.styleSheet.cssText = r;這個寫法是亮點,動態載入過css,但從來都沒這么載入過,在公司的項目中試了下,不一樣的感覺.很搞怪.

原帖:http://www.webchina110.cn/?p=328
今天網上閑逛,看到克軍寫了一句話,惡搞一下。憑著這句話的吸引力,我點進去了。

哇,居然頁面倒過來了,頁頭和頁腳翻了一個根頭,其實我以前還在想,背景圖片是不是可以這樣呢,今天至少在頁面上看到了。

真好奇,立即啟動火狐看了一下,哈哈,找到原因了,馬上在試了一下IE,搞定,也OK,哈哈,如果不知道的同學們,我想你們也想知道這是怎么回事吧。

其實就是這個東東在做怪。。。

-moz-transform: rotate(180deg);

-webkit-transform: rotate(180deg);

filter: progid:DXImageTransform.Microsoft.BasicImage(rotation=2);

三行代碼,分別用來支持moz,webkit和IE的內核的瀏覽器。

我們看看克軍是怎么做的呢,他使用了JS,讓代碼立即執行的方式。。給頁面增加一個<sytle>和相應的樣式,并且為body增加相應的class.下面我將他的js代碼貼出來。嘿嘿!

;(function(){
var d = document, n = d.createElement('style'), r='.flip { -moz-transform: rotate(180deg);-webkit-transform: rotate(180deg);filter: progid:DXImageTransform.Microsoft.BasicImage(rotation=2); }';
n.type = 'text/css';
if(n.styleSheet)
{
 n.styleSheet.cssText = r;
}
else
{
 n.appendChild(d.createTextNode(r))
}
d.getElementsByTagName('head')[0].appendChild(n);
d.body.className += ' flip';
}

為了使用document方便,他把document傳給了變量d,建立了一個style標簽為變量n,把樣式的內容傳給
了變量r,克軍的命名都很簡潔。
在為這個n其實就是style標簽了type這個屬性。
下面是進行判斷頁面中style有不有屬性,如果沒有,直接將r,也就是樣式的內容放入n中。
如果有,就得使用建立文本并且追加的方式。
我不知道我的解釋對不對哈,反正大概意思就是這樣的,歡迎指正。
一切準備就緒以后,就將n添加到head中去,在將class增加到body上,這樣頁面一加載。。。。你的頁面就會被旋轉180度,當然你可以旋轉90度,10度,數字是可以調整的喲。

2010.09.30  今天在使用過程中,最后發現,原來ie只支持4個值,分別旋轉值可以是1,2,3或4。這些數字分別代表90,180,270,或360度旋轉。



cAng^Er 2011-06-10 18:08 發表評論
]]>
<轉>Web前端工程師技能列表http://www.tkk7.com/xiaosao/archive/2011/06/10/352060.htmlcAng^ErcAng^ErFri, 10 Jun 2011 10:04:00 GMThttp://www.tkk7.com/xiaosao/archive/2011/06/10/352060.htmlhttp://www.tkk7.com/xiaosao/comments/352060.htmlhttp://www.tkk7.com/xiaosao/archive/2011/06/10/352060.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/352060.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/352060.html要打造一流的Web產品開發團隊,在團隊成員基礎能力上一定要下功夫。對于Web前端產品開發來說,僅僅掌握Web1.0時代簡單的”網頁套接”是完全不夠的。我結合自己的團隊配備,特此羅列了Web前端產品工程師所涉及的技能列表如下:

通過許多實際項目,個人認為一個完備的前端產品開發團隊,必須擁有如下的人才配備,也希望大家補充:

  • 團隊全體成員達到所有技能中的a級標準
  • 團隊全體成員必須掌握兩項技能中的b級標準,并保證所有的b級標準在該團隊中有50%以上成員能達到
  • 團隊全體成員必須掌握一項技能中的c級標準,并保證所有的c級標準在該團隊中有25%以上成員能達到

具體技能描述:

  • 【必備】UserInterface
    1. PhotoShop/Fireworks Design
      a – 配合美工將草圖形成具體的符合WebPage的設計
      b – 有快速制作分層高品質PSD、PNG的能力
      c – 能迅速將PSD、PNG的內容構思成div+css或者table等HTML代碼
    2. Flash Design
      a – 基本動畫效果
      b – 復雜的交互體系設計,了解第三方swf輔助設計軟件
      c – 復雜的交互體系設計以及較強的對各類外埠資源(PNG、JPG、MP3、WAV等)的整合能力。精通部分第三方輔助設計軟件(AE、SwishMax、Swift3D等)
  • 【必備】Browser-side (Web Application)
    1. XHTML/CSS
      a – 基本的layout實現
      b – 嚴格跨平臺的layout實現以
      c – 優雅的HTML code,盡可能符合標準并有SEO的考慮因素。在任何平臺、瀏覽器下基本保持一致。不要求了解各種CSS的hacks,但要求知道遇到問題應該如何查閱資料以在第一時間內解決。能夠為JavaScript開發人員提供最好操作的DOM結構,讓JS開發人員在開發的時候認為”一切都已經準備就緒了”,而不是”捉襟見肘”。
    2. JavaScript/Ajax/DOM
      a – 基本的DOM操作,了解AJAX,可以實現數據通信
      b – 基本的DOM操作,能寫高效率的OOP代碼,以降低維護成本
      c – 基于需求,進行不同的開發,選擇合適的框架,做到代碼效率最高,用戶體驗最好,代碼下載量最小,并且可以在單獨甚至更多產品線中最大限度重用代碼
    3. Flash Developement
      a – 基于Timeline的ActionScript操作,能實現簡單交互
      b – 掌握a外,能實現數據層通信(與服務器以及本地SharedObject)
      c – 精通AS1-3,能根據需求進行各類RIA開發。無論是要求支持FlashPlayer8的,還是FlashPlayer9的,都能做到開發效率最高、靈活性最大(比如對HTML層的接口設計,等等)。
  • 【必備】Client-side (Desktop Application)
    1. Apollo
      a – 產品級的封裝,基本技術了解(如何打包、如何加入HTML和JavaScript等)
      b – 掌握a的同時,能利用Apollo的API獨立設計、開發OS的文件I/O功能。
      c – 掌握基本技能的同時,對”3D概念體系”有所認知。這里”3D”即:Design(設計)、Development(開發)、Deploy(產品部署)。能用Apollo
    2. Windows Presentation Foundation、WPF/E(Silverlight)
      (待定,歡迎補充)
  • 【增補】Server-side (修改:經考慮,這個技能不參與評級)
    本來列舉了”1、Server端簡單的技術、腳本”和”2、MediaServer(Red5)接口”作為”Web前端工程師技能列表“的一種(服務器、數據邏輯層技能的)評判標準。但似乎很多朋友對于前端工程師是否應該掌握Server端技能的必要性表示懷疑。確實,要掌握好上述的展現層技能不是意見容易的事情,而且前端工程師的確非常辛苦。但是,站在另一方面來說,辛苦的原因是什么,我不知道在你日夜奮戰div+CSS的時候思考過沒有。就我的經驗,前端的辛苦在于以下幾個方面:

     

    1. 重復勞動多,大量的div+css都是重復的,即便可以復制粘貼,但幾千行的div海洋中去尋找一個入口恐怕都非常痛苦
    2. 需求變更多,往往你折騰幾個小時終于把跨平臺問題解決好了,而且在IE6、7和Firefox下面都能顯示同樣的效果了,甚至連JavaScript交互都已經快搞定了。突然上面說需求要變。這無疑是莫大的痛苦。

    也許表面上看,這跟Server端技能無關,但我覺得有好的Server端的意識,一定會有所幫助(當然不可能解決所有的問題)。畢竟信息結構和數據庫是密切相關的,而Server是連接數據庫的唯一渠道(至少大多數B/S應用是如此)。掌握Server端的基本技能,對于同邏輯層開發人員設計接口是非常重要的。而且HTML表現層在開發時與數據的分離,也與Server端的各種模板技術有關。例如PHP中的Smarty模板(我曾經用的)、jsp的model2概念等等。HTML結構如何設計,如何讓HTML重用,甚至在HTML層進行OOP的開發(我現在在新產品線中設計的前端開發流程),都需要Server端的支持。最起碼,你要告訴php程序員你需要什么。如果你完全對PHP一無所知的話,那也無從談起了。
    此外,對于創業團隊,往往人手非常有限。為了讓運營成本降到最低,所有的技術人員都有義務對Server端技術有所了解。如果為了修改一個網頁的標題還要跑去喊PHP程序員連接Remote Server的話,那實在是增加了整個公司的運營成本。
    總結:我認為,可以不了解技術細節,但應該知道原理,最好能掌握一兩套設計思想(畢竟數據邏輯都在這里走,光看HTML和JavaScript,對人的見識還是有局限的,這種局限限制了我自己很久的時間),那將是一比寶貴的財富。

  • 【增補】Mobile-side(不參與評級)
    1. Flashlite
      (待定,歡迎補充)
    2. Java?
      (待定,歡迎補充)

看到很多朋友留言說前端工程師沒前途,我在想,同時掌握移動設備的技能是否也是拓展前途的一個必要性?這里再多說幾句,關于技術人員的前途,目前在國內確實得用”慘淡”來形容。浮躁的氛圍讓技術人才往往過早放棄了自己的技術生涯,而爾虞我詐的整體道德水平也讓單純的技術人員痛不欲生(我身邊太多了,恩,不說具體細節了,呵呵)。

作為一個技術人員,開發人員,在保持純粹地敬業心態(這是前提,這么沒有,啥也別談)外,更要學會如何保護自己,如何壯大自身,社會不會同情你,只有你自己才能保護你自己。



cAng^Er 2011-06-10 18:04 發表評論
]]>
<轉>webkit webApp 開發技術要點總結http://www.tkk7.com/xiaosao/archive/2011/06/09/351982.htmlcAng^ErcAng^ErThu, 09 Jun 2011 06:24:00 GMThttp://www.tkk7.com/xiaosao/archive/2011/06/09/351982.htmlhttp://www.tkk7.com/xiaosao/comments/351982.htmlhttp://www.tkk7.com/xiaosao/archive/2011/06/09/351982.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/351982.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/351982.html閱讀全文

cAng^Er 2011-06-09 14:24 發表評論
]]>
<轉>Jquery插件Thickbox的使用總結及自定義設置 http://www.tkk7.com/xiaosao/archive/2011/04/26/349052.htmlcAng^ErcAng^ErTue, 26 Apr 2011 08:59:00 GMThttp://www.tkk7.com/xiaosao/archive/2011/04/26/349052.htmlhttp://www.tkk7.com/xiaosao/comments/349052.htmlhttp://www.tkk7.com/xiaosao/archive/2011/04/26/349052.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/349052.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/349052.html

ThickBox運行需要的文件

官方下載:

Download thickbox.js or thickbox-compressed.js, ThickBox.css, and the loading graphic (loadingAnimation.gif) to your local machine (or cut and paste the code from the tabs). Along with these three files, a copy of the jQuery JavaScript library is needed. For this site, and ThickBox, I am using the compressed version of jQuery.

首先在 HTML 文件的 head中導入jquery.js 和thickbox.js文件,導入 thickbox.css 文件;并且jquery.js 文件放在前面:

<script src="../Scripts/jquery-latest.pack.js" mce_src="Scripts/jquery-latest.pack.js" type="text/javascript"></script>
<script src="../Scripts/thickbox.js" mce_src="Scripts/thickbox.js" type="text/javascript"></script>
<link href="../Styles/thickbox.css" mce_href="Styles/thickbox.css" rel="stylesheet" type="text/css" />  

最后你只要給元素添加 class=”thickbox” 屬性就可以開始用 thickbox

實現了一張圖片的彈出展示功能:

<a href="”bg.jpg”" mce_href="”bg.jpg”" class=”thickbox” ><img src="”bg.jpg”" mce_src="”bg.jpg”" alt=”圖片”/></a>
//只需要指定圖片的class為thickbox
彈出框使用方法:
<a href="Default.aspx?keepThis=true&TB_iframe=true&height=400&width=500" title="主頁" class="thickbox" </a>
<input onclick="<web.path:path/>/bannedUserList!unBannedUserList?height=400&width=800&inlineId=myOnPageContent" title="彈出層" class="thickbox" type="button" value="Ban Another" />
//內嵌內容
<input alt="#TB_inline?height=300&width=400&inlineId=myOnPageContent" title="標題" class="thickbox" type="button" value="Show" />
<a href="#TB_inline?height=155&width=300&inlineId=hiddenModalContent&modal=true" class="thickbox">顯示隱藏內容a>
//遮罩層
URL后面加?KeepThis=true&TB_iframe=true&height=400&width=600
參數字符串中加 modal=true
?KeepThis=true&TB_iframe=true&height=400&width=600&modal=true
這樣當關閉ThickBox時會調用ThickBox iframe (self.parent.tb_remove())內部的一個tb_remove()函數
所有其他參數字符都必須在TB_iframe 參數之前。URL中所有"TB" 之后的將被移除。
<a href="index.html?keepThis=true&TB_iframe=true&height=250&width=400" title="標題" class="thickbox">打開一個頁面</a>
<a href="index.html?keepThis=true&TB_iframe=true&height=300&width=500" title="標題" class="thickbox">打開一個頁面</a>
<a href="index.html?placeValuesBeforeTB_=savedValues&TB_iframe=true&height=200&width=300&modal=true" title="標題" class="thickbox">打開一個頁面</a>  

 

自定義設置:

1、彈出窗口(div)右上角的關閉按鈕為顯示為"close or esc key",而不是中文的; 如果想把它變成[X]或"關閉"應該怎么來辦呢?

將thickbox.js文件打開,查找關鍵字"or esc key",將其刪除,并將前面的close更改為[X]或"關閉",然后把文件另存為UTF-8格式,如果不保存為UTF-8的話,將會出現亂碼。
2、thickbox 彈出層的遮住層透明度修改

打開thickbox.css查找.TB_overlayBG 進行更改

.TB_overlayBG {
background-color:#000;
filter:alpha(opacity=75);
-moz-opacity: 0.75;
opacity: 0.75;
}

3、關閉層:如果我們需要自己添加一個關閉按鈕或者圖片可以使用:

onclick="self.parent.tb_remove();"  

4、關閉層刷新父頁面,修改關閉方法 :

function tb_remove() {
$("#TB_imageOff").unbind("click");
$("#TB_closeWindowButton").unbind("click");
$("#TB_window").fadeOut("fast",function(){$('#TB_window,#TB_overlay,#TB_HideSelect').trigger("unload").unbind().remove();});
$("#TB_load").remove();
if (typeof document.body.style.maxHeight == "undefined") {//if IE 6
$("body","html").css({height: "auto", width: "auto"});
$("html").css("overflow","");
}
document.onkeydown = "";
document.onkeyup = "";
//刷新父頁面,未指定
window.location.reload();
return false;
}

5、thickbox插件默認情況是點擊灰色的遮罩層就會關閉取消

把兩個$("#TB_overlay").click(tb_remove);去掉就可以取消掉

6、updatepanel回發后thickbox失效的解決方法

只需把以下代碼粘貼至頁面中就OK了。
<script type="text/javascript" language="javascript">
function pageLoad()
{
var isAsyncPostback = Sys.WebForms.PageRequestManager.getInstance().get_isInAsyncPostBack();
if (isAsyncPostback)
{
tb_init('a.thickbox, area.thickbox, input.thickbox');
}
}
</script>


cAng^Er 2011-04-26 16:59 發表評論
]]>
<轉>ThickBox 3.1參數詳解http://www.tkk7.com/xiaosao/archive/2011/04/26/349046.htmlcAng^ErcAng^ErTue, 26 Apr 2011 08:15:00 GMThttp://www.tkk7.com/xiaosao/archive/2011/04/26/349046.htmlhttp://www.tkk7.com/xiaosao/comments/349046.htmlhttp://www.tkk7.com/xiaosao/archive/2011/04/26/349046.html#Feedback2http://www.tkk7.com/xiaosao/comments/commentRss/349046.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/349046.html 前幾天寫了一篇關于ThickBox 3.1的文章:[ThickBox 3.1完美修正版下載],今天在使用這個東西的時候發現里面有許多參數沒有詳細解釋,今天抽空整理出來,現和大家分享一下:

先說幾個參數:
class="thickbox" 調用特效;
height 打開頁面的高度;
width 打開頁面的寬度;
title="Iframe(Auto Hide)" title的內容;
keepThis=true TB_iframe=true 這兩個參數沒搞明白什么意思,但通過框架來調用頁面時不可缺少;
#TB_inline 調用當前頁面的層;
inlineId 當前頁面層的ID;
modal=true 表示禁用title,去掉即可顯示title及可自動關閉;

1、調用框架,固定寬度和高度,超出顯示滾動條 Iframe(auto hide):

<a >OECSPACE</a>


2、打開頁面,固定寬度和高度,無滾動條,禁用title, fixed width, fixed height

<a href="boxs.html?keepThis=true&TB_iframe=true&height=100&width=220&modal=true" title="ThickBox 3.1:modal=true表示禁用title,去掉即可顯示title及可自動關閉" class="thickbox">Open iFrame Modal</a>


3、Ajax載入,自動寬度和高度,禁用title,頁面無法查看源代碼

<a href="box.html?height=350&width=350&modal=true" title="ThickBox 3.1:Ajax載入,頁面無法查看源代碼" class="thickbox">Example</a>


4、鏈接顯示隱藏的層,層存在于頁面中。

<a href="#TB_inline?height=200&width=300&inlineId=hiddenModalContent&modal=true" title="ThickBox 3.1:鏈接顯示隱藏層" class="thickbox">Show hidden modal content</a>

<div id="hiddenModalContent" style="display:none">
<p>ThickBox hidden modal content. Click to hide.</p>
<p style="text-align:center"><input type="submit" value=" O K " onclick="tb_remove()" /></p>
</div>


5、按鈕顯示隱藏的層,層存在于頁面中。

<input alt="#TB_inline?height=150&width=400&inlineId=myOnPageContent " title="ThickBox 3.1:按鈕顯示隱藏層" class="thickbox" type="button" value="Show" />

<div id="myOnPageContent" style="display:none">
<p>ThickBox hidden modal content.Auto Hide.</p>
</div>


6、單張圖片調用

<a href="images/plant1.jpg" title="plant" class="thickbox"><img src="images/plant1_t.jpg" alt="ThickBox 3.1" /></a>


7、多張圖片調用

<a href="images/plant1.jpg" title="plant1" class="thickbox" rel="gallery-plants"><img src="images/plant1_t.jpg" alt="ThickBox 3.1 1" /></a>
<a href="images/plant2.jpg" title="plant2" class="thickbox" rel="gallery-plants"><img src="images/plant2_t.jpg" alt="ThickBox 3.1 2" /></a>
<a href="images/plant3.jpg" title="plant3" class="thickbox" rel="gallery-plants"><img src="images/plant3_t.jpg" alt="ThickBox 3.1 3" /></a>


另外,如果不想點擊圖片關閉的話,找到thickbox.js,到128行,把下面的代碼:

$("#TB_window").append("<a href='' id='TB_ImageOff' title='Close'><img id='TB_Image' src='"+url+"' width='"+imageWidth+"' height='"+imageHeight+"' alt='"+caption+"'/></a>" + "<div id='TB_caption'>"+caption+"<div id='TB_secondLine'>" + TB_imageCount + TB_PrevHTML + TB_NextHTML + "</div></div><div id='TB_closeWindow'><a href='#' id='TB_closeWindowButton' title='Close'>close</a> or Esc Key</div>");     

改成:

$("#TB_window").append("<img id='TB_Image' src='"+url+"' width='"+imageWidth+"' height='"+imageHeight+"' alt='"+caption+"'/>" + "<div id='TB_caption'>"+caption+"<div id='TB_secondLine'>" + TB_imageCount + TB_PrevHTML + TB_NextHTML + "</div></div><div id='TB_closeWindow'><a href='#' id='TB_closeWindowButton' title='Close'>close</a> or Esc Key</div>");     

即可,說白了就是把鏈接去掉。

cAng^Er 2011-04-26 16:15 發表評論
]]>
<轉>如何求素數http://www.tkk7.com/xiaosao/archive/2008/05/28/203457.htmlcAng^ErcAng^ErWed, 28 May 2008 04:34:00 GMThttp://www.tkk7.com/xiaosao/archive/2008/05/28/203457.htmlhttp://www.tkk7.com/xiaosao/comments/203457.htmlhttp://www.tkk7.com/xiaosao/archive/2008/05/28/203457.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/203457.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/203457.html1。自然數是0,1,2……
2。素數是2,3,5……(不包括1的只能背1和它本身整除的自然數)

import java.util.Scanner;

public class Prime {

    //最基本的做法

    private int prime1(int num) {
        int i = 0, s = 0;
        label1: for (int n = 2; n <= num; n++) {
            for (int m = 2; m * m <= n; m++) {
                if (n % m == 0)
                    continue label1;
            }
            s++;
            i++;
            //System.out.println("第" + i + "個素數是:" + n);
        }
        return s;
    }

    //6N±1法

    private int prime2(int num){
        int i = 0, s = 0;
        for(int n = 2; n <=3; n ++){
            s++;
            i++;
            //System.out.println("第" + i + "個素數是:" + n);
        }
        label1: for(int n = 1; ; n++) {
            label2: for (int m = 0; m <= 1; m++) {
                int tmp = 2 * (3 * n + m) - 1;
                if (tmp > num)
                    break label1;
                for(int k = 2; k * k <= tmp; k++)
                    if (tmp % k == 0)
                        if (m == 0)
                            continue label2;
                        else
                            continue label1;
                s++;
                i++;
                //System.out.println("第" + i + "個素數是:" + tmp);
            }
        }
        return s;
    }

    public static void main(String args[]) {
        Scanner in = new Scanner(System.in);
        int num = in.nextInt();
        long start = System.currentTimeMillis();
        int sum = new Prime().prime1(num);
        long end = System.currentTimeMillis();
        System.out.println("方法一共" + sum + "個素數");
        System.out.println("用時:" + (end - start));
        start = System.currentTimeMillis();
        sum = new Prime().prime2(num);
        end = System.currentTimeMillis();
        System.out.println("方法二共" + sum + "個素數");
        System.out.println("用時:" + (end - start));
       
    }
}

輸入:1000000

運行結果:

方法一共78498個素數
用時:3434
方法二共78498個素數
用時:3453

(看來基本方法比6N±1法還要更快些,奇怪了,我的程序寫的沒什么問題阿)


【1】求10000以內的所有素數。
素數是除了1和它本身之外再不能被其他數整除的自然數。由于找不到一個通項公式來表示所有的素數,所以對于數學家來說, 素數一直是一個未解之謎。像著名的 哥德巴赫猜想、孿生素數猜想,幾百年來不知吸引了世界上多少優秀的數學家。盡管他們苦心鉆研,嘔心瀝血,但至今仍然未見分曉。
自從有了計算機之后,人們借助于計算機的威力,已經找到了2216091以內的所有素數。
求素數的方法有很多種,最簡單的方法是根據素數的定義來求。對于一個自然數N,用大于1小于N的各個自然數都去除一下N,如果都除不盡,則N為素數,否則N為合數。
但是,如果用素數定義的方法來編制計算機程序,它的效率一定是非常低的,其中有許多地方都值得改進。
第一,對于一個自然數N,只要能被一個非1非自身的數整除,它就肯定不是素數,所以不
必再用其他的數去除。
第二,對于N來說,只需用小于N的素數去除就可以了。例如,如果N能被15整除,實際
上就能被3和5整除,如果N不能被3和5整除,那么N也決不會被15整除。
第三,對于N來說,不必用從2到N一1的所有素數去除,只需用小于等于√N(根號N)的所有素數去除就可以了。這一點可以用反證法來證明:
如果N是合數,則一定存在大于1小于N的整數d1和d2,使得N=d1×d2。
如果d1和d2均大于√N,則有:N=d1×d2>√N×√N=N。
而這是不可能的,所以,d1和d2中必有一個小于或等于√N。
基于上述分析,設計算法如下:
(1)用2,3,5,7逐個試除N的方法求出100以內的所有素數。
(2)用100以內的所有素數逐個試除的方法求出10000以內的素數。
首先,將2,3,5,7分別存放在a[1]、a[2]、a[3]、a[4]中,以后每求出一個素數,只要不大于100,就依次存放在A數組中的一個單元 中。當我們求100—10000之間的素數時,可依次用a[1]-a[2]的素數去試除N,這個范圍內的素數可以不保存,直接打印。

【2】用篩法求素數。
簡單介紹一下厄拉多塞篩法。厄拉多塞是一位古希臘數學家,他在尋找素數時,采用了一種與眾不同的方法:先將2-N的各數寫在紙上:

在2的上面畫一個圓圈,然后劃去2的其他倍數;第一個既未畫圈又沒有被劃去的數是3,將它畫圈,再劃去3的其他倍數;現在既未畫圈又沒有被劃去的第一個數 是5,將它畫圈,并劃去5的其他倍數……依次類推,一直到所有小于或等于N的各數都畫了圈或劃去為止。這時,表中畫了圈的以及未劃去的那些數正好就是小于 N的素數。

這很像一面篩子,把滿足條件的數留下來,把不滿足條件的數篩掉。由于這種方法是厄拉多塞首先發明的,所以,后人就把這種方法稱作厄拉多塞篩法。
在計算機中,篩法可以用給數組單元置零的方法來實現。具體來說就是:首先開一個數組:a[i],i=1,2,3,…,同時,令所有的數組元素都等于下標 值,即a[i]=i,當i不是素數時,令a[i]=0 。當輸出結果時,只要判斷a[i]是否等于零即可,如果a[i]=0,則令i=i+1,檢查下一個a[i]。
篩法是計算機程序設計中常用的算法之一。

【3】用6N±1法求素數。
任何一個自然數,總可以表示成為如下的形式之一:
6N,6N+1,6N+2,6N+3,6N+4,6N+5 (N=0,1,2,…)
顯然,當N≥1時,6N,6N+2,6N+3,6N+4都不是素數,只有形如6N+1和6N+5的自然數有可能是素數。所以,除了2和3之外,所有的素數都可以表示成6N±1的形式(N為自然數)。
根據上述分析,我們可以構造另一面篩子,只對形如6 N±1的自然數進行篩選,這樣就可以大大減少篩選的次數,從而進一步提高程序的運行效率和速度。

在程序上,我們可以用一個二重循環實現這一點,外循環i按3的倍數遞增,內循環j為0-1的循環,則2(i+j)-1恰好就是形如6N±1的自然數。

http://www.tkk7.com/renyangok/archive/2006/11/20/82278.html



cAng^Er 2008-05-28 12:34 發表評論
]]>
上傳圖片并生成略縮圖 http://www.tkk7.com/xiaosao/archive/2008/04/23/195226.htmlcAng^ErcAng^ErWed, 23 Apr 2008 11:10:00 GMThttp://www.tkk7.com/xiaosao/archive/2008/04/23/195226.htmlhttp://www.tkk7.com/xiaosao/comments/195226.htmlhttp://www.tkk7.com/xiaosao/archive/2008/04/23/195226.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/195226.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/195226.html參數說明
FileBox:上傳表單中的表單名
SavePath:保存路徑
ThumbnailWidth:略縮圖寬
ThumbnailHeight:略縮圖高
FilePath:文件保存路徑(輸出參數)
FileName:文件名稱(輸出參數)
LastError:(輸出錯誤)
public static bool UploadImages(System.Web.UI.HtmlControls.HtmlInputFile FileBox,string SavePath,Int32 ThumbnailWidth,Int32 ThumbnailHeight,out String FilePath,out String FileName,out string LastError)
{
DateTime datTime=System.DateTime.Now;
String strTemp;
string strFileName = FileBox.PostedFile.FileName;
bool blnResult;
LastError="";
System.Drawing.Image xImage;
System.Drawing.Bitmap xBitmap;
int PhotoHeight,PhotoWidth;
string BasePath=System.Web.HttpContext.Current.Server.MapPath("../");
Rectangle NewPhoto;
System.Drawing.Imaging.ImageFormat xObject;
string Extension="";
FileName="";
FilePath="";

try
{
if(strFileName=="")
{
LastError="請點擊瀏覽選擇要上傳的文件!";
return false;
}
if(Convert.ToDecimal(FileBox.PostedFile.ContentLength)<= 2000000 &&Convert.ToDecimal(FileBox.PostedFile.ContentLength)>1500)
{
//查看擴展名是否合法
Extension=Path.GetExtension(strFileName);
Extension=Extension.ToLower();
if(Extension==".gif" || Extension==".jpg" || Extension==".jpeg" || Extension==".bmp")
{
switch(Extension)
{
case ".gif":
xObject=System.Drawing.Imaging.ImageFormat.Gif;
break;
case ".bmp":
xObject=System.Drawing.Imaging.ImageFormat.Bmp;
break;
default:
xObject=System.Drawing.Imaging.ImageFormat.Jpeg;

break;
}
}
else
{
LastError="上傳的文件不是有效的圖片文件,請上傳格式為:.gif.jpg.bmp的文件!";
return false;
}
//-----------------------------
//生成保存的文件路徑及文件名
//-----------------------------
strTemp = datTime.ToShortDateString().Replace( "-", "");
FileName = datTime.ToLongTimeString().Replace(":","")+ Extension;
FilePath = SavePath + strTemp + "/";
Friendlib.Function.CreateDirectory(BasePath + FilePath);
//保存大圖
xImage=System.Drawing.Bitmap.FromStream(FileBox.PostedFile.InputStream);

xImage.Save(BasePath+FilePath+FileName);
xImage.Dispose();
//-----------------------------
//開始生成縮略圖
//-----------------------------
xBitmap=new Bitmap(BasePath+FilePath+FileName);//------------------

PhotoHeight=xBitmap.Height;
PhotoWidth=xBitmap.Width;
// //判斷圖片的形狀
if(Convert.ToDecimal(PhotoHeight)/Convert.ToDecimal(PhotoWidth)>Convert.ToDecimal(ThumbnailHeight)/Convert.ToDecimal(ThumbnailWidth))//高有余
{
PhotoHeight=Convert.ToInt16((Convert.ToDecimal(ThumbnailHeight)/Convert.ToDecimal(ThumbnailWidth))*Convert.ToDecimal(PhotoWidth));
NewPhoto=new Rectangle(0,0,PhotoWidth,PhotoHeight);
}
else if(Convert.ToDecimal(PhotoHeight)/Convert.ToDecimal(PhotoWidth)<Convert.ToDecimal(ThumbnailHeight)/Convert.ToDecimal(ThumbnailWidth))
{
PhotoWidth=Convert.ToInt16((Convert.ToDecimal(ThumbnailWidth)/Convert.ToDecimal(ThumbnailHeight))*Convert.ToDecimal(PhotoHeight));
NewPhoto=new
Rectangle(Convert.ToInt16((Convert.ToDecimal(xBitmap.Width)-Convert.ToDecimal(PhotoWidth))/2),0,PhotoWidth,PhotoHeight);
}
else
{
NewPhoto= newRectangle(0,0,PhotoWidth,PhotoHeight); < BR> }
System.Drawing.Image myBitmap;
myBitmap=xBitmap.Clone(NewPhoto,System.Drawing.Imaging.PixelFormat.DontCare);
System.Drawing.Image.GetThumbnailImageAbort myCallback = new
System.Drawing.Image.GetThumbnailImageAbort(ThumbnailCallback);
System.Drawing.Image myThumbnail =
myBitmap.GetThumbnailImage(ThumbnailWidth,ThumbnailHeight, myCallback,
IntPtr.Zero); myThumbnail.Save(BasePath + FilePath +"s"+FileName,xObject);
myThumbnail.Dispose(); myBitmap.Dispose(); xBitmap.Dispose(); return true; }
else
{
blnResult=false; LastError="不能上傳大于2M及小于1.5K的圖片!";
}
}< BR> catch (Exception ex) {
throw new System.Exception((ex.Message + ("\r\n" + ex.StackTrace)));
}
return blnResult;
}
private static bool ThumbnailCallback() { return false; }
//http://www.tkk7.com/happyfish/archive/2005/05/11/4162.html 原文地址



cAng^Er 2008-04-23 19:10 發表評論
]]>
java面試筆試題大匯總 及c/c++面試試題(轉載) http://www.tkk7.com/xiaosao/archive/2008/04/23/195223.htmlcAng^ErcAng^ErWed, 23 Apr 2008 11:08:00 GMThttp://www.tkk7.com/xiaosao/archive/2008/04/23/195223.htmlhttp://www.tkk7.com/xiaosao/comments/195223.htmlhttp://www.tkk7.com/xiaosao/archive/2008/04/23/195223.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/195223.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/195223.html閱讀全文

cAng^Er 2008-04-23 19:08 發表評論
]]>
變態級JAVA程序員面試32問http://www.tkk7.com/xiaosao/archive/2008/04/23/195220.htmlcAng^ErcAng^ErWed, 23 Apr 2008 10:34:00 GMThttp://www.tkk7.com/xiaosao/archive/2008/04/23/195220.htmlhttp://www.tkk7.com/xiaosao/comments/195220.htmlhttp://www.tkk7.com/xiaosao/archive/2008/04/23/195220.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/195220.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/195220.html     第二,Anonymous Inner Class (匿名內部類) 是否可以extends(繼承)其它類,是否可以implements(實現)interface(接口)?

     第三,Static Nested Class 和 Inner Class的不同,說得越多越好(面試題有的很籠統)。

     第四,&和&&的區別。

     第五,HashMap和Hashtable的區別。

     第六,Collection 和 Collections的區別。

     第七,什么時候用assert。

     第八,GC是什么? 為什么要有GC?

     第九,String s = new String("xyz");創建了幾個String Object?

     第十,Math.round(11.5)等於多少? Math.round(-11.5)等於多少?

     第十一,short s1 = 1; s1 = s1 + 1;有什么錯? short s1 = 1; s1 += 1;有什么錯?

     第十二,sleep() 和 wait() 有什么區別?

     第十三,Java有沒有goto?

     第十四,數組有沒有length()這個方法? String有沒有length()這個方法?

     第十五,Overload和Override的區別。Overloaded的方法是否可以改變返回值的類型?

     第十六,Set里的元素是不能重復的,那么用什么方法來區分重復與否呢? 是用==還是equals()? 它們有何區別?

     第十七,給我一個你最常見到的runtime exception。

     第十八,error和exception有什么區別?

     第十九,List, Set, Map是否繼承自Collection接口?

     第二十,abstract class和interface有什么區別?

     第二十一,abstract的method是否可同時是static,是否可同時是native,是否可同時是synchronized?

     第二十二,接口是否可繼承接口? 抽象類是否可實現(implements)接口? 抽象類是否可繼承實體類(concrete class)?

     第二十三,啟動一個線程是用run()還是start()?

     第二十四,構造器Constructor是否可被override?

     第二十五,是否可以繼承String類?

     第二十六,當一個線程進入一個對象的一個synchronized方法后,其它線程是否可進入此對象的其它方法?

     第二十七,try {}里有一個return語句,那么緊跟在這個try后的finally {}里的code會不會被執行,什么時候被執行,在return前還是后?

     第二十八,編程題: 用最有效率的方法算出2乘以8等於幾?

     第二十九,兩個對象值相同(x.equals(y) == true),但卻可有不同的hash code,這句話對不對?

     第三十,當一個對象被當作參數傳遞到一個方法后,此方法可改變這個對象的屬性,并可返回變化后的結果,那么這里到底是值傳遞還是引用傳遞?

     第三十一,swtich是否能作用在byte上,是否能作用在long上,是否能作用在String上?

     第三十二,編程題: 寫一個Singleton出來。

以下是答案

第一,談談final, finally, finalize的區別。
final—修飾符(關鍵字)如果一個類被聲明為final,意味著它不能再派生出新的子類,不能作為父類被繼承。因此一個類不能既被聲明為 abstract的,又被聲明為final的。將變量或方法聲明為final,可以保證它們在使用中不被改變。被聲明為final的變量必須在聲明時給定初值,而在以后的引用中只能讀取,不可修改。被聲明為final的方法也同樣只能使用,不能重載
finally—再異常處理時提供 finally 塊來執行任何清除操作。如果拋出一個異常,那么相匹配的 catch 子句就會執行,然后控制就會進入 finally 塊(如果有的話)。
finalize—方法名。Java 技術允許使用 finalize() 方法在垃圾收集器將對象從內存中清除出去之前做必要的清理工作。這個方法是由垃圾收集器在確定這個對象沒有被引用時對這個對象調用的。它是在 Object 類中定義的,因此所有的類都繼承了它。子類覆蓋 finalize() 方法以整理系統資源或者執行其他清理工作。finalize() 方法是在垃圾收集器刪除對象之前對這個對象調用的。

     第二,Anonymous Inner Class (匿名內部類) 是否可以extends(繼承)其它類,是否可以implements(實現)interface(接口)?
匿名的內部類是沒有名字的內部類。不能extends(繼承) 其它類,但一個內部類可以作為一個接口,由另一個內部類實現。

     第三,Static Nested Class 和 Inner Class的不同,說得越多越好(面試題有的很籠統)。
Nested Class (一般是C++的說法),Inner Class (一般是JAVA的說法)。Java內部類與C++嵌套類最大的不同就在于是否有指向外部的引用上。具體可見http: //www.frontfree.net/articles/services/view.asp?id=704&page=1
注: 靜態內部類(Inner Class)意味著1創建一個static內部類的對象,不需要一個外部類對象,2不能從一個static內部類的一個對象訪問一個外部類對象

     第四,&和&&的區別。
&是位運算符。&&是布爾邏輯運算符。

     第五,HashMap和Hashtable的區別。
都屬于Map接口的類,實現了將惟一鍵映射到特定的值上。
HashMap 類沒有分類或者排序。它允許一個 null 鍵和多個 null 值。
Hashtable 類似于 HashMap,但是不允許 null 鍵和 null 值。它也比 HashMap 慢,因為它是同步的。

     第六,Collection 和 Collections的區別。
Collections是個java.util下的類,它包含有各種有關集合操作的靜態方法。
Collection是個java.util下的接口,它是各種集合結構的父接口。

第七,什么時候用assert。
斷言是一個包含布爾表達式的語句,在執行這個語句時假定該表達式為 true。如果表達式計算為 false,那么系統會報告一個 AssertionError。它用于調試目的:
  assert(a > 0); // throws an AssertionError if a < = 0
斷言可以有兩種形式:    
  assert Expression1 ;
  assert Expression1 : Expression2 ;
Expression1 應該總是產生一個布爾值。
Expression2 可以是得出一個值的任意表達式。這個值用于生成顯示更多調試信息的 String 消息。
斷言在默認情況下是禁用的。要在編譯時啟用斷言,需要使用 source 1.4 標記: 
  javac -source 1.4 Test.java
要在運行時啟用斷言,可使用 -enableassertions 或者 -ea 標記。
要在運行時選擇禁用斷言,可使用 -da 或者 -disableassertions 標記。
要系統類中啟用斷言,可使用 -esa 或者 -dsa 標記。還可以在包的基礎上啟用或者禁用斷言。
可以在預計正常情況下不會到達的任何位置上放置斷言。斷言可以用于驗證傳遞給私有方法的參數。不過,斷言不應該用于驗證傳遞給公有方法的參數,因為不管是否啟用了斷言,公有方法都必須檢查其參數。不過,既可以在公有方法中,也可以在非公有方法中利用斷言測試后置條件。另外,斷言不應該以任何方式改變程序的狀態。

    
第八,GC是什么? 為什么要有GC? (基礎)。
GC是垃圾收集器。Java 程序員不用擔心內存管理,因為垃圾收集器會自動進行管理。要請求垃圾收集,可以調用下面的方法之一:
  System.gc()
  Runtime.getRuntime().gc()

     第九,String s = new String("xyz");創建了幾個String Object?
兩個對象,一個是“xyx”,一個是指向“xyx”的引用對象s。

     第十,Math.round(11.5)等於多少? Math.round(-11.5)等於多少?
Math.round(11.5)返回(long)12,Math.round(-11.5)返回(long)-11;

     第十一,short s1 = 1; s1 = s1 + 1;有什么錯? short s1 = 1; s1 += 1;有什么錯?
short s1 = 1; s1 = s1 + 1;有錯,s1是short型,s1+1是int型,不能顯式轉化為short型??尚薷臑閟1 =(short)(s1 + 1) 。short s1 = 1; s1 += 1正確。

     第十二,sleep() 和 wait() 有什么區別? 搞線程的最愛
sleep()方法是使線程停止一段時間的方法。在sleep 時間間隔期滿后,線程不一定立即恢復執行。這是因為在那個時刻,其它線程可能正在運行而且沒有被調度為放棄執行,除非(a)“醒來”的線程具有更高的優先級
(b)正在運行的線程因為其它原因而阻塞。
wait()是線程交互時,如果線程對一個同步對象x 發出一個wait()調用,該線程會暫停執行,被調對象進入等待狀態,直到被喚醒或等待時間到。

     第十三,Java有沒有goto?
Goto—java中的保留字,現在沒有在java中使用。

     第十四,數組有沒有length()這個方法? String有沒有length()這個方法?
數組沒有length()這個方法,有length的屬性。
String有有length()這個方法。

     第十五,Overload和Override的區別。Overloaded的方法是否可以改變返回值的類型?
方法的重寫Overriding和重載Overloading是Java多態性的不同表現。重寫Overriding是父類與子類之間多態性的一種表現,重載Overloading是一個類中多態性的一種表現。如果在子類中定義某方法與其父類有相同的名稱和參數,我們說該方法被重寫 (Overriding)。子類的對象使用這個方法時,將調用子類中的定義,對它而言,父類中的定義如同被“屏蔽”了。如果在一個類中定義了多個同名的方法,它們或有不同的參數個數或有不同的參數類型,則稱為方法的重載(Overloading)。Overloaded的方法是可以改變返回值的類型。

     第十六,Set里的元素是不能重復的,那么用什么方法來區分重復與否呢? 是用==還是equals()? 它們有何區別?
Set里的元素是不能重復的,那么用iterator()方法來區分重復與否。equals()是判讀兩個Set是否相等。
equals()和==方法決定引用值是否指向同一對象equals()在類中被覆蓋,為的是當兩個分離的對象的內容和類型相配的話,返回真值。

     第十七,給我一個你最常見到的runtime exception。
ArithmeticException, ArrayStoreException, BufferOverflowException, BufferUnderflowException, CannotRedoException, CannotUndoException, ClassCastException, CMMException, ConcurrentModificationException, DOMException, EmptyStackException, IllegalArgumentException, IllegalMonitorStateException, IllegalPathStateException, IllegalStateException,
ImagingOpException, IndexOutOfBoundsException, MissingResourceException, NegativeArraySizeException, NoSuchElementException, NullPointerException, ProfileDataException, ProviderException, RasterFormatException, SecurityException, SystemException, UndeclaredThrowableException, UnmodifiableSetException, UnsupportedOperationException

     第十八,error和exception有什么區別?
error 表示恢復不是不可能但很困難的情況下的一種嚴重問題。比如說內存溢出。不可能指望程序能處理這樣的情況。
exception 表示一種設計或實現問題。也就是說,它表示如果程序運行正常,從不會發生的情況。

第十九,List, Set, Map是否繼承自Collection接口?
List,Set是

     Map不是

     第二十,abstract class和interface有什么區別?
聲明方法的存在而不去實現它的類被叫做抽象類(abstract class),它用于要創建一個體現某些基本行為的類,并為該類聲明方法,但不能在該類中實現該類的情況。不能創建abstract 類的實例。然而可以創建一個變量,其類型是一個抽象類,并讓它指向具體子類的一個實例。不能有抽象構造函數或抽象靜態方法。Abstract 類的子類為它們父類中的所有抽象方法提供實現,否則它們也是抽象類為。取而代之,在子類中實現該方法。知道其行為的其它類可以在類中實現這些方法。
接口(interface)是抽象類的變體。在接口中,所有方法都是抽象的。多繼承性可通過實現這樣的接口而獲得。接口中的所有方法都是抽象的,沒有一個有程序體。接口只可以定義static final成員變量。接口的實現與子類相似,除了該實現類不能從接口定義中繼承行為。當類實現特殊接口時,它定義(即將程序體給予)所有這種接口的方法。然后,它可以在實現了該接口的類的任何對象上調用接口的方法。由于有抽象類,它允許使用接口名作為引用變量的類型。通常的動態聯編將生效。引用可以轉換到接口類型或從接口類型轉換,instanceof 運算符可以用來決定某對象的類是否實現了接口。

第二十一,abstract的method是否可同時是static,是否可同時是native,是否可同時是synchronized?
都不能

     第二十二,接口是否可繼承接口? 抽象類是否可實現(implements)接口? 抽象類是否可繼承實體類(concrete class)?
接口可以繼承接口。抽象類可以實現(implements)接口,抽象類是否可繼承實體類,但前提是實體類必須有明確的構造函數。

     第二十三,啟動一個線程是用run()還是start()?
啟動一個線程是調用start()方法,使線程所代表的虛擬處理機處于可運行狀態,這意味著它可以由JVM調度并執行。這并不意味著線程就會立即運行。run()方法可以產生必須退出的標志來停止一個線程。

    第二十四,構造器Constructor是否可被override?
構造器Constructor不能被繼承,因此不能重寫Overriding,但可以被重載Overloading。

     第二十五,是否可以繼承String類?
String類是final類故不可以繼承。

     第二十六,當一個線程進入一個對象的一個synchronized方法后,其它線程是否可進入此對象的其它方法?
不能,一個對象的一個synchronized方法只能由一個線程訪問。

     第二十七,try {}里有一個return語句,那么緊跟在這個try后的finally {}里的code會不會被執行,什么時候被執行,在return前還是后?
會執行,在return前執行。

    
第二十八,編程題: 用最有效率的方法算出2乘以8等於幾?
有C背景的程序員特別喜歡問這種問題。

     2 <<  3

     第二十九,兩個對象值相同(x.equals(y) == true),但卻可有不同的hash code,這句話對不對?
不對,有相同的hash code。

     第三十,當一個對象被當作參數傳遞到一個方法后,此方法可改變這個對象的屬性,并可返回變化后的結果,那么這里到底是值傳遞還是引用傳遞?
是值傳遞。Java 編程語言只由值傳遞參數。當一個對象實例作為一個參數被傳遞到方法中時,參數的值就是對該對象的引用。對象的內容可以在被調用的方法中改變,但對象的引用是永遠不會改變的。

    
第三十一,swtich是否能作用在byte上,是否能作用在long上,是否能作用在String上?
switch(expr1)中,expr1是一個整數表達式。因此傳遞給 switch 和 case 語句的參數應該是 int、 short、 char 或者 byte。long,string 都不能作用于swtich。

     第三十二,編程題: 寫一個Singleton出來。
     Singleton模式主要作用是保證在Java應用程序中,一個類Class只有一個實例存在。
一般Singleton模式通常有幾種種形式:
第一種形式: 定義一個類,它的構造函數為private的,它有一個static的private的該類變量,在類初始化時實例話,通過一個public的getInstance方法獲取對它的引用,繼而調用其中的方法。
public class Singleton {
private Singleton(){}
//在自己內部定義自己一個實例,是不是很奇怪?
//注意這是private 只供內部調用
private static Singleton instance = new Singleton();
//這里提供了一個供外部訪問本class的靜態方法,可以直接訪問
public static Singleton getInstance() {
return instance;
}
}
第二種形式:
public class Singleton {
private static Singleton instance = null;
public static synchronized Singleton getInstance() {
//這個方法比上面有所改進,不用每次都進行生成對象,只是第一次
//使用時生成實例,提高了效率!
if (instance==null)
instance=new Singleton();
return instance; }
}
其他形式:
定義一個類,它的構造函數為private的,所有方法為static的。
一般認為第一種形式要更加安全些
第三十三 Hashtable和HashMap
Hashtable繼承自Dictionary類,而HashMap是Java1.2引進的Map interface的一個實現

     HashMap允許將null作為一個entry的key或者value,而Hashtable不允許

     還有就是,HashMap把Hashtable的contains方法去掉了,改成containsvalue和containsKey。因為contains方法容易讓人引起誤解。

     最大的不同是,Hashtable的方法是Synchronize的,而HashMap不是,在
多個線程訪問Hashtable時,不需要自己為它的方法實現同步,而HashMap
就必須為之提供外同步。

     Hashtable和HashMap采用的hash/rehash算法都大概一樣,所以性能不會有很大的差異。



cAng^Er 2008-04-23 18:34 發表評論
]]>
SQL Server 2000/2005 分頁SQL — 單條SQL語句 http://www.tkk7.com/xiaosao/archive/2008/04/23/195199.htmlcAng^ErcAng^ErWed, 23 Apr 2008 10:15:00 GMThttp://www.tkk7.com/xiaosao/archive/2008/04/23/195199.htmlhttp://www.tkk7.com/xiaosao/comments/195199.htmlhttp://www.tkk7.com/xiaosao/archive/2008/04/23/195199.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/195199.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/195199.html有關分頁 SQL 的資料很多,有的使用存儲過程,有的使用游標。本人不喜歡使用游標,我覺得它耗資、效率低;使用存儲過程是個不錯的選擇,因為存儲過程是經過預編譯的,執行效率高,也更靈活。先看看單條 SQL 語句的分頁 SQL 吧。

方法1:
適用于 SQL Server 2000/2005
SELECT TOP 頁大小 *
FROM table1
WHERE id NOT IN
          (
          
SELECT TOP 頁大小*(-1) id FROM table1 ORDER BY id
          )
ORDER BY id

方法2:
適用于 SQL Server 2000/2005
SELECT TOP 頁大小 *
FROM table1
WHERE id >
          (
          
SELECT ISNULL(MAX(id),0
          FROM 
                (
               
SELECT TOP 頁大小*(-1) id FROM table1 ORDER BY id
                ) 
A
          )
ORDER BY id

方法3:
適用于 SQL Server 2005

SELECT TOP 頁大小 * 
FROM 
        (
        
SELECT ROW_NUMBER() OVER (ORDER BY id) AS RowNumber,* FROM table1
        ) A
WHERE RowNumber > 頁大小*(頁數-1)

說明,頁大?。好宽摰男袛?;頁數:第幾頁。使用時,請把“頁大小”和“頁大小*(頁數-1)”替換成數字。
http://www.cnblogs.com/anjou/archive/2007/10/17/926944.html


cAng^Er 2008-04-23 18:15 發表評論
]]>
解析Java對象的equals()和hashCode()的使用http://www.tkk7.com/xiaosao/archive/2008/04/11/192142.htmlcAng^ErcAng^ErFri, 11 Apr 2008 04:11:00 GMThttp://www.tkk7.com/xiaosao/archive/2008/04/11/192142.htmlhttp://www.tkk7.com/xiaosao/comments/192142.htmlhttp://www.tkk7.com/xiaosao/archive/2008/04/11/192142.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/192142.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/192142.html在Java語言中,equals()和hashCode()兩個函數的使用是緊密配合的,你要是自己設計其中一個,就要設計另外一個。在多數情況 下,這兩個函數是不用考慮的,直接使用它們的默認設計就可以了。但是在一些情況下,這兩個函數最好是自己設計,才能確保整個程序的正常運行。最常見的是當 一個對象被加入收集對象(collection object)時,這兩個函數必須自己設計。更細化的定義是:如果你想將一個對象A放入另一個收集對象B里,或者使用這個對象A為查找一個元對象在收集對 象B里位置的鑰匙,并支持是否容納,刪除收集對象B里的元對象這樣的操作,那么,equals()和hashCode()函數必須開發者自己定義。其他情 況下,這兩個函數是不需要定義的。

equals():

它是用于進行兩個對象的比較的,是對象內容的比 較,當然也能用于進行對象參閱值的比較。什么是對象參閱值的比較?就是兩個參閱變量的值得比較,我們 都知道參閱變量的值其實就是一個數字,這個數字可以看成是鑒別不同對象的代號。兩個對象參閱值的比較,就是兩個數字的比較,兩個代號的比較。這種比較是默 認的對象比較方式,在Object這個對象中,這種方式就已經設計好了。所以你也不用自己來重寫,浪費不必要的時間。

對象內容的比較才是設計equals()的真正目的,Java語言對equals()的要求如下,這些要求是必須遵循的。否則,你就不該浪費時間:
  • 對稱性:如果x.equals(y)返回是“true”,那么y.equals(x)也應該返回是“true”。
  • 反射性:x.equals(x)必須返回是“true”。
  • 類推性:如果x.equals(y)返回是“true”,而且y.equals(z)返回是“true”,那么z.equals(x)也應該返回是“true”。
  • 還有一致性:如果x.equals(y)返回是“true”,只要x和y內容一直不變,不管你重復x.equals(y)多少次,返回都是“true”。
  • 任何情況下,x.equals(null),永遠返回是“false”;x.equals(和x不同類型的對象)永遠返回是“false”。
hashCode():
這 個函數返回的就是一個用來進行赫希操作的整型代號,請不要把這個代號和前面所說的參閱變量所代表的代號弄混了。后者不僅僅是個代號還具有在內存中才查找對 象的位置的功能。hashCode()所返回的值是用來分類對象在一些特定的收集對象中的位置。這些對象是HashMap, Hashtable, HashSet,等等。這個函數和上面的equals()函數必須自己設計,用來協助HashMap, Hashtable, HashSet,等等對自己所收集的大量對象進行搜尋和定位。

這些收集對象究竟如何工作的,想象每個元對象hashCode是一個箱子的 編碼,按照編碼,每個元對象就是根據hashCode()提供的代號歸入相應的箱子里。所有的箱子加起來就是一個HashSet,HashMap,或 Hashtable對象,我們需要尋找一個元對象時,先看它的代碼,就是hashCode()返回的整型值,這樣我們找到它所在的箱子,然后在箱子里,每 個元對象都拿出來一個個和我們要找的對象進行對比,如果兩個對象的內容相等,我們的搜尋也就結束。這種操作需要兩個重要的信息,一是對象的 hashCode(),還有一個是對象內容對比的結果。

hashCode()的返回值和equals()的關系如下:
  • 如果x.equals(y)返回“true”,那么x和y的hashCode()必須相等。
  • 如果x.equals(y)返回“false”,那么x和y的hashCode()有可能相等,也有可能不等。


為 什么這兩個規則是這樣的,原因其實很簡單,拿HashSet來說吧,HashSet可以擁有一個或更多的箱子,在同一個箱子中可以有一個 或更多的獨特元對象(HashSet所容納的必須是獨特的元對象)。這個例子說明一個元對象可以和其他不同的元對象擁有相同的hashCode。但是一個 元對象只能和擁有同樣內容的元對象相等。所以這兩個規則必須成立。

設計這兩個函數所要注意到的:
如果你設計的對象類型并不使用于收集性對象,那么沒有必要自己再設計這兩個函數的處理方式。這是正確的面向對象設計方法,任何用戶一時用不到的功能,就先不要設計,以免給日后功能擴展帶來麻煩。

如果你在設計時想別出心裁,不遵守以上的兩套規則,那么勸你還是不要做這樣想入非非的事。我還沒有遇到過哪一個開發者和我說設計這兩個函數要違背前面說的兩個規則,我碰到這些違反規則的情況時,都是作為設計錯誤處理。

當 一個對象類型作為收集型對象的元對象時,這個對象應該擁有自己處理equals(),和/或處理hashCode()的設計,而且要遵守前面所說 的兩種原則。equals()先要查null和是否是同一類型。查同一類型是為了避免出現ClassCastException這樣的異常給丟出來。查 null是為了避免出現NullPointerException這樣的異常給丟出來。

如果你的對象里面容納的數據過多,那么這兩個函數 equals()和hashCode()將會變得效率低。如果對象中擁有無法serialized的數據,equals()有可能在操作中出現錯誤。想象 一個對象x,它的一個整型數據是transient型(不能被serialize成二進制數據流)。然而equals()和hashCode()都有依靠 這個整型數據,那么,這個對象在serialization之前和之后,是否一樣?答案是不一樣。因為serialization之前的整型數據是有效的 數據,在serialization之后,這個整型數據的值并沒有存儲下來,再重新由二進制數據流轉換成對象后,兩者(對象在serialization 之前和之后)的狀態已經不同了。這也是要注意的。

 
原文 :http://blog.csdn.net/RichardSundusky/archive/2007/02/12/1508028.aspx


cAng^Er 2008-04-11 12:11 發表評論
]]>
Groovy輕松入門——通過與Java的比較,迅速掌握Groovy (更新于2007.09.23)http://www.tkk7.com/xiaosao/archive/2008/02/29/182943.htmlcAng^ErcAng^ErFri, 29 Feb 2008 07:30:00 GMThttp://www.tkk7.com/xiaosao/archive/2008/02/29/182943.htmlhttp://www.tkk7.com/xiaosao/comments/182943.htmlhttp://www.tkk7.com/xiaosao/archive/2008/02/29/182943.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/182943.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/182943.html原文:http://www.tkk7.com/BlueSUN/archive/2007/03/10/103014.html | 山風小子
注意:本教程是針對Groovy最新版本編寫的


Groovy和Java的相同點有:
3+, 4+, 6+, 8+, 10+, 12+, 13, 14, 15, 18+, 20+, 21, 22, 23, 28+, 29+, 30+, 31+, 32+
+表示Groovy不但涵蓋了Java的語法,而且還有增強部分

Groovy和Java的不
點有:
0, 1, 2, 5, 7, 9, 11, 16, 17, 19, 24, 25, 26, 27
 

下面所列序號不分先后順序:

0. 在Groovy可以用def定義無類型的變量(定義變量方面def與JavaScript中的var相似),和返回值為無類型的方法,而在Java中沒有def
Groovy:
class Man {
  def name 
= "山風小子"
  def introduce() {
    
return "I'm $name" // return可以省略
  }
}

1. Java中的equals方法對應Groovy中的== , 而Java中的==(判斷是否引用同一對象)對應Groovy中的is方法
eg.
Test1.java :
public class Test {
  
public static void main(String[] args) {
    String name1 
= "山風小子";
    String name2 
= new String("山風小子");
    
// Groovy中寫為 name1 == name2
    if (name1.equals(name2)) {
      System.out.println(
"equal");
    } 
else {
      System.out.println(
"not equal"); 
    }
    
// Groovy中寫為 name1.is(name2)
    if (name1 == name2) {
      System.out.println(
"identical");
    } 
else {
      System.out.println(
"not identical"); 
    } 
}

與Test1.java相對應的
Test1.groovy :
String name1 = "山風小子" // 你也可以這樣寫: def name1 = "山風小子"  其中的def可以理解為JavaScript中定義變量的var
String name2 = new String("山風小子"//請注意Groovy的句尾的分號時可選的,即可要可不要,前提是一行代碼一條語句
// Java中寫為 name1.equals(name2)
if (name1  == name2) {
  System.out.println(
"equal");
else {
  System.out.println(
"not equal"); 
}
// Java中寫為 name1  == name2
if (name1.is(name2)) {
  System.out.println(
"identical");
else {
  System.out.println(
"not identical"); 
}

2. Java中的數組定義int[] a = {1, 2, 3}; 在Groovy寫成int[] a = [1, 2, 3]
3. Java中的for循環for (int i = 0; i < len; i++) {...} 在Groovy中還可以寫成for (i in 0..len-1) {...} 或者 for (i in 0..<len) {...}
Java:
for (int i =0; i < len; i++) {
  
// do something
}
Groovy:
for (int i =0; i < len; i++) {
  
// do something
}

// 或者
for (i in 0..len-1) {
  
// do something
}

// 或者
for (i in 0..<len) {
  
// do something
}

4. Java中的方法返回寫為return; 或者return obj; 在Groovy的方法中return是可選的
Java:
public String sayHello() {
  
return "Hello, 山風小子";
}
Groovy:
public String sayHello() {
  
return "Hello, 山風小子"
}
//或者
public String sayHello() {
  
"Hello, 山風小子"
}
//或者
String sayHello() {
  
"Hello, 山風小子"
}
//或者
public sayHello() {
  
"Hello, 山風小子"
}
// 或者
def sayHello() {
  
"Hello, 山風小子"
}

5. Java中的inner class即內部類,在Groovy中用Closure實現(Closure是Java7正在考慮的一個特性,比inner class在語義方面更完善)
6. Groovy中的注釋比Java多了首行注釋#!,其他與Java相同比如單行注釋://    多行注釋:/*  */   或者是  支持javadoc的/**   */
Java:
/*
 * 多行注釋
 
*/

/**
 * javadoc 注釋
 
*/

// 單行注釋

Groovy:
#! 首行注釋,使Unix shell能夠定位Groovy啟動程序以運行Groovy代碼,例如
#!/usr/bin/groovy

/*

 * 多行注釋
 
*/

/**
 * javadoc 注釋
 
*/

// 單行注釋


7. Java5中的for-each:for (Type t : iteratable) {...}  在Groovy中,for (t in iteratable) {...}
Java:
for (Type t : iterable) {
  
// do something
}

Groovy:
for (t in iterable) {
  
// do something
}

8. Groovy中switch語句與Java中相同,不過支持更多類型了,比如String
9. Groovy的while語句跟Java相同,但廢棄了do-while (考慮到語義方面的問題,而且do-while可以用其他形式的循環語句代替,使用頻率低)
10. Java中的String常量表示為"Hello, 山風小子",在Groovy中可如下表示
// 雙引號
"Hello, 山風小子"

// 單引號也可以
'Hello, 山風小子'

//多行字符串
"""Hello,
山風小子"""

//或者
'''Hello,
山風小子
'''

// 替代字符串
def name = "山風小子"
"Hello, ${name}"
//或者
"Hello, $name"
11.在Groovy中定義類,定義方法與Java中定義類相同,唯一區別在于Groovy中類,屬性以及方法默認都是public的,而在Java中默認是package的,另外,在Groovy中可以用def來定義方法,請看注釋。
Java:
public class Hello {
  
private String name = "山風小子";
  
public void sayHello() {
    System.out.println(
"Hello, " + name);
  }
}
Groovy:
class Hello {
  
private String name = "山風小子"
  
public void sayHello() {
    
//println與Java中System.out.println()相同
    println "Hello, $name"
  }
  
/* sayHello也可以這樣定義
  def sayHello() {
    println "Hello, $name"
  }
   
*/
}

12.對象創建在Java寫為Thought t = new Thought(); 在Groovy也可以這樣寫,不過還多了種寫法:def t = new Thought();
13.靜態方法調用在Java和Groovy中相同,即ClassName.staticMethodName();
14.實現接口和繼承父類方面Groovy也與Java完全相同,即實現接口class ClassName implements InterfaceName {...}
      繼承父類:class ClassName extends SuperClass {...}
15.定義接口方面Groovy與Java完全相同,即interface InterfaceName {...}  //在Groovy中默認為public的
16.正則表達式常量在Java中沒有,在Groovy中表示為 /pattern/
17.Hash常量(類型為java.util.HashMap)在Java沒有,在Groovy中表示為 def frequence = ["the": 5, "hello": 2, "world": 2]
18.類變量即static變量,Groovy與Java相同,static String name = "山風小子",在Groovy也可寫為static name = "山風小子"
19.在varargs方法方面,Groovy與Java稍有區別,如下所示:
Java:
// Java: 
public void varargsMethod(Type args) {
  
//do something
}

Groovy:
def varargsMethod(Type[] args) {
  
//do something
}

20.引用當前對象,Groovy和Java相同,在Java中用this表示,在Groovy中也用this表示,而且在Groovy中,this可以出現在static范圍中,指向所在類的類對象,本例中,this等同于
ThisInStaticScope.class(Java寫法ThisInStaticScope(Groovy寫法)
class ThisInStaticScope {
   
static {
        println 
this
    }
   
// 請不要詫異,參數類型可以省略。如果方法聲明中有修飾關鍵字比如public,synchronized,static等,則返回值類型可以省略。
    static main(args) {
        println 
this
    }
}

21.子類中調用父類方法,Groovy和Java也相同,在Java中 super.methodName() ,在Groovy中 super.methodName()
22.命名空間的定義,Groovy和Java相同,在Java中 package edu.ecust.bluesun;   在Groovy中 package edu.ecust.bluesun     (分號可省略)
23.在導入類方面,Groovy和Java相同,在Java中 import edu.ecust.bluesun.GroovyTest;  在Groovy中 import edu.ecust.bluesun.GroovyTest
24.List常量(類型為java.util.ArrayList)在Java中沒有, 在Groovy中表示為 def list = [3, 11, "Hello", "山風小子", "!"]
25.在異常處理方面,Groovy與Java相同,除了不強制程序員捕獲檢查異常(checked exception)外 (這跟C#很像,如果我沒記錯的話 :)
        并且在方法聲明時,也可以不寫throws語句。
26.方法的默認參數,Java中沒有,Groovy中表示如下:
class Hello {
  
//如果沒有參數傳入,默認打印出 Hello, 山風小子
  def greet(name="山風小子") {
    println(
"Hello, $name"//也可省略括號()
  }
}
27.在Groovy中,語句如果單獨占一行的話,句尾的分號(;)可以省略,而在Java中每條語句后面必須跟有分號(;)
28.在Groovy中,如果不是Boolean或boolean類型,非null或非空(空字符串,[],[:])為true,null為false,而Java中對象不可以表示true或false;如果是Boolean或boolean類型,與Java中的一樣。
29.在Groovy中,萬事萬物都是對象!而Java中不是這樣,基本類型(primitive type)就不是對象。
30.在Java中,Class對象表示為ClassName.class,而在Groovy中,可以直接用ClassName表示Class對象
31.Groovy會自動導入java.lang.*, java.util.*, java.net.*, java.io.*, java.math.BigInteger, java.math.BigDecimal,   groovy.lang.*, groovy.util.*,而Java則只自動導入java.lang.*
32.Groovy不僅有? :三元操作符,還有?:兩元操作符,但Java只有? :三元操作符。
Groovy:
def a = null;
// 如果a為“空”(null,空串"",[],[:]),那么結果為?:之后的那個值; 如果不為“空”,那么結果就是a
def result = a ?"default result"
println result

= "山風小子"
result 
= a ?"default result"
println result




由上可知,Groovy幾乎完全兼容Java的語法,難怪‘江南白衣’稱Groovy是Java的‘私生子’;但由于Groovy不僅借鑒了Java 95%以上的特性,而且還借鑒了許多卓越的動態語言,比如Python, Ruby等,使Groovy成為極其高效敏捷的編程語言,而不僅僅是Java的副本。所以其實Java++可以作為Groovy的別名,即具有動態特性的Java。

最后,我想補充一句:Groovy的特性遠不至所列的這些,比如還有Mixins,builder系列:MarkupBuilder,SwingBuilder等,很多都是Groovy中有而Java中沒有,因此就不一一列舉了,想繼續深入學習Groovy,可訪問Groovy官方網站:http://groovy.codehaus.org ,里面有很多例子和教程供大家參閱,也可以參考在下的Groovy高效編程系列(其中一些是在下的筆記)。網上也有《Groovy In Action》電子書下載,大家不妨搜一下。


參考文獻
Differences from Java:http://groovy.codehaus.org/Differences+from+Java


cAng^Er 2008-02-29 15:30 發表評論
]]>
Groovy和Grails簡介http://www.tkk7.com/xiaosao/archive/2008/02/21/181031.htmlcAng^ErcAng^ErThu, 21 Feb 2008 04:40:00 GMThttp://www.tkk7.com/xiaosao/archive/2008/02/21/181031.htmlhttp://www.tkk7.com/xiaosao/comments/181031.htmlhttp://www.tkk7.com/xiaosao/archive/2008/02/21/181031.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/181031.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/181031.html轉載:原文地址 http://dev2dev.bea.com.cn/techdoc/200601194089.html

摘要

  Java Web應用程序框架是企業Java得以成功的重要原因之一。人們懷疑如果沒有Apache Struts框架Java EE是否能夠如此成功。雖然底層編程語言很重要,但通常是框架使編程語言成為引人注目的中心的。如果您經常訪問討論論壇,就會注意到Ruby語言和Ruby On Rails框架之間也是這種情況。Ruby已經出現十多年了,然而只是在Ruby On Rails框架流行之后,開發人員才開始注意到Ruby語言。

  諸如Ruby、PHP和Python之類的腳本語言最近幾年越來越流行,因此,需要開發一個Java腳本備選語言和類似Rails的針對Java環境的框架。Groovy就是這個腳本語言,而Grails就是這個框架。

  在本文中我將討論Groovy的Web開發功能,然后繼續討論Grails框架。我將開發一個示例Grails Web應用程序,并討論此框架的各種特性。

Groovy是什么?

  Groovy是一種語言,其語法類似于Java,但比Java更簡單。它通常被視為腳本/靈活/動態的語言,但是我不喜歡這類形容詞,因為我認為它們只會令人困惑。如果說Java是一位明智的中年男子,那么Groovy就是他十幾歲的兒子。Groovy具有父親的許多特點,但是更為狂野且更為有趣。他們也可以很好地合作。

  Groovy的規則比Java少得多。例如,要在Java中獲得標準的"Hello World"輸出,您需要編寫一個類、一個具有合適參數的主方法,等等。但是在Groovy中,如果不想編寫所有樣板代碼,您可以拋開類定義和主方法,僅編寫一行代碼即可打印出"Hello World"。

  以下是打印Hello World的文件 Hello.groovy 的內容:

println "Hello World" 

  Java平臺僅關心使字節碼得到執行。同樣,此平臺不強迫您使用Java語言。只要提供了字節碼,工作就會進行。Groovy代碼會被編譯為字節碼,而對于Java平臺來說,字節碼是從Java代碼還是Groovy代碼生成的并沒有任何區別。

  以下是一個Groovy例子,它顯示了Groovy對清單、映射和范圍的內置支持,并證明了Groovy的簡單性及其利用Java的強大功能的能力:

// Print Date
def mydate = new java.util.Date()
println mydate
//Iterate through a map
def numbersMAP = ['1':'ONE', '2':'TWO']
for (entry in numbersMAP) {
println "${entry.key} = ${entry.value}"
}
//Introducing the range
def range = 'a'..'d'
//Lists
def numberlist = [1, 2, 3, 4, 5, 6, 7, 8]
println numberlist;
println "Maximum value: ${numberlist.max()}"

  請注意以上代碼直接使用java.util.Date ,對收集的內置支持減少了使用清單、映射和范圍所需的代碼。還有許多其他有趣的Groovy特性,例如閉包和簡化的XML處理。您可以在groovy.codehaus.org上找到詳細清單。

  現在讓我們來討論如何將Groovy用于Web開發。

使用Groovy進行Web開發

  大多數Java EE教程都從一個基本servlet例子開始。對于Groovy Web開發來說,您將從groovlet(在groovy中servlet的對應概念)開始。如果您在servlet中擺脫了類和doXX() 方法聲明,那么剩下的內容就與groovlet很像了。以下是一個名為 Login.groovy 的groovlet例子,您需要將它置于Web應用程序的最高級目錄:

def username= request.getParameter("username")
def password= request.getParameter("password")
if (username == "java" && password == "developer") {
response.sendRedirect("home.jsp")
session = request.getSession(true);
session.setAttribute("name", username)
}
else {
println """
<h1>Login Invalid</h1>
<p>Your IP has been logged > ${request.remoteHost}</p>
"""
paramMap = request.getParameterMap()
println "<p>You Submitted:</p>"
for (entry in paramMap) {
println "${entry.key} = ${entry.value}<br/>"
}
}

  您可以僅創建一個簡單的HTML表單,然后將此表單的行為屬性發送到 action="Login.groovy"。然后將以下標簽添加到web.xml:

<servlet>
<servlet-name>Groovy</servlet-name>
<servlet-class>groovy.servlet.GroovyServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>Groovy</servlet-name>
<url-pattern>*.groovy</url-pattern>
</servlet-mapping>

  現在只需將要求的Groovy jar 文件添加到WEB-INF/lib 目錄,您的Groovy Web應用程序就準備好在任意Java EE應用服務器上運行了。

  您應該已經注意到代碼中沒有分號,而且使用了隱式變量如request和response。其他隱式變量有context、application、session、out、sout和 html。

  GSP是JSP在groovy中的對應概念。您無需使用println生成HTML;只需將Groovy代碼嵌入HTML頁面。本文中的例子將在提到Grails時使用GSP。

  請注意,因為所有代碼最終都要轉換為字節碼,所以groovlet和GSP能夠與servlet和JSP輕松協作。因此您無需區分groovlet和GSP或者servlet和JSP。

  現在讓我們討論前途無量的Grails框架。如果成功的話,Grails能夠極大地改變開發Java Web應用程序的方式。Ruby on Rails對Ruby的影響,Grails也能夠對Groovy實現。

Grails特性和架構

  Grails試圖使用Ruby On Rails的“規約編程”(coding by convention)范例來降低對配置文件和其他樣板代碼的需求。使用“規約編程” ,如果文件的名稱本身就能說明此文件的用途,那么您就不需要在配置文件中再次聲明這些內容了。此框架會查看文件名,并自己弄清文件用途。通過使用“規約編程” ,Grails還將自動生成Web應用程序中需要的許多內容。通過使用Grails,您將能夠在很短的時間內、以最小的復雜性使Web應用程序就緒。請看以下例子。

  Grails基于開源技術,例如Spring、Hibernate和SiteMesh。如果您已經擅長這些技術,那么這是件好事;但是如果您由于某種原因不喜歡這些技術,或者您認為不僅需要學習Grails,還需要學習其他三種框架,那么這就不是件好事了。雖然這些技術能夠幫助Grails執行得更好,但是學習四種框架對于大多數人來說是一個很高的門檻。Grails文檔目前主要關注它與Spring、Hibernate和其他程序的集成,然而我認為它需要采用相反的方法,將Grails推行為一個簡單快速的Web應用程序開發框架。開發人員無需擔心或考慮底層發生了什么。

  幸運的是,一旦您開始使用Grails,您將發現Grails隱藏了這些框架的大多數底層復雜性。如果您忘掉在底層運行的是Spring、Hibernate和其他程序,那么事情就會變得簡單。

  Grails應用程序的三個層是:

  1. 由視圖和控制器組成的Web層
  2. 由域類和服務組成的業務邏輯層
  3. 由域類和數據源組成的持久層

  大多數框架都有數十種特性,其中只有很少幾種得到了廣泛使用。對于Grails來說,這種關鍵特性是指“規則編程”(coding by convention)范例和構件的自動生成。

  Grails的其他特性包括對Ajax、驗證、單元測試和功能測試的內置支持。它使用免費的開源Canoo WebTest項目來實現Web應用程序的功能測試。Grails還提供與Quartz Scheduler的集成。

  現在是時候安裝Grails框架并且編寫您的第一個應用程序了。

Grails安裝

  安裝過程非常簡單。以下是Grails下載頁面:http://grails.org/Download。您可以從http://dist.codehaus.org/grails/grails-bin-0.2.1.zip下載version 0.2.1。請注意Grails源代碼和文檔作為單獨的下載提供。下載zip文件之后,只需將其內容解壓縮到一個目錄即可,在我的案例中此目錄是 C:\groovy\grails-0.2.1\。

  創建一個名為GRAILS_HOME 的新環境變量,并將其值設為 C:\groovy\grails-0.2.1\。接下來將GRAILS_HOME\bin 添加到PATH 環境變量。這樣安裝就完成了。通過在命令提示符界面中運行grails 命令您可以檢查安裝是否成功。您應該獲得此命令的使用信息。

  既然您有了一個運行中的Grails安裝,那么您已經為創建Grails Web應用程序做好了準備。

開發Grails應用程序:應用程序結構

  多年來我一直計劃開發一個可以幫助我管理衣服的應用程序——這個應用程序應該能夠告訴我我最喜歡的T恤衫放在哪里、是否洗過、是否熨過,等等??傆幸惶煳視夸N售這個應用程序掙上幾百萬,但是現在我將把它用作Grails例子。

  第一步是創建一個Grails項目目錄結構。在這一步我將在C:\groovy\grailsapps 創建一個新目錄,并在此級別打開一個命令提示符窗口。在此窗口中,執行命令grails create-app。要求您輸入應用程序名稱。輸入 ClothesMgt。Grails將顯示它為您創建的全部目錄和文件。圖1顯示了最后得到的命令結構。

Groovy和Grails簡介圖-1

  圖1:Grails項目目錄結構

  此命令將創建約800 KB大小的文件和目錄。這里的想法是此框架遵循已經建立的Web應用程序開發慣例,因此它創建的文件和目錄在大多數Web應用程序中是有用的。雖然有些人可能不喜歡這種強制使用某種結構的想法,但是這種基于慣例的自動生成正是Grails的RAD特性的基礎。

  如果更仔細地看一下這些目錄,您就會發現存在用于諸如控制器、視圖、測試、配置文件和標簽庫之類東西的目錄。您還會發現存在一些基本JavaScript和CSS文件。那么現在應用程序的基本結構已經有了。您只需做些填空,應用程序即可就緒。

  請注意自動生成目錄和文件的命令是可選的。您可以手動創建全部文件和目錄。如果熟悉Apache Ant,那么您甚至可以打開GRAILS_HOME 目錄中的\src\grails\build.xml 文件,來仔細查看每個Grails命令的用途。

數據庫

  在此例中我將使用一個 運行于localhost的名為Clothes_Grails的MySQL數據庫。Grails內置一個HSQL數據庫,這對測試簡單的應用程序或僅試用Grails非常有用。如果您使用HSQL DB,那么無需執行以下幾步。我將使用MySQL來證明您能夠非常輕松地使用HSQL之外的數據庫。

  從http://www.mysql.com/products/connector/j/ 下載MySQL驅動器,并將mysql-connector-java-<version number>-stable-bin.jar 文件放置在ClothesMgt\lib 目錄中。接下來您需要編輯 ClothesMgt\grails-app\conf\ApplicationDataSource.groovy文件。

  現在此文件的內容應該類似以下內容:

class ApplicationDataSource {
boolean pooling = true
String dbCreate = "create-drop"
String url = "jdbc:mysql://localhost/Clothes_Grails"
String driverClassName = "com.mysql.jdbc.Driver"
String username = "grails"
String password = "groovy"
}

  現在讓我們看一下如何使用此數據庫和對象關系映射。

域類

  Grails的對象關系映射(GORM)功能在內部使用Hibernate 3,但是您無需了解或更改任何Hibernate設置。Grails具有稱為“域類”的東西,這些域類的對象被映射到數據庫。您可以使用關系來鏈接域類,它們也提供用于CRUD(創建/讀取/更新/刪除)操作的功能非常強大的動態方法。

  在此例中,我們將創建三個域類,其名稱分別是Shirt、Trouser和Cabinet。要創建域類,只需運行命令 grails create-domain-class。請記住在您的項目目錄(而不是它的上級目錄)內運行此命令。這是一個常見錯誤,雖然我已經提醒了您,您還是會犯至少一次這樣的錯誤。

  您必須提供給create-domain-class 命令的唯一輸入是類的名稱。運行此命令三次,將Shirt、Trouser和Cabinet作為三個域類的名稱。Grails現在將在目錄 grails-app/domain/中創建這些域類。它們將僅具有兩個屬性id 和 version。我將為這些類添加屬性,以便使它們更能代表襯衫、褲子和衣櫥。

  清單1:Cabinet.groovy

class Cabinet {
Long id
Long version
String name
String location
def relatesToMany = [ shirts : Shirt, trousers : Trouser ]
Set shirts = new HashSet()
Set trousers = new HashSet()
String toString() { "${this.class.name} :  $id" }
boolean equals(other) {
if(other?.is(this))return true
if(!(other instanceof Cabinet)) return false
if(!id || !other?.id || id!=other?.id) return false
return true
}
int hashCode() {
int hashCode = 0
hashCode = 29 * (hashCode + ( !id ? 0 : id ^ (id >>> 32)) )
}
}

  清單2: Trouser.groovy

class Trouser {
Long id
Long version
String name
String color
Cabinet cabinet
def belongsTo = Cabinet
String toString() { "${this.class.name} :  $id" }
boolean equals(other) {
if(other?.is(this))return true
if(!(other instanceof Trouser)) return false
if(!id || !other?.id || id!=other?.id) return false
return true
}
int hashCode() {
int hashCode = 0
hashCode = 29 * (hashCode + ( !id ? 0 : id ^ (id >>> 32) ) )
}
}

  清單3: Shirt.groovy

class Shirt {
Long id
Long version
String name
String color
Cabinet cabinet
def belongsTo = Cabinet
String toString() { "${this.class.name} :  $id" }
boolean equals(other) {
if(other?.is(this))return true
if(!(other instanceof Shirt)) return false
if(!id || !other?.id || id!=other?.id) return false
return true
}
int hashCode() {
int hashCode = 0
hashCode = 29 * (hashCode + ( !id ? 0 : id ^ (id >>> 32)))
}
}

  我添加的僅有的幾行聲明了字段名稱和顏色,然后聲明了Cabinet、Shirt和Trouser之間的關系。每個Shirt和Trouser都屬于Cabinet,而Cabinet具有shirt和trouser的集合。belongsTo 屬性在此案例中是可選的,因為在一對多關系中,Grails會將“一”這一方視為所有者。因此您就無需顯式聲明了。在這里我進行顯式聲明只是為了使這種關系更明顯。

  接下來我們將討論Grails應用程序的控制器和視圖部分。

控制器和視圖

  既然域類已經就緒,讓我們使用generate-all命令自動生成基本CRUD Web應用程序。運行grails generate-all 命令三次,當被詢問時提供域類名稱。generate-all 命令的目的是生成每個域類的控制器和視圖,但是由于bug-245,Grails 0.2.1不能生成控制器。您必須手動生成控制器,其方法是對每個域類使用generate-controller 命令。

  現在您應該在grails-app\controllers 目錄中看到三個控制器。這些控制器負責處理Web應用程序中針對特定域類的請求。因此ShirtController.groovy 將處理Web應用程序中與Shirt域類相關的CRUD請求,等等?,F在控制器具有多個閉包,每個閉包映射到一個URI。閉包是Groovy語言很好的一個特性,然而要習慣它還是需要一些時間的。清單4顯示了Shirtcontroller.groovy的一段摘錄。

  清單4:ShirtController.groovy 摘錄

class ShirtController {
def index = { redirect(action:list,params:params) }
def list = {
[ shirtList: Shirt.list( params ) ]
}
def show = {
[ shirt : Shirt.get( params.id ) ]
}
def delete = {
def shirt = Shirt.get( params.id )
if(shirt) {
shirt.delete()
flash.message = "Shirt ${params.id} deleted."
redirect(action:list)
}
else {
flash.message = "Shirt not found with id ${params.id}"
redirect(action:list)
}
}
// ...
}

  在此例中,ShirtController 中的list閉包將處理URI是/shirt/list的請求,等等。您可在控制器中使用您習慣在Java Web應用程序中使用的東西,例如請求、會話和servletContext。

  請注意:閉包也將值作為顯式返回語句返回,或者作為閉包體中的最后一個語句的值返回。不要因為Grails生成的代碼中沒有return 而困惑。

  一旦控制器完成了對請求的處理,它必須委托給合適的視圖。Grails使用慣例機制實現此操作。因此ShirtController 中的list閉包將委托給視圖 /grails-app/views/shirt/list.gsp 或 /grails-app/views/shirt/list.jsp。 盡管您在使用Grails,全部視圖可以是JSP文件而不是GSP。我幾乎沒有編寫任何代碼,但是我已經準備好了一個Web應用程序。

  讓我們嘗試部署和運行我們的應用程序。

在Java EE Server上部署和運行Grails

  Grails具有一個內置Resin服務器,您可使用grails run-app 命令運行應用程序。此命令會將應用程序部署到Resin服務器并啟動服務器。因此您現在可以在http://localhost:8080/ClothesMgt 訪問此應用程序。您還可以同樣輕松地將應用程序部署到任意JavaEE服務器。我嘗試將它部署到Tomcat。要實現此操作,我所需要做的是運行grails war 命令,將生成的war文件復制到Tomcat中的webapps目錄!

  在此案例中生成的war文件的名稱為 ClothesMgt.war。一旦部署到Tomcat,您就應該能夠在http://localhost:8080/ClothesMgt/ 上訪問它,并看到如圖2所示的屏幕。

Groovy和Grails簡介圖-2

  圖2:Grails 應用程序

  通過此應用程序,能夠獲得Shirt、Trouser和Cabinet的全部CRUD功能??梢燥@示衣櫥的全部數據、向衣櫥添加新襯衫和褲子、編輯它們的值和刪除記錄——實現這些操作都無需編寫任何業務邏輯、視圖或數據訪問代碼。僅在幾分鐘內您就在JavaEE服務器上部署好了一個合適的Web應用程序。很酷吧?!

  讓我們更進一步來定制Grails。

創建自定義控制器

  我現在將把新功能和頁面添加到Web應用程序,同時重用已經存在的域類。shirt/list 和 trouser/list 會分別顯示襯衫和褲子的清單,現在讓我們添加一個新的顯示,來同時顯示襯衫和褲子的清單。要創建一個新的顯示,您需要一個新的控制器和視圖。

  使用 generate-controller 和 generate-views 命令,可以輕松實現使用域類自動生成視圖和控制器。然而,在此案例中我希望創建一個與域類不直接關聯的控制器。因此我將使用grails create-controller命令。當被提示輸入控制器名稱時,聲明Display。Grails將在grails-app/controllers/ 目錄創建一個名為DisplayController.groovy 的控制器,在grails-tests 目錄創建一個測試套件。如清單5所示編輯控制器。

  清單5:DisplayController.groovy

class DisplayController {
def index = {redirect(action:list,params:params)}
def list = {
params['max'] = 10
return [ shirtList: Shirt.list( params ),
trouserList: Trouser.list( params )]
}
}

  index 閉包將請求重定向到清單。在list 閉包中我將最大參數設為10,然后使用動態方法Shirt.list 和 Trouser.list。然后返回Groovy Map,它有兩個清單——襯衫清單和褲子清單。

  作為Java開發人員,當看到Shirt.list()時會自然認為是在Shirt域類中的list 方法。然而,如果打開Shirt.groovy,會發現并沒有此方法。對于Java開發人員來說,不了解Groovy的特性就使用Grails不僅是令人困惑的,而且是死胡同。動態方法是Grails的特殊特性,它是構建于Groovy語言的一個非常特殊的特性元對象協議 (MOP)之上的。如此證明可以使用動態方法查詢域類。因此,在控制器中,您將注意到在域類上調用的方法似乎在域類中不存在。您可以在這里閱讀關于使用動態方法查詢的更多信息??梢栽?a >這里找到對Grails控制器和域類中可用的動態方法的參考資料。

  既然控制器能夠處理請求、獲取清單并轉發到視圖,我需要創建相應視圖。

創建自定義視圖

  當創建控制器時,Grails還在grails-app\views 目錄創建了一個新的顯示目錄,并將以下映射添加到web.xml 文件中。

<servlet-mapping>
<servlet-name>grails</servlet-name>
<url-pattern>/display/*</url-pattern>
</servlet-mapping>

  目前Grails有一個generate-views 命令,此命令能夠生成基于域類的視圖,然而沒有能夠自動生成視圖的create-view 命令。請看圖3中的例子。

Groovy和Grails簡介圖-3

  圖3:一個顯示Trousers的默認視圖

  因為我希望創建一個獨立于域類的視圖,所以讓我們手動創建視圖文件。在目錄grails-app\views\display\中,創建一個名為 list.gsp的文件,如清單6所示。

  清單6:list.gsp

<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
<title>Display Shirt And Trouser List</title>
<link rel="stylesheet" href="${createLinkTo(dir:'css',file:'main.css')}"></link>
</head>
<body>
<div class="nav">
<span class="menuButton"><a href="${createLinkTo(dir:'')}">Home</a></span>
</div>
<div class="body">
<h1>Shirt List</h1>
<table>
<tr>
<th>Id</th><th>Cabinet</th> <th>Color</th><th>Name</th>
</tr>
<g:each in="${shirtList}">
<tr>
<td>${it.id}</td> <td>${it.cabinet.name}</td>
<td>${it.color}</td> <td>${it.name}</td>
</tr>
</g:each>
</table>
<h1>Trouser List</h1>
<table>
<tr>
<th>Id</th> <th>Cabinet</th>
<th>Color</th> <th>Name</th>
</tr>
<g:each in="${trouserList}">
<tr>
<td>${it.id}</td> <td>${it.cabinet.name}</td>
<td>${it.color}</td> <td>${it.name}</td>
</tr>
</g:each>
</table>
</div>
</body>
</html>

  與我之前使用的方式類似,您現在也可以使用run-app 命令運行應用程序,或者創建一個war文件并將其部署到Tomcat。您應該在http://localhost:8080/ClothesMgt/display/下看到新的視圖,如圖4所示。

Groovy和Grails簡介圖-4

  圖4:新創建的列出襯衫和褲子清單的視圖

  現在讓我們快速討論一下Grails服務。

服務

  如果您想知道如何分開業務邏輯以及放置業務邏輯的位置,答案在Grails 服務中。服務以SomeNameService.groovy 格式命名,且被置于 /grails-app/services/目錄。服務可利用依賴注入特性,您能夠輕松地從控制器內部調用這些服務。

  讓我們來看一個使用服務的例子。首先,使用create-service 命令創建新服務。運行此命令并命名服務Order。Grails將創建兩個文件——grails-app/services/OrderService.groovy 和 grails-tests/OrderTests.groovy。

  現在編輯OrderService.groovy,如清單7所示。當引入新的orderGoods() 方法時會自動生成serviceMethod() 。

  清單7:OrderService.groovy

class OrderService {
boolean transactional = true
def serviceMethod() {
// TODO
}
def orderGoods() {
return "Order Placed - New shirts and trousers \
will be sent shortly."
}
}

  現在編輯DisplayController,如清單8所示。引入使用OrderService的重排閉包。請注意服務將由Groovy注入。

  清單8:DisplayController.groovy

class DisplayController {
OrderService orderService
def index = {redirect(action:list,params:params)}
def list = {
params['max'] = 10
return [ shirtList: Shirt.list( params )
, trouserList: Trouser.list( params )]
}
def reorder = {
render(orderService.orderGoods())
}
}

  現在當您訪問URL http://localhost:8080/ClothesMgt/display/reorder時,重排閉包將調用 OrderService,響應會被發回到瀏覽器。您能夠以類似方式將全部業務邏輯移入服務,然后使用Grails的注入功能非常輕松地使用它們。

動態方法和屬性

  正如之前提到的,域類沒有能夠從數據庫獲取數據或更新/刪除現有數據的任何方法,例如find()、 findAll() 或 save() 。在控制器中您也沒有編寫諸如 redirect() 或 render() 之類的方法。但是域類和控制器有它們的計劃目的,且允許所有要求的操作。原因是Grails中動態方法和屬性的存在。動態方法被動態添加到類,就好像功能是在程序中編譯的一樣。

  這些是可用的方法和屬性,無需編寫。這些動態方法涵蓋了大多數Web應用程序開發中會碰到的常見情況。對于域類來說,存在諸如find()、findAll()、list()、executeQuery()、save()和 delete()之類的動態方法??刂破骶哂兄T如session、request和response之類的動態屬性,以及諸如chain()、render()和 redirect()之類的方法。要真正利用Grails的強大功能,您需要了解所有這些動態方法和屬性的功能。

順便介紹一下:自動重載和@Property

  Grails的一個重要特性是能夠在開發過程中進行了更改時自動重載文件。因此只需編輯和保存gsp文件,就會自動重載新文件。然而這里創建的類似OrderService 的事務服務不會被重載。您會在服務器控制臺看到以下消息"[groovy] Cannot reload class [class OrderService] reloading of transactional service classes is not currently possible. Set class to non-transactional first. "。

  Grails的自動重載功能會為您節省許多時間,您就無需浪費時間來重啟服務器了。我碰到過一些Grails不能自動重載的案例,例如將一個jsp文件重命名到gsp。然而,Grails的這項功能有望在未來版本中得到進一步改進。

  在Groovy JSR 06 的之前版本中,您必須使用@Property 來定義Groovy中的新屬性。因此您會在線看到許多使用@Property的舊的Groovy例子。然而請注意,@Property已經從Groovy JSR 06中移除,在Grails 0.2和之后的版本中也不會再需要它。請參閱@Property 建議來獲得更多細節。

結束語

  在本文中,我介紹了Grails框架的基本特性,并使用Grails創建了一個應用程序。Groovy和Grails最大的好處是一切都運行在優秀的舊Java和Java EE上——因此您能夠使用Groovy和Grails的RAD特性快速開發應用程序,然后將應用程序部署到可靠的Java EE服務器上??紤]到關于Ruby和Rails的宣傳噪音,顯然需要一個Java備選方案。Groovy和Grails看起來非常適合這個角色。

下載

  下載本文中的代碼:

參考資料



cAng^Er 2008-02-21 12:40 發表評論
]]>
刪除.svn文件夾的JS腳本http://www.tkk7.com/xiaosao/archive/2008/02/19/180700.htmlcAng^ErcAng^ErTue, 19 Feb 2008 07:59:00 GMThttp://www.tkk7.com/xiaosao/archive/2008/02/19/180700.htmlhttp://www.tkk7.com/xiaosao/comments/180700.htmlhttp://www.tkk7.com/xiaosao/archive/2008/02/19/180700.html#Feedback2http://www.tkk7.com/xiaosao/comments/commentRss/180700.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/180700.html今天在網上游蕩發現了個好東西~保存下來!一直都苦于無法一下將舊項目下面的.SVN文件夾利落刪除~今天恰巧發現了一個方法,在此很是感謝~~
  原文如下:http://www.ljnid.cn/?id=261
   寫了個JS腳本,可以刪除指定文件夾下面的所有.SVN文件夾
   在windows server 2003下測試通過,不保證XP可以運行
   使用方法:把下面的代碼保存為xxx.js
  在控制臺用:cscript.exe xxx.js X:\xxx\xx
  其中X:\xxx\xx是你指定的路徑


if(WScript.Arguments.length==0)
WScript.quit();

var fso = new ActiveXObject("Scripting.FileSystemObject");
var fc,folder,folderList=[];
var rootFolder = fso.getFolder(WScript.Arguments(0));

folderList.push(rootFolder);
while(folderList.length>0){
folder = folderList.pop();
fc = new Enumerator(folder.SubFolders);
while(!fc.atEnd())
{
if(fc.item().name =='.svn')
fso.deleteFolder(fc.item().path,true);
else
folderList.push(fc.item());

fc.moveNext();
}
}


cAng^Er 2008-02-19 15:59 發表評論
]]>
<又是轉來的> 面試java高級工程師、項目經理等的常見問題http://www.tkk7.com/xiaosao/archive/2007/11/09/159386.htmlcAng^ErcAng^ErFri, 09 Nov 2007 08:02:00 GMThttp://www.tkk7.com/xiaosao/archive/2007/11/09/159386.htmlhttp://www.tkk7.com/xiaosao/comments/159386.htmlhttp://www.tkk7.com/xiaosao/archive/2007/11/09/159386.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/159386.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/159386.htmlhttp://paulandjoy.javaeye.com/blog/126315
1、 類、對象的概念:
1) 類:具有共同屬性和行為的對象的抽象。類是創建對象的模板。
2) 對象:現實世界中的實體。在計算機中,是指可標識的存儲區域。
3) 類是對象的抽象、對象是類的實例。
2、 抽象:是從特定的實例中抽取共同性質形成一般化概念的過程。
3、 接口與抽象類:
1)接口和抽象類都用于抽象,接口是抽象類的抽象。
2)接口中只有方法聲明,沒有實現(無方法體);在接口中聲明的方法具有public和abstract屬性,一個類可以實現多個接口(即多繼承),接口以‘,’分隔;接口中的方法必須全部實現。
3)抽象類可以有部分方法實現,抽象類必須通過繼承才能使用。
4、內部類(Inner Class):
1)內部類是嵌套在另一個類中的類。
2)內部類用于名稱隱藏和程序代碼的組織,另外內部類擁有直接訪問其外部類所有成員(包括private的)的權限(無需任何關鍵字修飾)。
3)內部類不可以在其他類或main方法里實例化,必須使用如下方法(非靜態內部類)
外部類.內部類 對象名=new 外部類().new 內部類();
靜態內部類調用方式:
外部類.內部類 對象名=new 外部類.內部類();
4)非靜態內部類不可以聲明靜態成員;靜態內部類的非靜態成員可以訪問其外部類的靜態成員,聲明為靜態的成員不可以訪問外部的非靜態成員。
5、訪問修飾符限制:
Private protected friendly(default) public
同類 Y Y Y Y
同包不同類 N Y Y Y
同包子類 N Y Y Y
不同包不同類 N N N Y
不同包子類 N Y N Y
6、Static關鍵字的使用:
1)類成員,直接使用 類名.成員 調用。
2)靜態方法只能訪問靜態成員。
3)靜態方法不能使用this、super關鍵字。
4)靜態方法不能被非靜態方法重寫或重載。
7、final關鍵字:
1)被final修飾的變量為常量不能改變。
2)被final修飾的方法不可以重寫。
3)被final修飾的類不能被繼承。
8、abstract關鍵字:
1)被abstract修飾的類不能實例化。
2)被abstract修飾的方法只能在子類中實現。
9、native關鍵字:非Java語言的編寫,例如JNI技術。
10、synchronized關鍵字:多線程的同步訪問控制。
11、分類列舉服務器和組件技術:
1)服務器端技術:Jsp、Servlet;
2)組件技術:JavaBean、EJB。
12、Http與Https:Https即多了安全的Http,s(Security Socket Layer)指加密套接字協議層(簡寫SSL)。
13、OSI(Open System Interconnection)網絡抽象模型:
1)由國際標準化組織(ISO)提出。
2)將互聯網分為七層,從下至上分別為:物理層(physical)、數據鏈路層(data link)、網絡層(network)、傳送層(transport)、會話層(session)、表示層(presentation)、應用層(application)。底層通過提供接口支持上層功能。各層詳解:
物理層:LAN/ATM,為硬件層。
數據鏈路層:LAN/ATM
網絡層:IP協議,IOS
傳輸層:TCP/UDP協議,支持Java Socket。
會話層:
表示層:HTML、XML
應用層:HTTP協議,使用Java Servlet/JSP
<第八層(Web服務層):SOAP/UDDI>
14、J2EE的容器與服務器:
容器負責EJB組件中生命周期的控制;
服務器包含在容器外,提供系統級操作底層服務,包括事務、事件、多線程……。
15、繼承限制:
父類對象不可以賦給子類對象,因為子類可能具有更多的成員,反之可以。
16、邏輯操作:c=(a>b)?a:b;等同于下式
if(a>b) c=a; else c=b;
17、列舉常見集合框架類型:
1)List、Set、Map。由這三個接口實現出ArrayList、LinkedList、HashSet、TreeSet、HashMap、TreeMap等常用集合框架。
2)Vector屬于重量級組件不推薦使用。
3)Map類型維護鍵/值對,Hashtable與HashMap相近但效率略低于HashMap、高于TreeMap,TreeMap優點是可以排序。
4)Set類型可裝入唯一值,HashSet效率高于TreeSet但TreeSet可以維護內部元素的排序狀態。
5)List類型可按某種特定順序維護元素。ArrayList允許快速隨機訪問,但如果添加或刪除位于中間的元素時效率很低;LikedList提供最佳循序訪問及快速的中間位置添加刪除元素,并有addFirst、addLast、getFirst、getLast、removeFirst、removeLast方法。
18、面向對象的特征:
1)繼承:通過子類可以實現繼承,子類繼承父類的所有狀態和行為,同時添加自身的狀態和行為。
2)封裝:將代碼及處理數據綁定在一起的一種編程機制,該機制保證程序和數據不受外部干擾。
3)多態:包括重載和重寫。重載為編譯時多態,重寫是運行時多態。重載必須是同類中名稱相同參數不同(包括個數不同和類型不同),但返回類型不同不構成重載;重寫發生于子類對父類的覆蓋,子類繼承父類方法名相同、參數列表相同、返回類型相同才構成重寫。
19、Java命名規范:必須以英文字母、下劃線(’_’)或’$’開始,其余可以有數字但不允許 包含空格,且組合后的名稱不能是Java關鍵字或保留字。
匈牙利命名法:以m開始為類成員變量,以g開始為全局變量,以v開始為本地局部變量,常量命名一般不以下劃線、美元符開始。
駝峰命名:一般稱由多個單詞或縮寫組成的變量名,并且該變量名每個單詞首字母均為大寫(一般類名全部首字母大寫,方法或屬性名第一個字母小寫)的稱為駝峰命名。
20、Java語言共包含47個關鍵字。
21、設計模式:
一個設計模式描述了一個被證實可行的方案。這些方案非常普遍,是具有完整定義的最常用的模式。一般模式有4個基本要素:模式名稱(pattern name)、問題(problem)、解決方案(solution)、效果(consequences)。
常見23種模式概述:
1) 抽象工廠模式(Abstract Factory):提供一個創建一系列相關或相互依賴對象的接口,而無需指定它們具體的類。
2) 適配器模式(Adapter):將一個類的接口轉換成客戶希望的另外一個接口。適配器模式使得原本由于接口不兼容而不能一起工作的類可以一起工作。
3) 橋梁模式(Bridge):將抽象部分與它的實現部分分離,使它們都可以獨立地變化。
4) 建造模式(Builder):將一個復雜對象的構建與它的表示分離,使同樣的構建過程可以創建不同的表示。
5) 責任鏈模式(Chain of Responsibility):為解除請求的發送者和接收者之間耦合,而使多個對象都有機會處理這個請求。將這些對象連成一條鏈,并沿著這條鏈傳遞該請求,直到有一個對象處理它。
6) 命令模式(Command):將一個請求封裝為一個對象,從而可用不同的請求對客戶進行參數化;對請求排隊或記錄請求日志,以及支持可取消的操作。
7) 合成模式(Composite):將對象組合成樹形結構以表示“部分-整體”的層次結構。它使得客戶對單個對象和復合對象的使用具有一致性。
8) 裝飾模式(Decorator):動態地給一個對象添加一些額外的職責。就擴展功能而言,它能生成子類的方式更為靈活。
9) 門面模式(Facade):為子系統中的一組接口提供一個一致的界面,門面模式定義了一個高層接口,這個接口使得這一子系統更加容易使用。
10) 工廠方法(Factory Method):定義一個用于創建對象的接口,讓子類決定將哪一個類實例化。Factory Method 使一個類的實例化延遲到其子類。
11) 享元模式(Flyweight):運用共享技術以有效地支持大量細粒度的對象。
12) 解釋器模式(Interpreter):給定一個語言,定義它的語法的一種表示,并定義一個解釋器,該解釋器使用該表示解釋語言中的句子。
13) 迭代子模式(Iterator):提供一種方法順序訪問一個聚合對象中的各個元素,而又不需暴露該對象的內部表示。
14) 調停者模式(Mediator):用一個中介對象來封裝一系列的對象交互。中介者使各對象不需要顯式的內部表示。
15) 備忘錄模式(Memento):在不破壞封裝性的前提下,捕獲一個對象的內部狀態,并在該對象之外保存這個狀態。這樣以后就可將該對象恢復到保存的狀態。
16) 觀察者模式(Observer):定義對象間的一種一對多的依賴關系,以便當一個對象的狀態發生改變時,所有依賴于它的對象都得到通知并自動刷新。
17) 原始模型模式(Prototype):用原型實例指定創建對象的種類,并且通過拷貝這個原型創建新的對象。
18) 代理模式(Proxy):為其他對象提供一個代理以控制對這個對象的訪問。
19) 單例模式(Singleton):保證一個類僅有一個實例,并提供一個訪問它的全局訪問點。
20) 狀態模式(State):允許一個對象在其內部狀態改變時改變它的行為。對象看起來似乎修改了它所屬的類。
21) 策略模式(Strategy):定義一系列的算法,把它們一個個封裝起來,并且使它們可相互替換。本模式使得算法的變化可獨立于使用它的客戶。
22) 模板模式(Template Method):定義一個操作中的算法的骨架,而將一些步驟延遲到子類中。模板方法使得子類可以不改變一個算法的結構即可重定義該算法的某些特定步驟。
23) 訪問者模式(Visitor):表示一個作用于某對象結構中的各元素的操作。該模式可以實現在不改變各元素的類的前提下定義作用于這些元素的新操作。
1、 你認為一個項目如何進行才正確?(比如:盡一切可能快的完成任務或完全按照類似CMM來操作)
根據林銳博士的觀點:企業的根本目標是合法地賺取盡可能多的利潤,使企業利益最大化。企業所有的特定目標和行動都是圍繞上述根本目標開展的,任何背離根本目標的行動都將對企業造成傷害,應當杜絕。
基于此任何人都不要強調我將嚴格遵守XX模式,帶領團隊開發出具有XX等級的產品,企業需要的是能夠帶領團隊按時、合格的開發出產品的Manager。
2、 你經??椿蜃屑氀凶x過的書有哪些?
不用回答你看過的課本,枚舉幾個經典的當然前提是必須真的看過至少瀏覽過主題和目錄。比如《Java編程思想》、《Java模式》、《人月神話》等,由于將來要做的是team中的替補leader或真正的leader所以你必須說出軟工的東西。
3、 你認為你應聘我們公司的項目經理,你自身的優勢在哪?
1)融洽,沒有領導希望你帶領團隊每天打嘴仗然后他還要去開屁股(Sorry,這似乎不很文明),你必須說明你能在以往團隊中與其他人和諧相處。
2)技術,千萬不要謙虛,對方要的就是技術過硬、能力出眾的人才,你只需要說明你成功解決過什么難題并且你對J2EE、XX中間件、XX系統有多么的熟悉。
4、 如果給你一個Team,公司決定讓你的Team開發A產品,OK這恰好是你的強項,你們很快開發出來了,但很沮喪的結果是它(你們的產品)沒有銷路;經過討論公司又決定讓你的Team開發B產品,OK這恰好又是你的強項,你們很快開發出來了,但很沮喪的結果是它(你們的產品)又沒有銷路。請問你怎么面對這個問題,你是否覺得決策層很無能甚至要推翻他或者去一個更有前途的公司?
不要以為這個故事很單純,這應該是人力在考核你。你一定要告訴她(人力多半是女的),我個人對于失敗的項目肯定會難過(是的,我想了很久才想出“難過”這個詞,它很中性),不過我認為這恰恰認證了公司在革新和新技術探索方面的魄力(自己想怎么說,如果天下人都說這句那才是悲劇呢),我肯定會以個人的名義向領導層提出我自己的建議和看法當然它未必正確,我不會離開公司,因為有點小挫折未嘗不是好事。(你應該在這個問題上好好想想,盡量發揮到10分鐘)
5、 你認為項目中最重要的是哪些過程?
分析、設計階段(也可以加上測試,但千萬別說編碼或開發階段),根據《人月神話》的觀點:1/3 計劃;1/6 編碼;1/4 構件測試和早期系統測試;1/4 系統測試,所有的構件已完成
但根據國內目前的狀況一般公司不會有很多的分析與設計時間(這取決于公司規模和時間成本),這樣在一個工期很緊張的項目中我們應該盡量分配出進度優先級來,首先拿出客戶最希望看到的和最能證明成果的東西來,其他的留待2期甚至3期去作,你可以告訴客戶需要進一步調試(專業人員的欺騙手段,實際上就是在進行后續的開發)。
6、 如果給你一個4-6個人的Team,那么你怎么分配他們、管理他們?
管理能力和經驗的綜合題,可能沒有人有相同的觀點,那你可以按照某些思路來側面解答:我會挑選一個技術過硬的人作為我的替補和項目的輕騎兵,是的團隊中必須有機動人員,否則你的項目十有八九會夭折。其他的人會被平均的分配任務。
我們會在每周進行全面的任務分配,每個人獲取一周的大概工作,然后每天的工作由他自己完成并匯報。(很好,如果答出這些就差不多了,多說可能會出現漏洞)
7、 簡述常用的軟件開發文檔。
1) 可行性研究報告(某些公司或模型沒有)
2) 項目開發計劃
3) 軟件需求說明書(必有)
4) 數據要求說明書
5) 概要設計說明書(必有)
6) 詳細設計說明書(必有)
7) 數據庫設計說明書(必有)
8) 用戶手冊(一般會有)
9) 操作手冊(必有)
10) 模塊開發卷宗
11) 測試計劃(必有)
12) 測試分析報告
13) 開發進度月報
14) 項目開發總結報告
8、 簡述類的關系。
1) 當一個類是“一種”另一個類時:is-a關系
2) 當兩個類之間有關聯時:
一個類“包含”另一個類:has-a關系
一個類“使用”另一個類
還可以細分有聚合和組合(UML寶典)或聚集和組成(包括國內某些知名學術團體都這么說)。
聚集(aggregation)表示整體與各部分之間的關系。例如汽車與輪胎,沒有了汽車輪胎依然是一個整體。(用空心菱形表示)
組成是一種整體和部分所屬更強的聚集關系,每個部分只能屬于一個整體,沒有整體部分也就沒有存在的價值。比如桌子和桌腿,沒有桌子也就沒有桌腿的價值了。(用實心菱形表示)



cAng^Er 2007-11-09 16:02 發表評論
]]>
<轉>JavaScript substr() 和 substring() 方法的區別 http://www.tkk7.com/xiaosao/archive/2007/10/18/153968.htmlcAng^ErcAng^ErThu, 18 Oct 2007 09:55:00 GMThttp://www.tkk7.com/xiaosao/archive/2007/10/18/153968.htmlhttp://www.tkk7.com/xiaosao/comments/153968.htmlhttp://www.tkk7.com/xiaosao/archive/2007/10/18/153968.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/153968.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/153968.html又是轉過來的文章,最近在學JavaScripts,用到了substr()和substring(),上網找了下,看到了這個
 轉載到這里,以便瀏覽

  原文地址:http://www.cnblogs.com/tishifu/archive/2007/08/03/841429.html

substr 方法
返回一個從指定位置開始的指定長度的子字符串。

stringvar.substr(start [, length ])

參數
stringvar

必選項。要提取子字符串的字符串文字或 String 對象。

start

必選項。所需的子字符串的起始位置。字符串中的第一個字符的索引為 0。

length

可選項。在返回的子字符串中應包括的字符個數。

說明
如果 length 為 0 或負數,將返回一個空字符串。如果沒有指定該參數,則子字符串將延續到 stringvar 的最后。

示例
下面的示例演示了substr 方法的用法。

function SubstrDemo(){
   var s, ss;                // 聲明變量。
   var s = "The rain in Spain falls mainly in the plain.";
   ss = s.substr(12, 5);  // 獲取子字符串。
   return(ss);               // 返回 "Spain"。
}


substring 方法
返回位于 String 對象中指定位置的子字符串。

strVariable.substring(start, end)
"String Literal".substring(start, end)

參數
start

指明子字符串的起始位置,該索引從 0 開始起算。

end

指明子字符串的結束位置,該索引從 0 開始起算。

說明
substring 方法將返回一個包含從 start 到最后(不包含 end )的子字符串的字符串。

substring 方法使用 start 和 end 兩者中的較小值作為子字符串的起始點。例如, strvar.substring(0, 3) 和 strvar.substring(3, 0) 將返回相同的子字符串。

如果 start 或 end 為 NaN 或者負數,那么將其替換為0。

子字符串的長度等于 start 和 end 之差的絕對值。例如,在 strvar.substring(0, 3) 和 strvar.substring(3, 0) 返回的子字符串的的長度是 3。

示例
下面的示例演示了 substring 方法的用法。

function SubstringDemo(){
   var ss;                         // 聲明變量。
   var s = "The rain in Spain falls mainly in the plain..";
   ss = s.substring(12, 17);   // 取子字符串。
   return(ss);                     // 返回子字符串。
}



cAng^Er 2007-10-18 17:55 發表評論
]]>
<轉>實戰SVN For Apache2(二)http://www.tkk7.com/xiaosao/archive/2007/10/17/153667.htmlcAng^ErcAng^ErWed, 17 Oct 2007 11:47:00 GMThttp://www.tkk7.com/xiaosao/archive/2007/10/17/153667.htmlhttp://www.tkk7.com/xiaosao/comments/153667.htmlhttp://www.tkk7.com/xiaosao/archive/2007/10/17/153667.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/153667.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/153667.html 在上一篇文章中,我們已經實現了對SVN和Apache的安裝,并將它們集成在一起工作。這一篇則主要介紹如何對SVN For Apache進行一些配置。
  1.    在Apache下為SVN建立多個庫。上篇中雖然也建立了一個SVN的庫在Apache下,但是我們不可能一個SVN服務器上只有一個項目,或者所有項目共用一個SVN庫進行開發,在Apache上為SVN建立多個庫可以通過建立多個虛擬目錄的方式實現。下面我們就來講講具體的步驟(注意,此篇中的配置和上篇中的配置方式有很大區別)

    1. 到F盤建立我們的SVN代碼存放位置目錄:SubversionFiles,并在其中建立一個conf目錄,將會把對Apache對SVN的配置文件都放在這里
    2. 再在F:\SubversionFiles下目錄Test1和Test2,并將它們變成SVN庫(用svnadmin create命令或者用TortoiseSVN來創建都可以)
    3. 在httpd.conf的最后添加如下代碼

      #
      # SVN 項目配置文件,格式:Include 配置文件名
      #為每個虛擬目錄設置一個配置文件,用Include命令加載

      #
      Include F:/SubversionFiles/conf/*.project

      即把F:/SubversionFiles/conf/目錄下的所有*.project文件都包到httpd.conf中來


    4. 在F:/SubversionFiles/conf/創建Test1.project和Test2.project文件,Test1內容如下,Test2可以做相應的修改

      #SVN項目名

           DAV svn

          #SVN項目的路徑
           SVNPath F:\SubversionFiles\Test1

          #SVN項目的路徑權限授權文件存放路徑
          AuthzSVNAccessFile F:\SubversionFiles\conf\Test1.access

      #對于所有用戶都需要身份驗證
          Satisfy Any
           Require valid-user

      #驗證方式
          AuthType Basic

          #項目的名稱
           AuthName "My Test1
      "

          #用戶信息
           AuthUserFile  F:\SubversionFiles\conf\user




      以上配置含義是在Apache下建立了一個叫作Test1的SVN虛擬目錄,指向F:\SubversionFiles\Test1,授權配置文件是F:\SubversionFiles\conf\Test1.access,用戶信息和密碼文件是F:\SubversionFiles\conf\user


    5.  通過以上配置,我們就在Apache上建立了Test1和Test2兩個虛擬目錄,也就是兩個獨立的SVN庫,我們還需要兩個文件訪問權限配置的.access和包含登錄用戶的user。 在F:\SubversionFiles\conf目錄中創建用戶列表文件user,方法在上篇中已經說過,也就是htpasswd命令,我們需要創建Xrinehart, X-Killer, Guest三個用戶。

    6. 接下來,我們為Test1庫創建它的訪問權限文件F:\SubversionFiles\conf\Test1.access。在上篇文章中我們是采用對每個用戶分別賦予對應權限的方式,如果用戶比較多,管理起來就會比較麻煩,這時我們需要用用戶組來對用戶進行管理。下面我們就來嘗試用用戶組配置訪問權限。Test1.access文件內容如下:


      [groups]
      AdminGroup = Xrinehart
      GuestGroup = guest

      [/]
      @AdminGroup = rw
      @GuestGroup = r

      X-Killer = rw


              在上面的配置中,通過[groups]節定義了兩個用戶組:AdminGroup和GuestGroup,并將Xrinehart用戶歸入AdminGroup用戶組,guest用戶歸入GuestGroup。再在下面的[/]定義了他們對Test1庫的根目錄的操作權限:AdminGroup組有讀寫權限,GuestGroup組只有讀權限,X-Killer用戶擁有讀寫權限。這里需要注意的是定義權限的時候,用戶組需要在前面加@符號來標示。

    7. 為了測試效果,我們可以將Test2.access文件寫為:

      [groups]
      AdminGroup = Xrinehart, X-Killer
      GuestGroup = guest

      [/]
      @AdminGroup = r
      @GuestGroup = rw


    8. 保存好上面的所有配置文件之后,重新啟動Apache服務器。此時我們就可以在IE里面通過http://localhost:8080/Test1http://localhost:8080/Test2 來訪問Test1和Test2庫了,你可以測試一下,用戶權限是否符合下面這個列表所示?做到這里,SVN在Apache上建立多個庫的配置也就算完成了。


                                                       Test1                                            Test2

      Xrinehart                                    rw                                                  r
      X-Killer                                       rw                                                  r
      guest                                          r                                                    rw


cAng^Er 2007-10-17 19:47 發表評論
]]>
<轉>實戰SVN For Apache2(一) http://www.tkk7.com/xiaosao/archive/2007/10/17/153664.htmlcAng^ErcAng^ErWed, 17 Oct 2007 11:39:00 GMThttp://www.tkk7.com/xiaosao/archive/2007/10/17/153664.htmlhttp://www.tkk7.com/xiaosao/comments/153664.htmlhttp://www.tkk7.com/xiaosao/archive/2007/10/17/153664.html#Feedback0http://www.tkk7.com/xiaosao/comments/commentRss/153664.htmlhttp://www.tkk7.com/xiaosao/services/trackbacks/153664.html原文地址:http://www.cnblogs.com/Xrinehart/archive/2005/10/27/262759.aspx
  SVN最常見的配置模式便是與Apache集成,這樣SVN可以采用WebDev協議方式提供服務。主要的好處有:通過Apache以HTTP方式穿透防火墻,便于在互聯網環境使用;提供目錄的版本控制;純2進制的文件對比方式等。下面將介紹如何將SVN集成在Apache上。

  1. 準備工作:

    1. 下載軟件:
      1. SVN服務器端程序。到官方網站 的下載二進制安裝文件,來到二進制包下載部分  ,找到 Windows NT, 2000, XP and 2003部分,然后選擇"this directory",這樣我們可以看到許多下載的內容,目前可以下載 svn-1.2.3-setup.exe

      2. TortoiseSVN,客戶端程序。TortoiseSVN是擴展Windows Shell的一套工具,可以看作Windows資源管理器的插件,安裝之后Windows就可以識別Subversion的工作目錄。
        官方網站是TortoiseSVN,下載方式和前面的svn服務器類似,在Download頁面的我們選擇Official version for Win2k/XP or higher的版本,然后在sourceforge的下載頁面選擇目前的最高穩定版本的安裝文件TortoiseSVN-1.2.5.4719-svn-1.2.3.msi。(注意:TortoiseSVN有一個特殊對VS.Net運行時一個BUG修正的版本,在下載頁面有選擇:Special version for Win2k/XP or higher: (We provide NO support for this!) uses _svn folders instead of .svn to work around the VS.NET bug with web projects. If you don't use web projects then please use the official version. )

      3. TortoiseSVN的中文語言包,注意這個需要下和客戶端程序版本相同的。可以在Download頁面內的Language Packs選擇下載。

      4. Apache服務器程序??梢缘?a >Apache官方網站下載最新版本的Apache,并且SVN必須運行在Apache2以上版本,在下載頁面選擇Windows的安裝包文件apache_2.0.55-win32-x86-no_ssl.msi

      5. AnkhSVN,可選安裝VS.Net的SVN插件。有0.5和0.6版本選擇,0.6版支持VS2005 Beta2,下載頁面。

    2. 停止IIS,因為Apache默認安裝為80端口,需要關閉IIS避免端口沖突??梢栽诎惭b好之后再調整端口。


  2. 開始安裝:

    1. 先安裝Apache服務器。在到填寫ServerInfo的時候,若機器已經啟用了AD,安裝程序會自動填寫,否則自己手動填寫。安裝完成之后到安裝目錄下找到conf目錄下的httpd.config文件,搜索Listen,并將Listen后面的80改為8080(或者你希望的端口),然后重新啟動Apache服務器,這時可以用http://localhost:8080 來測試是否安裝正確。

    2. 再安裝SVN服務器,沒有什么好說的,一路回車就可以了。(安裝SVN時,請確保Apache服務器正在運行狀態,安裝過程中可能會將Apache關閉,重新啟動屬于正常現象)

  3. 配置Apache服務器與SVN集成:


    1. 首先將Subversion安裝目錄bin\下面的兩個文件:mod_authz_svn.somod_dav_svn.so復制到Apache安裝目錄modules\目錄下。
    2. 有些文章說需要再修改httpd.conf,并做如下修改:
      找到Apache安裝目錄下的conf目錄,用文本編輯器打開httpd.conf,找到一下兩行:

      #LoadModule dav_module modules/mod_dav.so
      #LoadModule dav_fs_module modules/mod_dav_fs.so

      將每行前面的注釋符“#”去掉。再在所有LoadModule語句的最后添加一下幾行:

      #SVN
      LoadModule dav_svn_module modules/mod_dav_svn.so
      LoadModule authz_svn_module modules/mod_authz_svn.so

      但是我安裝完之后發現這個文件已經被做了那些修改,大概是新版的SVN安裝程序做了一些優化吧。其實在成功安裝SVN之后,Apache服務器已經和SVN初步集成。如圖:

      Apache.jpg

      注意下面的狀態欄已經是Apache/2.0.55 SVN1.2.3 DAV/2

    3. 建立一個SVN存放文件的目錄,我在F盤下創建一個文件夾:F:/SubversionFiles
    4. 接下來我們必須告訴Apache我們的資源庫所在的路徑,可以通過Location指令來完成這個設置。因為不希望為每個單獨的項目都進行單獨的設置,所以我們把所有項目都存放在統一的資源庫目錄,那么可以使用SVNParentPath指令來指定存放所有項目的路徑。在httpd.conf文件最后添加下面配置:

      #SVN

      <Location /svn>
          DAV svn
          SVNParentPath "F:/SubversionFiles"
      </Location>

    這樣我們就可以通過http://myhost:8080/svn/<項目名> 來訪問存放于資源庫F:/SubversionFiles中的指定項目。當然有可能你并不希望某個項目提供這樣一種訪問方式,這時候你可以使用SVNPath為每個項目進行單獨的設置,SVNPath的使用方法如下:

        DAV svn
        SVNPath "F:/SubversionFiles/project1"

    同樣把這段配置放在httpd.conf最后,重啟Apache HTTP服務即可通過http://myhost:8080/svn/project1 來訪問project1項目的資源庫。

    1. 下面作一個測試:

      1. 到F:\SubversionFiles下建立一個子目錄Test,然后到Aapche下的Bin目錄下執行svnadmin create F:\SubversionFiles\Test

      2. 打開瀏覽器輸入網址:http://localhost:8080/svn/Test ,應該可以正確訪問,但當前該項目下還沒有加入任何內容,所以顯示為空的

      3. 有一點需要提示的是,現在的訪問是完全匿名的,任何人都可以對SVN進行操作。所以我們接下來利用Apache的權限管理功能來對SVN進行用戶驗證集成

    2. 加入用戶身份驗證:

      在確定對訪問用戶的權限控制之前,你必須規劃好是對整個資源庫中的所有項目還是單獨的某一個項目進行統一的身份驗證, 也就是我們前面講到的是使用SVNParentPath還是SVNPath的問題。

      最簡單的身份驗證方式是使用Basic HTTP Authentication機制,該方式通過用戶名和口令對訪問用戶進行身份驗證。我們可以直接通過Apache提供的支持進行設置。Apache提供一個htpasswd工具來管理用戶名和口令。接下來我們利用這個工具來添加兩個用戶。

      在命令行窗口中轉到Apache所在的目錄,執行下列命令

      說明:創建用戶Xrinehart
      輸入:htpasswd –c F:\SubversionFiles\svn_auth_passwd Xrinehart
      說明:使用-c參數來創建一個passwd文件
      輸出:
      New password: *****
      Re-type new password: *****
      Adding password for user Xrinehart

      再創建用戶的時候就不用-c參數,而是用-m參數,因為文件svn_auth_passwd經創建。

      打開svn_auth_passwd文件,密碼使用MD5加密過了,而且同樣的密碼加密出來的內容卻不相同

      接下來我們必須告訴Apache服務器如何使用這個passwd文件,打開httpd.conf找到剛才我們添加的Location配置的位置,修改如下:

      #
      # SVN
      #

        DAV svn
        SVNParentPath "F:/SubversionFiles"

      # how to authenticate a user
       AuthType Basic
       AuthName "Subversion repository"
       AuthUserFile "F:/SubversionFiles/svn_auth_passwd"
       
        # only authenticated users may access the repository
      Require valid-user


      重新啟動Apache HTTP服務器,使用瀏覽器打開 http://localhost:8080/svn/Test 你將會看到要求登錄的對話框,輸入你剛設置的用戶名和口令即可。

      這樣Apache和SVN的集成就基本告成了



cAng^Er 2007-10-17 19:39 發表評論
]]>
主站蜘蛛池模板: 成人免费区一区二区三区| 亚洲精品福利网泷泽萝拉| 免费一级毛片女人图片| 日本黄色免费观看| 日韩a在线观看免费观看| 国产免费一区二区三区VR| 国内自产拍自a免费毛片| 韩国18福利视频免费观看| 成年男女男精品免费视频网站 | 亚洲国产精品无码专区| 色久悠悠婷婷综合在线亚洲| 亚洲色图综合在线| 亚洲中文字幕无码不卡电影| 国产亚洲一区二区在线观看 | www视频在线观看免费| 99re在线免费视频| 最近中文字幕电影大全免费版| 亚洲精品免费观看| 1区2区3区产品乱码免费| 99re热免费精品视频观看| 毛片大全免费观看| 永久免费看bbb| 免费在线黄色网址| 国产成人亚洲精品狼色在线| 亚洲国产精品嫩草影院在线观看| 久久青草亚洲AV无码麻豆| 亚洲国产精品线观看不卡| 亚洲日韩乱码中文字幕| 全部一级一级毛片免费看| 成人无码区免费A∨直播| 久久久精品2019免费观看 | 国产亚洲精品成人久久网站 | 亚洲国产美女福利直播秀一区二区| 亚洲一卡二卡三卡| 国产精品日本亚洲777| 国产日韩精品无码区免费专区国产| 国内精品一级毛片免费看| 2021久久精品免费观看| 国产真实伦在线视频免费观看| 亚洲伊人成无码综合网 | 久久精品国产亚洲AV麻豆~ |