REPLACE INTO 为什么返回"2 rows affected"

表的结构:

desc urls;
+——-+———————+——+—–+———+——-+
| Field | Type | Null | Key | Default | Extra |
+——-+———————+——+—–+———+——-+
| id | bigint(16) unsigned | NO | PRI | NULL | |
| url | varchar(255) | NO | | NULL | |
| md516 | varchar(32) | NO | UNI | | |
+——-+———————+——+—–+———+——-+
3 rows in set (0.01 sec)


原有数据:

+—-+———————–+——————+
| id | url | md516 |
+—-+———————–+——————+
| 0 | http://www.162cm.com/ | 49f6f2aaa26c124a |
+—-+———————–+——————+
1 row in set (0.00 sec)

执行的语句:

REPLACE INTO urls (`url`,`md516`,`id`) VALUES (‘http://www.162cm.com/’,’49f6f2aaa26c124a’,0);

而执行这条语句的返回是:

Query OK, 2 rows affected (0.00 sec)

为啥呢….看起来数据啥也没变,不是吗?
看文档:http://au2.php.net/mysql_affected_rows:

Returns the number of affected rows on success, and -1 if the last query failed.

If the last query was a DELETE query with no WHERE clause, all of the records will have been deleted from the table but this function will return zero with MySQL versions prior to 4.1.2.

When using UPDATE, MySQL will not update columns where the new value is the same as the old value. This creates the possibility that mysql_affected_rows() may not actually equal the number of rows matched, only the number of rows that were literally affected by the query.

The REPLACE statement first deletes the record with the same primary key and then inserts the new record. This function returns the number of deleted records plus the number of inserted records.

我的vimrc 文件

嗯,给大家共享一下。都是从各地copy来的。

“vimim 相关
let g:vimim_enable_static_enu=1
“let g:vimim_enable_sexy_input_style=1
” 关闭中文标点
let g:vimim_disable_chinese_punctuation=1
“中英文之间不加空格
let g:vimim_disable_seamless_english_input=1

set dictionary-=/etc/funclist.txt dictionary+=/etc/funclist.txt
“Use the dictionary completion
set complete-=k complete+=k

“自动补全..
set completeopt=longest,menu

imap
set nu
autocmd FileType python set complete+=k~/.vim/tools/pydiction

filetype plugin indent on
set nocp
filetype plugin on
set helplang=cn
map :Project
“autocmd BufNewFile *.py 0r ~/.vim/template/simple.py


set ignorecase
set shiftwidth=4
set softtabstop=4
set tabstop=4

set nocompatible ” Use Vim defaults (much better!)
“set ai ” always set autoindenting on
“set backup ” keep a backup file
set viminfo=’200,f1,<1500 " read/write a .viminfo file, don't store more " than 50 lines of registers set history=1500 " keep 50 lines of command line history set ruler " show the cursor position all the time if has("autocmd") filetype plugin indent on autocmd FileType python set omnifunc=pythoncomplete#Complete autocmd FileType ruby set omnifunc=rubycomplete#Complete autocmd FileType c set omnifunc=ccomplete#Complete autocmd FileType ada set omnifunc=adacomplete#Complete autocmd FileType php set omnifunc=phpcomplete#CompletePHP autocmd FileType javascript set omnifunc=javascriptcomplete#CompleteJS autocmd FileType html set omnifunc=htmlcomplete#CompleteTags autocmd FileType css set omnifunc=csscomplete#CompleteCSS autocmd FileType xml set omnifunc=xmlcomplete#CompleteTags autocmd FileType sql set omnifunc=sqlcomplete#Complete endif set ru "该命令打开 VIM 的状态栏标尺。 默认情况下, VIM 的状态栏标尺在屏幕底部, 它能即时显示当前光标所在位置在文件中的行号、列号, 以及对应的整个文件的百分比。 打开标尺可以给文件的编辑工作带来一定方便。 set nocp "该命令指定让 VIM 工作在不兼容模式下。 在 VIM 之前, 出现过一个非常流行的编辑器叫 vi。 VIM 许多操作与 vi 很相似,但也有许多操作与 vi 是不一样的。 如果使用“:set cp”命令打开了兼容模式开关的话, VIM 将尽可能地模仿 vi 的操作模式。 "也许有许多人喜欢“最正统的 vi”的操作模式, 对于初学者来说, vi 里许多操作是比较不方便的。 "举一个例子, VIM 里允许在 Insert 模式下使用方向键移动光标, 而 vi 里在 Insert 模式下是不能移动光标的, 必须使用 ESC 退回到 Normal 模式下才行。 "再举一个例子, vi 里使用 u 命令可以撤消一次先前的操作, 再次按下 u 时, 将撤消“撤消”这个动作本身,也就是我们常说的“重复”(redo)。 而 VIM 里可以使用 u 命令撤消多步操作, “重复”使用的快捷键是 Ctrl + R。 "使用兼容模式后, VIM 将放弃这些新的功能, 尽量模仿 vi 的各种操作方式。 只有在不兼容模式下, 才能更好地发挥 VIM 自身的特点。 Bram 爷爷强烈推荐大家使用 VIM 的不兼容模式, 滇狐也是这样推荐的。 请务必在你的 .vimrc 中的第一行写上: “set nocp”。 set hls "搜索时高亮显示被找到的文本。 该指令的功能在 vimtutor 中已经有过介绍, 这里就不多说了。 其实似乎许多人并不喜欢这个功能。 "set is "搜索时在未完全输入完毕要检索的文本时就开始检索。 vimtutor 对该命令也有过介绍, 滇狐并不喜欢这个功能, 因此滇狐自己的配置文件里是没有这条命令的。 但是周围有朋友很喜欢这个, 因此滇狐还是将它列在这里。 syntax on "打开关键字上色。 进行程序设计的朋友应该都知道关键字上色是什么东西, 因此这里就不多说了。 不进行程序设计的朋友不妨也打开这个功能, 虽然不一定能够用得着, 但这个功能其实也是很好玩的。 set backspace=indent,eol,start "设想这样一个情况: 当前光标前面有若干字母, 我们按下 i 键进入了 Insert 模式, 然后输入了 3 个字母, 再按 5 下删除(Backspace)。 默认情况下, VIM 仅能删除我们新输入的 3 个字母, 然后喇叭“嘟嘟”响两声。 如果我们“set backspace=start”, 则可以在删除了新输入的 3 个字母之后, 继续向前删除原有的两个字符。 "再设想一个情况: 有若干行文字, 我们把光标移到中间某一行的行首, 按 i 键进入 Insert 模式, 然后按一下 Backspace。默认情况下, 喇叭会“嘟”一声, 然后没有任何动静。 如果我们“set backspace=eol”, 则可以删除前一行行末的回车,也就是说将两行拼接起来。 "当我们设置了自动缩进后, 如果前一行缩进了一定距离, 按下回车后, 下一行也会保持相同的缩进。 默认情况下, 我们不能在 Insert 模式下直接按 Backspace 删除行首的缩进。 如果我们“set backspace=indent”, 则可以开启这一项功能。 "上述三项功能, 你可以根据自己的需要, 选择其中一种或几种, 用逗号分隔各个选项。 建议把这三个选项都选上。 set whichwrap=b,s,<,>,[,]
“默认情况下, 在 VIM 中当光标移到一行最左边的时候, 我们继续按左键, 光标不能回到上一行的最右边。 同样地, 光标到了一行最右边的时候,我们不能通过继续按右跳到下一行的最左边。 但是, 通过设置 whichwrap 我们可以对一部分按键开启这项功能。如果想对某一个或几个按键开启到头后自动折向下一行的功能, 可以把需要开启的键的代号写到 whichwrap 的参数列表中,各个键之间使用逗号分隔。 以下是 whichwrap 支持的按键名称列表:

” * b
” 在 Normal 或 Visual 模式下按删除(Backspace)键。
” * s
” 在 Normal 或 Visual 模式下按空格键。
” * h
” 在 Normal 或 Visual 模式下按 h 键。
” * l
” 在 Normal 或 Visual 模式下按 l 键。
” * < " 在 Normal 或 Visual 模式下按左方向键。 " * >
” 在 Normal 或 Visual 模式下按右方向键。
” * ~
” 在 Normal 模式下按 ~ 键(翻转当前字母大小写)。
” * [
” 在 Insert 或 Replace 模式下按左方向键。
” * ]
” 在 Insert 或 Replace 模式下按右方向键。
set encoding=utf-8
“设置当前字符编码为 UTF-8。 UTF-8 是支持字符集最多的编码之一, 在 UTF-8 下进行工作, 会带来许多方便之处。 由于 VIM 在运行过程中切换 encoding 会造成许多问题, 如提示信息乱码、 register 丢失等, 因此强烈建议大家在启动 VIM 的时候把 encoding 设置为 UTF-8, 在编辑非 UTF-8 的文件时, 通过 fileencoding 来进行转码。
set langmenu=zh_CN.UTF-8
“使用中文菜单, 并使用 UTF-8 编码。 如果没有这句的话, 在非 UTF-8 的系统, 如 Windows 下, 用了 UTF-8 的 encoding 后菜单会乱码。
“language message en_US.UTF-8
“使用中文提示信息, 并使用 UTF-8 编码。 如果没有这句的话, 在非 UTF-8 的系统, 如 Windows 下, 用了 UTF-8 的 encoding 后系统提示会乱码。
set fileencodings=ucs-bom,utf-8,cp936,gb18030,big5,euc-jp,euc-kr,latin1
“设置编码的自动识别。 关于这条设置的详细含义, 请参看这篇文章。
“set ambiwidth=double
“防止特殊符号无法正常显示。 在 Unicode 中, 许多来自不同语言的字符, 如果字型足够近似的话, 会把它们放在同一个编码中。但在不同编码中, 字符的宽度是不一样的。 例如中文汉语拼音中的 ā 就很宽, 而欧洲语言中同样的字符就很窄。 当 VIM 工作在 Unicode 状态时, 遇到这些宽度不明的字符时, 默认使用窄字符, 这会导致中文的破折号“——”非常短, 五角星“★”等符号只能显示一半。因此, 我们需要设置 ambiwidth=double 来解决这个问题。
filetype plugin indent on
“开启文件类型自动识别, 启用文件类型插件, 启用针对文件类型的自动缩进
set smarttab
“当使用 et 将 Tab 替换为空格之后, 按下一个 Tab 键就能插入 4 个空格,
“但要想删除这 4 个空格, 就得按 4 下 Backspace, 很不方便。 设置 smarttab
“之后, 就可以只按一下 Backspace 就删除 4 个空格了。
“set spell
“打开拼写检查。 拼写有错的单词下方会有红色波浪线, 将光标放在单词上, 按 z= 就会出现拼写建议, 按 ]s 可以直接跳到下一个拼写错误处。

