亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > Ruby > 正文

Ruby語言中的String深入理解

2020-10-29 19:46:21
字體:
來源:轉載
供稿:網友
Ruby語言中的String是mutable的,不像java、C#中的String是immutable的。比如
復制代碼 代碼如下:

str1="abc"
str2="abc"

  在java中,對于字面量的字符串,jvm內部維持一張表,因此如果在java中,str1和str2是同一個String對象。而在Ruby中, str1和str2是完全不同的對象。同樣,在java中對于String對象的操作都將產生一個新的對象,而Ruby則是操縱同一個對象,比如:
復制代碼 代碼如下:

str="abc"
str.concat("cdf")

  此時str就是"abccdf"。Ruby對String是怎么處理的呢?我們只談談c ruby中的實現,有興趣的先看看這篇文章《管窺Ruby――對象基礎》。在ruby.h中我們可以看到String對象的結構,Ruby中的對象(包括類也是對象)都是一個一個的struct,String也不能例外:
復制代碼 代碼如下:

struct RString {
struct RBasic basic;
long len;
char *ptr;
union {
long capa;
VALUE shared;
} aux;
};
//ruby.h

 顯然,len是String的長度;ptr是一個char類型的指針,指向實際的字符串;然后是一個聯合,這個稍后再說。如果你看看ruby.h可以發 現,幾乎所有定義的對象結構都有一個struct RBasic。顯然,struct RBasic包含由所有對象結構體共享的一些重要信息的。看看RBasic:
復制代碼 代碼如下:

struct RBasic {
unsigned long flags;
VALUE klass;
};

  其中的flags是一個多用途的標記,大多數情況下用于記錄結構體的類型,ruby.h中預定義了一些列的宏,比如T_STRING(表示struct RString),T_ARRAY(表示struct RArray)等。Klass是一個VALUE類型,VALUE也是unsigned long,可以地將它當成指針(一個指針4字節,綽綽有余了),它指向的是一個Ruby對象,這里以后再深入。

  那么聯合aux中的capa和shared是干什么用的呢?因為Ruby的String是可變的,可變意味著len可以改變,我們需要每次都根據len的 變換來增減內存(使用c中的realloc()函數),這顯然是一個很大的開銷,解決辦法就是預留一定的空間,ptr指向的內存大小略大于len,這樣就 不需要頻繁調用realloc了,aux.capa就是一個長度,包含額外的內存大小。那么aux.shared是干什么的呢?這是一個VALUE類型, 說明它是指向某個對象。aux.shared其實是用于加快字符串的創建速度,在一個循環中:

  ruby 代碼

  whiletruedo重復 a="str"#以“str”為內容創建字符串,賦值給a a.concat("ing")#為a所指向的對象添加“ing” p(a)#顯示“string” end

  每次都重新創建一個"str"對象,內部就是重復創建一個char[],這是相當奢侈,aux.shared就是用于共享char[],以字面量創建的字符串會共享一個char[],當要發生變化時,將字符串復制到一個非共享的內存中,變化針對這個新拷貝進行,這就是所謂的“copy-on-write"技術。解釋了String的內部構造,貌似還沒有介紹String是怎么實現mutable,我們寫一個Ruby擴展測試下,我們想寫這樣一個Ruby類:

  ruby 代碼

classTestdefteststr="str"str.concat("ing")endend

  對應的c語言代碼就是:

  cpp 代碼
復制代碼 代碼如下:

#include
#include"ruby.h"staticVALUEt_test(VALUEself){
VALUEstr;str=rb_str_new2("str");
printf("beforeconcat:str:%p,
str.aux.shared:%p,str.ptr:%s"n",str,(RSTRING(str)->aux).shared,RSTRING(str)->ptr);
rb_str_cat2(str,"ing");
printf("afterconcat:str:%p,str.aux.shared:%p,str.ptr:%s"n",
str,(RSTRING(str)->aux).shared,RSTRING(str)->ptr);returnself;
}
VALUEcTest;
voidInit_string_hack(){
cTest=rb_define_class("Test",rb_cObject);
rb_define_method(cTest,"test",t_test,0);
}//string_hack.c

  rb_define_class函數定義了一個類Test,rb_define_method將t_test方法以test的名稱添加到Test類。在t_test中,通過rb_str_new2每次生成一個RString結構,然后通過rb_str_cat2將str與"ing"連接起來,添加了一些打印用于跟蹤。利用mkmf產生Makefile,寫一個extconf.rb

  ruby 代碼

