成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

正確理解Ruby String用法

開發 開發工具
Ruby String不向JAVA和C#中的String一樣,它是操縱同一個對象。其中參數len是Ruby String的長度,而ptr則是一個char類型的指針。

Ruby語言是一個運用靈活的編程語言,通常都會被人們理解為一款解釋型腳本語言。在接下來的文章中,我們將會了解到一些關于Ruby String的相關知識。#t#

Ruby語言中的String是mutable的,不像java、C#中的String是immutable的。比如

 

  1. str1="abc" 
  2. str2="abc" 

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

 

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

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

 

  1. struct RString {  
  2. struct RBasic basic;  
  3. long len;  
  4. char *ptr;  
  5. union {  
  6. long capa;  
  7. VALUE shared;  
  8. } aux;  
  9. };  
  10. //ruby.h 

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

 

  1. struct RBasic {   
  2. unsigned long flags;   
  3. VALUE klass;   
  4. }; 

其中的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其實是用于加快字符串的創建速度,在一個循環中。

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

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

對應的c語言代碼就是:

  1. #include  
  2. #include"ruby.h"staticVALUEt_test(VALUEself){  
  3. VALUEstr;str=rb_str_new2("str");  
  4. printf("beforeconcat:str:%p,  
  5. str.aux.shared:%p,str.ptr:%s"n",str,
    (RSTRING(str)-
    >aux).shared,RSTRING(str)->ptr);  
  6. rb_str_cat2(str,"ing");  
  7. printf("afterconcat:str:%p,str.aux.
    shared:%p,str.ptr:%s"n",  
  8. str,(RSTRING(str)->aux).shared,RSTRING
    (str)-
    >ptr);returnself;  
  9. }  
  10. VALUEcTest;  
  11. voidInit_string_hack(){  
  12. cTest=rb_define_class("Test",rb_cObject);  
  13. rb_define_method(cTest,"test",t_test,0);  
  14. }//string_hack.c 

在t_test中,通過rb_str_new2每次生成一個RString結構,然后通過rb_str_cat2將str與"ing"連接起來,添加
了一些打印用于跟蹤。利用mkmf產生Makefile,寫一個extconf.rb

require'mkmf'create_makefile("string_hack");

執行ruby extconf.rb,將產生一個Makefile,執行make,生成一個string_hack.so的鏈接庫。擴展寫完了,通過
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函數的實現就一目了然了:

  1. VALUErb_str_cat(str,ptr,len)VALUEstr;  
  2. constchar*ptr;  
  3. longlen;  
  4. {  
  5. if(len<0){rb_raise(rb_eArgError,
    "negativestringsize(orsizetoobig)");  
  6. }  
  7. if(FL_TEST(str,STR_ASSOC))  
  8. {  
  9. rb_str_modify(str);  
  10. REALLOC_N(RSTRING(str)->ptr,char,
    RSTRING(str)-
    >len+len);  
  11. memcpy(RSTRING(str)->ptr+RSTRING
    (str)-
    >len,ptr,len);  
  12. RSTRING(str)->len+=len;  
  13. RSTRING(str)->ptr[RSTRING(str)->len]='"0';  
  14. /*sentinel*/  
  15. returnstr;  
  16. }  
  17. returnrb_str_buf_cat(str,ptr,len);  
  18. }  
  19. VALUErb_str_cat2(str,ptr)VALUEstr;  
  20. constchar*ptr;  
  21. {  
  22. returnrb_str_cat(str,ptr,strlen(ptr));  
  23. }  
  24. //string.c 

 

責任編輯:曹凱 來源: 68design.net
相關推薦

2009-12-16 10:33:31

Ruby更新文件

2009-12-17 11:36:55

Ruby輸入輸出

2009-12-16 17:00:43

Ruby on Rai

2009-06-18 10:29:24

Hibernate I

2020-01-07 15:10:32

Linuxinode命令

2009-12-04 18:00:46

PHP開發MVC模型

2010-07-20 12:35:33

SQL Server索

2010-02-01 10:54:37

C++框架

2024-01-29 00:35:00

Go并發開發

2023-12-27 19:52:08

Go模塊命令

2010-02-04 15:05:00

C++ cpuid指令

2010-01-18 17:29:35

VB.NET函數調用

2009-12-09 14:04:45

PHP include

2009-12-03 18:07:47

PHP轉義

2009-12-07 14:53:13

PHP抽象類應用

2009-12-04 17:16:41

PHP析構函數

2013-08-06 10:40:38

大數據數據

2010-08-05 15:40:07

DB2快照函數

2010-02-23 17:55:24

WCF雙向通信

2010-01-05 14:01:27

JSON.NET
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品一二三区 | 看片91 | 国产农村妇女毛片精品久久麻豆 | 91精品国产一区二区在线观看 | 蜜桃av一区二区三区 | 激情欧美一区二区三区中文字幕 | 欧美精品一区二区三区四区 在线 | 国产成人av电影 | 精品久久久久国产免费第一页 | 人人干人人舔 | www国产成人 | 91久久久久久久 | 国产一二三区精品视频 | 久久小视频 | 久久99国产精品 | 超碰导航| 久久久女女女女999久久 | 免费午夜视频 | 欧美日韩国产三级 | 国产91在线 | 亚洲 | 欧美日韩精品一区 | 一区二区免费看 | 国产精品久久久亚洲 | 免费在线播放黄色 | 欧美激情亚洲天堂 | 久久91av| 午夜二区 | 亚洲国产精品久久久 | 一区免费观看 | 999久久久| 日本在线免费看最新的电影 | 91麻豆产精品久久久久久 | 欧美成人一区二区 | 精品1区2区3区4区 | 亚洲成人黄色 | 久久精品国产精品青草 | 雨宫琴音一区二区在线 | 国产精品久久久久久久免费观看 | 欧美午夜精品久久久久久浪潮 | 久草网址| 欧美国产一区二区 |