“set tw=78
“设置光标超过 78 列的时候折行。
set lbr
“不在单词中间断行。 设置了这个选项后, 如果一行文字非常长, 无法在一行内显示完的话, 它会在单词与单词间的空白处断开, 尽量不会把一个单词分成两截放在两个不同的行里。
set fo+=mB
“打开断行模块对亚洲语言支持。 m 表示允许在两个汉字之间断行, 即使汉字之间没有出现空格。 B 表示将两行合并为一行的时候, 汉字与汉字之间不要补空格。 该命令支持的更多的选项请参看用户手册。


” c/c++
set sm
“显示括号配对情况。 打开这个选项后, 当输入后括号 (包括小括号、中括号、大括号)
“的时候, 光标会跳回前括号片刻, 然后跳回来, 以此显示括号的配对情况。
set ai
“打开普通文件类型的自动缩进。 该自动缩进不如 cindent 智能, 但它可以为你编辑非
“C/C++ 文件提供一定帮助。
set selectmode=
“不使用 selectmode。

set keymodel=
“不使用“Shift + 方向键”选择文本, “Shift + 方向键”代表向指定方向跳一个单词。 如果你喜欢这项功能的话, 可以使用“set keymodel=startsel,stopsel”打开它。
set keymodel=startsel,stopsel

