
2008年7月31日
內容如下:

Code
<script>
setInterval( function() {
var date = new Date();
if (date.getMinutes() == 0 || date.getMinutes() == 30) {
alert("老大,喝點水,休息一下!");
}
}, 1000 * 60);
</script>
大家注意身體啊

--------------------------
09博客園紀念T恤新聞:
網站評測:搜狐博客 VS 我燒網網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2009/08/25/1553578.html
posted @
2009-08-25 13:33 破名超難起 閱讀(157) |
評論 (0) |
編輯 收藏
防盜鏈原理:
http標準協議中有專門的字段記錄referer
一來可以追溯上一個入站地址是什么
二來對于資源文件,可以跟蹤到包含顯示他的網頁地址是什么。
因此所有防盜鏈方法都是基于這個Referer字段
網上比較多的2種
一種是使用apache文件FileMatch限制,在httpd.conf中增加 ( 其實也可以將把下面的語句存成一個.htaccess文件),并放到你的網站的根目錄(就是www/html目錄),這樣子別人就沒有辦法盜連你的東東了~~
SetEnvIfNoCase Referer "^
Order Allow,Deny
Allow from env=local_ref
Allow from 127.0.0.1
這種很方便禁止非允許訪問URL引用各種資源文件
請大家注意,把第一句"^http://www.linji.cn
我應該這么寫的
"^
第二種是使用rewrite,需要增加apache的mode_rewrite,支持.htaccess文件目錄權限限制
在虛擬主機根目錄增加.htaccess文件,描述從定向,把非本地地址refer的圖片文件都從定向到警告圖片或者警告網頁上。
首先要確認你的服務器或空間的服務器解譯引擎為Apache2,還有支持.htaccess客戶設置文件,
如果你有自己的服務器就請先對./conf/httpd.conf 文件做以下修改
找到:#LoadModule rewrite_module modules/mod_rewrite.so
把前面的 # 給去丟
找到等一個 AllowOverride None 改為 AllowOverride All
重啟Apache2服務器
接下就是做一個 .htaccess 文件了,其 .htaccess 文件內容為
RewriteEngine on
RewriteCond %{HTTP_REFERER} !^http://aaoo.net/.*$ [NC]
RewriteCond %{HTTP_REFERER} !^http://aaoo.net$ [NC]
RewriteCond %{HTTP_REFERER} !^http://www.aaoo.net/.*$ [NC]
RewriteCond %{HTTP_REFERER} !^http://www.aaoo.net$ [NC]
RewriteRule .*.(jpg|jpeg|gif|png|bmp|rar|zip|exe)$ http://down.yoyo.com.ru/err.html [R,NC]
其中有色的地方都是要改為你的:
紅色:就是改為你提供下載頁面的地址,也就是只有通過這個地址才可以下載你所提供的東東。
藍色:就是要保護文件的擴展名(以|分開),也就是說以這些為擴展名的文件只有通過紅色的地址才可以訪問。
綠色:如果不是通過紅色的地址訪問藍色這些為擴展名的文件時就回重定向到綠色地址上。
這個方法有個好處是,不同的虛擬主機用不同的描述定義。
接下就是怎么用 .htaccess 文件來實現防盜鏈了。
首先要在空間上建兩個目錄(當然目錄名隨你),一個為 web 另一個為 down ,
web 是用來放下載頁面的(或下載程序),down 當然就是放你提供的東東的啦,
把 .htaccess 文件的紅色部分改一下,改為http://你的域名/web。藍色部分
改為你要保護文件的擴展名。綠色部分改為http://你的域名/web。改后保存
.htaccess 文件把它上傳到 down 目錄。
還有第三種:
我在解決plog禁止盜鏈的時候,發現個問題,也算個好方法。
plog把所有資源都自己管理起來,用resserver.php來動態顯示,這樣統一的入口方便添加權限操作。
同時造成上面2種方法無法使用,因為不再是apache直接訪問資源文件,而是php通過文件讀取。
因此只能在代碼中做手腳:在讀取資源文件輸出之前,加如下判斷代碼
引用
$referer = $_SERVER['HTTP_REFERER'];
$selfurl = $_SERVER['HTTP_HOST'];
if(false == strpos($referer,$selfurl))
{
echo '非法盜鏈!';
exit(1);
}
這里有些偷懶,直接看引用地址中是否包含host地址,不過原理就是這樣,判斷referer是否是本站地址。
我們常常在下載的時候,也碰到盜鏈網站無法下載,報盜鏈的問題。要下載這類文件最簡單的方法就是改referer
比方flashget中,網址下面的"引用"一欄中,直接填寫下載地址就可以了。

--------------------------
09博客園紀念T恤新聞:
微軟:不一樣的PowerPoint 2010網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2009/08/21/1551756.html
posted @
2009-08-21 23:30 破名超難起 閱讀(330) |
評論 (0) |
編輯 收藏
這兩天完善了一下視頻轉換這個東西,以前做的那套東西上傳完之后就開始轉換,無法適應大并發量下的視頻轉換(我覺得同時有10個ffmpeg進程在轉視頻服務器肯定要掛掉),所以我用了另一套方案,視頻上傳之后先不轉換,把視頻的基本信息存到數據庫中,然后由程序定時的從數據庫中讀取數據,依次轉換視頻。具體做法是:1、上傳文件,將文件名存入數據庫,同時在數據庫標明videostat字段為0(表示未轉換)
2、通過程序,每隔30秒(根據不同情況可以改變)取出一個未轉換(videostat=0)且失敗次數小于5(confailtime<5)的紀錄
3、開始轉換視頻,這里要先將數據庫中videostat字段改為2(表示正在轉換)不然30秒鐘轉換不完下個進程又會讀到這條紀錄開始轉換......
4、等待轉換進程結束,如果成功將相應紀錄的videostat字段的值改為1(表示轉換成功),若轉換失敗則將失敗次數字段加一(confailtime=confailtime+1)
在鼓搗這東西的過程中,遇到了一個問題,如果一個進程先執行p.waitFor();方法而后輸出命令行中的結果是不行的,即:

Code
int flag = p.waitFor();
InputStream inErr = p.getErrorStream();
InputStream inIns = p.getInputStream();
BufferedReader brErr = new BufferedReader(new InputStreamReader(
inErr));
BufferedReader brIns = new BufferedReader(new InputStreamReader(
inIns));
// inErr讀取輸出信息開始
String strsErr = "";
String strErr = brErr.readLine();
while (strErr != null) {
strsErr = strsErr + strErr + "\n";
System.out.println(strErr);
strErr = brErr.readLine();
}
// inErr讀取輸出信息結束
// inIns讀取輸出信息開始
String strsIns = "";
String strIns = brIns.readLine();
while (strIns != null) {
strsIns = strsIns + strIns + "\n";
System.out.println(strIns);
strIns = brErr.readLine();
}
如果這樣執行的話進程會掛起,無法繼續下午,而正確的方法是先讀取命令行的數據,再來waitFor();
還有一點需要注意的是獲得命令行的輸出結果先要從錯誤流中獲得,即(ErrorStream),而非從輸入流(InputStream)中獲得,很有用的經驗。

--------------------------
09博客園紀念T恤新聞:
自由軟件基金會列舉Windows 7之七宗罪網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2009/08/04/1538824.html
posted @
2009-08-04 17:45 破名超難起 閱讀(272) |
評論 (0) |
編輯 收藏
很久不在這里發帖子了,最近在家閑來無事,想到轉視頻的那東西又要做了,遂翻出以前的帖子,看到一年前豐哥讓我做一下文件上傳的進度條....額~~那就做一下吧。
東西很簡單,主要用到commons-fileupload,其中有一個progressListener的接口,該接口可以實現實時更新已上傳文件的大小,有了這個還說什么呢?
給出代碼

代碼
package lc.progress;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import lc.progress.vo.fileUploadStatus;
import org.apache.commons.fileupload.ProgressListener;
public class myProgressListener implements ProgressListener {
private HttpSession session;
public myProgressListener(HttpServletRequest req) {
session=req.getSession();
fileUploadStatus status = new fileUploadStatus();
session.setAttribute("status", status);
}
/* pBytesRead 到目前為止讀取文件的比特數
* pContentLength 文件總大小
* pItems 目前正在讀取第幾個文件
* 只要在session中實時保存文件上傳的狀態(這里我用fileUploadStatus類來封裝)
*/
public void update(long pBytesRead, long pContentLength, int pItems) {
// TODO Auto-generated method stub
fileUploadStatus status = (fileUploadStatus) session.getAttribute("status");
status.setPBytesRead(pBytesRead);
status.setPContentLength(pContentLength);
status.setPItems(pItems);
}
}
然后在上傳得servlet或action中加入這樣一段代碼,就可以把自定義的progressListener添加進去

代碼
myProgressListener getBarListener = new myProgressListener(req);
ServletFileUpload upload = new ServletFileUpload(factory);
upload.setProgressListener(getBarListener);
最后就是通過js來不斷的訪問另一個servlet來實時返回上傳狀態就可以了,限于篇幅我就不再貼代碼了,有興趣的讀者可以自己下載來看。
代碼下載(目標另存為就行了)

