|
2013年3月13日
現(xiàn)在做互聯(lián)網(wǎng)產(chǎn)品的團(tuán)隊(duì)都比較小,也可能沒(méi)有特別多運(yùn)維人員。因此特別需要用一些系統(tǒng)或是工具來(lái)監(jiān)控服務(wù)器或者是服務(wù)是否正常。之前比較直接的做法是自己搭建一套開(kāi)源的監(jiān)控系統(tǒng),現(xiàn)在隨著云服務(wù)器的流行,也有越來(lái)越多的人用戶(hù)會(huì)使用云端的監(jiān)控平臺(tái)。
從我的經(jīng)驗(yàn)來(lái)看,云服務(wù)器監(jiān)控是有些特別的好處的:
1.自建的監(jiān)控平臺(tái),有一部分問(wèn)題是沒(méi)辦法發(fā)出警報(bào)。比如,一般監(jiān)控服務(wù)器也會(huì)在內(nèi)部網(wǎng)絡(luò)中,如果出現(xiàn)外部網(wǎng)絡(luò)問(wèn)題,監(jiān)控服務(wù)是沒(méi)辦法通知到相關(guān)人員(因?yàn)榘l(fā)郵件或者是發(fā)短信的通路也會(huì)出現(xiàn)問(wèn)題)。
2.自建的監(jiān)控平臺(tái),只能檢測(cè)到一個(gè)點(diǎn)的訪問(wèn)情況。阿里云監(jiān)控平臺(tái)可以檢測(cè)杭州和青島兩個(gè)節(jié)點(diǎn)(這是我的帳號(hào)看到的),可以比較有效地了解非監(jiān)測(cè)點(diǎn)的一個(gè)訪問(wèn)情況。
3.云監(jiān)控平臺(tái)都有一套不錯(cuò)的管理界面,可有效減少部署維護(hù)和使用成本。
下面我介紹一下,我使用過(guò)的一些云監(jiān)控平臺(tái),并對(duì)他們的優(yōu)點(diǎn)和缺點(diǎn)進(jìn)行比較。
阿里云監(jiān)控
阿里的云服務(wù)器在市場(chǎng)中是做的很不錯(cuò)的,我的很多朋友都購(gòu)買(mǎi)了阿里的云服務(wù)器。以阿里的公司實(shí)力,做一款云監(jiān)控的產(chǎn)品應(yīng)該不會(huì)差到哪里去。
優(yōu)點(diǎn):
1.產(chǎn)品體驗(yàn)好,進(jìn)入住界面就能輕而易舉找到你想要的功能。
2.監(jiān)控功能全,包括站點(diǎn)監(jiān)控、服務(wù)器監(jiān)控和自定義監(jiān)控。
3.多點(diǎn)監(jiān)控,可以在全國(guó)提供幾個(gè)點(diǎn)的監(jiān)控。從用戶(hù)的角度了解服務(wù)是否可用。
4.站點(diǎn)監(jiān)控功能也很全面,包含SMTP、POP3、FTP監(jiān)控。
5.免費(fèi),且監(jiān)控站點(diǎn)數(shù)沒(méi)限制。
缺點(diǎn):
1.服務(wù)器監(jiān)控和自定義監(jiān)控,要求是云服務(wù)器,如果不是云服務(wù)器,只能使用站點(diǎn)檢測(cè)功能。
360網(wǎng)站服務(wù)監(jiān)控
360的個(gè)人用戶(hù)產(chǎn)品很少使用,但是360的企業(yè)產(chǎn)品還真有不少做的不錯(cuò)的。說(shuō)說(shuō)360監(jiān)控的優(yōu)缺點(diǎn)。
優(yōu)點(diǎn):
1.提供服務(wù)器監(jiān)控。可以監(jiān)控到服務(wù)器的磁盤(pán),CPU,內(nèi)存等情況。
2.UI和告警都設(shè)計(jì)的不錯(cuò)。
3.免費(fèi)。
缺點(diǎn):
1.需要在服務(wù)器上開(kāi)SNMP協(xié)議。(不過(guò)360提供很多腳本工具,可以一鍵安裝)
2.有20臺(tái)服務(wù)器的限制。(這個(gè)基本上夠用,超過(guò)20臺(tái)服務(wù)器,可以購(gòu)買(mǎi)服務(wù)了)
監(jiān)控寶
名字很專(zhuān)一,一看就知道是專(zhuān)業(yè)做監(jiān)控的。
優(yōu)點(diǎn):
1.專(zhuān)注監(jiān)控,界面設(shè)計(jì)還算可以。
2.監(jiān)控功能全面,內(nèi)網(wǎng)采集的方式較多??梢圆杉瘮?shù)據(jù)庫(kù)數(shù)據(jù)。
缺點(diǎn):
1.免費(fèi)用戶(hù),服務(wù)器和網(wǎng)站監(jiān)控都有限制,而且數(shù)量極少2臺(tái)服務(wù)器監(jiān)控,5臺(tái)網(wǎng)站監(jiān)控。
作者簡(jiǎn)介: qiyadeng(www.qiyadeng.com)對(duì)互聯(lián)網(wǎng)技術(shù)、運(yùn)營(yíng)及市場(chǎng)領(lǐng)域有濃厚的興趣,喜愛(ài)思考、閱讀、討論;擅長(zhǎng)Java開(kāi)發(fā)及分布式技術(shù)?,F(xiàn)專(zhuān)注于互聯(lián)網(wǎng)的創(chuàng)新產(chǎn)品– 老來(lái)寶(http://www.laolaibao.com),幫助年輕用戶(hù)獲得補(bǔ)充養(yǎng)老金,并提供養(yǎng)老金增值產(chǎn)品。
比較少參加這類(lèi)大型的會(huì)議,進(jìn)到會(huì)場(chǎng)的第一眼,發(fā)現(xiàn)會(huì)議室已經(jīng)全部坐滿(mǎn),后來(lái)主辦方發(fā)現(xiàn)站的人太多了,找來(lái)了一些小板凳,我快速找到一個(gè)小板凳坐下。坐下開(kāi)始認(rèn)真聽(tīng),非常開(kāi)心地聽(tīng)到廣告時(shí)間結(jié)束,和我計(jì)劃的時(shí)間完美一致。
回顧一下我比較關(guān)系的幾個(gè)主題
基于用戶(hù)畫(huà)像的大數(shù)據(jù)實(shí)例
演講嘉賓是聯(lián)通沃商店的大數(shù)據(jù)技術(shù)經(jīng)理,該大數(shù)據(jù)實(shí)例主要是通過(guò)聯(lián)通營(yíng)運(yùn)商的數(shù)據(jù)和沃商店進(jìn)行分析,通過(guò)繪制用戶(hù)畫(huà)像的形式,在其他應(yīng)用場(chǎng)景,如廣告、游戲下載中為用戶(hù)推薦用戶(hù)喜歡的產(chǎn)品??梢钥闯鰜?lái)嘉賓技術(shù)實(shí)例及基礎(chǔ)功是十分不錯(cuò),至少是一個(gè)碩士畢業(yè)。近些年被大家掛在嘴邊的機(jī)器學(xué)習(xí)算法、推薦算法、語(yǔ)義分析都有部分介紹,實(shí)在是接受不過(guò)來(lái);比較熟悉的還是我們當(dāng)初碩士的專(zhuān)業(yè)方向推薦算法,看到了簡(jiǎn)單的介紹覺(jué)得很親切。不過(guò)后來(lái)提問(wèn)環(huán)節(jié)看,現(xiàn)場(chǎng)還是很多高人,有不少是做這個(gè)領(lǐng)域的。不過(guò)歸根是國(guó)有企業(yè)和類(lèi)似研究機(jī)構(gòu),是否能產(chǎn)生非常大的價(jià)值,我表示懷疑,不過(guò)這些算法一羅列,對(duì)經(jīng)費(fèi)的分配還是很有好處的。
電商系統(tǒng)的心得分享
這又是一個(gè)國(guó)有企業(yè),號(hào)稱(chēng)是線上賣(mài)大力丸的人(國(guó)藥1健康)。從技術(shù)成長(zhǎng)為總經(jīng)理,有很多心得體會(huì)。感覺(jué)和我有那么一點(diǎn)像,有一些體會(huì)也迫不及待的分析給這些年輕的IT從業(yè)者,為人嚴(yán)肅,總是會(huì)把困難估計(jì)的充分一點(diǎn)(估計(jì)年輕也沒(méi)少教學(xué)費(fèi))。演講中說(shuō)了構(gòu)建系統(tǒng)中的四個(gè)原則
權(quán)限獨(dú)立,相互制約
非常務(wù)實(shí)的看到某些大型企業(yè)的,部門(mén)斗爭(zhēng)。從系統(tǒng)層面開(kāi)始設(shè)計(jì)制約(這個(gè)應(yīng)該非常符合老板心意)。這個(gè)對(duì)很多小型企業(yè)在成才過(guò)程中是非常有幫助的。
設(shè)計(jì)流程 減少犯錯(cuò)
在電商行業(yè)非常清楚客服和倉(cāng)庫(kù)的員工流動(dòng)性,以及普遍受教育程度偏低,通過(guò)流程設(shè)計(jì),而不是提高對(duì)用人的要求。這也是非常務(wù)實(shí)的方法?;仡欀霸谙到y(tǒng)層面獨(dú)自設(shè)計(jì)支持中央預(yù)訂系統(tǒng),設(shè)計(jì)出來(lái)的自動(dòng)傳真(當(dāng)時(shí)網(wǎng)絡(luò)不如現(xiàn)在易得)及新訂單提醒(感謝施總的支持,增加音響進(jìn)行聲音)等等,簡(jiǎn)直覺(jué)得找到了知音。
多了解一些財(cái)務(wù)知識(shí)。
談到的兩點(diǎn)是數(shù)據(jù)之間需要有勾稽關(guān)系和不能修改歷史數(shù)據(jù),很驕傲我對(duì)財(cái)務(wù)的理解還是不錯(cuò),從未犯過(guò)這種不靠譜的錯(cuò)誤。
跨平臺(tái)大型在線客服系統(tǒng)的技術(shù)構(gòu)架
嘉賓談了的是一套客服系統(tǒng),比較多的關(guān)鍵字是客服妹子,可以看出IT從業(yè)者苦中作樂(lè)的精神。給我的體會(huì)是,客服系統(tǒng)都可以做成這樣。從一個(gè)項(xiàng)目到一個(gè)產(chǎn)品,在云計(jì)算的世界,可以好一個(gè)客服的組件,也是有很大的價(jià)值。和我的理想事業(yè)很接近,可以花上一生中最精華的時(shí)間,做好一個(gè)有價(jià)值的小眾專(zhuān)業(yè)的行業(yè)。
阿里分布式數(shù)據(jù)庫(kù)服務(wù)實(shí)踐
阿里的人就是高調(diào),上場(chǎng)就調(diào)戲京東雙11前系統(tǒng)崩潰。我也經(jīng)歷過(guò)很多系統(tǒng)崩潰,簡(jiǎn)直是開(kāi)發(fā)人員的噩夢(mèng),也是IT人員信用受損的嚴(yán)重事件(因此我一直比較注意防止崩潰及崩潰后的快速恢復(fù))。回到分布式數(shù)據(jù)庫(kù),這個(gè)是收獲最大的一個(gè)演講。雖然這個(gè)演講看上去是再給阿里云的DRDS做宣傳,但是嘉賓演講的很進(jìn)行,深入淺出地介紹了分布式數(shù)據(jù)庫(kù)和單機(jī)數(shù)據(jù)庫(kù)的區(qū)別。對(duì)分布式事務(wù)的重新認(rèn)識(shí)是一個(gè)很大的收獲,以前一直把教程中的數(shù)據(jù)庫(kù)原理中的事務(wù)定義,作為分布式事務(wù)需要解決的問(wèn)題,其實(shí)不是。需要更加務(wù)實(shí),在淘寶阿里這類(lèi)訂單處理系統(tǒng)中,有一類(lèi)對(duì)分布式事務(wù)的模式(異步消息機(jī)制);在其他領(lǐng)域會(huì)有其他模式分布式的事務(wù)模型,這些分布式模型肯定都不滿(mǎn)足單機(jī)的事務(wù)模型,但是可以滿(mǎn)足和解決相應(yīng)領(lǐng)域的問(wèn)題。
平臺(tái)架構(gòu)的服務(wù)器監(jiān)控
一個(gè)APP的監(jiān)控模型,猜測(cè)項(xiàng)目立項(xiàng)的原因,有兩個(gè)。一個(gè)是和競(jìng)爭(zhēng)對(duì)手的數(shù)據(jù)比較(UPYUN的對(duì)手主要是七牛),一個(gè)其實(shí)可以真正從用戶(hù)的角度看,用戶(hù)的體檢速度如何,以及影響用戶(hù)體驗(yàn)速度的真實(shí)原因。目前一般行業(yè)還不會(huì)做的這么細(xì),因?yàn)閁PYUN是技術(shù)支持公司,因此一定需要用這些數(shù)據(jù)去說(shuō)服和支持用戶(hù)。我們現(xiàn)在做的比較多的服務(wù)器的可用性、性能和應(yīng)用的可用性、性能監(jiān)控。前端時(shí)間剛好再比較,發(fā)現(xiàn)互聯(lián)網(wǎng)上有不少好的監(jiān)控平臺(tái),一般的創(chuàng)業(yè)公司,可以無(wú)需自己搭建監(jiān)控平臺(tái),接入到相應(yīng)的監(jiān)控平臺(tái)即可。下次再開(kāi)文進(jìn)行討論。
作者簡(jiǎn)介:qiyadeng(www.qiyadeng.com)對(duì)互聯(lián)網(wǎng)技術(shù)、運(yùn)營(yíng)及市場(chǎng)領(lǐng)域有濃厚的興趣,喜愛(ài)思考、閱讀、討論;擅長(zhǎng)Java開(kāi)發(fā)及分布式技術(shù)?,F(xiàn)專(zhuān)注于互聯(lián)網(wǎng)的創(chuàng)新產(chǎn)品--老來(lái)寶(http://www.laolaibao.com),立志于幫助廣大鳳凰(diao)男(si)提供補(bǔ)充養(yǎng)老金管理平臺(tái)。
摘要: guava是Java的一個(gè)擴(kuò)展類(lèi)庫(kù),在google的許多項(xiàng)目中使用過(guò)了,現(xiàn)在最為一個(gè) 開(kāi)源的Java類(lèi)庫(kù)廣泛使用(http://code.google.com/p/guava-libraries/)。
guava類(lèi)庫(kù)擴(kuò)展的主要是這些相關(guān)類(lèi):collections(集合類(lèi)),concurrency(并發(fā)),primitives,reflection(反射),comparison,I/O,hashi... 閱讀全文
簡(jiǎn)單介紹一下8個(gè)Java牛人,他們?yōu)镴ava社區(qū),創(chuàng)建了框架(framework),產(chǎn)品或者是寫(xiě)書(shū),影響甚至改變了Java開(kāi)發(fā)的方法(根據(jù)個(gè)人喜好排序)。 8.Tomcat創(chuàng)始人
James Duncan Davidson,是當(dāng)時(shí)Sun公司的軟件工程師(1997-2001),創(chuàng)建了Java的Web服務(wù)器Tomcat,Tomcat廣泛應(yīng)用于Java Web開(kāi)發(fā)的各個(gè)領(lǐng)域。 7.測(cè)試驅(qū)動(dòng)開(kāi)發(fā)JUnit創(chuàng)始人
Kent Beck,極限編程和測(cè)試驅(qū)動(dòng)開(kāi)發(fā)方法的締造者。此外,他還創(chuàng)造了JUnit,JUnit目前一次成為Java開(kāi)發(fā)測(cè)試的事實(shí)標(biāo)準(zhǔn)?;跍y(cè)試驅(qū)動(dòng)的開(kāi)發(fā)方法和JUnit給Java開(kāi)發(fā)的方法帶了巨大的變化。 6.Java Collections框架設(shè)計(jì)者
Joshua Bloch,領(lǐng)導(dǎo)設(shè)計(jì)了Java平臺(tái)的許多功能,包括Java 5.0 版本中飽受贊譽(yù)的Java Collections框架。2004年他離開(kāi)Sun公司,成為Google的首席Java架構(gòu)師,此外他的著作“Effective Java”基本上是學(xué)習(xí)Java的必讀之書(shū)。 5.JBoss創(chuàng)始人
Marc Fleury,在2001年創(chuàng)造了JBoss,JBoss是一個(gè)Java開(kāi)源的應(yīng)用服務(wù)器,也已經(jīng)成為Java Web應(yīng)用部署中的事實(shí)標(biāo)準(zhǔn)。后來(lái)他把JBoss買(mǎi)給了RedHat,之后繼續(xù)從事JBoss的開(kāi)發(fā)工作。不過(guò)2007年他離開(kāi)了RedHat去追求他的個(gè)人愛(ài)好。 4.Struts創(chuàng)始人
Craig Mcclanahan,創(chuàng)建了Struts,一個(gè)流行的基于Java的MVC開(kāi)源框架,基本上很多Java開(kāi)發(fā)者都知道如何開(kāi)發(fā)Struts的應(yīng)用程序。 3.Spring創(chuàng)始人
Rod Johnson,Spring框架的創(chuàng)始人,Spring Source的CEO。Spring是一個(gè)非常流行的Java應(yīng)用程序開(kāi)發(fā)的開(kāi)源框架。此外,他的著作Expert One-to-One J2EE Design and Development,是J2EE最有影響力的一本書(shū)。 2.Hibernate創(chuàng)始人
Gavin King,Hibernate的創(chuàng)始人,一個(gè)流行的Java ORM解決方案;同時(shí)他也是Seam的創(chuàng)始人,此外他為EJB3.0和JPA也做出了突出的貢獻(xiàn)。 1.Java之父
James Gosling,1994年發(fā)明了Java語(yǔ)言,他創(chuàng)建了Java編譯器和虛擬機(jī)。在2010年,當(dāng)Oracle收購(gòu)Sun公司時(shí),他離開(kāi)了Sun公司。 原創(chuàng)文章,作者:qiyadeng,轉(zhuǎn)載請(qǐng)注明: 轉(zhuǎn)載自http://www.qiyadeng.com/ 本文鏈接地址: 你應(yīng)該知道的8個(gè)Java牛人
1.申請(qǐng)開(kāi)發(fā)者帳號(hào) 首先注冊(cè)百度的帳號(hào),然后申請(qǐng)成為百度開(kāi)發(fā)者(需要通過(guò)手機(jī)進(jìn)行身份證驗(yàn)證)。   2.新建應(yīng)用 點(diǎn)擊菜單中的創(chuàng)建應(yīng)用,我們目前選擇的是Web應(yīng)用。   應(yīng)用創(chuàng)建之后,選擇左邊菜單的云環(huán)境,環(huán)境類(lèi)型需要選擇JAVA。  并新創(chuàng)建一個(gè)版本,輸入1作為版本號(hào)  3.在百度集成開(kāi)發(fā)環(huán)境中開(kāi)始開(kāi)發(fā) 百度提供了基于Eclipse的插件,由于該插件不能支持最新的Eclipse版本。建議下載百度的一鍵安裝版本。百度文檔中介紹了如何使用開(kāi)發(fā)環(huán)境,詳細(xì)請(qǐng)看集成開(kāi)發(fā)環(huán)境使用。 打開(kāi)百度集成開(kāi)發(fā)環(huán)境,在Eclipse左下角點(diǎn)擊Login to Baidu,使用你的賬號(hào)登陸。然后點(diǎn)擊工具欄中百度Logo,選擇Import BAE Project,填入application和version  之后選擇Java作為Project Language。 4.解決項(xiàng)目錯(cuò)誤 剛導(dǎo)入的BAE project,在Eclipse中會(huì)報(bào)錯(cuò)。通過(guò)problems view可以看到是因?yàn)镴RE環(huán)境配置不正確和Web運(yùn)行環(huán)境設(shè)置不正確。 A.右鍵項(xiàng)目屬性--選擇JavaBuildPath,在Libraries中選擇Add Library,之后再選擇JRE System Library。  B.接下來(lái)把Java project轉(zhuǎn)換為Java Web Project(Eclipse中Java Project轉(zhuǎn)換為Java web Project),注意如果你的tomcat是6版本的話(huà),請(qǐng)注意選擇Dynamic web Module的版本不超過(guò)2.5。 設(shè)置Web應(yīng)用的運(yùn)行環(huán)境,在servers view中新建一個(gè)tomcat服務(wù)器。  C.和A類(lèi)似,在Java Build Path中加入 Server Runtime,選擇Tomcat。 D.修改hello.jsp,在hello.jsp中加入如下代碼 <% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; %>
E.把項(xiàng)目部署到tomcat中。
至此項(xiàng)目錯(cuò)誤全部解決,應(yīng)該可以看到運(yùn)行結(jié)果。