set wildmenu
“在命令模式下使用 Tab 自动补全的时候,
“将补全内容使用一个漂亮的单行菜单形式显示出来。
set sw=4
“自动缩进的时候, 缩进尺寸为 4 个空格。
set ts=4
“Tab 宽度为 4 个字符。
set et
“编辑时将所有 Tab 替换为空格。
“该选项只在编辑时将 Tab 替换为空格, 如果打开一个已经存在的文件, 并不会将已有的 Tab 替换为空格。 如果希望进行这样的替换的话, 可以使用这条命令“:retab”。
retab
“set paste
“set nopaste
set pastetoggle=
“很多兄弟都碰到过这样一个问题,在vim中粘贴代码有时会自动增加缩进,造成代码排版
“的混乱。如何让它不缩进,保持原格式?其实vim有一个paste开关。
“输入 :set paste
“需要关闭时
“输入:set nopaste
“我是在vimrc中加入了:
“set pastetoggle=
“这样就可以用F3来切换了。
if (has(“win32”))

“————————————————————————-
” Win32
“————————————————————————-
if (has(“gui_running”))
set guifont=Bitstream_Vera_Sans_Mono:h9:cANSI
set guifontwide=NSimSun:h9:cGB2312
endif

else
if (has(“gui_running”))
set guifont=Bitstream Vera Sans Mono 9
endif
endif