--------------------------
09博客園紀念T恤新聞:
網速調查報告:韓國互聯網網速全球居首網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2009/08/03/1537870.html
posted @
2009-08-03 19:46 破名超難起 閱讀(4898) |
評論 (7) |
編輯 收藏
http://family168.com/oa/springsecurity/html/index.html
posted @
2009-07-19 18:28 破名超難起 閱讀(197) |
評論 (0) |
編輯 收藏
package lc.util;
import java.math.BigDecimal;
public class MathHelper {
private static final int DEF_DIV_SCALE = 10;
private MathHelper() {
}
/**
* 提供精確的加法運算。
*
* @param v1
* 被加數
* @param v2
* 加數
* @return 兩個參數的和
*/
public static double add(double v1, double v2) {
BigDecimal b1 = new BigDecimal(Double.toString(v1));
BigDecimal b2 = new BigDecimal(Double.toString(v2));
return b1.add(b2).doubleValue();
}
/**
* 提供精確的減法運算。
*
* @param v1
* 被減數
* @param v2
* 減數
* @return 兩個參數的差
*/
public static double sub(double v1, double v2) {
BigDecimal b1 = new BigDecimal(Double.toString(v1));
BigDecimal b2 = new BigDecimal(Double.toString(v2));
return b1.subtract(b2).doubleValue();
}
/**
* 提供精確的乘法運算。
*
* @param v1
* 被乘數
* @param v2
* 乘數
* @return 兩個參數的積
*/
public static double mul(double v1, double v2) {
BigDecimal b1 = new BigDecimal(Double.toString(v1));
BigDecimal b2 = new BigDecimal(Double.toString(v2));
return b1.multiply(b2).doubleValue();
}
/**
* 提供(相對)精確的除法運算,當發生除不盡的情況時,精確到 小數點以后10位,以后的數字四舍五入。
*
* @param v1
* 被除數
* @param v2
* 除數
* @return 兩個參數的商
*/
public static double div(double v1, double v2) {
return div(v1, v2, DEF_DIV_SCALE);
}
/**
* 提供(相對)精確的除法運算。當發生除不盡的情況時,由scale參數指 定精度,以后的數字四舍五入。
*
* @param v1
* 被除數
* @param v2
* 除數
* @param scale
* 表示表示需要精確到小數點以后幾位。
* @return 兩個參數的商
*/
public static double div(double v1, double v2, int scale) {
if (scale < 0) {
throw new IllegalArgumentException(
"The scale must be a positive integer or zero");
}
BigDecimal b1 = new BigDecimal(Double.toString(v1));
BigDecimal b2 = new BigDecimal(Double.toString(v2));
return b1.divide(b2, scale, BigDecimal.ROUND_HALF_UP).doubleValue();
}
/**
* 提供精確的小數位四舍五入處理。
*
* @param v
* 需要四舍五入的數字
* @param scale
* 小數點后保留幾位
* @return 四舍五入后的結果
*/
public static double round(double v, int scale) {
if (scale < 0) {
throw new IllegalArgumentException(
"The scale must be a positive integer or zero");
}
BigDecimal b = new BigDecimal(Double.toString(v));
BigDecimal one = new BigDecimal("1");
return b.divide(one, scale, BigDecimal.ROUND_HALF_UP).doubleValue();
}
}

--------------------------
09博客園紀念T恤新聞:
王建宙臺灣布道TD:痛快,一起努力的感覺網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2009/05/20/1469589.html
posted @
2009-05-20 17:13 破名超難起 閱讀(311) |
評論 (0) |
編輯 收藏
import java.sql.*;
public class Access
{
public static void main(String args[])
{
try
{
String strurl="jdbc:odbc:driver={Microsoft Access Driver (*.mdb)};DBQ=books.mdb";
Class.forName("sun.jdbc.odbc.JdbcOdbcDriver");
Connection conn=DriverManager.getConnection(strurl) ;
Statement stmt=conn.createStatement();
ResultSet rs=stmt.executeQuery("select * from books");
if(rs.next())
{
System.out.println(rs.getString("簡介"));
}
}catch(Exception e)
{
System.out.println(e);
}
}
}

--------------------------
09博客園紀念T恤新聞:
微軟正式對Word禁售令提起上訴 稱法官不懂法網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2009/05/08/1452661.html
posted @
2009-05-08 15:08 破名超難起 閱讀(184) |
評論 (0) |
編輯 收藏
又一個新聞發布......比以前的更加人性化了,用到了Struts2、Struts2的JSON插件,前臺js用了Jquery庫,自己看吧。歡迎大家提出自己的寶貴意見。
關于顯示新聞的功能我還沒有找到一個合適的AJAX表現形式,所以還沒有完成,歡迎大家給我個建議
點擊下載源碼

--------------------------
09博客園紀念T恤新聞:
IBM 2009年北美裁員近1萬人網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2009/03/16/1413631.html
posted @
2009-03-16 17:48 破名超難起 閱讀(198) |
評論 (0) |
編輯 收藏
最近搞的一個通訊錄想要加一個Ajax的自動完成功能,看起來功能雖小,可給用戶的體驗會改進不少。在一個介紹了幾十種java的Ajax框架的網頁里面,我找到了AjaxTags這個小東西,開始了我的第一次Ajax之旅。
從AjaxTags的官方網站上面http://ajaxtags.sourceforge.net/可以下載到其最新的版本,目前是AjaxTags1.3順便下載了一個官方的小例子看了看,確實很好阿,例子中使用Ajax完成了11種功能,然而我需要的是自動完成(autocomplete)部分的代碼,所以重點研究這一部分
jsp頁面中,首先當然是要添加AjaxTags的標簽支持,需要如下語句
Code

<%@ taglib uri="http://ajaxtags.org/tags/ajax" prefix="ajax"%>
然后在頁面中加入如下的標簽
Code

<ajax:autocomplete
source="model"//從控件"model"中獲得輸入的字符
target="make"//通過Ajax自動完成控件"make"的內容
baseUrl="${contextPath}/autocomplete.view"//Ajax執行時調用的請求路徑
className="autocomplete"//css類名
indicator="indicator"
minimumCharacters="1"//Ajax執行需要輸入的最小字符數
/>
在官方的例子中使用的是Servlet來完成Ajax,而對于使用struts就不適用了,后面說這個問題。
在 autocomplete.view對應的Servlet類中需要建立xml來供頁面調取,代碼如下

Code
public class AutocompleteServlet extends BaseAjaxServlet {
public String getXmlContent(HttpServletRequest request, HttpServletResponse response)
throws Exception {
String model = request.getParameter("model");//從頁面獲取控件"model"的輸入值
CarService service = new CarService();
List<Car> list = service.getModelsByName(model);//調用CarService的方法獲得汽車的List
// Create xml schema
return new AjaxXmlBuilder().addItems(list, "model", "make",true).toString();//生成xml
}
}
其實還是很簡單的,接下來看CarService的代碼吧,重點其實只有一段,然后在下面顯示出自動提示,于是可以把剛才的那個ajax標簽修改如下

Code
public class CarService {
private static List<Car> cars = new ArrayList<Car>();
static {
cars.add(new Car("Ford", "Escape"));
cars.add(new Car("Ford", "Expedition"));
cars.add(new Car("Ford", "Explorer"));
cars.add(new Car("Ford", "Focus"));
cars.add(new Car("Ford", "Mustang"));
cars.add(new Car("Ford", "Thunderbird"));
cars.add(new Car("Honda", "Accord"));
cars.add(new Car("Honda", "Civic"));
cars.add(new Car("Honda", "Element"));
cars.add(new Car("Honda", "Ridgeline"));
cars.add(new Car("Mazda", "Mazda 3"));
cars.add(new Car("Mazda", "Mazda 6"));
cars.add(new Car("Mazda", "RX-8"));
}//其實把上面這個地方改為數據庫查詢就可以從數據庫中得到List來供頁面顯示
public CarService() {
super();
}
public List<Car> getModelsByMake(String make) {
.
}
public List<Car> getModelsByName(String name) {
..
}
public List<Car> getAllCars() {
return cars;
}
}
例子看完了,開始實際操作吧,在我的頁面中,需要通過一個名為"name"的文本框輸入要查詢的人的姓名

Code
<ajax:autocomplete
source="name"
target="name"
baseUrl="ajaxfinduser.do"
className="autocomplete"
indicator="indicator"
minimumCharacters="1"
/>
接下來是我的Action,在寫Action的時候,我以為只需要把原來Servlet繼承的BaseAjaxServlet改為BaseAjaxAction就可以了,可后來才發現,jar包中根本就沒有BaseAjaxAction這個類,無語,去官方網站上看了后才知道,在1.2更新到1.3的時候,把對Struts的支持去掉了,如果想支持Struts的話需要建立自己的BaseAjaxAction
(Struts removed, to use it create your own BaseAjaxAction.java and implement BaseAjaxXmlAction then just call xml = AjaxActionHelper.invoke(this, request, response);)
下載了一份AjaxTags的源碼來看,原來AjaxActionHelper.invoke();這個方法需要調用Action中的getXmlContent方法來完成xml的寫入,那就好說了,代碼如下:

Code
package com.txl.action;
import java.io.IOException;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.ajaxtags.servlets.BaseAjaxXmlAction;
import org.ajaxtags.xml.AjaxXmlBuilder;
import org.apache.struts.action.Action;
import org.apache.struts.action.ActionForm;
import org.apache.struts.action.ActionForward;
import org.apache.struts.action.ActionMapping;
import org.ajaxtags.servlets.AjaxActionHelper;
import org.springframework.context.ApplicationContext;
import org.springframework.web.context.support.WebApplicationContextUtils;
import com.txl.service.AjaxFindUserService;
import com.vo.User;
public class AjaxFindUserAction extends Action implements BaseAjaxXmlAction {
public ActionForward execute(ActionMapping mapping, ActionForm form,
HttpServletRequest request, HttpServletResponse response)
throws IOException, Exception {
response.setCharacterEncoding("utf-8");
response.getOutputStream().print(new String(AjaxActionHelper.invoke(this, request, response).getBytes("UTF-8"),"ISO-8859-1"));//這里需要轉換編碼,不然無法支持中文查詢
response.getOutputStream().flush();
return null;
}
public String getXmlContent(HttpServletRequest request,
HttpServletResponse response) throws Exception {
String realname = request.getParameter("name");
ApplicationContext ctx = WebApplicationContextUtils
.getRequiredWebApplicationContext(servlet.getServletContext());//用ssh時需要在這里引入spring的配置文件,不然會在調用dao的時候報空指針異常
AjaxFindUserService service = (AjaxFindUserService) ctx
.getBean("AjaxFindUserService");
List<User> list = service.getUsersByRealname(realname);
return new AjaxXmlBuilder()
.addItems(list, "realname", "realname", true).toString();
}
}
剩下工作就是在struts-config.xml中配置對應的action,在spring中配置對應的bean就ok拉

--------------------------
09博客園紀念T恤新聞:
Office 2010雙拳出擊加強反盜版網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/09/22/1295661.html
posted @
2008-09-22 09:42 破名超難起 閱讀(207) |
評論 (0) |
編輯 收藏
我一直有通宵下東西的習慣,可又不忍心讓筆記本下載完成后一直等到我睡醒,所以我經常用迅雷給我們提供的下載完成后自動關機這個功能。
然而現在迅雷運行在了虛擬機里面,這就郁悶了,即使自動關機也只能關掉一個虛擬機的進程,無奈我想到可不可以用shell程序來監測虛擬機的進程,當他關閉后就運行關機的命令,經過測試,果然成功了,具體代碼如下:

Code
#!/bin/bash
while [ `pidof -s $1` ]
do
sleep 2 #&& echo "ok"
done
date
echo "Done..shuting down in 60 seconds." && sleep 60
shutdown -h -P now
將代碼保存在shutdown.sh文件中
使用起來也很方便,用"ps ux"找到vbox運行的進程,將其對應的time command值作為參數輸入進來,例如我的是:
lichao 10659 91.7 52.0 1224284 1079860 ? Sl 21:49 11:01 /opt/VirtualBox-1.6.4/VirtualBox -comment winxp -startvm 5e04c66f-82f7-4df0-0f92-8b9113be6f2
則執行上述腳本文件:
./ shutdown.sh /opt/VirtualBox-1.6.4/VirtualBox -comment winxp -startvm 5e04c66f-82f7-4df0-0f92-8b9113be6f2
監測開始
當虛擬機關機后命令行出現如下提示:
Done..shuting down in 60 seconds.
60秒后就會關機拉,如果你還不想關機,可以Ctrl+Alt+C停止當前任務,再進行其他操作,怎么樣,挺好使的