5.新建Servert測(cè)試
新建一個(gè)Servlet,HomeServlet,Eclipse會(huì)自動(dòng)在web.xml中加入配置信息,HomeServet.java和web.xml的部分代碼如下:
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); }
/** * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response) */ protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { PrintWriter out = response.getWriter(); out.println("<h1>BAE Servlet Test.</h1>"); }
web.xml中部分代碼
<servlet> <description></description> <display-name>HomeServlet</display-name> <servlet-name>HomeServlet</servlet-name> <servlet-class>com.qiyadeng.HomeServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>HomeServlet</servlet-name> <url-pattern>/HomeServlet</url-pattern> </servlet-mapping> 運(yùn)行tomcat,可以看到如下運(yùn)行成功,這樣你就可以像一般的Java Web Project一樣進(jìn)行開(kāi)發(fā)。

6.最后
BAE中Java環(huán)境中百度使用的Jetty,而不是tomcat,Jetty的好處是不需要頻繁的重啟,修改的代碼即時(shí)就可看到運(yùn)行結(jié)果。
通過(guò)SVN提交代碼到BAE,如果有需要做小的修改,可以通過(guò)百度的在線編輯工具直接修改。

原創(chuàng)文章,轉(zhuǎn)載請(qǐng)注明: 轉(zhuǎn)載自http://www.qiyadeng.com/
本文鏈接地址: 百度開(kāi)發(fā)者中心BAE新建Java應(yīng)用
1.申請(qǐng)開(kāi)發(fā)者帳號(hào) 首先注冊(cè)百度的帳號(hào),然后申請(qǐng)成為百度開(kāi)發(fā)者(需要通過(guò)手機(jī)進(jìn)行身份證驗(yàn)證)。
2.新建應(yīng)用 點(diǎn)擊菜單中的創(chuàng)建應(yīng)用,我們目前選擇的是Web應(yīng)用。
應(yīng)用創(chuàng)建之后,選擇左邊菜單的云環(huán)境,環(huán)境類(lèi)型需要選擇JAVA。
并新創(chuàng)建一個(gè)版本,輸入1作為版本號(hào)
3.在百度集成開(kāi)發(fā)環(huán)境中開(kāi)始開(kāi)發(fā) 百度提供了基于Eclipse的插件,由于該插件不能支持最新的Eclipse版本。建議下載百度的一鍵安裝版本。百度文檔中介紹了如何使用開(kāi)發(fā)環(huán)境,詳細(xì)請(qǐng)看集成開(kāi)發(fā)環(huán)境使用。 打開(kāi)百度集成開(kāi)發(fā)環(huán)境,在Eclipse左下角點(diǎn)擊Login to Baidu,使用你的賬號(hào)登陸。然后點(diǎn)擊工具欄中百度Logo,選擇Import BAE Project,填入application和version
之后選擇Java作為Project Language。 4.解決項(xiàng)目錯(cuò)誤 剛導(dǎo)入的BAE project,在Eclipse中會(huì)報(bào)錯(cuò)。通過(guò)problems view可以看到是因?yàn)镴RE環(huán)境配置不正確和Web運(yùn)行環(huán)境設(shè)置不正確。 A.右鍵項(xiàng)目屬性--選擇JavaBuildPath,在Libraries中選擇Add Library,之后再選擇JRE System Library。
B.接下來(lái)把Java project轉(zhuǎn)換為Java Web Project(Eclipse中Java Project轉(zhuǎn)換為Java web Project),注意如果你的tomcat是6版本的話(huà),請(qǐng)注意選擇Dynamic web Module的版本不超過(guò)2.5。 設(shè)置Web應(yīng)用的運(yùn)行環(huán)境,在servers view中新建一個(gè)tomcat服務(wù)器。
C.和A類(lèi)似,在Java Build Path中加入 Server Runtime,選擇Tomcat。 D.修改hello.jsp,在hello.jsp中加入如下代碼 <% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; %>
E.把項(xiàng)目部署到tomcat中。
至此項(xiàng)目錯(cuò)誤全部解決,應(yīng)該可以看到運(yùn)行結(jié)果。
5.新建Servert測(cè)試
新建一個(gè)Servlet,HomeServlet,Eclipse會(huì)自動(dòng)在web.xml中加入配置信息,HomeServet.java和web.xml的部分代碼如下:
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); }
/** * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response) */ protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { PrintWriter out = response.getWriter(); out.println("<h1>BAE Servlet Test.</h1>"); }
web.xml中部分代碼
<servlet> <description></description> <display-name>HomeServlet</display-name> <servlet-name>HomeServlet</servlet-name> <servlet-class>com.qiyadeng.HomeServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>HomeServlet</servlet-name> <url-pattern>/HomeServlet</url-pattern> </servlet-mapping> 運(yùn)行tomcat,可以看到如下運(yùn)行成功,這樣你就可以像一般的Java Web Project一樣進(jìn)行開(kāi)發(fā)。
6.最后
BAE中Java環(huán)境中百度使用的Jetty,而不是tomcat,Jetty的好處是不需要頻繁的重啟,修改的代碼即時(shí)就可看到運(yùn)行結(jié)果。
通過(guò)SVN提交代碼到BAE,如果有需要做小的修改,可以通過(guò)百度的在線編輯工具直接修改。