“如果去掉这一行,默认值set mouse=a就生效了,这时不能用鼠标选中复制.
set mouse=v
“打开文件时自动到达上次浏览时的位置
” Only do this part when compiled with support for autocommands
if has(“autocmd”)
” ” In text files, always limit the width of text to 78 characters
autocmd BufRead *.txt set tw=78
” ” When editing a file, always jump to the last cursor position
autocmd BufReadPost *
if line(“‘””) > 0 && line (“‘””) <= line("$") | exe "normal g'"" | endif endif au BufReadPost * if line("'"") > 0|if line(“‘””) <= line("$")|exe("norm '"")|else|exe "norm $"|endif|endif " "其中字符串 “string” 规定了什么要储存。该字符串的语法为一个选项字符跟一个参数. ""选项和参数组成的对子之间由逗号分隔. ""来看一下你可以怎样构建你自己的 viminfo 字符串。首先,选项 ‘ 用于规定你为多 "少个文件保存标记 (a-z)。为此选项挑一个美妙的偶数 (比如 1000). 你的命令现在看 "起来像这样: " ":set viminfo=’1000 " "选项 f 控制是否要储存全局标记 (A-Z 和 0-9)。如果这个选项设为 0,那么什么也不 "存储。如果设为 1 ,或你对 f 选项不作规定, 那么标记就被存储. 你要这个功能, 现 "在你有了: " ":set viminfo=’1000,f1 " "选项 < 控制着每个寄存器内保存几行文本。默认情况下,所有的文本行都被保存. 如果 "设为 0,则什么也不保存。为了避免成千上万行文本被加入你的信息文件 (那些文本可能 "永远也没用,徒然使 Vim 起动得更慢), 你采用 500 行的上限: " ":set viminfo='1000,f1,<500 " "你也许用得着的其它选项: ": 保存命令行历史记录内的行数 "@ 保存输入行历史记录内的行数 "/ 保存搜索历史记录内的行数 "r 可移介质,其中没有任何标记存入 (可用多次) "! 以大写字母开头的全局变数,并且不含有小写字母 "h 起动时解除选项 'hlsearch' 的高亮度显示 "% 缓冲列表 (只有当不带参数起动 Vim 时才还原) "c 用编码 'encoding' 转换文本 "n 用于 viminfo 文件的名称 (必须为最后一项选项) set viminfo='1000,f1,<500

python多线程程序的中断(信号)处理

python程序很容易进行多线程处理,也很好进行signal的处理。但是,在多线程程序中进行信号处理,却不太好搞。我琢磨出了一个可行的办法。
看第一个程序:singlethread.py

#!/usr/bin/python
#coding:utf-8

import signal,os,sys,threading,time
import wk
def sig_exit():
    print "[end time]:"+str(time.time())
    print "exited."
    sys.exit()

def handler(signum, frame):
    print "got an signal",signum,frame
    if signum == 3:
        sig_exit()
    if signum == 2:
        sig_exit()
    if signum == 9:
        sig_exit()
        return None
signal.signal(signal.SIGINT,handler)
signal.signal(signal.SIGTERM,handler)
signal.signal(3,handler)




TIME=3

time.sleep(TIME)
print 'normally exit'