--------------------------
09博客園紀念T恤新聞:
中國聯通:國際通訊已恢復至震前水平網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/21/1273554.html
posted @
2008-08-21 22:25 破名超難起 閱讀(517) |
評論 (0) |
編輯 收藏
查看軟件xxx安裝內容
查找軟件
查找文件屬于哪個包
- dpkg -S filename
apt-file search filename
查詢軟件xxx依賴哪些包
查詢軟件xxx被哪些包依賴
增加一個光盤源
系統升級
- sudo apt-get update
sudo apt-get upgrade
sudo apt-get dist-upgrade
清除所以刪除包的殘余配置文件
- dpkg -l |grep ^rc|awk '{print $2}' |tr ["n"] [" "]|sudo xargs dpkg -P -
編譯時缺少h文件的自動處理
- sudo auto-apt run ./configure
查看安裝軟件時下載包的臨時存放目錄
- ls /var/cache/apt/archives
備份當前系統安裝的所有包的列表
- dpkg --get-selections | grep -v deinstall > ~/somefile
從上面備份的安裝包的列表文件恢復所有包
- dpkg --set-selections < ~/somefile
sudo dselect
清理舊版本的軟件緩存
清理所有軟件緩存
刪除系統不再使用的孤立軟件
查看包在服務器上面的地址
- apt-get -qq --print-uris install ssh | cut -d' -f2
系統 查看內核
查看Ubuntu版本
查看內核加載的模塊
查看PCI設備
查看USB設備
查看網卡狀態
查看CPU信息
顯示當前硬件信息
硬盤 查看硬盤的分區
查看IDE硬盤信息
查看STAT硬盤信息
- sudo hdparm -I /dev/sda
或
sudo apt-get install blktool
sudo blktool /dev/sda id
查看硬盤剩余空間
查看目錄占用空間
優盤沒法卸載
- sync
fuser -km /media/usbdisk
內存 查看當前的內存使用情況
進程 查看當前有哪些進程
中止一個進程
- kill 進程號(就是ps -A中的第一列的數字)
或者 killall 進程名
強制中止一個進程(在上面進程中止不成功的時候使用)
- kill -9 進程號
或者 killall -9 進程名
圖形方式中止一個程序
- xkill 出現骷髏標志的鼠標,點擊需要中止的程序即可
查看當前進程的實時狀況
查看進程打開的文件
ADSL 配置 ADSL
ADSL手工撥號
激活 ADSL
- sudo /etc/ppp/pppoe_on_boot
斷開 ADSL
查看撥號日志
如何設置動態域名
- #首先去 http://www.3322.org 申請一個動態域名
#然后修改 /etc/ppp/ip-up 增加撥號時更新域名指令
sudo vim /etc/ppp/ip-up
#在最后增加如下行
w3m -no-cookie -dump 'http://username:password@members.3322.org/dyndns/update?system=dyndns&hostname=yourdns.3322.org'
網絡 根據IP查網卡地址
查看當前IP地址
- ifconfig eth0 |awk '/inet/ {split($2,x,":");print x[2]}'
查看當前外網的IP地址
- w3m -no-cookie -dump www.ip138.com|grep -o '[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}'
w3m -no-cookie -dump www.123cha.com|grep -o '[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}'
w3m -no-cookie -dump ip.loveroot.com|grep -o '[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}'
查看當前監聽80端口的程序
查看當前網卡的物理地址
- arp -a | awk '{print $4}'
ifconfig eth0 | head -1 | awk '{print $5}'
立即讓網絡支持nat
- echo 1 | sudo tee /proc/sys/net/ipv4/ip_forward
sudo iptables -t nat -I POSTROUTING -j MASQUERADE
查看路由信息
- netstat -rn
sudo route -n
手工增加刪除一條路由
- sudo route add -net 192.168.0.0 netmask 255.255.255.0 gw 172.16.0.1
sudo route del -net 192.168.0.0 netmask 255.255.255.0 gw 172.16.0.1
修改網卡MAC地址的方法
- sudo ifconfig eth0 down #關閉網卡
sudo ifconfig eth0 hw ether 00:AA:BB:CC:DD:EE #然后改地址
sudo ifconfig eth0 up #然后啟動網卡
統計當前IP連接的個數
- netstat -na|grep ESTABLISHED|awk '{print $5}'|awk -F: '{print $1}'|sort|uniq -c|sort -r -n
netstat -na|grep SYN|awk '{print $5}'|awk -F: '{print $1}'|sort|uniq -c|sort -r -n
統計當前20000個IP包中大于100個IP包的IP地址
- tcpdump -tnn -c 20000 -i eth0 | awk -F "." '{print $1"."$2"."$3"."$4}' | sort | uniq -c | sort -nr | awk ' $1 > 100 '
屏蔽IPV6
- echo "blacklist ipv6" | sudo tee /etc/modprobe.d/blacklist-ipv6
服務 添加一個服務
- sudo update-rc.d 服務名 defaults 99
刪除一個服務
- sudo update-rc.d 服務名 remove
臨時重啟一個服務
臨時關閉一個服務
臨時啟動一個服務
設置 配置默認Java使用哪個
- sudo update-alternatives --config java
修改用戶資料
給apt設置代理
- export http_proxy=http://xx.xx.xx.xx:xxx
修改系統登錄信息
中文 轉換文件名由GBK為UTF8
- sudo apt-get install convmv
convmv -r -f cp936 -t utf8 --notest --nosmart *
批量轉換src目錄下的所有文件內容由GBK到UTF8
- find src -type d -exec mkdir -p utf8/{} ;
find src -type f -exec iconv -f GBK -t UTF-8 {} -o utf8/{} ;
mv utf8/* src
rm -fr utf8
轉換文件內容由GBK到UTF8
- iconv -f gbk -t utf8 $i > newfile
轉換 mp3 標簽編碼
- sudo apt-get install python-mutagen
find . -iname “*.mp3” -execdir mid3iconv -e GBK {} ;
控制臺下顯示中文
- sudo apt-get install zhcon
使用時,輸入zhcon即可
文件 快速查找某個文件
- whereis filename
- find 目錄 -name 文件名
查看文件類型
顯示xxx文件倒數6行的內容
讓tail不停地讀地最新的內容
- tail -n 10 -f /var/log/apache2/access.log
查看文件中間的第五行(含)到第10行(含)的內容
- sed -n '5,10p' /var/log/apache2/access.log
查找包含xxx字符串的文件
查找關于xxx的命令
通過ssh傳輸文件
- scp -rp /path/filename username@remoteIP:/path #將本地文件拷貝到服務器上
scp -rp username@remoteIP:/path/filename /path #將遠程文件從服務器下載到本地
查看某個文件被哪些應用程序讀寫
把所有文件的后輟由rm改為rmvb
把所有文件名中的大寫改為小寫
刪除特殊文件名的文件,如文件名:--help.txt
- rm -- --help.txt 或者 rm ./--help.txt
查看當前目錄的子目錄
將當前目錄下最近30天訪問過的文件移動到上級back目錄
- find . -type f -atime -30 -exec mv {} ../back ;
將當前目錄下最近2小時到8小時之內的文件顯示出來
- find . -mmin +120 -mmin -480 -exec more {} ;
刪除修改時間在30天之前的所有文件
- find . -type f -mtime +30 -mtime -3600 -exec rm {} ;
查找guest用戶的以avi或者rm結尾的文件并刪除掉
- find . -name '*.avi' -o -name '*.rm' -user 'guest' -exec rm {} ;
查找的不以java和xml結尾,并7天沒有使用的文件刪除掉
- find . ! -name *.java ! -name ‘*.xml’ -atime +7 -exec rm {} ;
統計當前文件個數
統計當前目錄個數
- ls -l /usr/bin|grep ^d|wc -l
顯示當前目錄下2006-01-01的文件名
- ls -l |grep 2006-01-01 |awk '{print $8}'
壓縮 解壓縮 xxx.tar.gz
解壓縮 xxx.tar.bz2
壓縮aaa bbb目錄為xxx.tar.gz
- tar -zcvf xxx.tar.gz aaa bbb
壓縮aaa bbb目錄為xxx.tar.bz2
- tar -jcvf xxx.tar.bz2 aaa bbb
Nautilus 顯示隱藏文件
顯示地址欄
特殊 URI 地址
- * computer:/// - 全部掛載的設備和網絡
* network:/// - 瀏覽可用的網絡
* burn:/// - 一個刻錄 CDs/DVDs 的數據虛擬目錄
* smb:/// - 可用的 windows/samba 網絡資源
* x-nautilus-desktop:/// - 桌面項目和圖標
* file:/// - 本地文件
* trash:/// - 本地回收站目錄
* ftp:// - FTP 文件夾
* ssh:// - SSH 文件夾
* fonts:/// - 字體文件夾,可將字體文件拖到此處以完成安裝
* themes:/// - 系統主題文件夾
查看已安裝字體
- 在nautilus的地址欄里輸入”fonts:///“,就可以查看本機所有的fonts
程序 詳細顯示程序的運行信息
- strace -f -F -o outfile <cmd>
日期和時間 設置日期
設置時間
將時間寫入CMOS
讀取CMOS時間
從服務器上同步時間
- sudo ntpdate time.nist.gov
sudo ntpdate time.windows.com
控制臺 不同控制臺間切換
- Ctrl + ALT + ←
Ctrl + ALT + →
指定控制臺切換
控制臺下滾屏
控制臺抓圖
數據庫 mysql的數據庫存放在地方
從mysql中導出和導入數據
- mysqldump 數據庫名 > 文件名 #導出數據庫
mysqladmin create 數據庫名 #建立數據庫
mysql 數據庫名 < 文件名 #導入數據庫
忘了mysql的root口令怎么辦
- sudo /etc/init.d/mysql stop
sudo mysqld_safe --skip-grant-tables &
sudo mysqladmin -u user password 'newpassword''
sudo mysqladmin flush-privileges
修改mysql的root口令
- sudo mysqladmin -uroot -p password '你的新密碼'
其它 下載網站文檔
- wget -r -p -np -k http://www.21cn.com
· -r:在本機建立服務器端目錄結構;
· -p: 下載顯示HTML文件的所有圖片;
· -np:只下載目標站點指定目錄及其子目錄的內容;
· -k: 轉換非相對鏈接為相對鏈接。
如何刪除Totem電影播放機的播放歷史記錄
如何更換gnome程序的快捷鍵
- 點擊菜單,鼠標停留在某條菜單上,鍵盤輸入任意你所需要的鍵,可以是組合鍵,會立即生效;
如果要清除該快捷鍵,請使用backspace
vim 如何顯示彩色字符
- sudo cp /usr/share/vim/vimcurrent/vimrc_example.vim /usr/share/vim/vimrc
如何在命令行刪除在會話設置的啟動程序
- cd ~/.config/autostart
rm 需要刪除啟動程序
如何提高wine的反應速度 sudo sed -ie '/GBK/,/^}/d' /usr/share/X11/locale/zh_CN.UTF-8/XLC_LOCALE

--------------------------
09博客園紀念T恤新聞:
分析稱RSS閱讀器是Web 1.0工具 已逐漸被拋棄網站導航:
博客園首頁 個人主頁 新聞 社區 博問 閃存 找找看文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/20/1272372.html
posted @
2008-08-20 16:27 破名超難起 閱讀(129) |
評論 (0) |
編輯 收藏
Hibernate獲取數據的方式有不同的幾種,其與緩存結合使用的效果也不盡相同,而Hibernate中具體怎么使用緩存其實是我們很關心的一個問題,直接涉及到性能方面。
緩 存在Hibernate中主要有三個方面:一級緩存、二級緩存和查詢緩存;一級緩存在Hibernate中對應的即為session范圍的緩存,也就是當 session關閉時緩存即被清除,一級緩存在Hibernate中是不可配置的部分;二級緩存在Hibernate中對應的即為 SessionFactory范圍的緩存,通常來講SessionFactory的生命周期和應用的生命周期相同,所以可以看成是進程緩存或集群緩存,二 級緩存在Hibernate中是可以配置的,可以通過class-cache配置類粒度級別的緩存(class-cache在class中數據發生任何變 化的情況下自動更新),同時也可通過collection-cache配置集合粒度級別的緩存(collection-cache僅在 collection中增加了元素或者刪除了元素的情況下才自動更新,也就是當collection中元素發生值的變化的情況下它是不會自動更新的),緩 存自然會帶來并發的訪問問題,這個時候相應的就要根據應用來設置緩存所采用的事務隔離級別,和數據庫的事務隔離級別概念基本一樣,沒什么多介紹的, ^_^;查詢緩存在Hibernate同樣是可配置的,默認是關閉的,可以通過設置cache.use_ query_cache為true來打開查詢緩 存。根據緩存的通常實現策略,我們可以來理解Hibernate的這三種緩存,緩存的實現通過是通過key/value的Map方式來實現,在 Hibernate的一級、二級和查詢緩存也同樣如此,一級、二級緩存使用的key均為po的主鍵ID,value即為po實例對象,查詢緩存使用的則為 查詢的條件、查詢的參數、查詢的頁數,value有兩種情況,如果采用的是select po.property這樣的方式那么value為整個結果集,如采用的是from這樣的方式那么value為獲取的結果集中各po對象的主鍵ID,這樣 的作用很明顯,節省內存,^_^
簡單介紹完Hibernate的緩存后,再結合Hibernate的獲取數據方式來說明緩存的具體使用方式,在Hibernate中獲取數據常用的方式主要有四種:Session.load、Session.get、Query.list、Query.iterator。
1、Session.load
在執行session.load時,Hibernate首先從當前session的一級緩存中獲取id對應的值,在獲取不到的情況下,將根據該對象是否配 置了二級緩存來做相應的處理,如配置了二級緩存,則從二級緩存中獲取id對應的值,如仍然獲取不到則還需要根據是否配置了延遲加載來決定如何執行,如未配 置延遲加載則從數據庫中直接獲取,在從數據庫獲取到數據的情況下,Hibernate會相應的填充一級緩存和二級緩存,如配置了延遲加載則直接返回一個代 理類,只有在觸發代理類的調用時才進行數據庫查詢的操作。
在這樣的情況下我們就可以看到,在session一直打開的情況下,要注意在適當的時候對一級緩存進行刷新操作,通常是在該對象具有單向關聯維護的時候, 在Hibernate中可以使用象session.clear、session.evict的方式來強制刷新一級緩存。
二級緩存則在數據發生任何變化(新增、更新、刪除)的情況下都會自動的被更新。
2、Session.get
在執行Session.get時,和Session.load不同的就是在當從緩存中獲取不到時,直接從數據庫中獲取id對應的值。
3、Query.list
在執行Query.list時,Hibernate的做法是首先檢查是否配置了查詢緩存,如配置了則從查詢緩存中查找key為查詢語句+查詢參數+分頁條 件的值,如獲取不到則從數據庫中進行獲取,從數據庫獲取到后Hibernate將會相應的填充一級、二級和查詢緩存,如獲取到的為直接的結果集,則直接返 回,如獲取到的為一堆id的值,則再根據id獲取相應的值(Session.load),最后形成結果集返回,可以看到,在這樣的情況下,list也是有 可能造成N次的查詢的。
查詢緩存在數據發生任何變化的情況下都會被自動的清空。
4、Query.iterator
在執行Query.iterator時,和Query.list的不同的在于從數據庫獲取的處理上,Query.iterator向數據庫發起的是 select id from這樣的語句,也就是它是先獲取符合查詢條件的id,之后在進行iterator.next調用時才再次發起session.load的調用獲取實 際的數據。
可見,在擁有二級緩存并且查詢參數多變的情況下,Query.iterator會比Query.list更為高效。
這 四種獲取數據的方式都各有適用的場合,要根據實際情況做相應的決定,^_^,最好的方式無疑就是打開show_sql選項看看執行的情況來做分析,系統結 構上只用保證這種調整是容易實現的就好了,在cache這個方面的調整自然是非常的容易,只需要調整配置文件里的設置,而查詢的方式則可對外部進行屏蔽, 這樣要根據實際情況調整也非常容易。
推薦三篇關于Hibernate緩存機制介紹的文章:
http://gocom.primeton.com/blog/index.php?op=ViewArticle&articleId=467&blogId=37&src=jdon&srcforum=62
http://club.gamvan.com/club/clubPage.jsp?ccStyle=0&tID=10456&ccID=37
http://www.devx.com/dbzone/Article/29685/1954?pf=true文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/16/1269160.html
posted @
2008-08-16 11:03 破名超難起 閱讀(83) |
評論 (0) |
編輯 收藏
雖然VirtualBox是個很好用的虛擬機,但是安裝完成之后虛擬機仍然無法使用usb設備,沒有聲音還是需要作進一步配置的,下面是具體配置方法:
1、配置VirtualBox使其識別usb設備
如果不配置VirtualBox的話,每次虛擬機啟動時他都會報錯的,大體意思是當前虛擬機沒有安裝使用usb的服務
解決方法如下:
增加用戶組usbfs
$ sudo groupadd usbfs
2. 查看usbfs用戶組的gid
$ cat /etc/group | grep usbfs
usbfs:x:1002:
把當前用戶增加到usbfs組
$ sudo gedit /etc/group
把
usbfs:x:1001:
修改為
usbfs:x:1001:lichao
4. 為USB設備重新設置權限編輯/etc/fstab文件,添加下面兩行,注意你的gid可能不是1001
$ sudo gedit /etc/fstab
在末尾加上
# 1001 is the USB group IDI
none /proc/bus/usb usbfs devgid=1001,devmode=666 0 0
重新啟動后,應該就可以在客戶機中使用USB設備了。
方法:插入一個USB設備后,如U盤,右鍵點擊虛擬機里右下腳的USB圖標,選擇已經識別的U盤,就可以正常使用了。
注意:(1)在客戶機里使用USB設備前要先在主機里卸載。 (2)完成后重啟系統
2、使虛擬機支持聲音
使用了一段時間后,我突然發現虛擬機居然沒有聲音,而在操作系統中是有聲卡的,而且右下角有調節聲音的圖標,應該有聲音阿
經過仔細查找,我發現在虛擬機聲音設置中,雖然控制芯片被設置為了“ICH AC97”,但是聲卡類型卻被設置為了Null Audio Driver,只要將聲卡類型設置為OSS Audio Driver就可以了
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/15/1268427.html
posted @
2008-08-15 10:01 破名超難起 閱讀(358) |
評論 (0) |
編輯 收藏
今天很郁悶阿,昨天剛裝好的系統,今天就因為一次莫名的操作......重裝了
裝系統倒不是很費時間,麻煩的是等待系統升級,很郁悶阿,將近一小時,什么也作不了,所以這次重裝之后,我決定找到一種一勞永逸的方法,備份升級文件,找了找,還真有,具體過程如下:
備份:

Code
tar cizvf backup.tar.gz /var/cache/apt/archives --exclude=/var/cache/apt/archives/partial/* --exclude=/var/cache/apt/archives/loc
備份成功,在你所對應的/home/user文件夾中可以找到這個名為backup.tar.gz的文件。
清除:

Code
sudo apt-get clean
rm -rf ~/.thumbnails/fail/gnome-thumbnail-factory/*
還原:

Code
sudo apt-get update && sudo tar xzvf backup.tar.gz -C /
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/13/1267439.html
posted @
2008-08-13 23:32 破名超難起 閱讀(107) |
評論 (0) |
編輯 收藏
由于linux下沒有像windows的ALL IN ONE安裝包,而我又不想用eclipse+myeclipse的方式來安裝,就在這時,我發現在myeclipse的官方網站上有一個安裝包叫pulse
上網查了一下,這個安裝包本身沒有任何ide安張程序,他可以讓你來選擇哪些工具你需要下載安裝,哪些你不需要,似乎很好用哦,下載下來試試。
裝上之后,有很多ide提供下載安裝,就選擇我最熟悉的MyEclipse6.5吧,下載速度很快,不到10分鐘就裝好了,打開一看,和windows的界面一樣,爽??!以后就用他了
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/12/1266299.html
posted @
2008-08-12 20:09 破名超難起 閱讀(188) |
評論 (0) |
編輯 收藏
早就想把微軟的東西仍進虛擬機,只是一直沒下決心,這次實踐了一回,呵呵,還真好使
我用的ubuntu 7.10+sun Virtual Box
去sun的官方網站下載吧,
VirtualBox-1.6.4-Linux_x86.run
這個是最新的,下載后照常安裝,其間碰到幾個依賴包的安裝,在“新立得”里面找一下就好了....
5分鐘后,虛擬機裝好,開始裝windows拉,用了許久的Vista,回顧一下經典的XP,哈哈,蠻不錯的,感覺比純運行xp還要快,一下是截圖 :
這下爽拉,在linux下玩魔獸完全沒難度拉
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/12/1266297.html
posted @
2008-08-12 20:00 破名超難起 閱讀(203) |
評論 (0) |
編輯 收藏
首先去sun公司的網站下載jdk安裝文件 jdk-x.bin不要下載jdk-x.rpm.bin,不知為什么裝不上,
接下來在終端中執行
./jdk-x.bin
看過協議后鍵入"yes"同意許可
安裝完成后需要修改環境變量
在終端中輸入:
gksu gedit /etc/profile
在文件末尾加上如下幾句話即可:
#set java environment
JAVA_HOME=/home/lichao/Program_File/jdk
export JRE_HOME=/home/lichao/Program_File/jdk/jre
export CLASSPATH=.:$JAVA_HOME/lib:$JRE_HOME/lib:$CLASSPATH
export PATH=$JAVA_HOME/bin:$JRE_HOME/bin:$PATH
javac一下試試看,應該沒問題了
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/12/1266215.html
posted @
2008-08-12 17:48 破名超難起 閱讀(85) |
評論 (0) |
編輯 收藏
在終端中鍵入
sudo apt-get install mysql-server
ok,等一會吧,設置一下root的密碼,
安裝成功,簡單吧,哈哈!!
安裝完后最要命的是.....不知道mysql安裝到那里了,網上找到了答案,若干個目錄阿
ubuntu下mysql安裝布局:
/usr/bin 客戶端程序和mysql_install_db
/db 數據庫和日志文件(自定義的)
/var/run mysqld 服務器
/etc/mysql mysql 配置文件my.cnf
/usr/share/mysql 字符集,基準程序和錯誤消息
/etc/init.d/mysql 啟動mysql服務器
成功后在“系統 -> 系統管理 -> 服務”就可以看到mysql的服務了。
下面要配置了!
mysql默認只能在本機登錄,這時就需要設置一下了
在終端中輸入
mysql -u root -p
輸入正確密碼后執行如下sql語句:
- mysql -u root -pvmwaremysql>use mysql;
- mysql>update user set host = '%' where user = 'root';
- mysql>select host, user from user;
可以了,試試看吧!
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/12/1266213.html
posted @
2008-08-12 17:40 破名超難起 閱讀(191) |
評論 (0) |
編輯 收藏
這里的方法同樣適用于Debian下的Scim輸入法的安裝。
根據SCIM的文檔說明,裝SCIM最好用UTF-8的locale。(你也可以使用GB2312以及GBK的locale,使用scim都一樣的)
對于一個新裝好的沒有安裝SCIM的Ubuntu系統(新的Ubuntu安裝器選擇中文安裝后,一邊都已經裝好了scim),只要
apt-get install scim(當然前提是你要設置好源,具體見http://wiki.ubuntu.org.cn/%E5%BF%AB%E9%80%9F%E8%AE%BE%E7%BD%AE%E6%8C%87%E5%8D%97, 有些高校自己內部也有源的,好好找找)
就已經安裝好SCIM了!當然還不能用。因為SCIM只是一個輸入法平臺,還要在上面安裝輸入法(或者碼表)。例如,
apt-get install scim-chinese
就可以使用智能拼音輸入法了!當然在使用之前還要先配置一下,配置方法很簡單,在/etc/X11/Xsession.d/里新建一個名叫95xinput的文件,文件內容如下
/usr/bin/scim -d
XMODIFIERS="@im=SCIM"
export XMODIFIERS
export GTK_IM_MODULE=scim
保存文件,確認內容無誤后,退出X(建議退出X后運行exit命令重新login一次),再進入X的時候就可以用Ctrl+Space調出SCIM了!
就我的試驗情況,這樣已經可以在qterm里使用SCIM了。
如果你需要其它中文輸入法,可以這樣
apt-get install scim-tables-zh
這包括了簡體中文的五筆、二筆、廣東拼音、自然碼,和繁體中文的行列、 }頡五代、大易、注音等輸入法了。
我現在使用五筆輸入法,能輸入簡繁漢字,詞匯也豐富,很好用。
但是現在還不能在基于GTK的軟件中調出SCIM,例如我就不能在leafpad里使用SCIM。解決辦法很簡單,只要安裝scim-gtk2-immodule就可以了。
apt-get install scim-gtk2-immodule
安裝后無須重啟X,只要重新打開基于GTK的軟件就可以了,比如我新開一個leafpad,馬上就可以使用SCIM了!
這個命令會根據依賴關系自動安裝 scim-server-socket, scim-frontend-socket, scim-config-socket,如果沒有安裝scim,也會自動安裝。
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/12/1265876.html
posted @
2008-08-12 13:12 破名超難起 閱讀(81) |
評論 (0) |
編輯 收藏

Code
import java.util.*;
import java.io.*;
import java.security.*;
public class MD5 {
public static String hex(byte[] array) {
StringBuffer sb = new StringBuffer();
for (int i = 0; i < array.length; ++i) {
sb.append(Integer.toHexString((array[i] & 0xFF) | 0x100).toUpperCase().substring(1,3));
}
return sb.toString();
}
public static String md5 (String message) {
try {
MessageDigest md = MessageDigest.getInstance("MD5");\\想用SHA加密的話就把MD5換成SHA吧
return hex (md.digest(message.getBytes("CP1252")));
} catch (NoSuchAlgorithmException e) {
} catch (UnsupportedEncodingException e) {
}
return null;
}
public static void main(String[] args) {
System.out.println (md5 ("LiChao"));
}
}
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/04/1260376.html
posted @
2008-08-04 20:25 破名超難起 閱讀(111) |
評論 (0) |
編輯 收藏
經過若干天斷斷續續地研究,終于做出了第一個spring security的實例,真是艱難啊,配置太復雜了,若干個Bean之間存在著這樣或那樣的關系......
下面給出我的小例子,主要是配置文件拉~~別的東西自己看源碼吧!

Code
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd">
<!-- 過濾器鏈配置,其中filterInvocationDefinitionSource屬性為配置過濾器的種類與先后順序,注意,順序不能配置錯誤哦 -->
<bean id="filterChainProxy"
class="org.springframework.security.util.FilterChainProxy">
<property name="filterInvocationDefinitionSource">
<value><![CDATA[
CONVERT_URL_TO_LOWERCASE_BEFORE_COMPARISON
PATTERN_TYPE_APACHE_ANT
/**=httpSessionIntegrationFilter,authenticationProcessingFilter,exceptionTranslationFilter,filterSecurityInterceptor]]>
</value>
</property>
</bean>
<!-- 看看你是否已經登錄了,如果登錄了就略過下面的過濾器了,直接訪問資源 -->
<bean id="httpSessionIntegrationFilter"
class="org.springframework.security.context.HttpSessionContextIntegrationFilter" />
<!-- 安全驗證入口 -->
<bean id="authenticationEntryPoint"
class="org.springframework.security.ui.webapp.AuthenticationProcessingFilterEntryPoint">
<property name="loginFormUrl" value="/index.jsp" /><!--默認登錄頁面-->
<property name="forceHttps" value="true" /><!--使登錄頁面通過HTTPS安全地進行顯示-->
</bean>
<!-- 身份驗證過濾器,就是驗證身份用的嘛 -->
<bean id="authenticationProcessingFilter"
class="org.springframework.security.ui.webapp.AuthenticationProcessingFilter">
<!-- 驗證連接名稱,對應表單的action -->
<property name="filterProcessesUrl"
value="/j_spring_security_check" />
<!-- 驗證失敗后去哪 -->
<property name="authenticationFailureUrl"
value="/index.jsp?error=1" />
<!-- 驗證成功后去哪 -->
<property name="defaultTargetUrl"
value="/security/security.jsp" />
<!--依靠一個身份驗證管理器來驗證身份 其實這個才是干活的BEAN-->
<property name="authenticationManager"
ref="authenticationManager" />
</bean>
<!-- 用于處理登錄失敗異常和權限不足異常 -->
<bean id="exceptionTranslationFilter"
class="org.springframework.security.ui.ExceptionTranslationFilter">
<!--配置出現exception時跳轉到登錄頁-->
<property name="authenticationEntryPoint"
ref="authenticationEntryPoint" />
<!--配置403(權限不足)錯誤后跳轉的頁面-->
<property name="accessDeniedHandler" ref="accessDeniedHandler" />
</bean>
<!-- 配置權限不足時跳轉到的頁面 -->
<bean id="accessDeniedHandler"
class="org.springframework.security.ui.AccessDeniedHandlerImpl">
<property name="errorPage" value="/error.jsp" />
</bean>
<!-- 安全攔截器,下面看看它是干嘛的 -->
<bean id="filterSecurityInterceptor"
class="org.springframework.security.intercept.web.FilterSecurityInterceptor">
<!-- 驗證管理者 -->
<property name="authenticationManager"
ref="authenticationManager" />
<!-- 權限決定管理者,他手下的一幫人投票決定登錄者是否有權訪問該資源 -->
<property name="accessDecisionManager"
ref="accessDecisionManager" />
<!--受保護資源-->
<property name="objectDefinitionSource">
<!-- 下面表示/security/security.jsp需要ROLE_ADMIN權限才能訪問 -->
<value><![CDATA[
CONVERT_URL_TO_LOWERCASE_BEFORE_COMPARISON
PATTERN_TYPE_APACHE_ANT
/security/security.jsp=ROLE_ADMIN]]>
</value>
</property>
</bean>
<!-- 驗證管理者,他管理DAO驗證提供者來驗證 -->
<bean id="authenticationManager"
class="org.springframework.security.providers.ProviderManager">
<property name="providers">
<list>
<!-- DAO驗證提供者,SPRING SECURITY支持各種驗證,這里可以添加相應配置 -->
<ref local="daoAuthenticationProvider" />
</list>
</property>
</bean>
<!-- -->
<bean id="accessDecisionManager"
class="org.springframework.security.vote.AffirmativeBased">
<!-- 如果所有投票者都棄權則不讓訪問 -->
<property name="allowIfAllAbstainDecisions">
<value>false</value>
</property>
<!-- 參加投票的BEAN -->
<property name="decisionVoters">
<list>
<bean class="org.springframework.security.vote.RoleVoter">
<!-- 權限的前綴 -->
<property name="rolePrefix" value="ROLE_" />
</bean>
<bean class="org.springframework.security.vote.AuthenticatedVoter" />
</list>
</property>
</bean>
<!-- DAO驗證提供者依靠userDetailsService獲得一個userDetails實例,進而驗證權限 -->
<bean id="daoAuthenticationProvider"
class="org.springframework.security.providers.dao.DaoAuthenticationProvider">
<!-- jdbcDaoImpl實現了userDetailsService接口 -->
<property name="userDetailsService">
<ref local="jdbcDaoImpl" />
</property>
</bean>
<bean id="jdbcDaoImpl"
class="org.springframework.security.userdetails.jdbc.JdbcDaoImpl">
<!-- 根據用戶名獲得用戶名、密碼、用戶是否啟用等信息 -->
<property name="usersByUsernameQuery">
<value>
select username,password,enabled from user where
username=?
</value>
</property>
<!-- 通過用戶名獲取用戶權限 -->
<property name="authoritiesByUsernameQuery">
<value>
select username,authority from authentication where
username=?
</value>
</property>
<!-- DataSource,不用我說了吧 -->
<property name="dataSource">
<ref local="dataSource" />
</property>
</bean>
<bean id="dataSource"
class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName"
value="org.gjt.mm.mysql.Driver">
</property>
<property name="url" value="jdbc:mysql://localhost:3306/user">
</property>
<property name="username" value="root"></property>
<property name="password" value="hicc"></property>
</bean>
</beans>
這個是最簡單的一個例子,配了141行,呼~~繼續研究其深入功能,離成功越來越近了
ps.傳說spring security2.0有了超級簡單的配置方法,還沒有學到手,努力ing
源碼下載(需要自己添加spring和mysql的jar包)
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/04/1259523.html
posted @
2008-08-04 00:46 破名超難起 閱讀(665) |
評論 (0) |
編輯 收藏
你覺得自己是一個Java專家嗎?是否肯定自己已經全面掌握了Java的異常處理機制?在下面這段代碼中,你能夠迅速找出異常處理的六個問題嗎?

Code
OutputStreamWriter out = 
java.sql.Connection conn = 
try { // ⑸
Statement stat = conn.createStatement();
ResultSet rs = stat.executeQuery(
"select uid, name from user");
while (rs.next())
{
out.println("ID:" + rs.getString("uid") // ⑹
",姓名:" + rs.getString("name"));
}
conn.close(); // ⑶
out.close();
}
catch(Exception ex) // ⑵
{
ex.printStackTrace(); // ⑴,⑷
}
作為一個Java程序員,你至少應該能夠找出兩個問題。但是,如果你不能找出全部六個問題,請繼續閱讀本文。
本文討論的不是Java異常處理的一般性原則,因為這些原則已經被大多數人熟知。我們要做的是分析各種可稱為“反例”(anti-pattern)的違背優秀編碼規范的常見壞習慣,幫助讀者熟悉這些典型的反面例子,從而能夠在實際工作中敏銳地察覺和避免這些問題。
反例之一:丟棄異常
代碼:15行-18行。
這段代碼捕獲了異常卻不作任何處理,可以算得上Java編程中的殺手。從問題出現的頻繁程度和禍害程度來看,它也許可以和C/C++程序的一個惡名遠播的問題相提并論——不檢查緩沖區是否已滿。如果你看到了這種丟棄(而不是拋出)異常的情況,可以百分之九十九地肯定代碼存在問題(在極少數情況下,這段代碼有存在的理由,但最好加上完整的注釋,以免引起別人誤解)。
這段代碼的錯誤在于,異常(幾乎)總是意味著某些事情不對勁了,或者說至少發生了某些不尋常的事情,我們不應該對程序發出的求救信號保持沉默和無動于衷。調用一下printStackTrace算不上“處理異常”。不錯,調用printStackTrace對調試程序有幫助,但程序調試階段結束之后,printStackTrace就不應再在異常處理模塊中擔負主要責任了。
丟棄異常的情形非常普遍。打開JDK的ThreadDeath類的文檔,可以看到下面這段說明:“特別地,雖然出現ThreadDeath是一種‘正常的情形’,但ThreadDeath類是Error而不是Exception的子類,因為許多應用會捕獲所有的Exception然后丟棄它不再理睬。”這段話的意思是,雖然ThreadDeath代表的是一種普通的問題,但鑒于許多應用會試圖捕獲所有異常然后不予以適當的處理,所以JDK把ThreadDeath定義成了Error的子類,因為Error類代表的是一般的應用不應該去捕獲的嚴重問題??梢?,丟棄異常這一壞習慣是如此常見,它甚至已經影響到了Java本身的設計。
那么,應該怎樣改正呢?主要有四個選擇:
1、處理異常。針對該異常采取一些行動,例如修正問題、提醒某個人或進行其他一些處理,要根據具體的情形確定應該采取的動作。再次說明,調用printStackTrace算不上已經“處理好了異常”。
2、重新拋出異常。處理異常的代碼在分析異常之后,認為自己不能處理它,重新拋出異常也不失為一種選擇。
3、把該異常轉換成另一種異常。大多數情況下,這是指把一個低級的異常轉換成應用級的異常(其含義更容易被用戶了解的異常)。
4、不要捕獲異常。
結論一:既然捕獲了異常,就要對它進行適當的處理。不要捕獲異常之后又把它丟棄,不予理睬。
反例之二:不指定具體的異常
代碼:15行。
許多時候人們會被這樣一種“美妙的”想法吸引:用一個catch語句捕獲所有的異常。最常見的情形就是使用catch(Exception ex)語句。但實際上,在絕大多數情況下,這種做法不值得提倡。為什么呢?
要理解其原因,我們必須回顧一下catch語句的用途。catch語句表示我們預期會出現某種異常,而且希望能夠處理該異常。異常類的作用就是告訴Java編譯器我們想要處理的是哪一種異常。由于絕大多數異常都直接或間接從java.lang.Exception派生,catch(Exception ex)就相當于說我們想要處理幾乎所有的異常。
再來看看前面的代碼例子。我們真正想要捕獲的異常是什么呢?最明顯的一個是SQLException,這是JDBC操作中常見的異常。另一個可能的異常是IOException,因為它要操作OutputStreamWriter。顯然,在同一個catch塊中處理這兩種截然不同的異常是不合適的。如果用兩個catch塊分別捕獲SQLException和IOException就要好多了。這就是說,catch語句應當盡量指定具體的異常類型,而不應該指定涵蓋范圍太廣的Exception類。
另一方面,除了這兩個特定的異常,還有其他許多異常也可能出現。例如,如果由于某種原因,executeQuery返回了null,該怎么辦?答案是讓它們繼續拋出,即不必捕獲也不必處理。實際上,我們不能也不應該去捕獲可能出現的所有異常,程序的其他地方還有捕獲異常的機會——直至最后由JVM處理。
結論二:在catch語句中盡可能指定具體的異常類型,必要時使用多個catch。不要試圖處理所有可能出現的異常。
反例之三:占用資源不釋放
代碼:3行-14行。
異常改變了程序正常的執行流程。這個道理雖然簡單,卻常常被人們忽視。如果程序用到了文件、Socket、JDBC連接之類的資源,即使遇到了異常,也要正確釋放占用的資源。為此,Java提供了一個簡化這類操作的關鍵詞finally。
finally是樣好東西:不管是否出現了異常,Finally保證在try/catch/finally塊結束之前,執行清理任務的代碼總是有機會執行。遺憾的是有些人卻不習慣使用finally。
當然,編寫finally塊應當多加小心,特別是要注意在finally塊之內拋出的異?!@是執行清理任務的最后機會,盡量不要再有難以處理的錯誤。
結論三:保證所有資源都被正確釋放。充分運用finally關鍵詞。
反例之四:不說明異常的詳細信息
代碼:3行-18行。
仔細觀察這段代碼:如果循環內部出現了異常,會發生什么事情?我們可以得到足夠的信息判斷循環內部出錯的原因嗎?不能。我們只能知道當前正在處理的類發生了某種錯誤,但卻不能獲得任何信息判斷導致當前錯誤的原因。
printStackTrace的堆棧跟蹤功能顯示出程序運行到當前類的執行流程,但只提供了一些最基本的信息,未能說明實際導致錯誤的原因,同時也不易解讀。
因此,在出現異常時,最好能夠提供一些文字信息,例如當前正在執行的類、方法和其他狀態信息,包括以一種更適合閱讀的方式整理和組織printStackTrace提供的信息。
結論四:在異常處理模塊中提供適量的錯誤原因信息,組織錯誤信息使其易于理解和閱讀。
反例之五:過于龐大的try塊
代碼:3行-14行。
經常可以看到有人把大量的代碼放入單個try塊,實際上這不是好習慣。這種現象之所以常見,原因就在于有些人圖省事,不愿花時間分析一大塊代碼中哪幾行代碼會拋出異常、異常的具體類型是什么。把大量的語句裝入單個巨大的try塊就象是出門旅游時把所有日常用品塞入一個大箱子,雖然東西是帶上了,但要找出來可不容易。
一些新手常常把大量的代碼放入單個try塊,然后再在catch語句中聲明Exception,而不是分離各個可能出現異常的段落并分別捕獲其異常。這種做法為分析程序拋出異常的原因帶來了困難,因為一大段代碼中有太多的地方可能拋出Exception。
結論五:盡量減小try塊的體積。
反例之六:輸出數據不完整
代碼:7行-11行。
不完整的數據是Java程序的隱形殺手。仔細觀察這段代碼,考慮一下如果循環的中間拋出了異常,會發生什么事情。循環的執行當然是要被打斷的,其次,catch塊會執行——就這些,再也沒有其他動作了。已經輸出的數據怎么辦?使用這些數據的人或設備將收到一份不完整的(因而也是錯誤的)數據,卻得不到任何有關這份數據是否完整的提示。對于有些系統來說,數據不完整可能比系統停止運行帶來更大的損失。
較為理想的處置辦法是向輸出設備寫一些信息,聲明數據的不完整性;另一種可能有效的辦法是,先緩沖要輸出的數據,準備好全部數據之后再一次性輸出。
結論六:全面考慮可能出現的異常以及這些異常對執行流程的影響。
改寫后的代碼
根據上面的討論,下面給出改寫后的代碼。也許有人會說它稍微有點啰嗦,但是它有了比較完備的異常處理機制。

Code
OutputStreamWriter out = 
java.sql.Connection conn = 
try {
Statement stat = conn.createStatement();
ResultSet rs = stat.executeQuery("select uid, name from user");
while (rs.next())
{
out.println("ID:" + rs.getString("uid") + ",姓名: " + rs.getString("name"));
}
}
catch(SQLException sqlex)
{
out.println("警告:數據不完整");
throw new ApplicationException("讀取數據時出現SQL錯誤", sqlex);
}
catch(IOException ioex)
{
throw new ApplicationException("寫入數據時出現IO錯誤", ioex);
}
finally
{
if (conn != null) {
try {
conn.close();
}
catch(SQLException sqlex2)
{
System.err(this.getClass().getName() + ".mymethod - 不能關閉數據庫連接: " + sqlex2.toString());
}
}
if (out != null) {
try {
out.close();
}
catch(IOException ioex2)
{
System.err(this.getClass().getName() + ".mymethod - 不能關閉輸出文件" + ioex2.toString());
}
}
}
本文的結論不是放之四海皆準的教條,有時常識和經驗才是最好的老師。如果你對自己的做法沒有百分之百的信心,務必加上詳細、全面的注釋。
一方面,不要笑話這些錯誤,不妨問問你自己是否真地徹底擺脫了這些壞習慣。即使最有經驗的程序員偶爾也會誤入歧途,原因很簡單,因為它們確確實實帶來了“方便”。所有這些反例都可以看作Java編程世界的惡魔,它們美麗動人,無孔不入,時刻誘惑著你。也許有人會認為這些都屬于雞皮蒜毛的小事,不足掛齒,但請記?。何鹨詯盒《鵀橹?,勿以善小而不為。
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/01/1258247.html
posted @
2008-08-01 15:57 破名超難起 閱讀(79) |
評論 (0) |
編輯 收藏
請求有效性處理,使用令牌可以有效的防止重復提交。
protected String generateToken(HttpServletRequest request) 創建一個令牌.
protected boolean isTokenValid(HttpServletRequest request) 檢查令牌是否有效
protected boolean isTokenValid(HttpServletRequest request,Boolean reset) 檢查令牌是否有效,并且重置令牌(如果reset 是true)
protected void resetToken(HttpServletRequest request) 重置令牌
protected void saveToken(HttpServletRequest request) 添加令牌
基本原理:
服務器端在處理到達的請求之前,會將請求中包含的令牌值與保存在當前用戶會話中的令牌值進行比較,
看是否匹配。在處理完該請求后,且在答復發送給客戶端之前,將會產生一個新的令牌,該令牌除傳給
客戶端以外,也會將用戶會話中保存的舊的令牌進行替換。這樣如果用戶回退到剛才的提交頁面并再次
提交的話,客戶端傳過來的令牌就和服務器端的令牌不一致,從而有效地防止了重復提交的發生。
實例:

Code
/*
* Generated by MyEclipse Struts
* Template path: templates/java/JavaClass.vtl
*/
package com.yourcompany.struts.action;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.struts.action.Action;
import org.apache.struts.action.ActionForm;
import org.apache.struts.action.ActionForward;
import org.apache.struts.action.ActionMapping;
/**
* MyEclipse Struts
* Creation date: 08-01-2008
*
* XDoclet definition:
* @struts.action validate="true"
* @struts.action-forward name="add" path="/add.jsp"
*/
public class ToAddAction extends Action {
/*
* Generated Methods
*/
/**
* Method execute
* @param mapping
* @param form
* @param request
* @param response
* @return ActionForward
*/
public ActionForward execute(ActionMapping mapping, ActionForm form,
HttpServletRequest request, HttpServletResponse response) {
// TODO Auto-generated method stub
saveToken(request);
return mapping.findForward("add");
}
}
這個Action主要作用就是在跳轉的頁面上加入Token,只有加入Token才能實現Token的驗證。執行完這個Action后,跳轉到的頁面會出現類似如下的一個hidden控件