require'mkmf'create_makefile("string_hack");

  執行ruby extconf.rb,將產生一個Makefile,執行make,生成一個string_hack.so的鏈接庫。擴展寫完了,通過ruby調用:

  ruby 代碼

require'string_hack"t=Test.new(1..3).each{|i|t.test}

  輸出:

before concat: str:0x40098a40, str.aux.shared:0x3, str.ptr:str
after concat: str:0x40098a40, str.aux.shared:0x8, str.ptr:string
before concat: str:0x40098a2c, str.aux.shared:0x3, str.ptr:str
after concat: str:0x40098a2c, str.aux.shared:0x8, str.ptr:string
before concat: str:0x40098a18, str.aux.shared:0x3, str.ptr:str
after concat: str:0x40098a18, str.aux.shared:0x8, str.ptr:string

  從結果可以看出,在str concat之前之后,str指向的位置沒有改變,改變的僅僅是str中ptr指向的字符串的值,看看rb_str_cat2函數的實現就一目了然了:

  cpp 代碼
復制代碼 代碼如下:

VALUErb_str_cat(str,ptr,len)VALUEstr;
constchar*ptr;
longlen;
{
if(len<0){rb_raise(rb_eArgError,"negativestringsize(orsizetoobig)");
}
if(FL_TEST(str,STR_ASSOC))
{
rb_str_modify(str);
REALLOC_N(RSTRING(str)->ptr,char,RSTRING(str)->len+len);
memcpy(RSTRING(str)->ptr+RSTRING(str)->len,ptr,len);
RSTRING(str)->len+=len;
RSTRING(str)->ptr[RSTRING(str)->len]='"0';
/*sentinel*/
returnstr;
}
returnrb_str_buf_cat(str,ptr,len);
}
VALUErb_str_cat2(str,ptr)VALUEstr;
constchar*ptr;
{
returnrb_str_cat(str,ptr,strlen(ptr));
}
//string.c
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品av免费在线观看| 在线成人激情黄色| 国产精品一区二区在线| 欧美视频在线看| 欧美激情视频在线免费观看 欧美视频免费一| 欧美激情一区二区三区高清视频| 亚洲男人天堂视频| 久久人人爽人人爽爽久久| 日韩av中文字幕在线免费观看| 97精品久久久中文字幕免费| 91九色在线视频| 久久综合免费视频影院| 96精品视频在线| 亚洲韩国日本中文字幕| 亚洲天堂2020| 成人福利视频在线观看| 久久99亚洲热视| 亚洲精品福利免费在线观看| 性夜试看影院91社区| 欧美电影免费观看高清| 亚洲第一精品自拍| 成人激情综合网| 97视频在线观看网址| 亚州国产精品久久久| 久久精品91久久久久久再现| 中文字幕不卡av| 欧美—级a级欧美特级ar全黄| 亚洲一区亚洲二区| 国产成人精品免高潮费视频| 国产精品一区二区av影院萌芽| 日韩免费av一区二区| 91九色国产在线| 97精品视频在线| 欧美黄色成人网| 91精品国产91| 国产日韩欧美电影在线观看| 欧美日韩福利在线观看| 91中文在线观看| 奇米成人av国产一区二区三区| 欧美在线观看视频| 亚洲一区二区久久久久久久| 亚洲视频第一页| 成人性生交大片免费看小说| 久久精品视频一| 麻豆乱码国产一区二区三区| 亚洲石原莉奈一区二区在线观看| 欧美午夜xxx| 日韩精品极品毛片系列视频| 久久精视频免费在线久久完整在线看| 久久av在线看| 97人洗澡人人免费公开视频碰碰碰| 国产日韩欧美在线看| 精品调教chinesegay| 欧美性在线视频| 91经典在线视频| 日韩精品极品在线观看播放免费视频| 久久中国妇女中文字幕| 欧美性极品少妇精品网站| 久久国产精品影视| 国产精品九九久久久久久久| 久久91精品国产| 亚洲精品乱码久久久久久金桔影视| 日韩欧亚中文在线| 色悠悠久久88| 国产精品美女免费视频| 亚洲tv在线观看| 久久久亚洲精品视频| 精品免费在线视频| 国产精品久久久久久久app| 日本不卡免费高清视频| 久久久久久久亚洲精品| 久久国产精品网站| 538国产精品视频一区二区| 欧美资源在线观看| 国产婷婷97碰碰久久人人蜜臀| 欧美黑人xxx| 国产精品三级久久久久久电影| 国产精品无码专区在线观看| 日韩欧美国产高清91| 欧美午夜丰满在线18影院| 精品视频久久久久久久| 国产日韩在线精品av| 亚洲成人精品久久久| 欧美中文在线观看国产| 韩国欧美亚洲国产| 91久久综合亚洲鲁鲁五月天| 国产精品第一第二| 成人黄色午夜影院| 91久久久久久久久久久久久| 久久久久久网站| 91免费人成网站在线观看18| 青草青草久热精品视频在线观看| 色樱桃影院亚洲精品影院| 欧美激情综合亚洲一二区| 日韩有码视频在线| 91成人免费观看网站| 成人精品aaaa网站| 成人国产亚洲精品a区天堂华泰| 亚洲电影免费观看高清完整版在线观看| 亚洲网站在线播放| 亚洲va码欧洲m码| 欧美激情综合色综合啪啪五月| 欧美最猛性xxxxx(亚洲精品)| 欧美日韩亚洲国产一区| 日韩高清电影免费观看完整版| 91探花福利精品国产自产在线| 欧美性高跟鞋xxxxhd| 8090理伦午夜在线电影| 国产成人亚洲综合91精品| 97在线观看免费| 精品一区二区三区四区| 中文字幕不卡av| 国产在线精品播放| 欧美精品精品精品精品免费| 中文字幕在线亚洲| 久久精品国产一区二区三区| 欧美日韩亚洲天堂| 国产精品日韩在线一区| 久久乐国产精品| 久久久亚洲网站| 国产98色在线| 久久久久久久久久婷婷| 国产精品视频公开费视频| 亚洲精品v欧美精品v日韩精品| 一道本无吗dⅴd在线播放一区| 国产在线拍揄自揄视频不卡99| 欧美午夜精品久久久久久浪潮| 亚洲伊人成综合成人网| 97精品视频在线| 成人精品在线视频| 91在线观看免费网站| 欧美精品在线免费观看| 国产91色在线免费| 欧美大片大片在线播放| 久久伊人精品一区二区三区| 午夜精品久久久久久久99黑人| 国产精品久久久久久久久久久不卡| 在线亚洲男人天堂| 精品久久久久久久久久久久| 亚洲аv电影天堂网| 久久亚洲精品小早川怜子66| 91精品国产综合久久久久久久久| 一本色道久久88精品综合| 97精品国产aⅴ7777| 日韩av手机在线看| 一区二区三区回区在观看免费视频| 91精品在线观看视频| 欧美成人免费观看| 欧美中文在线观看| 欧美激情极品视频| 国产在线拍揄自揄视频不卡99| 日韩久久免费电影| 亚洲欧美福利视频| 97视频在线观看视频免费视频| 97精品在线观看| 国产一区欧美二区三区| 国产欧美在线观看| 精品偷拍各种wc美女嘘嘘| 欧美成人sm免费视频| 国产一区二区三区在线视频| 91精品视频免费观看| 久久精品国产亚洲精品| 亚洲欧美激情一区| 人人爽久久涩噜噜噜网站|