• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            糯米

            TI DaVinci, gstreamer, ffmpeg
            隨筆 - 167, 文章 - 0, 評(píng)論 - 47, 引用 - 0
            數(shù)據(jù)加載中……

            置頂隨筆

            [置頂]個(gè)人說(shuō)明


            關(guān)于寫博客的目的
            引用一句話“如果每個(gè)程序員都寫博客,中國(guó)的技術(shù)水平就不是現(xiàn)在這個(gè)樣子”。
            對(duì),這就是我寫博客的目的。我必須盡量保證博客里的每一篇文章都清晰明了,適于閱讀,能讓他人在最短時(shí)間能獲得想要的東西。
            因此我以后不會(huì)把廢話放上來(lái),保證發(fā)表的文章都是關(guān)于技術(shù),并且有利于他人。

            關(guān)于我
            豆瓣:http://www.douban.com/people/nuomihaochi/

            posted @ 2011-08-24 09:55 糯米 閱讀(363) | 評(píng)論 (1)編輯 收藏

            2012年3月25日

            TI8168EVM 調(diào)試總結(jié)


            硬件:
            是TI原廠的EVM8168開發(fā)板,公司買的,價(jià)格貌似近1w,國(guó)內(nèi)貌似還沒有人做開發(fā)板出來(lái)賣啊。(但貌似已經(jīng)有產(chǎn)品使用它了)

            單路解碼:
            將hdmi線插入四個(gè)音頻接口旁邊的那個(gè)hdmi口。
            終端執(zhí)行命令:runDecodeDisplayHDMI
            代碼位置:/usr/bin/runDecodeDisplayHDMI

            posted @ 2012-03-25 09:14 糯米 閱讀(635) | 評(píng)論 (0)編輯 收藏

            2012年2月13日

            使用 VNC display 0


            用 x11vnc 軟件包代替 vnc4server 軟件包即可。
            在 ubuntu 下 apt-get install x11vnc 。
            然后 x11vnc :0 即可。

            posted @ 2012-02-13 09:29 糯米 閱讀(648) | 評(píng)論 (0)編輯 收藏

            2012年2月1日

            一次難忘的調(diào)試


            今天在 buildroot 上交叉編譯 gstreamer 的時(shí)候,發(fā)現(xiàn)一個(gè)詭異的問(wèn)題。
            版本:
            buildroot 2011.11-rc3
            gstreamer-0.10.35
            arm-none-linux-gnueabi-gcc-4.5.2

            編譯出來(lái)之后調(diào)用 gst-inspect shmsrc,發(fā)現(xiàn)信息沒有打印完就出現(xiàn) Abort 錯(cuò)誤。
            而 gst-inspect mpegtsmux 或者 v4l2src 還有其他的組件也有類似的錯(cuò)誤。
            如下所示:


            這可能是收到了一個(gè) Abort 信號(hào)導(dǎo)致退出的,而不是程序正常結(jié)束。
            于是在 buildroot 上選中 gdb 并且選擇帶調(diào)試信息編譯(-g3)。
            用 gdb 調(diào)試的時(shí)候,定位到出錯(cuò)位置,堆棧打印如下:



            可見它是在 print_element_info 內(nèi)發(fā)生的錯(cuò)誤,而上面打印的信息也是在 “Element Implementation:” 這一句話后終止的。
            接下來(lái)的 g_strdup_printf 以及一系列的 g 打頭的函數(shù)都是 glib 里面的代碼。
            奇怪了,這類常用的函數(shù)怎么可能會(huì)出錯(cuò)呢?
            歷經(jīng)周折,最后發(fā)現(xiàn)最底層調(diào)用的是 sprintf 函數(shù)。


            而在 sprintf 的斷點(diǎn)處,確發(fā)現(xiàn)函數(shù)的參數(shù)都是正常的。
            真是十分詭異的現(xiàn)象。


            而 lib 的 sprintf 已經(jīng)沒有源碼了,于是用 si 逐指令跟蹤。按了半分多種忽然有重大發(fā)現(xiàn)!
            一開始一直在 vfprintf 里面,然后跳轉(zhuǎn)到一個(gè)底層的 __parse_one_specmb 函數(shù)。
            居然瞬間又跳轉(zhuǎn)到一個(gè) gstreamer 中的函數(shù) _gst_info_printf_extension_arginfo!


            這個(gè)函數(shù)究竟是何方神圣,看起來(lái)像是一個(gè) printf 擴(kuò)展類的回調(diào)函數(shù)。
            發(fā)現(xiàn)了真身,這到底是什么東西?

            搜了一下,找到了 GNU 的官網(wǎng),發(fā)現(xiàn)是 printf 的擴(kuò)展功能。是 glibc 2.0 版本以后的新東西。

            發(fā)現(xiàn)它是可有可無(wú)的。
            在 configure 里注釋掉它,在 buildroot 的 package 里加了一個(gè) patch,然后重新 make。



            一切正常了!哈哈!


            最后發(fā)現(xiàn)早在2011年1月份就有老外發(fā)現(xiàn)了這個(gè)問(wèn)題。https://bugzilla.gnome.org/show_bug.cgi?id=638599
            貌似是跟 glib 的一些機(jī)制相關(guān)。
            可能是影響范圍不夠廣,還沒有導(dǎo)致到要取消掉這個(gè)機(jī)制。



            雖然說(shuō)為了一個(gè)莫名其妙的問(wèn)題忙活了半天,但學(xué)了一些 gdb 命令,而且還知道了 printf 原來(lái)還能擴(kuò)展。很神奇的東西。

            posted @ 2012-02-01 12:43 糯米 閱讀(726) | 評(píng)論 (0)編輯 收藏

            2011年9月14日

            str_fastcmp

            摘自:luajit源碼lj_str.c

            /* Fast string data comparison. Caveat: unaligned access to 1st string! */
            static LJ_AINLINE int str_fastcmp(const char *a, const char *b, MSize len)
            {
              MSize i 
            = 0;
              lua_assert(len 
            > 0);
              lua_assert((((uintptr_t)a 
            + len) & (LJ_PAGESIZE-1)) <= LJ_PAGESIZE-4);
              
            do {  /* Note: innocuous access up to end of string + 3. */
                uint32_t v 
            = lj_getu32(a+i) ^ *(const uint32_t *)(b+i);
                
            if (v) {
                  i 
            -= len;
            #if LJ_LE
                  
            return (int32_t)i >= -3 ? (v << (32+(i<<3))) : 1;
            #else
                  
            return (int32_t)i >= -3 ? (v >> (32+(i<<3))) : 1;
            #endif
                }
                i 
            += 4;
              } 
            while (i < len);
              
            return 0;
            }

            posted @ 2011-09-14 22:00 糯米 閱讀(438) | 評(píng)論 (0)編輯 收藏

            2011年9月13日

            標(biāo)記-清除( Mark-Sweep )算法

            第一種實(shí)用和完善的垃圾收集算法是 J. McCarthy 等人在 1960 年提出并成功地應(yīng)用于 Lisp 語(yǔ)言的標(biāo)記-清除算法。仍以餐巾紙為例,標(biāo)記-清除算法的執(zhí)行過(guò)程是這樣的:  

            午餐過(guò)程中,餐廳里的所有人都根據(jù)自己的需要取用餐巾紙。當(dāng)垃圾收集機(jī)器人想收集廢舊餐巾紙的時(shí)候,它會(huì)讓所有用餐的人先停下來(lái),然后,依次詢問(wèn)餐廳里的每一個(gè)人:“你正在用餐巾紙嗎?你用的是哪一張餐巾紙?”機(jī)器人根據(jù)每個(gè)人的回答將人們正在使用的餐巾紙畫上記號(hào)。詢問(wèn)過(guò)程結(jié)束后,機(jī)器人在餐廳里尋找所有散落在餐桌上且沒有記號(hào)的餐巾紙(這些顯然都是用過(guò)的廢舊餐巾紙),把它們統(tǒng)統(tǒng)扔到垃圾箱里。  

            正如其名稱所暗示的那樣,標(biāo)記-清除算法的執(zhí)行過(guò)程分為“標(biāo)記”和“清除”兩大階段。這種分步執(zhí)行的思路奠定了現(xiàn)代垃圾收集算法的思想基礎(chǔ)。與引用計(jì)數(shù)算法不同的是,標(biāo)記-清除算法不需要運(yùn)行環(huán)境監(jiān)測(cè)每一次內(nèi)存分配和指針操作,而只要在“標(biāo)記”階段中跟蹤每一個(gè)指針變量的指向——用類似思路實(shí)現(xiàn)的垃圾收集器也常被后人統(tǒng)稱為跟蹤收集器( Tracing Collector )  

            伴隨著 Lisp 語(yǔ)言的成功,標(biāo)記-清除算法也在大多數(shù)早期的 Lisp 運(yùn)行環(huán)境中大放異彩。盡管最初版本的標(biāo)記-清除算法在今天看來(lái)還存在效率不高(標(biāo)記和清除是兩個(gè)相當(dāng)耗時(shí)的過(guò)程)等諸多缺陷,但在后面的討論中,我們可以看到,幾乎所有現(xiàn)代垃圾收集算法都是標(biāo)記-清除思想的延續(xù),僅此一點(diǎn), J. McCarthy 等人在垃圾收集技術(shù)方面的貢獻(xiàn)就絲毫不亞于他們?cè)?nbsp;Lisp 語(yǔ)言上的成就了。  

            posted @ 2011-09-13 10:34 糯米 閱讀(1861) | 評(píng)論 (0)編輯 收藏

            2011年8月27日

            去除dvsdk安裝的Ubuntu版本限制

            今天偶然間發(fā)現(xiàn)的一個(gè)小hack:)

            我要安裝dvsdk的4.0版本。它的安裝包是一個(gè)單個(gè)的可執(zhí)行文件:


            它提示,只能在Ubuntu 10.04上面安裝。


            我自己的Ubuntu是11.04,文件系統(tǒng)resierfs,如果resize的話貌似會(huì)清空。當(dāng)然也不愿意再裝一個(gè)舊版的。
            所以琢磨著怎么把這個(gè)限制去掉。
            首先我估計(jì)它是讀了某個(gè)文件來(lái)獲取的發(fā)行版本號(hào)。
            所以先用 strace 命令看一下。加一個(gè) -f 表示跟蹤 fork 出來(lái)的子進(jìn)程。
            這個(gè)工具可以打印出程序所有的 syscall。因此 open,read,write,select 之類的調(diào)用都會(huì)被打印出來(lái)。



            這時(shí)候出來(lái)很多結(jié)果,好比大海撈針一般,要仔細(xì)看哦。
            突然發(fā)現(xiàn)一行顯眼的:


            “natty”是Ubuntu 11.04的代號(hào),也就是說(shuō)這個(gè)程序在某個(gè)地方讀到了Ubuntu的版本號(hào)。
            我對(duì)這些不了解,于是google了一下,發(fā)現(xiàn)是通過(guò)調(diào)用 lsb_release 這個(gè)工具來(lái)讀到的。


            然后又發(fā)現(xiàn)一個(gè)好消息,這個(gè)工具是 python 寫的。
            于是加了半句優(yōu)雅的代碼:


            其中“lucid”是 Ubuntu 10.04 的版本號(hào)。如果 /tmp/lucidhack 這個(gè)文件存在,就輸出 lucid 。
            看看效果吧:



            創(chuàng)建 /tmp/lucidhack,然后重新運(yùn)行安裝程序。
            大功告成!




            其實(shí)很多時(shí)候問(wèn)題很簡(jiǎn)單,完全沒有必要大動(dòng)干戈。

            posted @ 2011-08-27 09:43 糯米 閱讀(2011) | 評(píng)論 (1)編輯 收藏

            2011年8月26日

            WTFPL - Do What The Fuck You Want To Public License


            翻譯自:http://en.wikipedia.org/wiki/WTFPL

            WTFPL (Do What The Fuck You Want To Public License) 是一個(gè)很少使用的,極端自由的自由軟件協(xié)議。
            WTFPL 最初的 1.0 版本許可證發(fā)布于 2000 年 3 月份。發(fā)布者是 Banlu Kemiyatorn,最初用于 WindowMaker 軟件。
            Samuel “Sam” Hocevar,2007 年和 2008 年間的 Debian 項(xiàng)目負(fù)責(zé)人,發(fā)布了 WTFPL 的 2.0 版本。
            WTFPL 允許代碼在任何協(xié)議下都被能任意修改和發(fā)布 --- 也就是“do what the fuck [they] want to do”(做你他媽想做的)。
            WTFPL 被自由軟件組織認(rèn)可為 GPL 兼容。

            協(xié)議內(nèi)容
                       DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
                               Version 2, December 2004
            Copyright (C) 2004 Sam Hocevar <sam@hocevar.net>
            Everyone is permitted to copy and distribute verbatim or modified
            copies of this license document, and changing it is allowed as long
            as the name is changed.
                       DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
              TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
             0. You just DO WHAT THE FUCK YOU WANT TO.
            

            posted @ 2011-08-26 18:56 糯米 閱讀(2056) | 評(píng)論 (0)編輯 收藏

            perl 常用函數(shù)


            摘自 perldoc.perl.org

            map BLOCK LIST
            map EXPR, LIST
            對(duì)于 LIST 里的每一個(gè)元素按照 BLOCK 或者 EXPR 里的操作進(jìn)行求值。
            返回求值結(jié)果組成的數(shù)組。
            如果返回值是 scalar 類型,則返回結(jié)果數(shù)組的大小。
            例子:
            返回一組數(shù)字對(duì)應(yīng)的字符
            @chars = map(chr, @numbers);
            
            返回一組數(shù)字的平方
            my @squares = map { $_ > 5 ? ($_ * $_) : () } @numbers;
            
            返回大于5的數(shù)字的平方
            my @squares = map { $_ > 5 ? ($_ * $_) : () } @numbers;
            
            由于 map 總是返回一個(gè)列表,因此可以賦值給哈希類型的變量:
            %hash = map { get_a_key_for($_) => $_ } @array;
            
            也可以這樣寫:
            %hash = ();
            foreach (@array) {
                $hash{get_a_key_for($_)} = $_;
            }
            

            -X FILEHANDLE
            文件測(cè)試函數(shù):
            -r  File is readable by effective uid/gid.
            -w  File is writable by effective uid/gid.
            -x  File is executable by effective uid/gid.
            -o  File is owned by effective uid.
            -R  File is readable by real uid/gid.
            -W  File is writable by real uid/gid.
            -X  File is executable by real uid/gid.
            -O  File is owned by real uid.
            -e  File exists.
            -z  File has zero size (is empty).
            -s  File has nonzero size (returns size in bytes).
            -f  File is a plain file.
            -d  File is a directory.
            -l  File is a symbolic link.
            -p  File is a named pipe (FIFO), or Filehandle is a pipe.
            -S  File is a socket.
            -b  File is a block special file.
            -c  File is a character special file.
            -t  Filehandle is opened to a tty.
            -u  File has setuid bit set.
            -g  File has setgid bit set.
            -k  File has sticky bit set.
            -T  File is an ASCII text file (heuristic guess).
            -B  File is a "binary" file (opposite of -T).
            -M  Script start time minus file modification time, in days.
            -A  Same for access time.
            -C  Same for inode change time (Unix, may differ for other platforms)
            

            delete EXPR
            EXPR 為哈希變量的 slice 或者是單個(gè)元素。
            返回值為刪除掉元素的值,可以是列表。
            如果返回值被要求為 scalar 類型,則返回被刪除的最后一個(gè)值。
            delete 也可以用于數(shù)組類型的變量,不過(guò)它的行為可能不是你所預(yù)想的那樣。
            例子:
            %hash = (foo => 11, bar => 22, baz => 33);
            $scalar = delete $hash{foo};             # $scalar is 11
            $scalar = delete @hash{qw(foo bar)};     # $scalar is 22
            @array  = delete @hash{qw(foo bar baz)}; # @array  is (undef,undef,33)
            

            each HASH
            each ARRAY
            each EXPR
            返回哈希的每個(gè) (key, value) 所組成的數(shù)組。
            例子:
            while (($key, $value) = each %hash) {
                print $key, "\n";
                delete $hash{$key};   # This is safe
            }
            

            eof FILEHANDLE
            eof ()
            eof
            注意 eof() 與 eof 的區(qū)別:
            eof():如果到達(dá)了 <> 的最后一個(gè)文件的末尾,則返回1
            eof:如果到達(dá)了當(dāng)前文件的末尾,則返回1
            # reset line numbering on each input file
            while (<>) {
                next if /^\s*#/;  # skip comments
                print "$.\t$_";
            } continue {
                close ARGV if eof;  # Not eof()!
            }
            # insert dashes just before last line of last file
            while (<>) {
                if (eof()) {  # check for end of last file
                    print "--------------\n";
                }
                print;
                last if eof();          # needed if we're reading from a terminal
            }
            


            posted @ 2011-08-26 18:25 糯米 閱讀(604) | 評(píng)論 (0)編輯 收藏

            2011年8月24日

            perl 特殊變量

            翻譯自:perldoc.perl.org

            == 常用變量 ==

            $ARG
            $_
            默認(rèn)模式搜索變量
            下列函數(shù)使用它作為默認(rèn)參數(shù):
            abs, alarm, chomp, chop, chr, chroot, cos, defined, eval, exp, glob, hex, int, lc, lcfirst,
            length, log, lstat, mkdir, oct, ord, pos, print, quotemeta, readlink, readpipe, ref, require,
            reverse (in scalar context only), rmdir, sin, split (on its second argument), sqrt, stat,
            study, uc, ucfirst, unlink, unpack.
            用于文件測(cè)試 (-f, -d)
            用于模式匹配操作 m//, s/// 和 tr// (又名 y///)
            用于 foreach, grep, map
            用于 given 語(yǔ)句

            @ARG
            @_
            對(duì)于函數(shù)來(lái)說(shuō),@_ 包含了傳遞給該函數(shù)的參數(shù)。在函數(shù)中,@_ 是 push, pop, shift, unshift 的默認(rèn)參數(shù)。

            $LIST_SEPARATOR
            $"
            當(dāng)一個(gè)數(shù)組被引用在 "" 中或者 // 中。數(shù)組的值變?yōu)樗性乇?$" 連接起來(lái)的值。
            $" 默認(rèn)值為空格。
            print "The array is: @array\n";
            print "The array is: " . join($", @array) . "\n";
            
            它們是相等的。
            @a = (1, 3, 4);
            print "@a\n";
            $" = ",";
            print "@a\n";
            print "1,3,4" =~ /@a/;
            
            輸出為
            1 3 4
            1,3,4
            1
            

            $PROCESS_ID
            $PID
            $$
            運(yùn)行 perl 腳本的當(dāng)前進(jìn)程的 pid 值。等同于 getpid()。

            $REAL_GROUP_ID
            $GID
            $(
            該進(jìn)程的 real gid 。由 getgid() 獲得。
            如果運(yùn)行在一個(gè)支持用戶同時(shí)屬于多個(gè)組的操作系統(tǒng)上,$( 將返回由空格分隔的多個(gè) gid。由 getgroups() 獲得。
            只能給 $( 賦予單個(gè)整數(shù)。

            $EFFECTIVE_GROUP_ID
            $EGID
            $)
            該進(jìn)程的 effective gid。
            $< = $>;            # set real to effective uid
            ($<,$>) = ($>,$<);  # swap real and effective uids
            

            $PROGRAM_NAME
            $0
            當(dāng)前程序的名字。

            $SUBSCRIPT_SEPARATOR
            $SUBSEP
            $;
            如果這樣訪問(wèn)一個(gè)哈系表:
            $foo{$a,$b,$c}
            
            等同于:
            $foo{join($;, $a, $b, $c)}
            
            例子:
            %h = ('123', 'test');
            $; = '';
            print $h{1,2,3}; # 輸出 test
            

            %ENV
            存放環(huán)境變量哈希表。

            @F
            當(dāng)指定了 -a 參數(shù)的時(shí)候,@F 中保存著每一行被分隔后的數(shù)組:
            perl -ane 'print pop(@F), "\n";'
            
            等同于
            while (<>) {
                @F = split(' ');
                print pop(@F), "\n";
            }
            

            $OSNAME
            $^O
            當(dāng)前操作系統(tǒng)的名字。
            # perl -e "print $^O"
            # linux
            

            %SIG
            哈希 %SIG 中保存了信號(hào)的處理函數(shù)。例子如下:
            sub handler {   # 1st argument is signal name
                my($sig) = @_;
                print "Caught a SIG$sig--shutting down\n";
                close(LOG);
                exit(0);
                }
            $SIG{'INT'}  = \&handler;
            $SIG{'QUIT'} = \&handler;
            ...
            $SIG{'INT'}  = 'DEFAULT';   # restore default action
            $SIG{'QUIT'} = 'IGNORE';    # ignore SIGQUIT
            
            賦值為 'IGNORE' 則忽略該信號(hào),除了 CHLD 信號(hào)。
            下面是指定處理函數(shù)的其他方式:
            $SIG{"PIPE"} = "Plumber";   # 指定為 main::Plumber (不推薦)
            $SIG{"PIPE"} = \&Plumber;   # 推薦的方式
            $SIG{"PIPE"} = *Plumber;    # 有點(diǎn)奇怪
            $SIG{"PIPE"} = Plumber();   # 錯(cuò)誤的做法,Plumber() 會(huì)返回什么??
            
            部分內(nèi)部的 hook 也可以通過(guò)設(shè)置 SIG 來(lái)指定。比如說(shuō):
            local $SIG{__WARN__} = sub { die $_[0] };
            eval $proggie;
            
            當(dāng)一個(gè) warning 被拋出的時(shí)候,$SIG{__WARN__} 所指向的函數(shù)將被調(diào)用。
            warning 消息的內(nèi)容為函數(shù)的第一個(gè)參數(shù)。

            $BASETIME
            $^T
            程序開始運(yùn)行的時(shí)間(時(shí)間戳)
            # perl -e "print $^T"
            1314234806
            

            $PERL_VERSION
            $^V
            perl 的版本號(hào)
            warn "Hashes not randomized!\n" if !$^V or $^V lt v5.8.1
            

            == 和正則表達(dá)式相關(guān)的變量 ==

            $<digits> ($1, $2, ...)
            匹配中 () 匹配的部分。

            $&
            $MATCH
            前一次成功匹配的字符串。

            $`
            $PREMATCH
            位于前一次成功匹配字符串前面的字符串。

            $'
            $POSTMATCH
            位于前一次成功匹配字符串后面的字符串。

            local $_ = 'abcdefghi';
            /def/;
            print "$`:$&:$'\n";      # 輸出 abc:def:ghi
            

            $LAST_PAREN_MATCH
            $+
            前一次成功匹配中最后一個(gè) () 中的內(nèi)容。
            /Version: (.*)|Revision: (.*)/ && ($rev = $+);
            

            @LAST_MATCH_END
            @+
            前一次成功匹配中每個(gè) () 匹配的偏移量。
            $+[0] 為整個(gè)匹配末尾的偏移量
            $+[1] 為 $1 末尾的偏移量
            $+[2] 為 $2 末尾的偏移量
            ...
            $#+ 為前一次成功匹配的 () 數(shù)量

            %LAST_PAREN_MATCH
            %+
            前一次成功匹配中命名匹配(即為 (?<name>...) 的形式)的哈系表。
            如:
            'foo' =~ /(?<foo>foo)/; # $+{foo} 和 $1 中都存放著 'foo'
            

            另外一個(gè)例子:
            $_ = "testing1234end";
            /^testing(.)(.)(..)end/;
            print "@+"; # 輸出 14 8 9 11
            /^(?<foo>[a-z]+)(?<num>\d+)/;
            print "$+{foo}"; # 輸出 testing
            print "$+{num}"; # 輸出 1234
            

            @LAST_MATCH_START
            @-
            和 @+ 類似:
            $-[0] 為整個(gè)匹配頭部的偏移量
            $-[1] 為 $1 頭部的偏移量
            $-[2] 為 $2 頭部的偏移量
            ...
            $` 等同于 substr($var, 0, $-[0])
            $& 等同于 substr($var, $-[0], $+[0] - $-[0])
            $' 等同于 substr($var, $+[0])
            $1 等同于 substr($var, $-[1], $+[1] - $-[1])
            $2 等同于 substr($var, $-[2], $+[2] - $-[2])
            $3 等同于 substr($var, $-[3], $+[3] - $-[3])

            %LAST_MATCH_START
            %-
            類似于 %+,通過(guò) %- 可以訪問(wèn)到上一次成功匹配中所有的命名匹配。
            每個(gè)命名匹配的名字都與一個(gè)數(shù)組相關(guān)聯(lián)。
            比如說(shuō):
            if ('1234' =~ /(?<A>1)(?<B>2)(?<A>3)(?<B>4)/) {
                foreach my $bufname (sort keys %-) {
                    my $ary = $-{$bufname};
                    foreach my $idx (0..$#$ary) {
                        print "\$-{$bufname}[$idx] : ",
                              (defined($ary->[$idx]) ? "'$ary->[$idx]'" : "undef"),
                              "\n";
                    }
                }
            }
            
            會(huì)輸出:
            $-{A}[0] : '1'
            $-{A}[1] : '3'
            $-{B}[0] : '2'
            $-{B}[1] : '4'
            

            == 和文件句柄相關(guān)的變量 ==

            $ARGV
            <> 中正在被讀取的文件的名字
            例子:
            # cat c
            1
            2
            # cat d
            1
            2
            3
            4
            # cat a.pl 
            while (<>) {
                print "$ARGV\n";
            }
            # perl a.pl c d
            c
            c
            d
            d
            d
            d
            

            @ARGV
            命令行參數(shù),$ARGV[0] 代表第一個(gè)參數(shù)(而不是程序的名字)。

            $OUTPUT_FIELD_SEPARATOR
            $OFS
            $,
            print 函數(shù)的輸出分隔符,默認(rèn)值為 undef。
            例子:
            print "a", "c"; # 輸出 ac
            $, = "|";
            print "a", "c"; # 輸出 a|c
            

            $INPUT_LINE_NUMBER
            $NR
            $.
            上一次讀取文件的行號(hào)。
            $. 在文件句柄關(guān)閉的時(shí)候重置。
            由于 <> 操作符不會(huì)關(guān)閉文件,因此在連續(xù)從 ARGV 讀取文件的時(shí)候,$. 會(huì)不斷增加。
            例子:
            # cat a.pl
            while (<>) {
                print "$.\n";
            }
            # perl a.pl c d
            1
            2
            3
            4
            5
            6
            
            其中 c d 的長(zhǎng)度分別為 2, 4 行。

            $INPUT_RECORD_SEPARATOR
            $RS
            $/
            類似于 awk 的 RS 變量。它決定了 perl 如何分割行。
            例子:
            # cat c
            1|2|3
            # cat a.pl 
            $/ = "|";
            while (<>) {
                print "line:$_\n";
            }
            # perl a.pl c
            line:1|
            line:2|
            line:3
            

            $OUTPUT_RECORD_SEPARATOR
            $ORS
            $\
            print 函數(shù)輸出的分隔符。默認(rèn)值為 undef,如果定義了,則在 print 完所有參數(shù)后輸出 $\。
            例子:
            $\ = "|";
            print "a";
            print "b";
            # 將輸出 a|b|
            

            $OUTPUT_AUTOFLUSH
            $|
            如果設(shè)置為非0的值,將強(qiáng)制的 flush 當(dāng)前 selected 的文件句柄。
            (類似于 setbuf(xx, NULL) ?)

            == 和錯(cuò)誤信息有關(guān)的變量 ==

            $ERRNO
            $!
            等同于 libc 中的 errno??梢赞D(zhuǎn)換為 int 和 string 兩種形式:
            open '/a';
            print int($!) . ": $!\n"; # 輸出 2: 沒有那個(gè)文件或目錄
            

            $CHILD_ERROR
            $?
            類似于 shell 中的 $???梢詾橄铝胁僮鞯姆祷刂担?/br> - 最后一次管道關(guān)閉
            - `` 或者 system() 語(yǔ)句
            - wait() 或者 waitpid()
            $? >> 8 為子進(jìn)程的返回值

            $EVAL_ERROR
            $@
            上一次 eval() 操作的語(yǔ)法錯(cuò)誤信息。
            例子:
            # cat a.pl
            eval "my a";
            print "$@";
            # perl a.pl
            No such class a at (eval 1) line 1, near "my a"
            syntax error at (eval 1) line 2, at EOF
            

            posted @ 2011-08-24 23:45 糯米 閱讀(865) | 評(píng)論 (0)編輯 收藏

            perl 范圍操作符

            摘自:perldoc.perl.org

            當(dāng)".."操作符返回bool型變量的時(shí)候,有如下幾個(gè)例子:
            while (<>) {
                print if (2 .. 6);
            }
            
            將會(huì)打印文件的 2 ~ 6 行。

            同樣的:
            next if (1 .. /^$/); # 跳過(guò)前面的空白行
            s/^/> / if (/^$/ .. eof); # 在正文前面加上 > 
            

            還可以這樣使用:
            # parse mail messages
            while (<>) {
                $in_header =   1  .. /^$/;
                $in_body   = /^$/ .. eof;
                if ($in_header) {
                    # do something
                } else { # in body
                    # do something else
                }
            } 
            

            下面的例子展示了".."和"..."的區(qū)別:
            @lines = ("   - Foo",
                      "01 - Bar",
                      "1  - Baz",
                      "   - Quux");
            foreach (@lines) {
                if (/0/ .. /1/) {
                    print "$_\n";
                }
            }
            
            這個(gè)程序只輸出Bar那行。如果將".."換成"...",則還會(huì)輸出Baz那行。

            下面展示".."操作符返回列表型變量時(shí)候的幾個(gè)例子:
            for (101 .. 200) { print; }    # print $_ 100 times
            @foo = @foo[0 .. $#foo];    # an expensive no-op
            @foo = @foo[$#foo-4 .. $#foo];    # slice last 5 items
            

            同樣可以用于字符串:
            @alphabet = ("A" .. "Z");
            $hexdigit = (0 .. 9, "a" .. "f")[$num & 15];
            @z2 = ("01" .. "31");  print $z2[$mday];
            

            還可以用于單個(gè)字符:
            use charnames "greek";
            my @greek_small =  ("\N{alpha}" .. "\N{omega}");
            

            use charnames "greek";
            my @greek_small =  map { chr } ord "\N{alpha}" .. ord "\N{omega}";
            

            posted @ 2011-08-24 17:00 糯米 閱讀(725) | 評(píng)論 (0)編輯 收藏

            亚洲欧美日韩久久精品| 久久国产精品波多野结衣AV| 一本久久a久久精品综合香蕉 | 久久久久国产精品| 婷婷综合久久狠狠色99h| 久久久久综合国产欧美一区二区 | 99热精品久久只有精品| 亚洲国产精品无码久久青草| 亚洲精品高清国产一线久久| 国内精品久久久久久麻豆| 中文字幕日本人妻久久久免费 | 国内精品久久人妻互换| 久久精品国产一区二区三区| 精品熟女少妇a∨免费久久| 久久综合五月丁香久久激情| 精品久久久久久中文字幕| 一本久久知道综合久久| 久久久久亚洲AV无码专区网站| 久久夜色精品国产欧美乱| 日本亚洲色大成网站WWW久久| 99re久久精品国产首页2020| 18岁日韩内射颜射午夜久久成人| 久久人人爽人人爽人人片AV东京热| 久久精品国产91久久综合麻豆自制| 久久精品国产久精国产一老狼| 久久er国产精品免费观看8| 国产成人精品久久二区二区| 久久久精品国产sm调教网站| 伊人久久综合成人网| 国产精品久久久久a影院| 亚洲日韩欧美一区久久久久我| 国产精品久久久天天影视香蕉| 丁香五月网久久综合| 国产精品女同久久久久电影院| 久久久久亚洲精品无码蜜桃| 久久久一本精品99久久精品88| 一本久道久久综合狠狠爱| 伊人久久大香线蕉综合Av| 欧美噜噜久久久XXX| 99久久精品毛片免费播放| 久久精品9988|