空指針異常(Null Pointer Exception)是我們平時(shí)最容易碰到的,也是最令人討厭的異常。本文介紹如何避免出現(xiàn)空指針異常。 首先我們看如下的示例 private Boolean isFinished(String status) { if (status.equalsIgnoreCase("Finish")) { return Boolean.TRUE; } else { return Boolean.FALSE; } } 如果status的值為空的話(huà),那么將會(huì)出現(xiàn)空指針異常(本例第2行)。所以我們應(yīng)該使用如下的方法
private Boolean isFinished(String status) { if ("Finish".equalsIgnoreCase(status)) { return Boolean.TRUE; } else { return Boolean.FALSE; } } 這樣的話(huà),如果status為空,也不會(huì)出現(xiàn)空指針異常。相信我們大多數(shù)朋友已經(jīng)知道這樣的方法了,如果一個(gè)對(duì)象可能為null,那么不需要直接調(diào)用它的方法。
接下來(lái)我將接著提供幾種避免空指針的建議。
1.判斷Collection是否為空。
2.使用一些判斷方法。
3.assert關(guān)鍵字。
4.Assert類(lèi)。
5.異常處理。
6.太多的點(diǎn).操作語(yǔ)法。
7.使用StringUtils類(lèi)
1.判斷Collection是否為空
Collection 為空是指Collection中沒(méi)有元素。一些開(kāi)發(fā)者如果碰到Collection中沒(méi)有元素的時(shí)候,經(jīng)常return null,更好的做法是,你應(yīng)該return Collections.EMPTY_LIST,Collections.EMPTY_SET或者是Collections.EMPTY_MAP.
錯(cuò)誤的代碼
public static List getEmployees() { List list = null; return list; }
正確的代碼
public static List getEmployees() { List list = Collections.EMPTY_LIST; return list; }
2.使用一些判斷方法
使用一些方法如contains(),indexOf(),isEmpty(),containsKey(),ContainsValue和hasNext()等來(lái)判斷,確保不存在空值。
示例:
String myName = "qiyadeng"; List list = Collections.EMPTY_LIST; boolean exist = list.contains(myName); int index = list.indexOf(myName); boolean isEmpty =list.isEmpty(); Map map =Collections.EMPTY_MAP; exist=map.containsKey(myName); exist=map.containsValue(myName); isEmpty=map.isEmpty(); Set set=Collections.EMPTY_SET; exist=set.contains(myName); isEmpty=set.isEmpty(); Iterator iterator; exist = iterator.hasNext();
3.assert關(guān)鍵字
在Java1.4版本之后,提供了斷言assert來(lái)確定你的代碼中的假設(shè)。使用的語(yǔ)法如下:
expression1是一個(gè)boolean表達(dá)式,如果expression1返回的false,系統(tǒng)將會(huì)拋出AssertError(沒(méi)有詳細(xì)信息)。
另外一種使用方法
assert expression1:expression2 如果expression1返回false,那么系統(tǒng)將會(huì)拋出AssertError,并且詳細(xì)信息為expression2。
示例:
public static String getManager(String employeeId) { assert (employeeId != null) : "employeeId must be not null"; return "qiyadeng"; } 我使用getManager(null)來(lái)調(diào)用getManger方法,最后運(yùn)行的結(jié)果是"java.lang.AssertionError:employeedId must be not null"
注意記得使用java選項(xiàng)中加入-enableassertion開(kāi)啟assertion功能。
4.Assert類(lèi)
Assert類(lèi)在com.bea.core.repackaged.springframework.util包中,有許多方法可以用于斷言。
public static String getManager(String employeeId) { Assert.notNull(employeeId, "employeeId must be not null"); Assert.hasLength(employeeId, "employeeId must has length greater than 0"); return "qiyadeng"; } 當(dāng)我同樣使用getManager(null)來(lái)調(diào)用getManager方法,將獲得信息"java.lang.IllegalArgumentException: employeeId must be not null"。
5.異常處理
使用try catch處理異?;蚴菣z查變量是否為空。
public static String getManager(String employeeId) { return null; } 如上代碼,我使用下面方法調(diào)用
String managerId = getManager("A015"); System.out.println(managerId.toString()); 將會(huì)發(fā)生"java.lang.NullPointerException",為了處理這個(gè)異常,我們應(yīng)該使用try catch來(lái)處理異常或者是檢查變量是否為null。
try-catch方法
String managerId = getManager("A015"); try { System.out.println(managerId.toString()); } catch (NullPointerException npe) { //write your code here } 或者是對(duì)變量進(jìn)行檢查
String managerId = getManager("A015"); if (managerId != null) { System.out.println(managerId.toString()); } else { //write your code here }
6.不要太多的點(diǎn).操作語(yǔ)法
一些開(kāi)發(fā)者使用太多的這樣的方法來(lái)減少代碼,但是這個(gè)對(duì)后面的維護(hù)和異常處理都是不太好的。
錯(cuò)誤的寫(xiě)法
String attrValue = (String)findViewObject("VO_NAME").getCurrentRow().getAttribute("Attribute_NAME"); 正確的寫(xiě)法
ViewObject vo = findViewObject("VO_NAME"); Row row = vo.getCurrentRow(); String attrValue = (String)row.getAttribute("Attribute_NAME");
7.使用StringUtils類(lèi)
StringUtil是org.apache.commns.lang包中的類(lèi),我們可以使用該類(lèi)來(lái)避免空指針異常。
例如 StringUtils.isEmpty(),StringUtils.isBlank,StringUtils.equals()等等,更多的你可以參考文檔。
為了不出現(xiàn)空指針異常,在寫(xiě)代碼的過(guò)程中需要時(shí)刻檢查你的代碼是否會(huì)拋出NullPointerException,如果你沒(méi)有時(shí)間及時(shí)調(diào)整的話(huà),使用//TODO標(biāo)記,便于你后面解決問(wèn)題。
現(xiàn)在經(jīng)常需要根據(jù)用戶(hù)提供的位置,提供一些和位置相關(guān)的信息。有時(shí)可以直接確定用戶(hù)的經(jīng)度和緯度,有時(shí)不一定可以確定用戶(hù)的經(jīng)度和緯度信息,用戶(hù)是 通過(guò)輸入一些路名、標(biāo)志性建筑或是商場(chǎng)名等位置,但是我們的數(shù)據(jù)庫(kù)可能并沒(méi)有存法用戶(hù)可能輸入的這些位置信息的經(jīng)度緯度,這時(shí)候可以使用一些地圖提供的 API來(lái)確定,用戶(hù)所輸入的位置信息的經(jīng)度和緯度。 我們使用百度地圖提供的GeoCoding API實(shí)現(xiàn)從位置信息到經(jīng)度緯度的轉(zhuǎn)換,詳細(xì)的使用說(shuō)明可以參考 GeoCoding API。我們這里做一個(gè)簡(jiǎn)單的演示 public String getGeoCode(String query) throws ClientProtocolException, IOException{ HttpClient httpClient = new DefaultHttpClient(); String url = geoCodeRequestUrl(query); logger.log(Level.INFO, url); HttpGet httpget = new HttpGet(url); ResponseHandler<String> responseHandler = new BasicResponseHandler(); String responseBody = httpClient.execute(httpget, responseHandler);//百度返回的經(jīng)度緯度信息xml logger.log(Level.INFO,"baidu response:"+responseBody); return responseBody; } public String geoCodeRequestUrl(String query) throws UnsupportedEncodingException{ String url = WeChatConstant.BASEURL + "geocoder?address=" + URLEncoder.encode(query,"UTF-8") + "&key=" + WeChatConstant.MAPKEY + "&output=" + WeChatConstant.OUTPUTFORMAT; return url; } 使用JUnit進(jìn)行測(cè)試 @Test public void testGeoCode() throws Exception { BaiduMapService bms = new BaiduMapService(); String response = bms.getGeoCode("上地十街十號(hào)"); BaiduGeoCodeResponse res = BaiduGeoCodeResponse.getBaiduGeoCode(response);//解析xml System.out.println(res.toString()); } 輸出的結(jié)果 <GeocoderSearchResponse> <status>OK</status> <result> <location> <lat>40.057098</lat> <lng>116.307175</lng> </location> <precise>1</precise> <confidence>80</confidence> <level>道路</level> </result> </GeocoderSearchResponse> BaiduGeoCodeResponse [lat=40.057098, lng=116.307175]
到了一個(gè)較陌生的環(huán)境,經(jīng)常會(huì)在周邊找一些基礎(chǔ)設(shè)施,比如銀行,商場(chǎng),餐廳等(還有一種更急切的是找?guī)?。通過(guò)百度提供的地圖API,可以在你的應(yīng)用中簡(jiǎn)單做到,詳情可閱讀 Place API。我們以查找周邊銀行作為示例,需確定的參數(shù)至少有三個(gè),要查找的位置的經(jīng)度和緯度,需要查找的內(nèi)容的類(lèi)型或是關(guān)鍵字。 public String getPalace(String query,String lat,String lng) throws ClientProtocolException, IOException{ HttpClient httpClient = new DefaultHttpClient(); String url = palceRequestUrl(query,lat,lng); logger.log(Level.INFO, url); HttpGet httpget = new HttpGet(url); ResponseHandler<String> responseHandler = new BasicResponseHandler(); String responseBody = httpClient.execute(httpget, responseHandler);//位置xml logger.log(Level.INFO,"baidu response:"+responseBody); return responseBody; } public String palceRequestUrl(String query,String lat,String lng) throws UnsupportedEncodingException { String url = WeChatConstant.BASEURL + "place/search?query=" + URLEncoder.encode(query,"UTF-8") + "&key=" + WeChatConstant.MAPKEY +"&location="+lat+","+lng +"&radius=2000"+"&output=" + WeChatConstant.OUTPUTFORMAT; return url; } Junit測(cè)試 @Test public void testGetBaiduPlace() throws Exception{ BaiduMapService bms = new BaiduMapService(); String response = bms.getPalace("銀行", "39.915", "116.404"); List<BaiduPlaceResponse> list = BaiduPlaceResponse.getBaiduPlace(response); for(BaiduPlaceResponse res:list){ System.out.println(res.toString()); } } 輸出內(nèi)容(省略部分內(nèi)容) <?xml version="1.0" encoding="utf-8" ?> <PlaceSearchResponse> <status>OK</status> <results> <result> <name>中國(guó)工商銀行東長(zhǎng)安街支行</name> <location> <lat>39.915891</lat> <lng>116.41867</lng> </location> <address>東城區(qū)東長(zhǎng)安街1號(hào)東方廣場(chǎng)西三辦公樓1樓</address> <uid>a025683c73033c35a21de987</uid> <detail_url>http://api.map.baidu.com/place/detail?uid=a025683c73033c35a21de987&amp;output=html&amp;source=placeapi</detail_url> <tag>銀行,王府井/東單</tag> </result> </results> </PlaceSearchResponse> BaiduPlaceResponse [name=中國(guó)工商銀行東長(zhǎng)安街支行, telephone=null, address=東城區(qū)東長(zhǎng)安街1號(hào)東方廣場(chǎng)西三辦公樓1樓, lat=39.915891, lng=116.41867, tag=null, detailUrl=http://api.map.baidu.com/place/detail?uid=a025683c73033c35a21de987&output=html&source=placeapi]
位置識(shí)別這是實(shí)際應(yīng)用經(jīng)常應(yīng)用的消息,特別是很多商家,通過(guò)了解用戶(hù)位置,給用戶(hù)提供特別的產(chǎn)品或是商場(chǎng)的推薦。其中用戶(hù)可能發(fā)送兩種類(lèi)型的消息: 1.微信地理位置信息 2.路名、標(biāo)志性建筑或是商場(chǎng)名稱(chēng) 1.微信地理位置消息認(rèn)識(shí)一下,微信地理位置消息,包含一些什么信息 <xml> <ToUserName><![CDATA[toUser]]></ToUserName> <FromUserName><![CDATA[fromUser]]></FromUserName> <CreateTime>1351776360</CreateTime> <MsgType><![CDATA[location]]></MsgType> <Location_X>23.134521</Location_X> <Location_Y>113.358803</Location_Y> <Scale>20</Scale> <Label><![CDATA[位置信息]]></Label> <MsgId>1234567890123456</MsgId> </xml>
包含的主要信息有經(jīng)度緯度和Label的位置。可以根據(jù)label中描述的位置信息,提供給用戶(hù)對(duì)應(yīng)的服務(wù)。也可根據(jù)用戶(hù)的經(jīng)度緯度信息,提供你最近的產(chǎn)品或是有地域性的產(chǎn)品。
首先根據(jù)微信的地理位置信息,定義WeChatLocationMessage類(lèi),并能把Xml轉(zhuǎn)換為WeChatLocationMessage對(duì)象 public class WeChatLocationMessage { private String toUserName; private String fromUserName; private String createTime; private String msgType; private String locationx; private String localtiony; private String scale; private String label; private String msgId; public static WeChatLocationMessage getWeChatLocationMessage(String xml){ XStream xstream = new XStream(new DomDriver()); WeChatLocationMessage message = null; xstream.alias("xml", WeChatLocationMessage.class); xstream.aliasField("ToUserName", WeChatLocationMessage.class, "toUserName"); xstream.aliasField("FromUserName", WeChatLocationMessage.class, "fromUserName"); xstream.aliasField("CreateTime", WeChatLocationMessage.class, "createTime"); xstream.aliasField("MsgType", WeChatLocationMessage.class, "msgType"); xstream.aliasField("Location_X", WeChatLocationMessage.class, "locationx"); xstream.aliasField("Location_Y", WeChatLocationMessage.class, "localtiony"); xstream.aliasField("Scale", WeChatLocationMessage.class, "scale"); xstream.aliasField("Label", WeChatLocationMessage.class, "label"); xstream.aliasField("MsgId", WeChatLocationMessage.class, "msgId"); message = (WeChatLocationMessage)xstream.fromXML(xml); return message; } //getter and setter } 本文利用百度的地圖API,查找最近的銀行做為示例。 public String getPalace(String query,String lat,String lng) throws ClientProtocolException, IOException{ HttpClient httpClient = new DefaultHttpClient(); String url = palceRequestUrl(query,lat,lng); logger.log(Level.INFO, url); HttpGet httpget = new HttpGet(url); ResponseHandler<String> responseHandler = new BasicResponseHandler(); String responseBody = httpClient.execute(httpget, responseHandler); logger.log(Level.INFO,"baidu response:"+responseBody); return responseBody; } public String palceRequestUrl(String query,String lat,String lng) throws UnsupportedEncodingException { String url = WeChatConstant.BASEURL + "place/search?query=" + URLEncoder.encode(query,"UTF-8") + "&key=" + WeChatConstant.MAPKEY +"&location="+lat+","+lng +"&radius=2000"+"&output=" + WeChatConstant.OUTPUTFORMAT; return url; } 輸出的結(jié)果 <PlaceSearchResponse> <status>OK</status> <results> <result> <name>中國(guó)工商銀行東長(zhǎng)安街支行</name> <location> <lat>39.915891</lat> <lng>116.41867</lng> </location> <address>東城區(qū)東長(zhǎng)安街1號(hào)東方廣場(chǎng)西三辦公樓1樓</address> <uid>a025683c73033c35a21de987</uid> <detail_url>http://api.map.baidu.com/place/detail?uid=a025683c73033c35a21de987&amp;output=html&amp;source=placeapi </detail_url> <tag>銀行,王府井/東單</tag>
</result> </results> </PlaceSearchResponse> 接下來(lái),把百度地圖反映出來(lái)的最近位置信息,以圖文消息的格式展示給微信用戶(hù) public static String getWeChatReplyNewsMessageByBaiduPlace(List<BaiduPlaceResponse> placeList, double lat, double lng,String userName, int size){ WeChatReplyNewsMessage newsMessage = new WeChatReplyNewsMessage(); List<Item> items = new ArrayList<Item>(); StringBuffer strBuf = new StringBuffer(); logger.log(Level.INFO,"placeList count="+placeList.size()); newsMessage.setItems(items); if(placeList.size()>size){ newsMessage.setArticleCount(size); } else{ newsMessage.setArticleCount(placeList.size()); } logger.log(Level.INFO,"article count="+newsMessage.getArticleCount()); newsMessage.setCreateTime(new Date().getTime()+""); newsMessage.setMsgType("news"); newsMessage.setFuncFlag("0"); newsMessage.setToUserName(userName); newsMessage.setFromUserName(WeChatConstant.FROMUSERNAME); for(int i = 0;i <newsMessage.getArticleCount();i++){ BaiduPlaceResponse place = placeList.get(i); Double distance = GeoUtil.DistanceOfTwoPoints(Double.valueOf(place.getLng()), Double.valueOf(place.getLat()), lng, lat, GaussSphere.Beijing54); Item item = new Item(); item.setTitle(place.getName()+"["+distance+"米]"+"\n"+place.getAddress()+"\n"+place.getTelephone()); item.setPicUrl(""); item.setUrl(place.getDetailUrl()); item.setDescription(""); items.add(item); } logger.log(Level.INFO,"newMessage="+newsMessage.toString()); strBuf = strBuf.append(getWeChatNewsMessage(newsMessage)); return strBuf.toString(); } public static String getWeChatNewsMessage(WeChatReplyNewsMessage newsMessage){ XStream xstream = new XStream(new DomDriver()); xstream.alias("xml", WeChatReplyNewsMessage.class); xstream.aliasField("ToUserName", WeChatReplyNewsMessage.class, "toUserName"); xstream.aliasField("FromUserName", WeChatReplyNewsMessage.class, "fromUserName"); xstream.aliasField("CreateTime", WeChatReplyNewsMessage.class, "createTime"); xstream.aliasField("MsgType", WeChatReplyNewsMessage.class, "msgType"); xstream.aliasField("ArticleCount", WeChatReplyNewsMessage.class, "articleCount"); xstream.aliasField("Content", WeChatReplyNewsMessage.class, "content"); xstream.aliasField("FuncFlag", WeChatReplyNewsMessage.class, "funcFlag"); xstream.aliasField("Articles", WeChatReplyNewsMessage.class, "items"); xstream.alias("item", Item.class); xstream.aliasField("Title", Item.class, "title"); xstream.aliasField("Description", Item.class, "description"); xstream.aliasField("PicUrl", Item.class, "picUrl"); xstream.aliasField("Url", Item.class, "url"); return xstream.toXML(newsMessage); } 2.路名、標(biāo)志性建筑或是商場(chǎng)名稱(chēng)對(duì)路名、標(biāo)志性建筑等信息,方法還是通過(guò)第三方地圖信息,確定輸入的位置信息的經(jīng)度緯度。 本文使用百度地圖API,確定所查找的位置的經(jīng)度和緯度。 public String getGeoCode(String query) throws ClientProtocolException, IOException{ HttpClient httpClient = new DefaultHttpClient(); String url = geoCodeRequestUrl(query); logger.log(Level.INFO, url); HttpGet httpget = new HttpGet(url); ResponseHandler<String> responseHandler = new BasicResponseHandler(); String responseBody = httpClient.execute(httpget, responseHandler); logger.log(Level.INFO,"baidu response:"+responseBody); return responseBody; } public String geoCodeRequestUrl(String query) throws UnsupportedEncodingException{ String url = WeChatConstant.BASEURL + "geocoder?address=" + URLEncoder.encode(query,"UTF-8") + "&key=" + WeChatConstant.MAPKEY + "&output=" + WeChatConstant.OUTPUTFORMAT; return url; } 確定了經(jīng)度和緯度,問(wèn)題就變成和第1種消息類(lèi)型一致了,根據(jù)經(jīng)度緯度去做相應(yīng)處理。
3.源代碼本文的代碼較長(zhǎng),提供源代碼下載。 WeChatDemo下載
本文介紹,如果把Java Project轉(zhuǎn)換為Java Web Project,應(yīng)該在多數(shù)的Eclipse的版本都類(lèi)似。 1.Java Project一個(gè)Java Projec,在Eclipse中顯示的是一個(gè)“J”的藍(lán)色文件夾。
2.Project Facets右鍵項(xiàng)目屬性Properties,右側(cè)選擇菜單Project Facets,點(diǎn)擊converted to faceted form...
勾選dynamic web module
選擇下面的further configuration available
項(xiàng)目中的Web目錄和設(shè)置保持一致。
3.Java Web Project這樣你就轉(zhuǎn)換到了Java Web Project。 
1.新建Project 新建Java Project,并把mongo-java-driver驅(qū)動(dòng)加入到項(xiàng)目bulid path中,如果你使用的是maven增加依賴(lài)。 <dependency> <groupId>org.mongodb</groupId> <artifactId>mongo-java-driver</artifactId> <version>2.10.1</version> </dependency> 2.連接上MongoDB //>2.10版本 MongoClient mongo = new MongoClient( "localhost" , 27017 );
//老版本 Mongo mongo = new Mongo("localhost", 27017); 如果需要驗(yàn)證,需要輸入用戶(hù)名和密碼 MongoClient mongoClient = new MongoClient(); DB db = mongoClient.getDB("database name"); boolean auth = db.authenticate("username", "password".toCharArray()); 3.MongoDB數(shù)據(jù)庫(kù) 得到MongoDB中的數(shù)據(jù)庫(kù),如果數(shù)據(jù)庫(kù)名不存在,MongoDB會(huì)自動(dòng)創(chuàng)建 DB db = mongo.getDB("database name"); 顯示所有的數(shù)據(jù)庫(kù) List<String> dbs = mongo.getDatabaseNames(); for(String db : dbs){ System.out.println(db); } 4.MongoDB Collection(MongoDB表) 得到數(shù)據(jù)庫(kù)中的表 DB db = mongo.getDB("testdb"); DBCollection table = db.getCollection("user"); 顯示數(shù)據(jù)庫(kù)中的所有表 DB db = mongo.getDB("testdb"); Set<String> tables = db.getCollectionNames(); for(String coll : tables){ System.out.println(coll); } 5.插入、查找、更新、刪除操作 插入數(shù)據(jù),向Collection(表)中插入一個(gè)Document DBCollection table = db.getCollection("user"); BasicDBObject document = new BasicDBObject(); document.put("name", "qiyadeng"); document.put("age", 30); document.put("createdDate", new Date()); table.insert(document); 更新Document中的name="qiyadeng.com" DBCollection table = db.getCollection("user");
BasicDBObject query = new BasicDBObject(); query.put("name", "qiyadeng");
BasicDBObject newDocument = new BasicDBObject(); newDocument.put("name", "qiyadeng.com");
BasicDBObject updateObj = new BasicDBObject(); updateObj.put("$set", newDocument); table.update(query, updateObj); 從Collection中查找name="qiyadeng.com"的Document DBCollection table = db.getCollection("user"); BasicDBObject searchQuery = new BasicDBObject(); searchQuery.put("name", "qiyadeng.com"); DBCursor cursor = table.find(searchQuery); while (cursor.hasNext()) { System.out.println(cursor.next()); } 刪除name="qiyadeng"的Document DBCollection table = db.getCollection("user"); BasicDBObject searchQuery = new BasicDBObject(); searchQuery.put("name", "qiyadeng.com"); table.remove(searchQuery);
6.完整的例子 package com.qiyadeng.mongodb;
import java.util.Date;
import com.mongodb.BasicDBObject; import com.mongodb.DB; import com.mongodb.DBCollection; import com.mongodb.DBCursor; import com.mongodb.MongoClient;
public class MongoDBSample {
public static void main(String[] args) throws Exception{ /**** Connect to MongoDB ****/ //2.10.0后,使用MongoClient MongoClient mongo = new MongoClient("localhost", 27017); /**** Get database ****/ // if database doesn't exists, MongoDB will create it for you DB db = mongo.getDB("testdb"); /**** Get collection / table from 'testdb' ****/ // if collection doesn't exists, MongoDB will create it for you DBCollection table = db.getCollection("user"); /**** Insert ****/ // create a document to store key and value BasicDBObject document = new BasicDBObject(); document.put("name", "qiyadeng"); document.put("age", 30); document.put("createdDate", new Date()); table.insert(document); /**** Find and display ****/ BasicDBObject searchQuery = new BasicDBObject(); searchQuery.put("name", "qiyadeng"); DBCursor cursor = table.find(searchQuery); while (cursor.hasNext()) { System.out.println(cursor.next()); } /**** Update ****/ // search document where name="qiyadeng" and update it with new values BasicDBObject query = new BasicDBObject(); query.put("name", "qiyadeng"); BasicDBObject newDocument = new BasicDBObject(); newDocument.put("name", "qiyadeng.com"); BasicDBObject updateObj = new BasicDBObject(); updateObj.put("$set", newDocument); table.update(query, updateObj); /**** Find and display ****/ BasicDBObject searchQuery2 = new BasicDBObject().append("name", "qiyadeng.com"); DBCursor cursor2 = table.find(searchQuery2); while (cursor2.hasNext()) { System.out.println(cursor2.next()); } } }
輸出 { "_id" : { "$oid" : "51444c88874c79654063356b"} , "name" : "qiyadeng" , "age" : 30 , "createdDate" : { "$date" : "2013-03-16T10:42:16.555Z"}} { "_id" : { "$oid" : "51444c88874c79654063356b"} , "age" : 30 , "createdDate" : { "$date" : "2013-03-16T10:42:16.555Z"} , "name" : "qiyadeng.com"}
使用mongo驗(yàn)證創(chuàng)建的數(shù)據(jù)庫(kù)testdb,collection user是否存在。
本文介紹如何安裝在windows 7中安裝MongoDB。 注:MongoDB并不像Windows上安裝其他軟件,只需要下載Zip包并解壓,然后配置數(shù)據(jù)存放目錄并啟動(dòng)即可。 1.下載MongoDB從MongoDB官方網(wǎng)站,根據(jù)你的平臺(tái)選擇對(duì)應(yīng)的windows的壓縮包并解壓,本文解壓到D:\mongodb\。
注:如果需要在命令行中快速使用MongoDB bin目錄下的命令,可以將D:\mongoDB\bin加入到Window環(huán)境變量。 2.配置數(shù)據(jù)文件在D:\mongodb\創(chuàng)建mongo.config文件,如下(并在d:\mongodb目錄下新建data,log文件夾) ##數(shù)據(jù)存儲(chǔ)的位置 dbpath=D:\mongodb\data ##所有的輸出位置 logpath=D:\mongodb\log\mongo.log ##日志讀寫(xiě)操作 diaglog=3 3.運(yùn)行MongoDB Server在命令控制行,切換到d:\mongodb\bin目錄下,使用命令mongod.exe --config d:\mongdb\mongo.config啟動(dòng)MongoDb Server。 D:\mongodb\bin>mongod.exe --config d:\mongodb\mongo.config all output going to: D:\mongodb\log\mongo.log 4.連接MongoDB新開(kāi)啟一個(gè)命令行控制窗口,使用mongo.exe連接MongoDB Server. 5.設(shè)置MongoDB為Windows服務(wù)在命令行控制窗口,加入--install選項(xiàng)可以把MongoDB安裝為Windows服務(wù)。 D:\mongodb\bin>mongod.exe --config d:\mongodb\mongo.config 啟動(dòng)MongoDB的命令為:net start MongoDB 停車(chē)MongODB的命令為:net stop MongoDB 刪除MongoDB的命令為:mongod --remove
1.設(shè)置成為開(kāi)發(fā)者模式
登錄微信工作平臺(tái),選擇高級(jí)功能-進(jìn)入開(kāi)發(fā)模式,成為開(kāi)發(fā)者。需要做如下圖配置。URL配置的信息是指,微信的后臺(tái)服務(wù)器把您的用戶(hù)消息發(fā)送到該URL處理。Token是你和微信之間的一個(gè)密碼,用來(lái)驗(yàn)證消息是否是從微信的服務(wù)發(fā)送而來(lái),而不是其他來(lái)攻擊你的系統(tǒng)。
現(xiàn)在你還不能設(shè)置,在設(shè)置時(shí)微信會(huì)GET請(qǐng)求你設(shè)置的URL,已檢測(cè)接口是否可以使用。只有等你準(zhǔn)備好GET方法之后才可以進(jìn)行設(shè)置。
2.實(shí)現(xiàn)GET方法
從文檔中知道,我們需要實(shí)現(xiàn)POST和GET方法,GET方法用于驗(yàn)證微信和你的通訊驗(yàn)證,POST用于消息處理。
新建Servlet HelloWeChat,先實(shí)現(xiàn)其中的GET方法
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO 為了簡(jiǎn)單起見(jiàn),先不對(duì)消息來(lái)源進(jìn)行校驗(yàn)
response.setContentType("text/html;charset=UTF-8");
PrintWriter pw = response.getWriter();
String echo = request.getParameter("echostr");
echo = new String(echo.getBytes("ISO-8859-1"),"UTF-8");
pw.println(echo);
}
可以在本地使用http://localhost:8080/QiyadengWeb/HelloWeChat?echostr=hello中文,先進(jìn)行測(cè)試,如果沒(méi)有問(wèn)題,可以部署到服務(wù)器上,然后在微信公眾平臺(tái)進(jìn)行設(shè)置了。
3.實(shí)現(xiàn)POST方法
POST方法首先接收到微信公眾平臺(tái)傳送過(guò)來(lái)的XML,從中提取消息發(fā)送人和消息內(nèi)容。更加消息發(fā)送內(nèi)容,你可以增加自己的處理邏輯,最后拼裝成回復(fù)消息XML,返回給微信公眾平臺(tái)。
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.setContentType("text/html;charset=UTF-8");
PrintWriter pw = response.getWriter();
String wxMsgXml = IOUtils.toString(request.getInputStream(),"utf-8");
WeChatTextMessage textMsg = null;
try {
textMsg = getWeChatTextMessage(wxMsgXml);
} catch (Exception e) {
e.printStackTrace();
}
StringBuffer replyMsg = new StringBuffer();
if(textMsg != null){
//增加你所需要的處理邏輯,這里只是簡(jiǎn)單重復(fù)消息
replyMsg.append("您給我的消息是:");
replyMsg.append(textMsg.getContent());
}
else{
replyMsg.append(":)不是文本的消息,我暫時(shí)看不懂");
}
String returnXml = getReplyTextMessage(replyMsg.toString(), textMsg.getFromUserName());
pw.println(returnXml);
}
關(guān)于調(diào)試,這里推薦一個(gè)工具Fiddler,你可以模擬微信的POST消息到你的本地,而不必每次部署到服務(wù)器上進(jìn)行調(diào)試。關(guān)于Fiddler的POST數(shù)據(jù)使用方法,可以參考下圖標(biāo)注內(nèi)容。
4.部署并測(cè)試
完成第一步,并和你的公眾帳號(hào)好進(jìn)行對(duì)話(huà),回復(fù)消息沒(méi)有問(wèn)題的話(huà),那就恭喜你了 。
5.依賴(lài)庫(kù)
使用maven的同學(xué),添加以下依賴(lài)即可。非maven用戶(hù),找到這些庫(kù)添加到buider path中即可。
<dependency>
<groupId>joda-time</groupId>
<artifactId>joda-time</artifactId>
<version>2.2</version>
</dependency>
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-io</artifactId>
<version>1.3.2</version>
</dependency>
<dependency>
<groupId>com.thoughtworks.xstream</groupId>
<artifactId>xstream</artifactId>
<version>1.4.3</version>
</dependency>
6.完整的代碼
package com.qiyadeng.wechat;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Date;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.io.IOUtils;
import com.thoughtworks.xstream.XStream;
import com.thoughtworks.xstream.io.xml.DomDriver;
/**
* Servlet implementation class HelloWeChat
*/
public class HelloWeChat extends HttpServlet {
private static final long serialVersionUID = 1L;
/**
* @see HttpServlet#HttpServlet()
*/
public HelloWeChat() {
super();
}
/**
* @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response)
*/
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO 為了簡(jiǎn)單起見(jiàn),先不對(duì)消息來(lái)源進(jìn)行校驗(yàn)
response.setContentType("text/html;charset=UTF-8");
PrintWriter pw = response.getWriter();
String echo = request.getParameter("echostr");
echo = new String(echo.getBytes("ISO-8859-1"),"UTF-8");
pw.println(echo);
}
/**
* @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
*/
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.setContentType("text/html;charset=UTF-8");
PrintWriter pw = response.getWriter();
String wxMsgXml = IOUtils.toString(request.getInputStream(),"utf-8");
WeChatTextMessage textMsg = null;
try {
textMsg = getWeChatTextMessage(wxMsgXml);
} catch (Exception e) {
e.printStackTrace();
}
StringBuffer replyMsg = new StringBuffer();
if(textMsg != null){
//增加你所需要的處理邏輯,這里只是簡(jiǎn)單重復(fù)消息
replyMsg.append("您給我的消息是:");
replyMsg.append(textMsg.getContent());
}
else{
replyMsg.append(":)不是文本的消息,我暫時(shí)看不懂");
}
String returnXml = getReplyTextMessage(replyMsg.toString(), textMsg.getFromUserName());
pw.println(returnXml);
}
private WeChatTextMessage getWeChatTextMessage(String xml){
XStream xstream = new XStream(new DomDriver());
xstream.alias("xml", WeChatTextMessage.class);
xstream.aliasField("ToUserName", WeChatTextMessage.class, "toUserName");
xstream.aliasField("FromUserName", WeChatTextMessage.class, "fromUserName");
xstream.aliasField("CreateTime", WeChatTextMessage.class, "createTime");
xstream.aliasField("MsgType", WeChatTextMessage.class, "messageType");
xstream.aliasField("Content", WeChatTextMessage.class, "content");
xstream.aliasField("MsgId", WeChatTextMessage.class, "msgId");
WeChatTextMessage wechatTextMessage = (WeChatTextMessage)xstream.fromXML(xml);
return wechatTextMessage;
}
private String getReplyTextMessage(String content, String weChatUser){
WeChatReplyTextMessage we = new WeChatReplyTextMessage();
we.setMessageType("text");
we.setFuncFlag("0");
we.setCreateTime(new Long(new Date().getTime()).toString());
we.setContent(content);
we.setToUserName(weChatUser);
we.setFromUserName("shanghaiweather");//TODO 你的公眾帳號(hào)微信號(hào)
XStream xstream = new XStream(new DomDriver());
xstream.alias("xml", WeChatReplyTextMessage.class);
xstream.aliasField("ToUserName", WeChatReplyTextMessage.class, "toUserName");
xstream.aliasField("FromUserName", WeChatReplyTextMessage.class, "fromUserName");
xstream.aliasField("CreateTime", WeChatReplyTextMessage.class, "createTime");
xstream.aliasField("MsgType", WeChatReplyTextMessage.class, "messageType");
xstream.aliasField("Content", WeChatReplyTextMessage.class, "content");
xstream.aliasField("FuncFlag", WeChatReplyTextMessage.class, "funcFlag");
String xml =xstream.toXML(we);
return xml;
}
}
開(kāi)始微信公眾平臺(tái)的開(kāi)發(fā),我們首先要了解微信平臺(tái)可以幫助我們做哪些事情? 使用您的公眾賬號(hào)登陸http://mp.weixin.qq.com/,選擇菜單--高級(jí)功能-開(kāi)發(fā)模式--查看文檔,即能看到微信公眾平臺(tái)目前所能開(kāi)發(fā)的功能。 一、通訊機(jī)制 公眾平臺(tái)的主要內(nèi)容是 - 接受用戶(hù)發(fā)送給您公眾賬號(hào)的消息
- 給您的用戶(hù)回復(fù)消息
需要特別說(shuō)明的是,發(fā)送消息和回復(fù)消失是一個(gè)連貫的過(guò)程,只能在一個(gè)對(duì)話(huà)中完成。也就是說(shuō)您的用戶(hù)不找您說(shuō)話(huà),您是不能主動(dòng)發(fā)送消息給你的客戶(hù)(群發(fā)是另外一種情況,有次數(shù)限制。你也可以申請(qǐng)付費(fèi)使用微信CRM平臺(tái))。所有的發(fā)送消息和接受消息,都需要微信平臺(tái)進(jìn)行中轉(zhuǎn)。 二、消息類(lèi)型 下面介紹用戶(hù)能給您發(fā)送的消息類(lèi)型,也就是目前接受到的消息類(lèi)型。 1.接受消息類(lèi)型 1.1文本消息: 這也是我們平時(shí)碰到最多的,可以根據(jù)文本中提到的一些關(guān)鍵字,進(jìn)行判斷,判斷用戶(hù)的含義,并進(jìn)行回復(fù)。 1.2圖片消息: 目前通過(guò)圖片理解用戶(hù)想表達(dá)的意思,還是有較大難度,因此多數(shù)的公眾賬號(hào),會(huì)選擇忽略圖片信息或選擇由人工來(lái)處理。只能說(shuō)一句:圖片很美,但是我看不懂。 1.3地理位置消息: 用戶(hù)把他的位置發(fā)給您,這對(duì)大多數(shù)公眾賬號(hào)來(lái)說(shuō),是一個(gè)重要的信息??梢蕴峁┮恍┗谖恢眯畔⒌姆?wù),比如酒店預(yù)訂公眾賬號(hào),可以給你推薦你周邊的酒店。 另外一個(gè)補(bǔ)充是,可以在文本消息中分析出位置信息,并加以利用。比如用戶(hù)輸入“南京路步行街”,可以提供用戶(hù)南京路步行街的相關(guān)商戶(hù)。 1.4鏈接消息: 目前還沒(méi)有看到開(kāi)發(fā)模式中特別有效的使用方法。使用比較多的可能會(huì)是購(gòu)物時(shí)或是咨詢(xún)時(shí),對(duì)所談?wù)摰膶?duì)象進(jìn)行明確。 1.5事件推送消息: 當(dāng)用戶(hù)進(jìn)入到和你對(duì)話(huà)的過(guò)程中,可以先和用戶(hù)打招呼等。這個(gè)消息目前只支持4.5版本,且暫時(shí)還沒(méi)有開(kāi)發(fā)。后續(xù)可想想的空間很大,比如用戶(hù)進(jìn)入到會(huì)話(huà)之后,搖一搖會(huì)發(fā)生什么呢? 2.回復(fù)消息類(lèi)型 2.1文本消息 這是我們平時(shí)發(fā)送最多的一類(lèi)消息,當(dāng)只需要簡(jiǎn)單的文字即可回答用戶(hù)的消息時(shí),可用文本消息。文本消息中可以帶有鏈接地址。 2.2圖文消息 圖文消息,這是我們?cè)谕扑拖⒅薪?jīng)常看到的消息格式。每項(xiàng)內(nèi)容可以點(diǎn)擊查看更詳細(xì)信息(當(dāng)然你也可以把鏈接設(shè)置為空,使其不能跳轉(zhuǎn)) 2.3音樂(lè)消息 在你的答復(fù)中給用戶(hù)一個(gè)語(yǔ)音消息或是音樂(lè),可以獲得不少用戶(hù)的親睞。 了解了公眾平臺(tái)的通訊機(jī)制和消息類(lèi)型,接下來(lái),我們開(kāi)始準(zhǔn)備開(kāi)發(fā)環(huán)境了。
在類(lèi)似PHP的項(xiàng)目中,經(jīng)常會(huì)碰到項(xiàng)目開(kāi)發(fā)目錄和運(yùn)行目錄不是一個(gè)目錄的情況。 在window7下面有一個(gè)非常好的工具,可以做到自動(dòng)同步。 如: mklink /J D:\wamp\www\shanghaisales C:\Users\Admin\git\ShanghaiSales\ShanghaiSales
|