Code
<div><input type="hidden" name="org.apache.struts.taglib.html.TOKEN" value="b030d9188a218097211d9061907cde5d"></div>
那么恭喜你,Token生效拉,注意,跳轉到的頁面里面,表單指可以用Struts標簽來生成,不可以用HTML來生成,不然Token是無效的。
使用Token的第一步完成,第二步,在提交的Action中驗證Token是否符合,代碼如下:

Code
/*
* Generated by MyEclipse Struts
* Template path: templates/java/JavaClass.vtl
*/
package com.yourcompany.struts.action;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.struts.action.Action;
import org.apache.struts.action.ActionForm;
import org.apache.struts.action.ActionForward;
import org.apache.struts.action.ActionMapping;
import org.apache.struts.action.ActionMessage;
import org.apache.struts.action.ActionMessages;
import com.lc.sqlhelp.Access;
import com.yourcompany.struts.form.AddForm;
/**
* MyEclipse Struts Creation date: 08-01-2008
*
* XDoclet definition:
*
* @struts.action path="/add" name="addForm" input="/form/add.jsp"
* scope="request" validate="true"
* @struts.action-forward name="add_suc" path="/add_suc.jsp"
* @struts.action-forward name="add_fail" path="/add_fail.jsp"
*/
public class AddAction extends Action {
/*
* Generated Methods
*/
/**
* Method execute
*
* @param mapping
* @param form
* @param request
* @param response
* @return ActionForward
*/
public ActionForward execute(ActionMapping mapping, ActionForm form,
HttpServletRequest request, HttpServletResponse response) {
AddForm af = (AddForm) form;
if (isTokenValid(request, true))//驗證Token是否符合要求
{
String sql = "insert into user (username,password) values('"
+ af.getUsername() + "','" + af.getPassword() + "')";
System.out.println(sql);
Access sh = new Access();
sh.loadMyDriver();
sh.setMyConnection();
sh.createMyStatement();
sh.executeMyUpdata(sql);
sh.closeMyStatement();
sh.closeMyConnection();
return mapping.findForward("add_suc");
}
return mapping.getInputForward();
}
}
這樣就完成了防止表單重復提交的功能。
實例源碼下載
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/08/01/1258067.html
posted @
2008-08-01 12:07 破名超難起 閱讀(111) |
評論 (0) |
編輯 收藏
在用戶登錄輸入密碼時,常常會有因為大寫鎖定開啟而造成輸入密碼錯誤的情況,如果在用戶大寫鎖定開啟時給予提示,就可以盡量避免這種情況的發生。
一下是代碼:

Code
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title>檢測大寫鎖定鍵</title>
<style type="text/css">
<!--
*{margin:0;padding:0;}
body{padding:2em;background:#242424;color:#ccc;}
h1{text-align:center;line-height:200%;}
h3{text-indent:1em;line-height:160%;color:#666;font-weight:normal;font-size:1em;}
h3 a{color:#bbb; text-decoration:none;margin:0 0.5em;}
h3 a:hover{color:#fff;}
p{margin:5em;}
span{margin:0 0.5em;font-size:85.7%;}
-->
</style>
</head>
<body>
<h1>檢測大寫鎖定鍵 </h1>
<form action="#" method="post">
<p><label for="password">密碼:</label><input type="password" id="password" name="password" /><span style="display:none;">大寫鎖定鍵被按下,請注意大小寫</span></p>
</form>
<script type="text/javascript">
//<![CDATA[
function detectCapsLock(event){
var e = event||window.event;
var o = e.target||e.srcElement;
var oTip = o.nextSibling;
var keyCode = e.keyCode||e.which; // 按鍵的keyCode
var isShift = e.shiftKey ||(keyCode == 16 ) || false ; // shift鍵是否按住
if (
((keyCode >= 65 && keyCode <= 90 ) && !isShift) // Caps Lock 打開,且沒有按住shift鍵
|| ((keyCode >= 97 && keyCode <= 122 ) && isShift)// Caps Lock 打開,且按住shift鍵
){oTip.style.display = '';}
else{oTip.style.display = 'none';}
}
document.getElementById('password').onkeypress = detectCapsLock;
//]]>
</script>
</body>
</html>
文章來源:
http://www.cnblogs.com/xiaoao808/archive/2008/07/31/1257624.html
posted @
2008-07-31 21:18 破名超難起 閱讀(245) |
評論 (0) |
編輯 收藏
摘要: 前一段時間朱哥一直在搞視頻轉換這個東東,我也一直很有興趣,就嘗試了一下。
首先是文件上傳功能的完成,做得很粗糙,沒有驗證上傳文件是否為視頻文件。利用前一段時間看視頻學來的部分代碼很輕松搞定。
接下來,就是視頻轉換了,...
閱讀全文
posted @
2008-07-31 16:41 破名超難起 閱讀(144) |
評論 (0) |
編輯 收藏