运行这个程序,并马上按上Ctrl+C,发现程序这样输出:
<pre>
^Cgot an signal 2
[end time]:1245852618.36
exited.

而不做干扰,让它正常退出时,是这样的:

normally exit

现在我们改进到多线程版本。先写一个线程类:wk.py

import threading,time
class wk(threading.Thread):
def run(self):
for k in xrange(0,4):
print ‘sleep 1 second,’,self,”:”,k
time.sleep(1)

这个线程类啥也不干,就不断输出一个标识 ,然后再歇一秒。再输出,再歇一秒….
主程序:multithreads.py

#!/usr/bin/python
#coding:utf-8

import signal,os,sys,threading,time
from wk import wk

def sig_exit():
print “[end time]:”+str(time.time())
print “exited.”
sys.exit()

def handler(signum, frame):
print “got an signal”,signum,frame
sig_exit()
if signum == 3:
sig_exit()
if signum == 2:
sig_exit()
if signum == 9:
sig_exit()
return None
signal.signal(signal.SIGINT,handler)
signal.signal(signal.SIGTERM,handler)
signal.signal(3,handler)
signal.signal(signal.SIGALRM, handler)

threads=[]
for a in xrange(0,4):
temp=wk()
temp.setDaemon(True)
temp.setName(“thread:”+str(a))
temp.start()
threads.append(temp)

for thread in threads:
thread.join()
print “thread joined”,thread

print ‘normally exit’

现在我们运行它,并按下CTL+C.输出结果像这样:

sleep 1 second,sleep 1 second, : 0
sleep 1 second, : 0
: 0
sleep 1 second, : 0
sleep 1 second, : 1
sleep 1 second, : 1
sleep 1 second, : 1
sleep 1 second, : 1
^Csleep 1 second, : 2
sleep 1 second, : 2
sleep 1 second, : 2
sleep 1 second, : 2
^Csleep 1 second, : 3
sleep 1 second, : 3
sleep 1 second, : 3
sleep 1 second, : 3
got an signal 2
[end time]:1245852879.37
exited.

在按下CTRL+C后,程序并不是马上退出,而是又过了几秒钟才退出。
而运行时不按CTRL+C干扰它时,是这样的输出:

sleep 1 second, : 0
sleep 1 second, : 0
sleep 1 second, : 0
sleep 1 second, : 0
sleep 1 second, : 1
sleep 1 second, : 1
sleep 1 second, : 1
sleep 1 second, : 1
sleep 1 second, : 2
sleep 1 second, : 2
sleep 1 second, : 2
sleep 1 second, : 2
sleep 1 second, : 3
sleep 1 second, : 3
sleep 1 second, : 3
sleep 1 second, : 3
thread joined
thread joined
thread joined
thread joined
normally exit
</coolcode>
说明在这个程序里,中断信号能被捕获,但是并不即时。

而最后我的多线程程序是:last.py

#!/usr/bin/python
#coding:utf-8

import signal,os,sys,threading,time
from wk import wk

def sig_exit():
print “[end time]:”+str(time.time())
print “exited.”
sys.exit()

def handler(signum, frame):
print “got an signal”,signum,frame
if signum == 3:
sig_exit()
if signum == 2:
sig_exit()
if signum == 9:
sig_exit()
return None
signal.signal(signal.SIGINT,handler)
signal.signal(signal.SIGTERM,handler)
#signal.signal(3,handler)
signal.signal(signal.SIGALRM, handler)

class pause(threading.Thread):
threads=[]
def run(self):
print “pause running”
for thread in self.threads:
thread.join()
print “thread “,thread,” joined”
print “all joined”
signal.alarm(1)

threads=[]
for i in xrange(0,4):
j=wk()
j.setDaemon(True)
j.setName(“thread:”+str(i))
j.start()
threads.append(j)
print “total threads:”,threading.activeCount()

print “current thread:”,threading.activeCount()
print “parent process,main thread,sleeping”

k=pause()
k.setDaemon(True)
k.threads=threads
k.start()
signal.pause()
print “normally exit”

