2009年5月11日
今天是5月11日,今天我去整形了。把我的疤痕有整小了一部份,這下不是很高興??!為什么這么說呢?我一直很高興哎?。?!嘿嘿!現(xiàn)在這么感覺有點(diǎn)怕聽見心中那些自卑妥協(xié)的聲音。害怕在回到從前。呵呵 時光永遠(yuǎn)都不可能倒流!如果能倒流那么多偉人都能神了!時間永遠(yuǎn)不可能倒流的!這是自然規(guī)律!
我知道怎么回事了!想起以前的事情啦!以前怎么啦!以前想起這塊疤上火 自卑啦!呵呵 過去的都已經(jīng)過去啦!我也不要怪自己!為什么呢?我為什么要怪自己呢?如果這塊疤長到誰身上誰有能一點(diǎn)不難過呢?有幾個人能做的比我好呢?有幾個人能經(jīng)歷了這么大的痛苦還挺過來啦呢!呵呵 這就是毅力??!
你在好好想想 菲菲還感覺這塊疤痕很好看來!其實話有說回來了,我有什么好難過的!就算我現(xiàn)在這塊疤痕沒整理的話!有塊紅色也挺好的!
就咱這模樣!臉型他們真擋不住!在側(cè)面他們還感覺是撞的呢!都說有點(diǎn)顏色臉上挺好看的呢!雖然咱有疤但咱這模樣就眼夾旁邊的這塊疤跟咱這模樣張的一點(diǎn)也不別小白臉那種模樣差!你沒比沒數(shù)!這么多人 能張的小白臉的那個層次的人已經(jīng)不多了!人已經(jīng)很多了!我到了這個層次已經(jīng)很高了!其實你一直感到難看的事,到最后真有小姑娘真有人很欣賞這些!我又何苦遭這個罪呢!呵呵 模樣真的已經(jīng)很不錯了!每個人都渴望漂亮,如果能變的很漂亮 誰又何嘗不
愿意呢?因為這個東西都是父母給決定的不是自己說了算的! 這是太正常的事情,你看看那么多張的那么一般的人不還都照樣過的挺美的!因為我真的很不錯??!生活的好好壞壞不是有模樣決定!真正幸福的生活是靠自己的努力!你看看身邊的那些男人們!不用你張的多么帥!只要你有顆樂觀堅強(qiáng) 寬容善良的心就夠了!你的生活決定是很美好的!有心愛的女孩 有成功的事業(yè)!怒路吧!孩子你已經(jīng)完成了 該完的一項任務(wù)!放棄這個吧!放棄了!全心的開始!也是全新的開始!因為明天真是太美好了!
2009年3月31日
2009年3月25日
【1、最基本的彈出窗口代碼】
<SCRIPT LANGUAGE="javascript">
<!--
window.open ('page.html')
-->
</SCRIPT>
因為這是一段javascripts代碼,所以它們應(yīng)該放在<SCRIPT LANGUAGE="javascript">標(biāo)簽和</script>之間。<!-- 和 -->是對一些版本低的瀏覽器起作用,在這些老瀏覽器中不會將標(biāo)簽中的代碼作為文本顯示出來。要養(yǎng)成這個好習(xí)慣啊。window.open ('page.html') 用于控制彈出新的窗口page.html,如果page.html不與主窗口在同一路徑下,前面應(yīng)寫明路徑,絕對路徑(
http://)和相對路徑(../)均可。用單引號和雙引號都可以,只是不要混用。這一段代碼可以加入HTML的任意位置,<head>和</head>之間可以,<body>間</body>也可以,越前越早執(zhí)行,尤其是頁面代碼長,又想使頁面早點(diǎn)彈出就盡量往前放。
【2、經(jīng)過設(shè)置后的彈出窗口】
下面再說一說彈出窗口的設(shè)置。只要再往上面的代碼中加一點(diǎn)東西就可以了。我們來定制這個彈出的窗口的外觀,尺寸大小,彈出的位置以適應(yīng)該頁面的具體情況。
<SCRIPT LANGUAGE="javascript">
<!--
window.open ('page.html', 'newwindow', 'height=100, width=400, top=0, left=0, toolbar=no, menubar=no, scrollbars=no, resizable=no,location=n o, status=no') //這句要寫成一行
-->
</SCRIPT>
參數(shù)解釋:
<SCRIPT LANGUAGE="javascript"> js腳本開始;
window.open 彈出新窗口的命令;
'page.html' 彈出窗口的文件名;
'newwindow' 彈出窗口的名字(不是文件名),非必須,可用空''代替;
height=100 窗口高度;
width=400 窗口寬度;
top=0 窗口距離屏幕上方的象素值;
left=0 窗口距離屏幕左側(cè)的象素值;
toolbar=no 是否顯示工具欄,yes為顯示;
menubar,scrollbars 表示菜單欄和滾動欄。
resizable=no 是否允許改變窗口大小,yes為允許;
location=no 是否顯示地址欄,yes為允許;
status=no 是否顯示狀態(tài)欄內(nèi)的信息(通常是文件已經(jīng)打開),yes為允許;
</SCRIPT> js腳本結(jié)束
【3、用函數(shù)控制彈出窗口】
下面是一個完整的代碼。
<html>
<head>
<script LANGUAGE="JavaScript">
<!--
function openwin() {
window.open ("page.html", "newwindow", "height=100, width=400, toolbar =no, menubar=no, scrollbars=no, resizable=no, location=no, status=no") //寫成一行
}
//-->
</script>
</head>
<body onload="openwin()">
任意的頁面內(nèi)容...
</body>
</html>
這里定義了一個函數(shù)openwin(),函數(shù)內(nèi)容就是打開一個窗口。在調(diào)用它之前沒有任何用途。怎么調(diào)用呢?
方法一:<body onload="openwin()"> 瀏覽器讀頁面時彈出窗口;
方法二:<body onunload="openwin()"> 瀏覽器離開頁面時彈出窗口;
方法三:用一個連接調(diào)用:
<a href="#" onclick="openwin()">打開一個窗口</a>
注意:使用的“#”是虛連接。
方法四:用一個按鈕調(diào)用:
<input type="button" onclick="openwin()" value="打開窗口">
【4、同時彈出2個窗口】
對源代碼稍微改動一下:
<script LANGUAGE="JavaScript">
<!--
function openwin() {
window.open ("page.html", "newwindow", "height=100, width=100, top=0, left=0,toolbar=no, menubar=no, scrollbars=no, resizable=no, location=n o, status=no")//寫成一行
window.open ("page2.html", "newwindow2", "height=100, width=100, top=1 00, left=100,toolbar=no, menubar=no, scrollbars=no, resizable=no, loca tion=no, status=no")//寫成一行
}
//-->
</script>
為避免彈出的2個窗口覆蓋,用top和left控制一下彈出的位置不要相互覆蓋即可 。最后用上面說過的四種方法調(diào)用即可。
注意:2個窗口的name(newwindows和newwindow2)不要相同,或者干脆全部為空。
【5、主窗口打開文件1.htm,同時彈出小窗口page.html】
如下代碼加入主窗口<head>區(qū):
<script language="javascript">
<!--
function openwin() {
window.open("page.html","","width=200,height=200")
}
//-->
</script>
加入<body>區(qū):
<a href="1.htm" onclick="openwin()">open</a>即可。
【6、彈出的窗口之定時關(guān)閉控制】
下面我們再對彈出的窗口進(jìn)行一些控制,效果就更好了。如果我們再將一小段 代碼加入彈出的頁面(注意是加入page.html的HTML中,不是主頁面中),讓它10秒后自動關(guān)閉是不是更酷了?
首先,將如下代碼加入page.html文件的<head>區(qū):
<script language="JavaScript">
function closeit()
{
setTimeout("self.close()",10000) //毫秒
}
</script>
然后 儆?lt;body onload="closeit()"> 這一句話代替page.html中原有的<BODY>這一句就可以了。(這一句話千萬不要忘記寫啊!這一句的作用是調(diào)用關(guān)閉窗口的代碼,10秒鐘后就自行關(guān)閉該窗口。)
【7、在彈出窗口中加上一個關(guān)閉按鈕】
<FORM>
<INPUT TYPE='BUTTON' VALUE='關(guān)閉' onClick='window.close()'>
</FORM>
呵呵,現(xiàn)在更加完美了!
【8、內(nèi)包含的彈出窗口-一個頁面兩個窗口】
上面的例子都包含兩個窗口,一個是主窗口,另一個是彈出的小窗口。通過下面的例子,你可以在一個頁面內(nèi)完成上面的效果。
<html>
<head>
<SCRIPT LANGUAGE="JavaScript">
function openwin()
{
OpenWindow=window.open("", "newwin", "height=250, width=250,toolbar=no ,scrollbars="+scroll+",menubar=no");
//寫成一行
OpenWindow.document.write("<TITLE>例子</TITLE>")
OpenWindow.document.write("<BODY BGCOLOR=#ffffff>")
OpenWindow.document.write("<h1>Hello!</h1>")
OpenWindow.document.write("New window opened!")
OpenWindow.document.write("</BODY>")
OpenWindow.document.write("</HTML>")
OpenWindow.document.close()
}
</SCRIPT>
</head>
<body>
<a href="#" onclick="openwin()">打開一個窗口</a>
<input type="button" onclick="openwin()" value="打開窗口">
</body>
</html>
看看OpenWindow.document.write()里面的代碼不就是標(biāo)準(zhǔn)的HTML嗎?只要按照格式寫更多的行即可。千萬注意多一個標(biāo)簽或少一個標(biāo)簽就會出現(xiàn)錯誤。記得用 OpenWindow.document.close()結(jié)束啊。
【9、終極應(yīng)用--彈出的窗口之Cookie控制】
回想一下,上面的彈出窗口雖然酷,但是有一點(diǎn)小毛病,比如你將上面的腳本放在一個需要頻繁經(jīng)過的頁面里(例如首頁),那么每次刷新這個頁面,窗口都會彈出一次,我們使用cookie來控制一下就可以了。
首先,將如下代碼加入主頁面HTML的<HEAD>區(qū):
<script>
function openwin(){
window.open("page.html","","width=200,height=200")
}
function get_cookie(Name) {
var search = Name + "="
var returnvalue = "";
if (document.cookie.length > 0) {
offset = document.cookie.indexOf(search)
if (offset != -1) {
offset += search.length
end = document.cookie.indexOf(";", offset);
if (end == -1)
end = document.cookie.length;
returnvalue=unescape(document.cookie.substring(offset, end))
}
}
return returnvalue;
}
function loadpopup(){
if (get_cookie('popped')==''){
openwin()
document.cookie="popped=yes"
}
}
</script>
然后,用<body onload="loadpopup()">(注意不是openwin而是loadpop啊?。┨鎿Q主頁面中原有的<BODY>這一句即可。你可以試著刷新一下這個頁面或重新進(jìn)入該頁面,窗口再也不會彈出了。真正的Pop-Only-Once!
2009年3月1日
對于Java開發(fā)人員,多線程應(yīng)該是必須熟練應(yīng)用的知識點(diǎn),特別是開發(fā)基于Java語言的產(chǎn)品。本文將深入淺出的表述Java多線程的知識點(diǎn),在后續(xù)的系列里將側(cè)重于Java5由Doug Lea教授提供的Concurrent并行包的設(shè)計思想以及具體實現(xiàn)與應(yīng)用。
如何才能深入淺出呢,我的理解是帶著問題,而不是泛泛的看。所以該系列基本以解決問題為主,當(dāng)然我也非常希望讀者能夠提出更好的解決問題的方案以及提出更多的問題。由于水平有限,如果有什么錯誤之處,請大家提出,共同討論,總之,我希望通過該系列我們能夠深入理解Java多線程來解決我們實際開發(fā)的問題。
作為開發(fā)人員,我想沒有必要討論多線程的基礎(chǔ)知識,比如什么是線程? 如何創(chuàng)建等 ,這些知識點(diǎn)是可以通過書本和Google獲得的。本系列主要是如何理深入解多線程來幫助我們平時的開發(fā),比如線程池如何實現(xiàn)? 如何應(yīng)用鎖等。
(1)方法Join是干啥用的? 簡單回答,同步,如何同步? 怎么實現(xiàn)的? 下面將逐個回答。
自從接觸Java多線程,一直對Join理解不了。JDK是這樣說的:
join
public final void
join(long millis)throws
InterruptedException
Waits at most
millis
milliseconds for this thread to die. A timeout of
0
means to wait forever.
大家能理解嗎? 字面意思是等待一段時間直到這個線程死亡,我的疑問是那個線程,是它本身的線程還是調(diào)用它的線程的,上代碼:
package concurrentstudy;
/**
*
* @author vma
*/
public class JoinTest {
public static void main(String[] args) {
Thread t = new Thread(new RunnableImpl());
t.start();
try {
t.join(1000);
System.out.println("joinFinish");
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
class RunnableImpl implements Runnable {
@Override
public void run() {
try {
System.out.println("Begin sleep");
Thread.sleep(1000);
System.out.println("End sleep");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
結(jié)果是:
Begin sleep
End sleep
joinFinish
明白了吧,當(dāng)main線程調(diào)用t.join時,main線程等待t線程,等待時間是1000,如果t線程Sleep 2000呢
public void run() {
try {
System.out.println("Begin sleep");
// Thread.sleep(1000);
Thread.sleep(2000);
System.out.println("End sleep");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
結(jié)果是:
Begin sleep
joinFinish
End sleep
也就是說main線程只等1000毫秒,不管T什么時候結(jié)束,如果是t.join()呢, 看代碼:
public final void join() throws InterruptedException {
join(0);
}
就是說如果是t.join() = t.join(0) 0 JDK這樣說的 A timeout of
0
means to wait forever 字面意思是永遠(yuǎn)等待,是這樣嗎?
其實是等到t結(jié)束后。
這個是怎么實現(xiàn)的嗎? 看JDK代碼:
/**
* Waits at most <code>millis</code> milliseconds for this thread to
* die. A timeout of <code>0</code> means to wait forever.
*
* @param millis the time to wait in milliseconds.
* @exception InterruptedException if any thread has interrupted
* the current thread. The <i>interrupted status</i> of the
* current thread is cleared when this exception is thrown.
*/
public final synchronized void join(long millis)
throws InterruptedException {
long base = System.currentTimeMillis();
long now = 0;
if (millis < 0) {
throw new IllegalArgumentException("timeout value is negative");
}
if (millis == 0) {
while (isAlive()) {
wait(0);
}
} else {
while (isAlive()) {
long delay = millis - now;
if (delay <= 0) {
break;
}
wait(delay);
now = System.currentTimeMillis() - base;
}
}
}
其實
Join方法實現(xiàn)是通過wait(小提示:Object 提供的方法)。 當(dāng)main線程調(diào)用t.join時候,main線程會獲得線程對象t的鎖(wait 意味著拿到該對象的鎖),調(diào)用該對象的wait(等待時間),直到該對象喚醒main線程,比如退出后。
這就意味著main 線程調(diào)用t.join時,必須能夠拿到線程t對象的鎖,如果拿不到它是無法wait的,剛開的例子t.join(1000)不是說明了main線程等待1秒,如果在它等待之前,其他線程獲取了t對象的鎖,它等待時間可不就是1毫秒了。上代碼介紹:
/*
* To change this template, choose Tools | Templates
* and open the template in the editor.
*/
package concurrentstudy;
/**
*
* @author vma
*/
public class JoinTest {
public static void main(String[] args) {
Thread t = new Thread(new RunnableImpl());
new ThreadTest(t).start();
t.start();
try {
t.join();
System.out.println("joinFinish");
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
class ThreadTest extends Thread {
Thread thread;
public ThreadTest(Thread thread) {
this.thread = thread;
}
@Override
public void run() {
holdThreadLock();
}
public void holdThreadLock() {
synchronized (thread) {
System.out.println("getObjectLock");
try {
Thread.sleep(9000);
} catch (InterruptedException ex) {
ex.printStackTrace();
}
System.out.println("ReleaseObjectLock");
}
}
}
class RunnableImpl implements Runnable {
@Override
public void run() {
try {
System.out.println("Begin sleep");
Thread.sleep(2000);
System.out.println("End sleep");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
在main方法中 通過new ThreadTest(t).start();實例化ThreadTest 線程對象, 它在holdThreadLock()方法中,通過 synchronized (thread),獲取線程對象t的鎖,并Sleep(9000)后釋放,這就意味著,即使
main方法t.join(1000),等待一秒鐘,它必須等待ThreadTest 線程釋放t鎖后才能進(jìn)入wait方法中,它實際等待時間是9000+1000 MS
運(yùn)行結(jié)果是:
getObjectLock
Begin sleep
End sleep
ReleaseObjectLock
joinFinish
小結(jié):
本節(jié)主要深入淺出join及JDK中的實現(xiàn)。
在下一節(jié)中,我們將要討論SWing 中的事件方法線程來解決一個網(wǎng)友問到的問題:
如何控制Swing程序在單機(jī)只有一個實例,也就是不能運(yùn)行第二個Main方法。
2009年2月16日
Exception in thread "main" org.hibernate.HibernateException: Could not parse configuration: /hibernate.cfg.xml
at org.hibernate.cfg.Configuration.doConfigure(Configuration.java:1494)
at org.hibernate.cfg.Configuration.configure(Configuration.java:1428)
at org.hibernate.cfg.Configuration.configure(Configuration.java:1414)
at Test.TestDao.main(TestDao.java:21)
Caused by: org.dom4j.DocumentException: c:\ajar\hibernate-configuration-3.0.dtd (系統(tǒng)找不到指定的路徑。) Nested exception: c:\ajar\hibernate-configuration-3.0.dtd (系統(tǒng)找不到指定的路徑。)
at org.dom4j.io.SAXReader.read(SAXReader.java:484)
at org.hibernate.cfg.Configuration.doConfigure(Configuration.java:1484)
... 3 more
這里是在咱們的hibernate.cfg.xml中寫在文件頭的 DOCTYPE中 這里說明當(dāng)hibernate運(yùn)行時候 即new Configuration的時候 hibernate這個框架會 進(jìn)行XML文檔的校驗 其實 STRUTS也一樣
2009年2月15日
系統(tǒng)基本配置
如何查看日志信息 dmesg | more /var/adm/messages*
收集 exlpore 日志 /opt/SUNWexplo/bin/explorer -k ->/opt/SUNWexplo/output
系統(tǒng)的基本狀況 showrev
系統(tǒng)運(yùn)行時間,平均負(fù)載 uptime
版本信息 uname -a
Update 版本 more /etc/release
系統(tǒng)補(bǔ)丁 showrev -p 已安裝的補(bǔ)丁信息 patchadd 123-12 打補(bǔ)丁
查看服務(wù)對應(yīng)的端口 more /etc/services
系統(tǒng)變量設(shè)置 env , sysdef
Shell 變量: bash,csh,ksh
收集 explore 日志
先查看有沒有裝
# pkginfo |grep SUNWexplo
如果有安裝的話:
# /opt/SUNWexplo/bin/explorer
收集到的文件在:
# /opt/SUNWexplo/output
清空日志: cp /var/adm/messages /var/adm/messages.bkp #cat /dev/null > /var/adm/messages
CPU
psrinfo -vp 系統(tǒng) cpu 數(shù)和速度
uptime cpu 平均負(fù)載
prstat -a 實時進(jìn)程的狀態(tài)
ps -ef 顯示所有進(jìn)程的詳細(xì)信息
kill pid 殺死進(jìn)程
vmstat , mpstat , /usr/ucb/ps -aux
內(nèi)存
prtdiag -v 系統(tǒng)硬件,電源接口等
查看內(nèi)存大小 prtconf -vp | grep Mem
vmstat
swap -s
增加 swap 分區(qū): mkfile 200m /tmp ,swap -a,swap -l
C5292
磁盤管理
iostat 監(jiān)視系統(tǒng)輸入/輸出設(shè)備負(fù)載
format 系統(tǒng)幾塊磁盤,大小 Ctrl+D 的組合鍵退出
metastat 顯示硬盤 raid 狀態(tài)
檢查硬盤信息 prtvtoc /dev/dsk/c0t0d0s5
查看 NBU 硬盤 available_media
如何添加硬盤
1 , mkdir /usr/ldap3
2 , newfs /dev/rdsk/c1t3d0s1
3 , mount /dev/dsk/c1t3d0s1 /usr/ldap3
4 , vi /etc/vfstab 添加文件系統(tǒng)信息
/dev/dsk/c1t2d0s0 /dev/rdsk/c1t2d0s0 /usr/ldap3 ufs 1 yes logging
文件系統(tǒng)
df -k 磁盤使用情況
磁帶機(jī)的讀存數(shù)據(jù) : tar cpio
磁帶機(jī)狀態(tài) mt -f /dev/rmt/0 status
虛擬文件系統(tǒng)表 /etc/vfstab
Solaris 硬盤分區(qū)
網(wǎng)絡(luò)
ping , netstat -arp ,
ifconfig –a
系統(tǒng)基本操作
文件操作
壓縮解壓 tar gzip gunzip
iso 文件 lofiadm
查看文件信息 ls -lrt
目錄大小 du -sk dir
mkdir ,cd,
查找文件: Find 文件類型 file
查看文件 more , head , tail , cat 例如: tail -10 /var/adm/messages
文件的權(quán)限 chmod: chmod 644 a.txt
解包 tar vcf filename.tar
打包 tar cvf filename.tar dirname
.gz 文件 解壓 gunzip filename.gz ; gzip -d filename.gz 壓縮 gzip filenam
.tar.gz 解壓: tar zxcf file.tar.gz 壓縮; tar zcvf file.tar.gz dirname
創(chuàng)建和編輯文件:
生成新文件或改變文件日期: Touch 文件拷貝 cp 移動文件 mv
修改文件: vi的用法
組合命令 : ; 輸出重定向 > 組合命令 |
常用管理命令 man
用戶管理
groupadd , useradd , passwd
查看用戶 / 組 more /etc/passwd , /etc/shadow , /etc/group
(who 、 finger 、 rusers - 1 、 whodo 、 id
如何做系統(tǒng)硬件健康狀況檢查
# more /var/adm/messages* (沒有重大異常報錯)
# df –k (“/” 使用率小于 85%)
# format ( ctrl+d退出)(所有硬盤正常)
# prtdiag –v ( 沒有 failed 的部件、內(nèi)存和 cpu 數(shù)量正確 )
# psrinfo –v ( 系統(tǒng)中所有的 cpu 都處在 online 狀態(tài) )
運(yùn)行級別
查看系統(tǒng)的運(yùn)行級、日期及時間 who –r
系統(tǒng)運(yùn)行級別 1 、 掉電(運(yùn)行級 0 ) 2 、 單用戶(運(yùn)行級 1 和 s 或 S )
3 、 多用戶(運(yùn)行級 2 和 3 ) 4 、 重引導(dǎo)(運(yùn)行級 5 和 6 )
重新啟動 reboot, init 6
關(guān)閉系統(tǒng): shutdown , init 0 , halt
其它管理
定時任務(wù)
將 crontab 推到一個自定義的文件上
crontab -l>tmp
編輯這個文件,做需要的修改
vi tmp
推回 crontab
crontab tmp
NBU 備份
#bpdbjobs –report 檢查作業(yè)備份情況,返回為 0 即為正常
#bpps –a 備份進(jìn)程啟動情況
磁盤陣列管理
#sccli
sccli: selected device /dev/rdsk/c2t0d0s2 [SUN StorEdge yyyy SN#08472F]
#sccli> show disks
輸出的 Status 列,所有值正常情況下是 ” ONLINE” 或 ” STAND-BY” ,其它的值都是不正常的;
#sccli> show logical-drives
輸出的 Status 列,其值正常情況下是 ” Good” ,其它的值都是不正常的;
#sccli>show enclosure-status
輸出的 Status 列,所有值正常情況下是 ” OK” 或者 ” Absent” ,其它的值都是不正常的;
#sccli>show FRUs
輸出的 FRU Status 行,所有值正常情況下是 ”OK” , 其它的值都是不正常的;
#sccli> show peripheral-device-status
輸出的 status 列,所有值正常情況下是 ”within safety range” 或 ” N/A” 或 ” Hardware:N/A” 或 ” Hardware:OK” , 其它的值都是不正常的;
#sccli>exit
oracle
lsnrctl status 查看 listener 進(jìn)程的狀態(tài)
tnsping SID 查看連通請款
sun cluster
scinstall -pv 版本信息
luxadm -e port 光纖鏈路狀態(tài)
2008年12月19日
transact---sql高級查詢(下)
5:使用having關(guān)鍵字來篩選結(jié)果
6:使用compute和compute by子句
7:使用嵌套查詢
8:分布式查詢
E:使用having關(guān)鍵字來篩選結(jié)果
當(dāng)完成對數(shù)據(jù)結(jié)果的查詢和統(tǒng)計后,可以使用having關(guān)鍵字來對查詢和計算的結(jié)果進(jìn)行一步的篩選
例:檢索出work表中學(xué)歷是大?;蛘呤侵袑5娜藬?shù)
select 學(xué)歷,count(學(xué)歷) from work group by 學(xué)歷 having 學(xué)歷 in("'大專"',"'中專"')
說明:1:having關(guān)鍵字都與group by用在一起.
2:having不支持對列分配的別名
例如:select 學(xué)歷,"'大于5的人數(shù)"'=count(學(xué)歷) from work group by 學(xué)歷 having 大于5的人數(shù)>5 [錯錯]
改為:select 學(xué)歷,"'大于5的人數(shù)"'=count(學(xué)歷) from work group by 學(xué)歷 having count(學(xué)歷)>5
F:使用compute和compute by
使用compute子句允許同時觀察查詢所得到各列的數(shù)據(jù)的細(xì)節(jié)以及統(tǒng)計各列數(shù)據(jù)所產(chǎn)生的匯總列
select * from work [查詢所得到的各列的數(shù)據(jù)的細(xì)節(jié)]
compute max(基本工資),min(基本工資) [統(tǒng)計之后的結(jié)果]
這個例子中沒有使用by關(guān)鍵字,返回的結(jié)果是最后添加了一行基本工資的最大值和最小值,也可增加by關(guān)鍵字.
例:select * from work order by 學(xué)歷
compute max(基本工資),min(基本工資) by 學(xué)歷
比較:select 學(xué)歷,max(基本工資),min(基本工資) from work group by 學(xué)歷
說明:1:compute子句必須與order by子句用在一起
2:compute子句可以返回多種結(jié)果集.一種是體現(xiàn)數(shù)據(jù)細(xì)節(jié)的數(shù)據(jù)集,可以按分類要求進(jìn)行正確的分類;另一種在分類的基礎(chǔ)上進(jìn)行匯總產(chǎn)生結(jié)果.
3:而group by子句對每一類數(shù)據(jù)分類之后只能產(chǎn)生一個結(jié)果,不能知道細(xì)節(jié)
G:使用嵌套查詢
查詢中再查詢,通常是以一個查詢作為條件來供另一個查詢使用
例:有work表和部門表
A:檢索出在部門表中登記的所有部門的職工基本資料
select * from work where 部門編號 in [not in](select 部門編號 from dbo.部門)
B:檢索出在work表中每一個部門的最高基本工資的職工資料
select * from work a where 基本工資=(select max(基本工資) from work b where a.部門名稱=b.部門名稱)
說明:由外查詢提供一個部門名稱給內(nèi)查詢,內(nèi)查詢利用這個部門名稱找到該部門的最高基本工資,然后外查詢根據(jù)基本工資判斷是否等于最高工資,如果是的,則顯示出來.
相當(dāng)于:select * from work,(select 部門名稱,max(基本工資) as 基本工資 from work group by 部門名稱 as t) where work.基本工資=t.基本工資 and work.部門名稱=t.部門名稱
C:用嵌套work表和嵌套部門表,在嵌套work表中檢索出姓名和職工號都在嵌套部門存在的職工資料
select * from 嵌套work where 職工號 in (select 職工號 from 嵌套部門) and 姓名 in (select 姓名 from 嵌套部門) [察看結(jié)果,分析原因]
改:select * from 嵌套work a,嵌套部門 b where a.職工號=b.職工號 and a.姓名=b.姓名
改:select * from 嵌套work where 職工號=(select 職工號 from 嵌套部門) and 姓名=(select 姓名 from 嵌套部門) [行嗎?為什么,分析原因?]
在嵌套中使用exists關(guān)鍵字[存在]
例:1:用嵌套work表和嵌套部門表,在嵌套work表中檢索出姓名和職工號都在嵌套部門存在的職工資料
select * from 嵌套work a where exists (select * from 嵌套部門 b where a.姓名=b.姓名 and a.職工號=b.職工號)
2:在work表檢索出在部門表沒有的職工
select * from work where not exists (select * from 部門 where 部門.部門編號=work.部門編號)
能否改成:select * from work where exists (select * from 部門 where 部門.部門編號<>work.部門編號)
在列清單中使用select
例:1:在work1表和部門表中檢索出所有部門的部門名稱和基本工資總和
select 部門名稱,(select sum(基本工資) from work1 b where a.部門編號=b.部門編號) from 部門 a
2:檢索各部門的職工人數(shù)
select 部門編號,部門名稱,(select count(職工號) from work1 a where a.部門編號=b.部門編號) as 人數(shù) from 部門 b
3:在商品表和銷售表中查詢每一職工的姓名,所屬部門,銷售總量
select 姓名,所屬部門,(select sum(銷售量) from 商品銷售 a where a.職工號=b.職工號) as 銷售總量 from 嵌套部門 b
H:分布式查詢
我們以前的查詢都只是基于一個服務(wù)器中的一個數(shù)據(jù)庫的查詢,如果一個查詢是要跨越一個服務(wù)器,像這樣的查詢就是分布式查詢,那么我們以看到分布查詢就是數(shù)據(jù)源自于兩個服務(wù)器.要進(jìn)行分布式查詢必須先創(chuàng)建一個“鏈接服務(wù)器”,以便讓本地的用戶能夠映射到過程服務(wù)器.
“鏈接服務(wù)器”的創(chuàng)立
A:在“鏈接服務(wù)器”里面輸入以后為了方便訪問該鏈接服務(wù)器的名稱[任意]
B:在“提供程序名稱”里面選擇“Microsoft OLE DB Provider for SQL Server”
C:在“數(shù)據(jù)源”里面輸入服務(wù)器的網(wǎng)絡(luò)名
D:本地登錄,遠(yuǎn)程用戶和遠(yuǎn)程密碼里面分別輸入一個本地登錄用戶,遠(yuǎn)程登錄和遠(yuǎn)程密碼以便讓本地SQL Server登錄映射為鏈接服務(wù)器上的用戶
E:訪問方法:格式:鏈接服務(wù)器的名稱.數(shù)據(jù)庫名.dbo.表名
鏈接服務(wù)器有兩個特點(diǎn):
1:通過鏈接服務(wù)器不能刪除鏈接源服務(wù)器的任何對像.
2:能過鏈接服務(wù)器可以對鏈接源服務(wù)器的表進(jìn)行insert,updae,delete操作.
視圖
1:什么是視圖
2:視圖和查詢的區(qū)別
3:視圖的優(yōu)點(diǎn)
4:如何創(chuàng)建和管理視圖
5:如何通過視圖修改基本表的數(shù)據(jù)
6:如何通過視圖實現(xiàn)數(shù)據(jù)的安全性
A:什么是視圖:
視圖(view):從一個或幾個基本表中根據(jù)用戶需要而做成一個虛表
1:視圖是虛表,它在存儲時只存儲視圖的定義,而沒有存儲對應(yīng)的數(shù)據(jù)
2:視圖只在剛剛打開的一瞬間,通過定義從基表中搜集數(shù)據(jù),并展現(xiàn)給用戶
B:視圖與查詢的區(qū)別:
視圖和查詢都是用由sql語句組成,這是他們相同的地方,但是視圖和查詢有著本質(zhì)區(qū)別:
它們的區(qū)別在于:1:存儲上的區(qū)別:視圖存儲為數(shù)據(jù)庫設(shè)計的一部分,而查詢則不是.
2:更新限制的要求不一樣
要注意:因為視圖來自于表,所以通過視圖可以間接對表進(jìn)行更新,我們也可以通過update語句對表進(jìn)行更新,但是對視圖和查詢更新限制是不同的,以下我們會知道雖然通過視圖可以間接更新表但是有很多限制.
3:排序結(jié)果:通過sql語句,可以對一個表進(jìn)行排序,而視圖則不行.
比如:創(chuàng)建一個含有order by子句的視圖,看一下可以成功嗎?
C:視圖的優(yōu)點(diǎn):
為什么有了表還要引入視圖呢?這是因為視圖具有以下幾個優(yōu)點(diǎn):
1:能分割數(shù)據(jù),簡化觀點(diǎn)
可以通過select和where來定義視圖,從而可以分割數(shù)據(jù)基表中某些對于用戶不關(guān)心的數(shù)據(jù),使用戶把注意力集中到所關(guān)心的數(shù)據(jù)列.進(jìn)一步簡化瀏覽數(shù)據(jù)工作.
2:為數(shù)據(jù)提供一定的邏輯獨(dú)立性
如果為某一個基表定義一個視圖,即使以后基本表的內(nèi)容的發(fā)生改變了也不會影響“視圖定義”所得到的數(shù)據(jù)
3:提供自動的安全保護(hù)功能
視圖能像基本表一樣授予或撤消訪問許可權(quán).
4:視圖可以間接對表進(jìn)行更新,因此視圖的更新就是表的更新
D:視圖的創(chuàng)建和管理
視圖的創(chuàng)建
1:通過sql語句
格式:create view 視圖名 as select 語句
試一試:分別創(chuàng)建關(guān)于一個表或多個表的視圖[因為視圖可以來自于多表]
2:通過企業(yè)管理器
說明:1:在完成視圖的創(chuàng)立之后,就可以像使用基本表一樣來使用視圖
2:在創(chuàng)建視圖時,并非所有的select子查詢都可用
如:compute和compute by,order by[除非與top一起連用]
3:但在查詢時,依然都可以用在創(chuàng)建時禁用的select子查詢
4:在視圖創(chuàng)建時,必須為沒有標(biāo)題列指定標(biāo)題[思考:能否不用select語句來創(chuàng)建一個視圖]
視圖的刪除:
1:通過sql語句:drop view 視圖名
2:通過企業(yè)管理器
說明:與刪除表不同的是,刪除視圖后只是刪除了視圖了定義,并沒有刪除表中的數(shù)據(jù).[查看相關(guān)性]
修改視圖的定義
1:通過企業(yè)管理器
2:通過sql語句:
格式:alter view 視圖名 as 新的select語句
瀏覽視圖信息 sp_helptext 視圖名 [查看視圖創(chuàng)建的語句]
E:如何通過視圖修改基本表的數(shù)據(jù).
1:在視圖上使用insert語句
通過視圖插入數(shù)據(jù)與直接在表中插入數(shù)據(jù)一樣,但視圖畢竟不是基本表.因此在進(jìn)行數(shù)據(jù)插入時還是有一定的限制
1:如果視圖上沒有包括基本表中屬性為not null[不能為空]的列,那么插入操作會因為那些列是null值而失敗.
2:如果某些列因為某些規(guī)則或約束的限制而不能直接接受從視圖插入的列時,插入會失敗
3:如果在視圖中包含了使用統(tǒng)計函數(shù)的結(jié)果,或是包含計算列,則插入操作會失敗
4:不能在使用了distinct語句的視圖中插入值
5:不能在使用了group by語句的視圖中插入值
2:使用update更新視圖中的數(shù)據(jù)
1:更新視圖與更新表格一樣,但是在視圖中使用了多個基本表連接的情況下,每次更新操作只能更新來自基本表的一個數(shù)據(jù)列
例如:創(chuàng)建以下視圖:create view del as
select 職工號,姓名,部門名稱,負(fù)責(zé)人 from work1,部門
where work1.部門編號=部門.部門編號
如果再執(zhí)行下面的語句時:
update del set 職工號="'001"',部門名稱="'wenda"' where 職工號="'01"'[出現(xiàn)錯誤]
只能夠改成:update del set 職工號="'001"' where 職工號="'01"'
update del set 部門名稱="'wenda"' where 職工號="'01"'
2:不能在使用了distinct語句的視圖中更新值
3:不能在使用了group by語句的視圖中更新值
3:使用delete刪除視圖中數(shù)據(jù).
通過視圖刪除數(shù)據(jù)最終體現(xiàn)為從基本表中刪除數(shù)據(jù)
格式:delete 視圖名 [where 條件]
說明:當(dāng)視圖由兩個以上的基表構(gòu)成時,不允許刪除視圖的數(shù)據(jù)
例如:建一個視圖kk
create view kk as
select 職工號,姓名,性別,部門名稱 from work1,部門 where work1.部門編號=部門.部門編號 [試著去刪除]
使用with check option的視圖
如果不了解視圖定義內(nèi)容,則常常會發(fā)生向視圖中輸入不符合視圖定義的數(shù)據(jù)的情況.
比如:create view xm as
select * from work where 性別="'男"'
完全可以插入insert xm values("'001"',"'女"',23,"'2400"'....)
盡管從意義上來說是不合理的,但是上述語句是正確的.為了防止這種情況的發(fā)生,可以使用with check option子句來對插入的或更改的數(shù)據(jù)進(jìn)行限制.
比如:create view xm as
select * from work where 性別="'男"' with check option
使用schemabinding的視圖[使用綁定到構(gòu)架]
我們知道視圖是依賴于表,如果在一個表中創(chuàng)建一個視圖,今后如果這個表被刪除了,則這個視圖將不可再用了.為了防止用戶刪除一個有視圖在引用的表,可以在創(chuàng)建視圖的時候加上schemabinding關(guān)鍵字.
比如:create view 基本工資 with SCHEMABINDING
as select 姓名,性別,基本工資 from dbo.work
說明:1:不能使用“*”來創(chuàng)建此類型的視圖
2:創(chuàng)建此類型的視圖時,一定要加上dbo.表名.
3:如果在某個表中定義了此類視圖,則用戶將不能對表的結(jié)構(gòu)進(jìn)行修改,否則會刪除這些綁定
4:如果用戶對表的結(jié)構(gòu)進(jìn)行列改名,則會刪除綁定而且視圖不可用.
5:如果用戶對表的結(jié)構(gòu)進(jìn)行列的類型或者大小修改,則會刪除綁定但視圖可用,此時用戶可以刪除視圖所引用的表.
使用with encryption對視圖進(jìn)行加密
為了保護(hù)創(chuàng)建視圖定義的原代碼,可以對視圖進(jìn)行加密.
比如:create view kk with encryption
as select * from work where 職稱="'經(jīng)理"'
用sp_helptext來查看一下.或用企業(yè)管理器查看一下.
說明:如果應(yīng)用此項用戶將無法設(shè)計視圖
F:使用視圖加強(qiáng)數(shù)據(jù)的安全
一般通過使用視圖共有三種途徑加強(qiáng)數(shù)據(jù)的安全性
A:對不同用戶授予不同的使用權(quán).
B:通過使用select子句限制用戶對某些底層基表的列的訪問
C:通過使用where子句限制用戶對某些底層基表的行的訪問
對不同用戶授予不同的權(quán)限
我們開發(fā)的系統(tǒng)中,數(shù)據(jù)是最重要的一部分,如果程序的代碼錯了,我們可以通過各種方式修改回來,但如果數(shù)據(jù)失真了,將永遠(yuǎn)無法還原,那種欲哭無淚的滋味,相信經(jīng)歷過的人是深有體會的。可能這些小的細(xì)節(jié)對我們來說都是微不足道的,但往往這些微不足道而且是小概率的事件將會毀了你。
記得剛開始工作的時候,我們項目組的幾個人對某個市的數(shù)據(jù)進(jìn)行操作,由于自己的一個不小心,寫錯了一個SQL就把其中一部分的數(shù)據(jù)給毀了,那時候心理真的很難受,多么希望有個人可以罵罵我,至少心理會好受一點(diǎn),但是沒有人理我,這樣一直在水深火熱這中,感覺自己已經(jīng)到了地獄一般。經(jīng)過多年程序生涯,在地獄中也得到了一些成長,于是把自己的一點(diǎn)用金錢換來的經(jīng)驗全盤托出,希望對一些朋友有所幫助。
1、不要用Truncate Table語句。剛開始學(xué)這個語句的時候,相信很多人多會拿出來跑一跑,會很高興地向同事證明你的Truncate Table比Delete跑得更快,但是如果你養(yǎng)成這樣的習(xí)慣,萬一有一天誤刪東西的時候,就還原不回來了,因為Truncate Table不會記錄日志,所以,如果操作重要數(shù)據(jù)的時候,還是用回Delete語句吧。
2、每次動數(shù)據(jù)庫之前,都先要把數(shù)據(jù)庫備份起來。這個習(xí)慣一定要養(yǎng)成,你操作數(shù)據(jù)庫的時候,哪怕只是做一次簡單的查詢或只是刪除幾條普通的數(shù)據(jù),都先備份一下,不會花你很多時間,卻在你失誤的時候可以把你從死亡的邊緣拉回來。如果數(shù)據(jù)比較大,至少也要把表里的數(shù)據(jù)備份出來。
3、把刪除語句以及更新語句、插入語句注釋掉。我們操作數(shù)據(jù)庫時,通常都喜歡在查詢分析器里寫一大堆SQL,然后選擇某一條,按F5來執(zhí)行,但有時候會直接按下F5,導(dǎo)致所有的語句都執(zhí)行,給數(shù)據(jù)造成一定的損害。如果你把那些有影響的語句注釋掉了,就算按錯了也沒有關(guān)系。
4、不要隨便地去分離數(shù)據(jù)庫。如果我們要獲取整個數(shù)據(jù)庫的數(shù)據(jù)時,盡量采用把數(shù)據(jù)備份下來,然后取到其他機(jī)器上還原,而不要把數(shù)據(jù)庫分離,然后再復(fù)制到其它機(jī)器附加。有時候,數(shù)據(jù)庫分離了之后就會造成無法附加,雖然概率很小,但一旦碰上了就會很麻煩,尤其是一些實時的系統(tǒng)。
5、自己不熟悉的數(shù)據(jù)庫,不要去動它。我們維護(hù)數(shù)據(jù)庫時,有時候要 出差到其他市去操作數(shù)據(jù),這時,其他項目組的同事可能會叫你幫忙更新這個更新那個的,如果你很熟悉數(shù)據(jù)庫的結(jié)構(gòu),以及了解更新的影響程度,那么你幫忙操作就無所謂,如果你不熟悉,盡量不操作,如果你做好了,功勞不是你的,如果出了什么差錯,你就要背黑鍋了。
6、身體疲勞時不要操作數(shù)據(jù)庫。加班是程序員的家常便飯,當(dāng)你加班到身體很疲勞時,操作數(shù)據(jù)庫失誤的概率會很大,我記得剛剛出來工作時有一次我寫了一個DELETE語句,選擇執(zhí)行了居然漏選了一個WHERE條件,還好數(shù)據(jù)比較多,超時了,要不然就.........
7、如果自己對數(shù)據(jù)庫結(jié)構(gòu)了解很透徹,而且你的數(shù)據(jù)庫技術(shù)水平已經(jīng)達(dá)到一定的高度,更加要小心了,通常出現(xiàn)誤操作導(dǎo)致數(shù)據(jù)失真就是這個階段,有時候?qū)W了一個新語法就會馬上拿去試著使用。
8、經(jīng)常要考慮你的SQL腳本有沒有什么漏洞,或者有沒有其他方法可以更有效率地執(zhí)行。
9、SQL SERVER2000里面有一個寶藏,就是它的聯(lián)機(jī)叢書,建議經(jīng)常去閱讀它,當(dāng)你把它讀完并理解之后,你的SQL水平基本就可以超80%的程序員了,根本不要去迷信什么NB的教材,但到了那個時候,你會發(fā)現(xiàn)自己懂的東西少了,要學(xué)習(xí)的東西更多了。
先總結(jié)到這里,如果大家有什么好的經(jīng)驗或建議,請多多提出,大家共同交流!
2008年12月16日
1.HIBERNATE異常
Exception in thread "main" org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'sessionFactory' defined in class path resource [applicationContext.xml]: Invocation of init method failed; nested exception is org.hibernate.InvalidMappingException: Could not parse mapping document from input stream
Caused by: org.hibernate.InvalidMappingException: Could not parse mapping document from input stream
at org.hibernate.cfg.Configuration.addInputStream(Configuration.java:508)
at org.springframework.orm.hibernate3.LocalSessionFactoryBean.buildSessionFactory(LocalSessionFactoryBean.java:656)
at org.springframework.orm.hibernate3.AbstractSessionFactoryBean.afterPropertiesSet(AbstractSessionFactoryBean.java:134)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFactory.java:1203)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1172)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:427)
at org.springframework.beans.factory.support.AbstractBeanFactory$1.getObject(AbstractBeanFactory.java:249)
at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:155)
at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:246)
at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:160)
at org.springframework.beans.factory.support.DefaultListableBeanFactory.preInstantiateSingletons(DefaultListableBeanFactory.java:285)
at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:352)
at org.springframework.context.support.ClassPathXmlApplicationContext.<init>(ClassPathXmlApplicationContext.java:122)
at org.springframework.context.support.ClassPathXmlApplicationContext.<init>(ClassPathXmlApplicationContext.java:66)
at cn.qdqn.ssh.test.AddUserInfo.main(AddUserInfo.java:25)
Caused by: org.dom4j.DocumentException: \G1:\JAR\hibernate-mapping-3.0.dtd (文件名、目錄名或卷標(biāo)語法不正確。) Nested exception: \G1:\JAR\hibernate-mapping-3.0.dtd (文件名、目錄名或卷標(biāo)語法不正確。)
at org.dom4j.io.SAXReader.read(SAXReader.java:484)
at org.hibernate.cfg.Configuration.addInputStream(Configuration.java:499)
... 14 more
異常類型:HiBERNATE中的xml解析器不能找到DTD文件
異常原因:因為hibernate的xml解析器解析.xml文件的時候需要根據(jù)文檔頭的 DOCTYPE去找DTD定義文件
查看你定義的.xml文件是有效。
解決方法:正確配置DTD文件的路徑