上周六通宵打牌周日白天只睡3小時累成狗,從今天起以后不能玩太大的了,小賭怡情大賭傷身,和同事朋友有空玩玩還是好的。今天公司外面馬路上有人掛燈籠時死了一個人,哎,快過年了悲劇又發生了,真是生命是脆弱的。雖然這一兩個月貪玩了點,但技術也不能落下,周一到周五下班沒事的話有時候也會上網大致了解一些。今天就接着之前的博客繼續開始redis之路。
至於什么是redis,它是用來做什么的?它的用處有好多,這里我也不一一列舉,以后也會有說明,寫redis主要是需求驅動,漸進式的學習,因為前面學習了nginx,導致會有多個地方部署程序,那這樣如果用戶在A登錄了,可能再次訪問時又會訪問B,但這樣B可能會判斷同一個用戶沒有登錄,既然對用戶透明,這樣多次登錄也是不好的,而redis就可以解決這個問題。還有我們登錄app或下單什么的,經常會收到驗證碼什么的,有效期有的是120s或60s,雖然用數據庫也能實現這樣的功能,這樣的話就要連接數據庫做一系列的邏輯驗證,如果用redis就會很簡單。這些都是redis的用處。redis有五大數據類型,從今天開始以此學習這幾種數據類型。今天就先學習字符串類型。首先說明redis這一系列博客我也是參照別人的博客,然后自己動手敲的,地址http://www.cnblogs.com/stephen-liu74/archive/2012/04/16/2370212.html的內容是學習五大類型主要模仿的對象,大家可以看上面地址的博客。
一、前戲
字符串類型是Redis中最為基礎的數據存儲類型,它在Redis中是二進制安全的,這便意味着該類型可以接受任何格式的數據,如JPEG圖像數據或Json對象描述信息等。在Redis中字符串類型的Value最多可以容納的數據長度是512M。
二、理論
命令原型 |
時間復雜度 | 命令描述 | 返回值 |
APPEND key value |
O(1) | 如果該Key已經存在,APPEND命令將參數Value的數據追加到已存在Value的末尾。如果該Key不存在,APPEND命令將會創建一個新的Key/Value。 | 追加后Value的長度。 |
DECR key | O(1) | 將指定Key的Value原子性的遞減1。如果該Key不存在,其初始值為0,在decr之后其值為-1。如果Value的值不能轉換為整型值,如Hello,該操作將執行失敗並返回相應的錯誤信息。注意:該操作的取值范圍是64位有符號整型。 | 遞減后的Value值。 |
INCR key | O(1) | 將指定Key的Value原子性的遞增1。如果該Key不存在,其初始值為0,在incr之后其值為1。如果Value的值不能轉換為整型值,如Hello,該操作將執行失敗並返回相應的錯誤信息。注意:該操作的取值范圍是64位有符號整型。 | 遞增后的Value值。 |
DECRBY key decrement | O(1) | 將指定Key的Value原子性的減少decrement。如果該Key不存在,其初始值為0,在decrby之后其值為-decrement。如果Value的值不能轉換為整型值,如Hello,該操作將執行失敗並返回相應的錯誤信息。注意:該操作的取值范圍是64位有符號整型。 | 減少后的Value值。 |
INCRBY key increment | O(1) | 將指定Key的Value原子性的增加increment。如果該Key不存在,其初始值為0,在incrby之后其值為increment。如果Value的值不能轉換為整型值,如Hello,該操作將執行失敗並返回相應的錯誤信息。注意:該操作的取值范圍是64位有符號整型。 | 增加后的Value值。 |
GET key | O(1) | 獲取指定Key的Value。如果與該Key關聯的Value不是string類型,Redis將返回錯誤信息,因為GET命令只能用於獲取string Value。 | 與該Key相關的Value,如果該Key不存在,返回nil。 |
SET key value | O(1) |
設定該Key持有指定的字符串Value,如果該Key已經存在,則覆蓋其原有值。 | 總是返回"OK"。 |
GETSET key value | O(1) | 原子性的設置該Key為指定的Value,同時返回該Key的原有值。和GET命令一樣,該命令也只能處理string Value,否則Redis將給出相關的錯誤信息。 | 返回該Key的原有值,如果該Key之前並不存在,則返回nil。 |
STRLEN key | O(1) | 返回指定Key的字符值長度,如果Value不是string類型,Redis將執行失敗並給出相關的錯誤信息。 | 返回指定Key的Value字符長度,如果該Key不存在,返回0。 |
SETEX key seconds value | O(1) | 原子性完成兩個操作,一是設置該Key的值為指定字符串,同時設置該Key在Redis服務器中的存活時間(秒數)。該命令主要應用於Redis被當做Cache服務器使用時。 | #通過ttl命令查看一下指定Key的剩余存活時間(秒數),0表示已經過期,-1表示永不過期.在Redis 2.8以后,當key不存在時,返回-2。 |
SETNX key value | O(1) | 如果指定的Key不存在,則設定該Key持有指定字符串Value,此時其效果等價於SET命令。相反,如果該Key已經存在,該命令將不做任何操作並返回。 | 1表示設置成功,否則0。 |
SETRANGE key offset value | O(1) | 替換指定Key的部分字符串值。從offset開始,替換的長度為該命令第三個參數value的字符串長度,其中如果offset的值大於該Key的原有值Value的字符串長度,Redis將會在Value的后面補齊(offset - strlen(value))數量的0x00,之后再追加新值。如果該鍵不存在,該命令會將其原值的長度假設為0,並在其后添補offset個0x00后再追加新值。鑒於字符串Value的最大長度為512M,因此offset的最大值為536870911。最后需要注意的是,如果該命令在執行時致使指定Key的原有值長度增加,這將會導致Redis重新分配足夠的內存以容納替換后的全部字符串,因此就會帶來一定的性能折損。 | 修改后的字符串Value長度。 |
GETRANGE key start end | O(1) | 如果截取的字符串長度很短,我們可以該命令的時間復雜度視為O(1),否則就是O(N),這里N表示截取的子字符串長度。該命令在截取子字符串時,將以閉區間的方式同時包含start(0表示第一個字符)和end所在的字符,如果end值超過Value的字符長度,該命令將只是截取從start開始之后所有的字符數據。 | 子字符串 |
SETBIT key offset value | O(1) | 設置在指定Offset上BIT的值,該值只能為1或0,在設定后該命令返回該Offset上原有的BIT值。如果指定Key不存在,該命令將創建一個新值,並在指定的Offset上設定參數中的BIT值。如果Offset大於Value的字符長度,Redis將拉長Value值並在指定Offset上設置參數中的BIT值,中間添加的BIT值為0。最后需要說明的是Offset值必須大於0。 | 在指定Offset上的BIT原有值。 |
GETBIT key offset | O(1) | 返回在指定Offset上BIT的值,0或1。如果Offset超過string value的長度,該命令將返回0,所以對於空字符串始終返回0。 | 在指定Offset上的BIT值。 |
MGET key [key ...] | O(N) | N表示獲取Key的數量。返回所有指定Keys的Values,如果其中某個Key不存在,或者其值不為string類型,該Key的Value將返回nil。 | 返回一組指定Keys的Values的列表。 |
MSET key value [key value ...] | O(N) | N表示指定Key的數量。該命令原子性的完成參數中所有key/value的設置操作,其具體行為可以看成是多次迭代執行SET命令。 | 該命令不會失敗,始終返回OK。 |
MSETNX key value [key value ...] | O(N) | N表示指定Key的數量。該命令原子性的完成參數中所有key/value的設置操作,其具體行為可以看成是多次迭代執行SETNX命令。然而這里需要明確說明的是,如果在這一批Keys中有任意一個Key已經存在了,那么該操作將全部回滾,即所有的修改都不會生效。 | 1表示所有Keys都設置成功,0則表示沒有任何Key被修改。 |
三、實操
127.0.0.1:6379> append mykey helloworld (integer) 10 127.0.0.1:6379> get mykey "helloworld" 127.0.0.1:6379> set mykey 20 OK 127.0.0.1:6379> decr mykey (integer) 19 127.0.0.1:6379> incr mykey (integer) 20 127.0.0.1:6379> decrby mykey -5 (integer) 25 127.0.0.1:6379> incrby mykey -6 (integer) 19 127.0.0.1:6379> strlen mykey (integer) 2 127.0.0.1:6379> setex mykey 60 18575533*** OK 127.0.0.1:6379> ttl mykey (integer) 55 127.0.0.1:6379> ttl mykey (integer) 50 127.0.0.1:6379> ttl mykey (integer) 17 127.0.0.1:6379> ttl mykey (integer) -2 127.0.0.1:6379> setnx mykey 123 (integer) 1 127.0.0.1:6379> get mykey "123" 127.0.0.1:6379> setnx mykey 234 (integer) 0 127.0.0.1:6379> get mykey "123" 127.0.0.1:6379> setrange mykey 1 2 (integer) 3 127.0.0.1:6379> get mykey "123" 127.0.0.1:6379> setrange mykey 3 4 (integer) 4 127.0.0.1:6379> get mykey "1234" 127.0.0.1:6379> setrange mykey 6 3 (integer) 7 127.0.0.1:6379> get mykey "1234\x00\x003" 127.0.0.1:6379> getrange mykey 3 4 "4\x00" 127.0.0.1:6379> getrange mykey 9 10 "" 127.0.0.1:6379> del mykey (integer) 1 127.0.0.1:6379> get mykey (nil) 127.0.0.1:6379> setbit mykey 7 1 (integer) 0 127.0.0.1:6379> get mykey "\x01" 127.0.0.1:6379> set mykey 7 1 (error) ERR syntax error 127.0.0.1:6379> setbit mykey 6 1 (integer) 0 127.0.0.1:6379> get mykey "\x03" 127.0.0.1:6379> getbit mykey 6 (integer) 1 127.0.0.1:6379> getbit mykey 10 (integer) 0 127.0.0.1:6379> del mykey (integer) 1 127.0.0.1:6379> get mykey (nil) 127.0.0.1:6379> mset mykey a mykey1 b OK 127.0.0.1:6379> mget mykey mykey1 1) "a" 2) "b" 127.0.0.1:6379> msetnx mykey2 c mykey3 d (integer) 1 127.0.0.1:6379> mget mykey mykey1 mykey2 mykey3 1) "a" 2) "b" 3) "c" 4) "d" 127.0.0.1:6379> msetnx mykey e mykey4 f (integer) 0 127.0.0.1:6379> mget mykey mykey4 1) "a" 2) (nil) 127.0.0.1:6379>