这个程序里,我们的逻辑是这样的:主线程创建了其他工作线程,并且还创建了一个特殊的线程, 这个特殊的线程的任务就着等着其他的工作线程结束(调用了wk类的join方法)。当工作线程都结束后,这个线程就发一个alarm信号。而主线程创建完所有的其他线程后的工作,就是在坐在那儿专门等一个信号。因此当我们按CTRL+C时信号就成功地被捕获了。而没有CTRL+C产生的信号被捕获时,所有工作线程(wk类)结束时也会产生一个alarm信号,也能让程序顺利退出。
这个程序,运行时按下CTRL+C的输出是:

sleep 1 second, : 0
total threads: 2
sleep 1 second, : 0
total threads: 3
sleep 1 second, : 0
total threads: 4
sleep 1 second, : 0
total threads: 5
current thread: 5
parent process,main thread,sleeping
pause running
^Cgot an signal 2
[end time]:1245853073.05
exited.

CTRL+C给程序的信号马上得到了处理。运行这个程序时啥也不做,等着程序结束,输出则是:

sleep 1 second, : 0
total threads: 2
sleep 1 second, : 0
total threads: 3
sleep 1 second, : 0
total threads: 4
sleep 1 second, : 0
total threads: 5
current thread: 5
parent process,main thread,sleeping
pause running
^Cgot an signal 2
[end time]:1245853073.05
exited.
renlu@aragorn:~/Dropbox/Private/devel/python$ py lastprogram.py
sleep 1 second, total threads: : 0
2
sleep 1 second, : 0
total threads: 3
sleep 1 second, : 0
total threads: 4
sleep 1 second, : 0
total threads: 5
current thread: 5
parent process,main thread,sleeping
pause running
sleep 1 second, : 1
sleep 1 second, : 1
sleep 1 second, : 1
sleep 1 second, : 1
sleep 1 second, sleep 1 second, : 2
: 2
sleep 1 second, : 2
sleep 1 second, : 2
sleep 1 second, : 3
sleep 1 second, : 3
sleep 1 second, : 3
sleep 1 second, : 3
thread joined
thread joined
thread joined
thread joined
all joined
got an signal 14
normally exit

测试FusionIO:strict_sync太秀逗了…

这是一篇译文,原文地址是:   http://www.mysqlperformanceblog.com/2009/06/15/testing-fusionio-strict_sync-is-too-strict/

随着新的更新FusionIO驱动的更新,我能够在我的戴尔R900与Ubuntu 8.10上进行尝鲜,而不再需要痛苦地编译驱动或是降级kernel.现在我决定测试一下strict_sync模式。

据我所知FusionIO默认情况下,比如英特尔的SSD ,是对应用程序“撒谎”了,fsync()并不是真正的发生了,它仍然只是提交到了内存而不是最终的磁盘。FusionIO还有一个“ strict_sync ”模式,可以保证fsync函数可靠地操作。

好啊兄弟,让我们来压一下性能—通常我在100w数据(约9GB数据),O_DIRECT模式下开着3GB的buffer_pool来搞这个测试。
原始测试数量参见这里
pub
结果用TPM (每分钟的事务数)表示 ,这样更清楚一些。图形显示,结果是是随着时间[X轴]不断变化 。
这个图形已经很明显,我都不需要再用文字来说明了。。。

但是我没有测试,在断电的情况下,在默认模式下处理、结束事务的情况。这个需要读者您检查一下。

怎样用一行脚本美化/etc/my.cnf文件

原文地址:http://www.mysqlperformanceblog.com/2009/06/15/how-to-pretty-print-mycnf-with-a-one-liner/
当我在一台服务器上转悠时,我会经常想看看/etc/my.cnf文件格式,当然是想看格化美观,没有注释的。这样的一行Perl就能将美化输出文件:

1.perl -ne ‘m/^([^#][^s=]+)s*(=.*|)/ && printf(“%-35s%sn”, $1, $2)’ /etc/my.cnf
2.[client]
3.port = 3306
4.socket = /var/run/mysqld/mysqld.sock
5.[ mysqld_safe ]
6.socket = /var/run/mysqld/mysqld.sock
7.nice = 0
8.[ mysqld ]
9.user = mysql
10.pid-file = /var/run/mysqld/mysqld.pid
11.socket = /var/run/mysqld/mysqld.sock
12.port = 3306
13…..