• <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>

            舞影凌亂

            專注C/C++,Linux

            統(tǒng)計(jì)

            留言簿(1)

            積分與排名

            Linux好站點(diǎn)

            Oracle

            閱讀排行榜

            評論排行榜

            2010年8月15日 #

            一個(gè)非常有用的Linux命令--cut

                 摘要:   閱讀全文

            posted @ 2010-08-15 01:11 幽夢新影 閱讀(1814) | 評論 (1)編輯 收藏

            CentOS常用命令(收藏)

            在電腦常用操作中,我們經(jīng)常會用到CentOS常用命令。所以,我們對一些經(jīng)常使用又很重要的CentOS常用命令進(jìn)行了全面的整理。下面,就給大家介紹這些CentOS常用命令。

            一:使用CentOS常用命令查看cpu

            more /proc/cpuinfo | grep "model name"  
            grep "model name" /proc/cpuinfo  
            [root@localhost /]# grep "CPU" /proc/cpuinfo  
            model name      : Intel(R) Pentium(R) Dual CPU E2180 @ 2.00GHz  
            model name      : Intel(R) Pentium(R) Dual CPU E2180 @ 2.00GHz  
            如果覺得需要看的更加舒服
            grep "model name" /proc/cpuinfo | cut -f2 -d:

            二:使用CentOS常用命令查看內(nèi)存

            grep MemTotal /proc/meminfo  grep MemTotal /proc/meminfo | cut -f2 -d:  free -m |grep "Mem" | awk '{print $2}'

            三:使用CentOS常用命令查看cpu是32位還是64位

            查看CPU位數(shù)(32 or 64)
            getconf LONG_BIT

            四:使用CentOS常用命令查看當(dāng)前l(fā)inux的版本

            more /etc/redhat-release
            cat /etc/redhat-release

            五:使用CentOS常用命令查看內(nèi)核版本

            uname -r
            uname -a

            六:使用CentOS常用命令查看當(dāng)前時(shí)間

            date上面已經(jīng)介紹如何同步時(shí)間了

            七:使用CentOS常用命令查看硬盤和分區(qū)

            df -h
            fdisk -l
            也可以查看分區(qū)
            du -sh
            可以看到全部占用的空間
            du /etc -sh
            可以看到這個(gè)目錄的大小

            八:使用CentOS常用命令查看安裝的軟件包

            查看系統(tǒng)安裝的時(shí)候裝的軟件包
            cat -n /root/install.log
            more /root/install.log | wc -l
            查看現(xiàn)在已經(jīng)安裝了那些軟件包
            rpm -qa
            rpm -qa | wc -l
            yum list installed | wc -l
            不過很奇怪,我通過rpm,和yum這兩種方式查詢的安裝軟件包,數(shù)量并不一樣。沒有找到原因。

            九:使用CentOS常用命令查看鍵盤布局

            cat /etc/sysconfig/keyboard
            cat /etc/sysconfig/keyboard | grep KEYTABLE | cut -f2 -d=

            十:使用CentOS常用命令查看selinux情況

            sestatus
            sestatus | cut -f2 -d:
            cat /etc/sysconfig/selinux

            十一:使用CentOS常用命令查看ip,mac地址

            在ifcfg-eth0 文件里你可以看到mac,網(wǎng)關(guān)等信息。  ifconfig  cat /etc/sysconfig/network-scripts/ifcfg-eth0 | grep IPADDR  cat /etc/sysconfig/network-scripts/ifcfg-eth0 | grep IPADDR | cut -f2 -d=  ifconfig eth0 |grep "inet addr:" |awk '{print $2}'|cut -c 6-  ifconfig   | grep 'inet addr:'| grep -v '127.0.0.1' | cut -d: -f2 | awk '{ print $1}'  查看網(wǎng)關(guān)  cat /etc/sysconfig/network  查看dns  cat /etc/resolv.conf  十二:使用CentOS常用命令查看默認(rèn)語言
            echo $LANG $LANGUAGE
            cat /etc/sysconfig/i18n

            十二:使用CentOS常用命令查看所屬時(shí)區(qū)和是否使用UTC時(shí)間

            cat /etc/sysconfig/clock

            十三:使用CentOS常用命令查看主機(jī)名

            hostname
            cat /etc/sysconfig/network
            修改主機(jī)名就是修改這個(gè)文件,同時(shí)最好也把host文件也修改。

            十四:使用CentOS常用命令查看開機(jī)運(yùn)行時(shí)間

            uptime
            09:44:45 up 67 days, 23:32, ...
            看來剛才確實(shí)是網(wǎng)段的問題,我的機(jī)器還是67天前開機(jī)的。
            #系統(tǒng)資源使用情況

            vmstat 1 -S m  procs -----------memory---------- ---swap-- -----io---- --system-- -----cpu------  r b   swpd   free   buff cache   si   so    bi    bo   in   cs us sy id wa st  0 0      0    233    199    778    0    0     4    25    1    1 3 0 96 0 0  0 0      0    233    199    778    0    0     0     0 1029 856 13 1 86 0 0

            posted @ 2010-08-15 01:03 幽夢新影 閱讀(920) | 評論 (0)編輯 收藏

            2010年7月3日 #

            Oracle 10G Linux 安裝筆記

            第一部分 安裝linux

                1.  操作系統(tǒng)的安裝省略,注意配置圖形界面

                2.  驗(yàn)證安裝

                通過運(yùn)行以下命令檢查內(nèi)核版本:

                uname –r

                其他所需程序包的版本(或更高版本):

                gcc-3.2.3-2

                make-3.79

                binutils-2.11

                openmotif-2.2.2-16

                setarch-1.3-1

                compat-db-4.0.14.5

                以上為必需的程序包

                compat-gcc-7.3-2.96.122

                compat-gcc-c++-7.3-2.96.122

                compat-libstdc++-7.3-2.96.122

                compat-libstdc++-devel-7.3-2.96.122

                以上為可選程序包,最好安裝上

                要查看系統(tǒng)上安裝了這些程序包的哪些版本,以 root 用戶身份運(yùn)行以下命令:

                rpm -q gcc make binutils openmotif setarch compat-db compat-gcc compat-gcc-c++ compat-libstdc++ compat-libstdc++-devel

            第二部分:針對 Oracle 配置 Linux

                1.  驗(yàn)證系統(tǒng)要求

                要驗(yàn)證系統(tǒng)是否滿足 Oracle 10g 數(shù)據(jù)庫的最低要求,以 root 用戶身份登錄并運(yùn)行以下命令。

                要查看可用 RAM 和交換空間大小,運(yùn)行以下命令:

                grep MemTotal /proc/meminfo

                grep SwapTotal /proc/meminfo

                所需最小 RAM 為 512MB,而所需最小交換空間為 1GB.對于 RAM 小于或等于 2GB 的系統(tǒng),交換空間應(yīng)為 RAM 數(shù)量的兩倍;對于 RAM 大于 2GB 的系統(tǒng),交換空間應(yīng)為 RAM 數(shù)量的一到兩倍。

                Oracle 10g 軟件還需要 2.5GB 的可用磁盤空間,而數(shù)據(jù)庫則另需 1.2GB 的可用磁盤空間。/tmp 目錄至少需要 400MB 的可用空間。要檢查系統(tǒng)上的可用磁盤空間,運(yùn)行以下命令:

                df –h

                2.  創(chuàng)建 Oracle 組和用戶帳戶

                接下來,創(chuàng)建用于安裝和維護(hù) Oracle 10g 軟件的 Linux 組和用戶帳戶。用戶帳戶將稱為 oracle,而組將稱為 oinstall 和 dba.以 root 用戶身份執(zhí)行以下命令:

                /usr/sbin/groupadd oinstall

                /usr/sbin/groupadd dba

                /usr/sbin/useradd -m -g oinstall -G dba oracle

                設(shè)置 oracle 帳戶的口令:

                passwd oracle

                3.  創(chuàng)建目錄

                現(xiàn)在,創(chuàng)建存儲 Oracle 10g 軟件和數(shù)據(jù)庫文件的目錄。本指南在創(chuàng)建目錄結(jié)構(gòu)時(shí)所用的命名慣例符合最佳靈活結(jié)構(gòu) (OFA) 規(guī)范。有關(guān) OFA 標(biāo)準(zhǔn)的更多信息,請參閱針對 UNIX 系統(tǒng)的 Oracle 數(shù)據(jù)庫 10g 安裝指南 的附錄 D.

                以下假設(shè)在根文件系統(tǒng)中創(chuàng)建目錄。這樣做是為了簡便起見,不建議將其作為通用做法。這些目錄通常被創(chuàng)建為單獨(dú)的文件系統(tǒng)。

                以 root 用戶身份執(zhí)行以下命令:

                mkdir -p /u01/app/oracle

                mkdir -p /u02/oradata

                chown -R oracle:oinstall /u01/app/oracle /u02/oradata

                chmod -R 775 /u01/app/oracle /u02/oradata

                4.  配置 Linux 內(nèi)核參數(shù)

                Linux 內(nèi)核非常出色。與大多數(shù)其他 *NIX 系統(tǒng)不同,Linux 允許在系統(tǒng)啟動和運(yùn)行時(shí)修改大多數(shù)內(nèi)核參數(shù)。完成內(nèi)核參數(shù)更改后不必重新啟動系統(tǒng)。Oracle 數(shù)據(jù)庫 10g 需要以下所示的內(nèi)核參數(shù)設(shè)置。其中給出的是最小值,因此如果您的系統(tǒng)使用的值較大,則不要更改它。

                kernel.shmall = 2097152

                kernel.shmmax = 2147483648

                kernel.shmmni = 4096

                kernel.sem = 250 32000 100 128

                fs.file-max = 65536

                net.ipv4.ip_local_port_range = 1024 65000

                如果您正確的安裝了linux,且內(nèi)核參數(shù)全部采用默認(rèn)值,則只需在以 root 用戶身份登錄后執(zhí)行下命令。

                cat >> /etc/sysctl.conf <<EOF

                kernel.shmall = 2097152

                kernel.shmmax = 2147483648

                kernel.shmmni = 4096

                kernel.sem = 250 32000 100 128

                fs.file-max = 65536

                net.ipv4.ip_local_port_range = 1024 65000

                EOF

                /sbin/sysctl –p

                以 root 用戶身份運(yùn)行以下命令來驗(yàn)證您的設(shè)置:

                /sbin/sysctl -a | grep shm

                /sbin/sysctl -a | grep sem

                /sbin/sysctl -a | grep file-max

                /sbin/sysctl -a | grep ip_local_port_range

                如果系統(tǒng)的參數(shù)設(shè)置的比上述參數(shù)值小,則編輯 /etc/sysctl.conf 文件,添加或更改這些參數(shù)。完成后,運(yùn)行以下命令激活更改:

                /sbin/sysctl –p

                5.  為 oracle 用戶設(shè)置 Shell 限制

                Oracle 建議對每個(gè) Linux 帳戶可以使用的進(jìn)程數(shù)和打開的文件數(shù)設(shè)置限制。要進(jìn)行這些更改,以 root 用戶的身份執(zhí)行下列命令:

                cat >> /etc/security/limits.conf <<EOF

                oracle               soft    nproc   2047

                oracle               hard    nproc   16384

                oracle               soft    nofile  1024

                oracle               hard    nofile  65536

                EOF

                cat >> /etc/pam.d/login <<EOF

                session    required     /lib/security/pam_limits.so

                EOF

                cat >> /etc/profile <<EOF

                if [ \$USER = "oracle" ]; then

                if [ \$SHELL = "/bin/ksh" ]; then

                ulimit -p 16384

                ulimit -n 65536

                else

                ulimit -u 16384 -n 65536

                fi

                umask 022

                fi

                EOF

                cat >> /etc/csh.login <<EOF

                if ( \$USER == "oracle" ) then

                limit maxproc 16384

                limit descriptors 65536

                umask 022

                endif

                EOF

                6.  oracle 用戶的環(huán)境變量

                要使用 Oracle 產(chǎn)品,應(yīng)該或必須設(shè)置幾個(gè)環(huán)境變量。對于數(shù)據(jù)庫服務(wù)器,建議設(shè)置以下環(huán)境變量:

                ORACLE_BASE

                ORACLE_HOME

                ORACLE_SID

                PATH

                如果您在同一服務(wù)器上安裝了多個(gè) Oracle 產(chǎn)品或數(shù)據(jù)庫,則 ORACLE_HOME、ORACLE_SID 和 PATH 變量可能會更改。ORACLE_BASE 變量不應(yīng)更改,并可以在需要時(shí)在您的登錄配置文件中設(shè)置它。Oracle 提供了一個(gè)稱作 oraenv 的實(shí)用程序來設(shè)置其他變量。

                以 oracle 身份登錄,并通過在 .bash_profile中添加以下行:

                TMP=/tmp;

                export TMP

                TMPDIR=$TMP;

                export TMPDIR

                ORACLE_BASE=/u01/app/oracle;

                export ORACLE_BASE

                ORACLE_HOME=$ORACLE_BASE/oracle/product/10.2.0/db_1;

                export ORACLE_HOME

                ORACLE_SID=mydb;

                export ORACLE_SID

                ORACLE_TERM=xterm;

                export ORACLE_TERM

                PATH=/usr/sbin:$PATH;

                export PATH

                PATH=$ORACLE_HOME/bin:$PATH;

                export PATH

                LD_LIBRARY_PATH=$ORACLE_HOME/lib:/lib:/usr/lib;

                export LD_LIBRARY_PATH

                CLASSPATH=$ORACLE_HOME/JRE:$ORACLE_HOME/jlib:$ORACLE_HOME/rdbms/jlib;

                export CLASSPATH

                NLS_LANG="Simplified Chinese_china".ZHS16GBK;

                export NLS_LANG

                此更改將在您下次登錄到 oracle 帳戶時(shí)生效。

              第三部分:安裝 Oracle

                1.安裝軟件

                使用 oracle 帳戶登錄。

                將目錄更改為 Oracle 10g 軟件解壓縮到的位置。

                啟動 Oracle 通用安裝程序。

                $ ./runInstaller

                1)歡迎

                Ø         單擊 Next.

                2)指定清單目錄和證書

                Ø         如果您一直在依循本指南中的步驟,則使用默認(rèn)值即可。否則,編輯清單目錄的路徑,以指向正確目錄。

                Ø         操作系統(tǒng)組名稱應(yīng)為 oinstall.

                3)如果這是首次在此機(jī)器上安裝 Oracle,則您將收到一個(gè)彈出窗口,提示需要以 root 用戶身份運(yùn)行 orainstRoot.sh 腳本。以 root 用戶身份登錄,更改到窗口中指定的目錄,執(zhí)行該腳本,然后繼續(xù)操作。

                4)指定文件位置

                Ø         如果您一直在依循本指南中的步驟,則使用默認(rèn)值即可。否則,在繼續(xù)操作前確保源路徑和目標(biāo)路徑正確。

                5)選擇安裝類型

                Ø         接受默認(rèn)值 Enterprise Edition.

                6)與特定產(chǎn)品相關(guān)的先決條件的檢查

                Ø         如果您一直在依循本指南中的步驟,則所有檢查都應(yīng)順利通過。如果一個(gè)或多個(gè)檢查失敗,則在繼續(xù)操作前糾正該問題。

                7)選擇數(shù)據(jù)庫配置

                Ø         接受默認(rèn)值 Create a starter database 和 General Purpose.

                8)指定數(shù)據(jù)庫配置選項(xiàng)

                Ø         輸入數(shù)據(jù)庫的全局?jǐn)?shù)據(jù)庫名稱。該名稱應(yīng)包含 ORACLE_SID 和服務(wù)器域名(例如,demo1.orademo.org,其中 demo1 是 ORACLE_SID,orademo.org 是域名)。

                Ø         輸入全局?jǐn)?shù)據(jù)庫名稱時(shí),SID 框?qū)⒆詣犹畛洹?/p>

                Ø         接受默認(rèn)的數(shù)據(jù)庫字符集。

                Ø         選擇 Create database with sample schemas.

                9)選擇數(shù)據(jù)庫管理選項(xiàng)

                Ø         選擇 Use Database Control for Database Management.

                10)指定數(shù)據(jù)庫文件存儲選項(xiàng)

                Ø         選擇 File System,然后輸入數(shù)據(jù)庫文件要使用的路徑名(在本例中為 /u02/oradata)。

                11)指定備份和恢復(fù)選項(xiàng)

                Ø          選擇 Do not enable Automated backups.

                12)指定數(shù)據(jù)庫模式口令

                Ø         選擇 Use the same password for all the accounts.

                Ø         選擇一個(gè)口令,然后輸入兩次進(jìn)行確認(rèn)。

                13)摘要

                Ø         顯示已安裝產(chǎn)品的摘要。

                單擊 Install.

            posted @ 2010-07-03 00:40 幽夢新影 閱讀(631) | 評論 (0)編輯 收藏

            Oracle 語句大全

                 摘要: 1. Oracle安裝完成后的初始口令?  internal/oracle   sys/change_on_install   system/manager   scott/tiger   sysman/oem_temp2. ORACLE9IAS WEB CACHE的初始默認(rèn)用戶和密碼? administrator/administrat...  閱讀全文

            posted @ 2010-07-03 00:38 幽夢新影 閱讀(1088) | 評論 (0)編輯 收藏

            2010年6月24日 #

            中國佛學(xué)66句震撼世界的禪語

            1.人之所以痛苦,在于追求錯(cuò)誤的東西。
            2.如果你不給自己煩惱,別人也永遠(yuǎn)不可能給你煩惱。因?yàn)槟阕约旱膬?nèi)心,你放不下。
            3.你永遠(yuǎn)要感謝給你逆境的眾生。
            4.你永遠(yuǎn)要寬恕眾生,不論他有多壞,甚至他傷害過你,你一定要放下,才能得到真正的快樂。
            5.當(dāng)你快樂時(shí),你要想這快樂不是永恒的。當(dāng)你痛苦時(shí),你要想這痛苦也不是永恒的。
            6.今日的執(zhí)著,會造成明日的后悔。
            7.你可以擁有愛,但不要執(zhí)著,因?yàn)榉蛛x是必然的。
            8.不要浪費(fèi)你的生命在你一定會后悔的地方上。
            9.你什么時(shí)候放下,什么時(shí)候就沒有煩惱。
            10.每一種創(chuàng)傷,都是一種成熟。
            11.狂妄的人有救,自卑的人沒有救,認(rèn)識自己,降伏自己,改變自己,才能改變別人。
            12.你不要一直不滿人家,你應(yīng)該一直檢討自己才對。不滿人家,是苦了你自己。
            13.一個(gè)人如果不能從內(nèi)心去原諒別人,那他就永遠(yuǎn)不會心安理得。
            14.心中裝滿著自己的看法與想法的人,永遠(yuǎn)聽不見別人的心聲。
            15.毀滅人只要一句話,培植一個(gè)人卻要千句話,請你多口下留情。
            16.根本不必回頭去看咒罵你的人是誰?如果有一條瘋狗咬你一口,難道你也要趴下去反咬他一口嗎?
            17.永遠(yuǎn)不要浪費(fèi)你的一分一秒,去想任何不喜歡你的人。
            18.請你用慈悲心和溫和的態(tài)度,把你的不滿與委屈說出來,別人就容易接受。
            19.同樣的瓶子,你為什么要裝毒藥呢?同樣的心里,你為什么要充滿著煩惱呢?
            20.得不到的東西,我們會一直以為他是美好的,那是因?yàn)槟銓λ私馓伲瑳]有時(shí)間與他相處在一起。當(dāng)有一天,你深入了解后,你會發(fā)現(xiàn)原不是你想像中的那么美好。
            21.活著一天,就是有福氣,就該珍惜。當(dāng)我哭泣我沒有鞋子穿的時(shí)候,我發(fā)現(xiàn)有人卻沒有腳。
            22.多一分心力去注意別人,就少一分心力反省自己,你懂嗎?
            23.憎恨別人對自己是一種很大的損失。
            24.每一個(gè)人都擁有生命,但并非每個(gè)人都懂得生命,乃至于珍惜生命。不了解生命的人,生命對他來說,是一種懲罰。
            25.情執(zhí)是苦惱的原因,放下情執(zhí),你才能得到自在
            26.不要太肯定自己的看法,這樣子比較少后悔。
            27.當(dāng)你對自己誠實(shí)的時(shí)候,世界上沒有人能夠欺騙得了你。
            28.用傷害別人的手段來掩飾自己缺點(diǎn)的人,是可恥的。
            29.默默的關(guān)懷與祝福別人,那是一種無形的布施。
            30.不要刻意去猜測他人的想法,如果你沒有智慧與經(jīng)驗(yàn)的正確判斷,通常都會有錯(cuò)誤的。
            31.要了解一個(gè)人,只需要看他的出發(fā)點(diǎn)與目的地是否相同,就可以知道他是否真心的。
            32.人生的真理,只是藏在平淡無味之中。
            33.不洗澡的人,硬擦香水是不會香的。名聲與尊貴,是來自于真才實(shí)學(xué)的。有德自然香。
            34.時(shí)間總會過去的,讓時(shí)間流走你的煩惱吧!
            35.你硬要把單純的事情看得很嚴(yán)重,那樣子你會很痛苦。
            36.永遠(yuǎn)扭曲別人善意的人,無藥可救。
            37.說一句謊話,要編造十句謊話來彌補(bǔ),何苦呢?
            38.白白的過一天,無所事事,就像犯了竊盜罪一樣
            39.廣結(jié)眾緣,就是不要去傷害任何一個(gè)人。
            40.沉默是毀謗最好的答覆。
            41.對人恭敬,就是在莊嚴(yán)你自己。
            42.擁有一顆無私的愛心,便擁有了一切。
            43.來是偶然的,走是必然的。所以你必須,隨緣不變,不變隨緣。
            44.慈悲是你最好的武器。
            45.只要面對現(xiàn)實(shí),你才能超越現(xiàn)實(shí)。
            46.良心是每一個(gè)人最公正的審判官,你騙得了別人,卻永遠(yuǎn)騙不了你自己的良心。
            47.不懂得自愛的人,是沒有能力去愛別人的。
            48.有時(shí)候我們要冷靜問問自已,我們在追求什么?我們活著為了什么?
            49.不要因?yàn)樾⌒〉臓巿?zhí),遠(yuǎn)離了你至親的好友,也不要因?yàn)樾⌒〉脑购蓿浟藙e人的大恩。
            50.感謝上蒼我所擁有的,感謝上蒼我所沒有的。
            51.凡是能站在別人的角度為他人著想,這個(gè)就是慈悲。
            52.說話不要有攻擊性,不要有殺傷力,不夸已能,不揚(yáng)人惡,自然能化敵為友。
            53.誠實(shí)的面對你內(nèi)心的矛盾和污點(diǎn),不要欺騙你自己。
            54.因果不曾虧欠過我們什么,所以請不要抱怨。
            55.大多數(shù)的人一輩子只做了三件事;自欺、欺人、被人欺。
            56.心是最大的騙子,別人能騙你一時(shí),而它卻會騙你一輩子
            57.只要自覺心安,東西南北都好。如有一人未度,切莫自己逃了。
            58.當(dāng)你手中抓住一件東西不放時(shí),你只能擁有這件東西,如果你肯放手,你就有機(jī)會選擇別的。人的心若死執(zhí)自己的觀念,不肯放下,那么他的智慧也只能達(dá)到某種程度而已
            59.如果你能夠平平安安的渡過一天,那就是一種福氣了。多少人在今天已經(jīng)見不到明天的太陽,多少人在今天已經(jīng)成了殘廢,多少人在今天已經(jīng)失去了自由,多少人在今天已經(jīng)家破人亡。
            60.你有你的生命觀,我有我的生命觀,我不干涉你。只要我能,我就感化你。如果不能,那我就認(rèn)命。
            61.你希望掌握永恒,那你必須控制現(xiàn)在。
            62.惡口永遠(yuǎn)不要出自于我們的口中,不管他有多壞,有多惡。你愈罵他,你的心就被污染了,你要想,他就是你的善知識。
            63.別人可以違背因果,別人可以害我們,打我們,毀謗我們。可是我們不能因此而憎恨別人,為什么?我們一定要保有一顆完整的本性和一顆清凈的心。
            64.如果一個(gè)人沒有苦難的感受,就不容易對他人給予同情。你要學(xué)救苦救難的精神,就得先受苦受難。
            65.世界原本就不是屬于你,因此你用不著拋棄,要拋棄的是一切的執(zhí)著。萬物皆為我所用,但非我所屬。
            66.雖然我們不能改變周遭的世界,我們就只好改變自己,用慈悲心和智慧心來面對這一切。

            posted @ 2010-06-24 21:18 幽夢新影 閱讀(349) | 評論 (0)編輯 收藏

            2010年6月23日 #

            STL vector 容器介紹

            介紹

            這篇文章的目的是為了介紹std::vector,如何恰當(dāng)?shù)厥褂盟鼈兊某蓡T函數(shù)等操作。本文中還討論了條件函數(shù)和函數(shù)指針在迭代算法中使用,如在remove_if()for_each()中的使用。通過閱讀這篇文章讀者應(yīng)該能夠有效地使用vector容器,而且應(yīng)該不會再去使用C類型的動態(tài)數(shù)組了。

             

            Vector總覽

            vectorC++標(biāo)準(zhǔn)模板庫中的部分內(nèi)容,它是一個(gè)多功能的,能夠操作多種數(shù)據(jù)結(jié)構(gòu)和算法的模板類和函數(shù)庫。vector之所以被認(rèn)為是一個(gè)容器,是因?yàn)樗軌蛳袢萜饕粯哟娣鸥鞣N類型的對象,簡單地說,vector是一個(gè)能夠存放任意類型的動態(tài)數(shù)組,能夠增加和壓縮數(shù)據(jù)。

            為了可以使用vector,必須在你的頭文件中包含下面的代碼:

            #include <vector>

             

            vector屬于std命名域的,因此需要通過命名限定,如下完成你的代碼:

            using std::vector;

            vector<int> vInts;

             

            或者連在一起,使用全名:

            std::vector<int> vInts;

             

            建議使用全局的命名域方式:

            using namespace std;

             

            在后面的操作中全局的命名域方式會造成一些問題。vector容器提供了很多接口,在下面的表中列出vector的成員函數(shù)和操作。

             

            Vector成員函數(shù)

            函數(shù)

            表述

            c.assign(beg,end)

            c.assign(n,elem)

            [beg; end)區(qū)間中的數(shù)據(jù)賦值給c

            n個(gè)elem的拷貝賦值給c

            c.at(idx)

            傳回索引idx所指的數(shù)據(jù),如果idx越界,拋出out_of_range

            c.back()

            傳回最后一個(gè)數(shù)據(jù),不檢查這個(gè)數(shù)據(jù)是否存在。

            c.begin()

            傳回迭代器重的可一個(gè)數(shù)據(jù)。

            c.capacity()

            返回容器中數(shù)據(jù)個(gè)數(shù)。

            c.clear()

            移除容器中所有數(shù)據(jù)。

            c.empty()

            判斷容器是否為空。

            c.end()

            指向迭代器中的最后一個(gè)數(shù)據(jù)地址。

            c.erase(pos)

            c.erase(beg,end)

            刪除pos位置的數(shù)據(jù),傳回下一個(gè)數(shù)據(jù)的位置。

            刪除[beg,end)區(qū)間的數(shù)據(jù),傳回下一個(gè)數(shù)據(jù)的位置

            c.front()

            傳回地一個(gè)數(shù)據(jù)。

            get_allocator

            使用構(gòu)造函數(shù)返回一個(gè)拷貝。

            c.insert(pos,elem)

            c.insert(pos,n,elem)

            c.insert(pos,beg,end)

            pos位置插入一個(gè)elem拷貝,傳回新數(shù)據(jù)位置。

            pos位置插入n個(gè)elem數(shù)據(jù)。無返回值。

            pos位置插入在[beg,end)區(qū)間的數(shù)據(jù)。無返回值。

            c.max_size()

            返回容器中最大數(shù)據(jù)的數(shù)量。

            c.pop_back()

            刪除最后一個(gè)數(shù)據(jù)。

            c.push_back(elem)

            在尾部加入一個(gè)數(shù)據(jù)。

            c.rbegin()

            傳回一個(gè)逆向隊(duì)列的第一個(gè)數(shù)據(jù)。

            c.rend()

            傳回一個(gè)逆向隊(duì)列的最后一個(gè)數(shù)據(jù)的下一個(gè)位置。

            c.resize(num)

            重新指定隊(duì)列的長度。

            c.reserve()

            保留適當(dāng)?shù)娜萘俊?/span>

            c.size()

            返回容器中實(shí)際數(shù)據(jù)的個(gè)數(shù)。

            c1.swap(c2)

            swap(c1,c2)

            c1c2元素互換。

            同上操作。

            vector<Elem> c

            vector <Elem> c1(c2)

            vector <Elem> c(n)

            vector <Elem> c(n, elem)

            vector <Elem> c(beg,end)

            c.~ vector <Elem>()

            創(chuàng)建一個(gè)空的vector

            復(fù)制一個(gè)vector

            創(chuàng)建一個(gè)vector,含有n個(gè)數(shù)據(jù),數(shù)據(jù)均已缺省構(gòu)造產(chǎn)生

            創(chuàng)建一個(gè)含有n個(gè)elem拷貝的vector

            創(chuàng)建一個(gè)以[beg;end)區(qū)間的vector

            銷毀所有數(shù)據(jù),釋放內(nèi)存。

             

            Vector操作

            函數(shù)

            描述

            operator[]

            返回容器中指定位置的一個(gè)引用。

             

            創(chuàng)建一個(gè)vector

            vector容器提供了多種創(chuàng)建方法,下面介紹幾種常用的。

            創(chuàng)建一個(gè)Widget類型的空的vector對象:

            vector<Widget> vWidgets;

            //     ------

            //      |

            //      |- Since vector is a container, its member functions

            //         operate on iterators and the container itself so

            //         it can hold objects of any type.

             

            創(chuàng)建一個(gè)包含500個(gè)Widget類型數(shù)據(jù)的vector

            vector<Widget> vWidgets(500);

             

            創(chuàng)建一個(gè)包含500個(gè)Widget類型數(shù)據(jù)的vector,并且都初始化為0

            vector<Widget> vWidgets(500, Widget(0));

             

            創(chuàng)建一個(gè)Widget的拷貝:

            vector<Widget> vWidgetsFromAnother(vWidgets);

             

            vector添加一個(gè)數(shù)據(jù)

            vector添加數(shù)據(jù)的缺省方法是push_back()push_back()函數(shù)表示將數(shù)據(jù)添加到vector的尾部,并按需要來分配內(nèi)存。例如:向vector<Widget>中添加10個(gè)數(shù)據(jù),需要如下編寫代碼:

            for(int i= 0;i<10; i++)

                vWidgets.push_back(Widget(i));

             

            獲取vector中制定位置的數(shù)據(jù)

            很多時(shí)候我們不必要知道vector里面有多少數(shù)據(jù),vector里面的數(shù)據(jù)是動態(tài)分配的,使用push_back()的一系列分配空間常常決定于文件或一些數(shù)據(jù)源。如果你想知道vector存放了多少數(shù)據(jù),你可以使用empty()。獲取vector的大小,可以使用size()。例如,如果你想獲取一個(gè)vector v的大小,但不知道它是否為空,或者已經(jīng)包含了數(shù)據(jù),如果為空想設(shè)置為-1,你可以使用下面的代碼實(shí)現(xiàn):

            int nSize = v.empty() ? -1 : static_cast<int>(v.size());

             

            訪問vector中的數(shù)據(jù)

            使用兩種方法來訪問vector

            1、   vector::at()

            2、   vector::operator[]

            operator[]主要是為了與C語言進(jìn)行兼容。它可以像C語言數(shù)組一樣操作。但at()是我們的首選,因?yàn)?/span>at()進(jìn)行了邊界檢查,如果訪問超過了vector的范圍,將拋出一個(gè)例外。由于operator[]容易造成一些錯(cuò)誤,所有我們很少用它,下面進(jìn)行驗(yàn)證一下:

            分析下面的代碼:

            vector<int> v;

            v.reserve(10);

             

            for(int i=0; i<7; i++)

                v.push_back(i);

             

            try

            {

             int iVal1 = v[7]; // not bounds checked - will not throw

             int iVal2 = v.at(7); // bounds checked - will throw if out of range

            }

            catch(const exception& e)

            {

             cout << e.what();

            }

             

            我們使用reserve()分配了10個(gè)int型的空間,但并不沒有初始化。如下圖所示:


            你可以在這個(gè)代碼中嘗試不同條件,觀察它的結(jié)果,但是無論何時(shí)使用at(),都是正確的。

             

            刪除vector中的數(shù)據(jù)

            vector能夠非常容易地添加數(shù)據(jù),也能很方便地取出數(shù)據(jù),同樣vector提供了erase()pop_back()clear()來刪除數(shù)據(jù),當(dāng)你刪除數(shù)據(jù)的時(shí)候,你應(yīng)該知道要?jiǎng)h除尾部的數(shù)據(jù),或者是刪除所有數(shù)據(jù),還是個(gè)別的數(shù)據(jù)。在考慮刪除等操作之前讓我們靜下來考慮一下在STL中的一些應(yīng)用。

             

            Remove_if()算法

            現(xiàn)在我們考慮操作里面的數(shù)據(jù)。如果要使用remove_if(),我們需要在頭文件中包含如下代碼::

            #include <algorithm>

             

                     Remove_if()有三個(gè)參數(shù):

            1、   iterator _First:指向第一個(gè)數(shù)據(jù)的迭代指針。

            2、   iterator _Last:指向最后一個(gè)數(shù)據(jù)的迭代指針。

            3、   predicate _Pred:一個(gè)可以對迭代操作的條件函數(shù)。

             

            條件函數(shù)

            條件函數(shù)是一個(gè)按照用戶定義的條件返回是或否的結(jié)果,是最基本的函數(shù)指針,或者是一個(gè)函數(shù)對象。這個(gè)函數(shù)對象需要支持所有的函數(shù)調(diào)用操作,重載operator()()操作。remove_if()是通過unary_function繼承下來的,允許傳遞數(shù)據(jù)作為條件。

            例如,假如你想從一個(gè)vector<CString>中刪除匹配的數(shù)據(jù),如果字串中包含了一個(gè)值,從這個(gè)值開始,從這個(gè)值結(jié)束。首先你應(yīng)該建立一個(gè)數(shù)據(jù)結(jié)構(gòu)來包含這些數(shù)據(jù),類似代碼如下:

            #include <functional>

            enum findmodes

            {

             FM_INVALID = 0,

             FM_IS,

             FM_STARTSWITH,

             FM_ENDSWITH,

             FM_CONTAINS

            };

            typedef struct tagFindStr

            {

             UINT iMode;

             CString szMatchStr;

            } FindStr;

            typedef FindStr* LPFINDSTR;

             

            然后處理?xiàng)l件判斷:

            class FindMatchingString

                : public std::unary_function<CString, bool>

            {

               

            public:

             FindMatchingString(const LPFINDSTR lpFS) : m_lpFS(lpFS) {}

               

             bool operator()(CString& szStringToCompare) const

             {

                 bool retVal = false;

             

                 switch(m_lpFS->iMode)

                 {

                 case FM_IS:

                   {

                     retVal = (szStringToCompare == m_lpFDD->szMatchStr);

                     break;

                   }

                 case FM_STARTSWITH:

                   {

                     retVal = (szStringToCompare.Left(m_lpFDD->szMatchStr.GetLength())

                           == m_lpFDD->szWindowTitle);

                     break;

                   }

                 case FM_ENDSWITH:

                   {

                     retVal = (szStringToCompare.Right(m_lpFDD->szMatchStr.GetLength())

                           == m_lpFDD->szMatchStr);

                     break;

                   }

                 case FM_CONTAINS:

                   {

                     retVal = (szStringToCompare.Find(m_lpFDD->szMatchStr) != -1);

                     break;

                   }

                 }

                   

                 return retVal;

             }

                   

            private:

                LPFINDSTR m_lpFS;

            };

             

            通過這個(gè)操作你可以從vector中有效地刪除數(shù)據(jù):

            // remove all strings containing the value of

            // szRemove from vector<CString> vs.

             

            FindStr fs;

            fs.iMode = FM_CONTAINS;

            fs.szMatchStr = szRemove;

             

            vs.erase(std::remove_if(vs.begin(), vs.end(), FindMatchingString(&fs)), vs.end());

             

            Remove_if()能做什么?

            你可能會疑惑,對于上面那個(gè)例子在調(diào)用remove_if()的時(shí)候還要使用erase()呢?這是因?yàn)榇蠹也⒉皇煜?/span>STL中的算法。Remove(),remove_if()等所有的移出操作都是建立在一個(gè)迭代范圍上的,那么不能操作容器中的數(shù)據(jù)。所以在使用remove_if(),實(shí)際上操作的時(shí)容器里數(shù)據(jù)的上面的。思考上面的例子:

            1、   szRemove = “o”.

            2、   vs見下面圖表中的顯示。

            觀察這個(gè)結(jié)果,我們可以看到remove_if()實(shí)際上是根據(jù)條件對迭代地址進(jìn)行了修改,在數(shù)據(jù)的后面存在一些殘余的數(shù)據(jù),那些需要?jiǎng)h除的數(shù)據(jù)。剩下的數(shù)據(jù)的位置可能不是原來的數(shù)據(jù),但他們是不知道的。

            調(diào)用erase()來刪除那些殘余的數(shù)據(jù)。注意上面例子中通過erase()刪除remove_if()的結(jié)果和vs.enc()范圍的數(shù)據(jù)。

             

            壓縮一個(gè)臃腫的vector

            很多時(shí)候大量的刪除數(shù)據(jù),或者通過使用reserve(),結(jié)果vector的空間遠(yuǎn)遠(yuǎn)大于實(shí)際需要的。所有需要壓縮vector到它實(shí)際的大小。resize()能夠增加vector的大小。Clear()僅僅能夠改變緩存的大小,所有的這些對于vector釋放內(nèi)存等九非常重要了。如何來解決這些問題呢,讓我們來操作一下。

            我們可以通過一個(gè)vector創(chuàng)建另一個(gè)vector。讓我們看看這將發(fā)生什么。假定我們已經(jīng)有一個(gè)vector v,它的內(nèi)存大小為1000,當(dāng)我們調(diào)用size()的時(shí)候,它的大小僅為7。我們浪費(fèi)了大量的內(nèi)存。讓我們在它的基礎(chǔ)上創(chuàng)建一個(gè)vector

             

            std::vector<CString> vNew(v);

            cout << vNew.capacity();

             

            vNew.capacity()返回的是7。這說明新創(chuàng)建的只是根據(jù)實(shí)際大小來分配的空間。現(xiàn)在我們不想釋放v,因?yàn)槲覀円谄渌胤接玫剿覀兛梢允褂?/span>swap()vvNew互相交換一下?

             

                vNew.swap(v);

                cout << vNew.capacity();

                cout << v.capacity();

             

            有趣的是:vNew.capacity()1000v.capacity()7

            現(xiàn)在是達(dá)到我的目的了,但是并不是很好的解決方法,我們可以像下面這么寫:

             

                std::vector<CString>(v).swap(v);

                 

            你可以看到我們做了什么?我們創(chuàng)建了一個(gè)臨時(shí)變量代替那個(gè)命名的,然后使用swap(),這樣我們就去掉了不必要的空間,得到實(shí)際大小的v

             

            結(jié)論

            我希望這個(gè)文檔可以給那些使用STL vector容器的開發(fā)者很有價(jià)值的參考。我也希望通過閱讀這篇文章你可以放心地使用vector來代替C語言中的數(shù)據(jù)了。

             

            參考

            Plauger, P.J. Standard C++ Library Reference. February, 2003. MSDN.

            Schildt, Herbert. C++ from the Ground Up, Second Edition. Berkeley: 1998.

            Sutter, Herb. More Exceptional C++. Indianapolis: 2002.

            posted @ 2010-06-23 13:56 幽夢新影 閱讀(473) | 評論 (0)編輯 收藏

            2010年6月2日 #

            C++ STL編程輕松入門

              作為C++標(biāo)準(zhǔn)不可缺少的一部分,STL應(yīng)該是滲透在C++程序的角角落落里的。STL不是實(shí)驗(yàn)室里的寵兒,也不是程序員桌上的擺設(shè),她的激動人心并非曇花一現(xiàn)。本教程旨在傳播和普及STL的基礎(chǔ)知識,若能借此機(jī)會為STL的推廣做些力所能及的事情,到也是件讓人愉快的事情。 

              1 初識STL:解答一些疑問  1.1 一個(gè)最關(guān)心的問題:什么是STL科學(xué)領(lǐng)域里所常用的基本數(shù)據(jù)結(jié)構(gòu)和基本算法。為廣大C++程序員們提供了一個(gè)可擴(kuò)展的應(yīng)用框架,高度體現(xiàn)了軟件的可復(fù)用性。這種現(xiàn)象有些類似于Microsoft Visual C++中的MFC(Microsoft Foundation Class Library),或者是Borland C++ Builder中的VCL(Visual Component Library),對于此二者,大家一定不會陌生吧。

              "什么是STL?",假如你對STL還知之甚少,那么我想,你一定很想知道這個(gè)問題的答案,坦率地講,要指望用短短數(shù)言將這個(gè)問題闡述清楚,也決非易事。因此,如果你在看完本節(jié)之后還是覺得似懂非懂,大可不必著急,在閱讀了后續(xù)內(nèi)容之后,相信你對STL的認(rèn)識,將會愈加清晰、準(zhǔn)確和完整。不過,上述這番話聽起來是否有點(diǎn)像是在為自己糟糕的表達(dá)能力開脫罪責(zé)呢?:) 

              不知道你是否有過這樣的經(jīng)歷。在你準(zhǔn)備著手完成數(shù)據(jù)結(jié)構(gòu)老師所布置的家庭作業(yè)時(shí),或者在你為你所負(fù)責(zé)的某個(gè)軟件項(xiàng)目中添加一項(xiàng)新功能時(shí),你發(fā)現(xiàn)需要用到一個(gè)鏈表(List)或者是映射表(Map)之類的東西,但是手頭并沒有現(xiàn)成的代碼。于是在你開始正式考慮程序功能之前,手工實(shí)現(xiàn)List或者M(jìn)ap是不可避免的。于是……,最終你順利完成了任務(wù)。或許此時(shí),作為一個(gè)具有較高素養(yǎng)的程序員的你還不肯罷休(或者是一個(gè)喜歡偷懶的優(yōu)等生:),因?yàn)槟銜氲剑绻院筮€遇到這樣的情況怎么辦?沒有必要再做一遍同樣的事情吧! 

              如果說上述這種情形每天都在發(fā)生,或許有點(diǎn)夸張。但是,如果說整個(gè)軟件領(lǐng)域里,數(shù)十年來確實(shí)都在為了一個(gè)目標(biāo)而奮斗--可復(fù)用性(reusability),這看起來似乎并不夸張。從最早的面向過程的函數(shù)庫,到面向?qū)ο蟮?a style="FONT-SIZE: 12px; COLOR: rgb(78,78,78); WORD-BREAK: break-all; TEXT-DECORATION: none" >程序設(shè)計(jì)思想,到各種組件技術(shù)(如:COM、EJB),到設(shè)計(jì)模式(design pattern)等等。而STL也在做著類似的事情,同時(shí)在它背后蘊(yùn)涵著一種新的程序設(shè)計(jì)思想--泛型化設(shè)計(jì)(generic programming)。 

              繼續(xù)上面提到的那個(gè)例子,假如你把List或者map完好的保留了下來,正在暗自得意。且慢,如果下一回的List里放的不是浮點(diǎn)數(shù)而是整數(shù)呢?如果你所實(shí)現(xiàn)的Map在效率上總是令你不太滿意并且有時(shí)還會出些bug呢?你該如何面對這些問題?使用STL是一個(gè)不錯(cuò)的選擇,確實(shí)如此,STL可以漂亮地解決上面提到的這些問題,盡管你還可以尋求其他方法。

             說了半天,到底STL是什么東西呢? 

              STL(Standard Template Library),即標(biāo)準(zhǔn)模板庫,是一個(gè)具有工業(yè)強(qiáng)度的,高效的C++程序庫。它被容納于C++標(biāo)準(zhǔn)程序庫(C++ Standard Library)中,是ANSI/ISO C++標(biāo)準(zhǔn)中最新的也是極具革命性的一部分。該庫包含了諸多在計(jì)算機(jī) 

              從邏輯層次來看,在STL中體現(xiàn)了泛型化程序設(shè)計(jì)的思想(generic programming),引入了諸多新的名詞,比如像需求(requirements),概念(concept),模型(model),容器(container),算法(algorithmn),迭代子(iterator)等。與OOP(object-oriented programming)中的多態(tài)(polymorphism)一樣,泛型也是一種軟件的復(fù)用技術(shù)。 

              從實(shí)現(xiàn)層次看,整個(gè)STL是以一種類型參數(shù)化(type parameterized)的方式實(shí)現(xiàn)的,這種方式基于一個(gè)在早先C++標(biāo)準(zhǔn)中沒有出現(xiàn)的語言特性--模板(template)。如果查閱任何一個(gè)版本的STL源代碼,你就會發(fā)現(xiàn),模板作為構(gòu)成整個(gè)STL的基石是一件千真萬確的事情。除此之外,還有許多C++的新特性為STL的實(shí)現(xiàn)提供了方便。 

              不知你對這里一下子冒出這么多術(shù)語做何感想,希望不會另你不愉快。假如你對它們之中的大多數(shù)不甚了解,敬請放心,在后續(xù)內(nèi)容中將會對這些名詞逐一論述。正如開頭所提到的。 

              有趣的是,對于STL還有另外一種解釋--STepanov & Lee,前者是指Alexander Stepanov,STL的創(chuàng)始人;而后者是Meng Lee,她也是使STL得以推行的功臣,第一個(gè)STL成品就是他們合作完成的。這一提法源自1995年3月,Dr.Dobb's Journal特約記者, 著名技術(shù)書籍作家Al Stevens對Alexander Stepanov的一篇專訪。 

              1.2 追根溯源:STL的歷史學(xué)院教授David Musser共同開發(fā)了一種叫做Tecton的語言。盡管這次嘗試最終沒有取得實(shí)用性的成果,但卻給了Stepanov很大的啟示。 

              在結(jié)識新朋友的時(shí)候,大多數(shù)人總是忍不住想了解對方的過去。本節(jié)將帶您簡單回顧一下STL的過去。 

              被譽(yù)為STL之父的Alexander Stepanov,出生于蘇聯(lián)莫斯科,早在20世紀(jì)70年代后半期,他便已經(jīng)開始考慮,在保證效率的前提下,將算法從諸多具體應(yīng)用之中抽象出來的可能性,這便是后來泛型化思想的雛形。為了驗(yàn)證自己的思想,他和紐約州立大學(xué)教授Deepak Kapur,倫塞里爾技術(shù) 

              在隨后的幾年中,他又和David Musser等人先后用Schema語言(一種Lisp語言的變種)和Ada語言建立了一些大型程序庫。這其間,Alexander Stepanov開始意識到,在當(dāng)時(shí)的面向?qū)ο蟪绦蛟O(shè)計(jì)思想中所存在的一些問題,比如抽象數(shù)據(jù)類型概念所存在的缺陷。Stepanov希望通過對軟件領(lǐng)域中各組成部分的分類,逐漸形成一種軟件設(shè)計(jì)的概念性框架。 

              1987年左右,在貝爾實(shí)驗(yàn)室工作的Alexander Stepanov開始首次采用C++語言進(jìn)行泛型軟件庫的研究。但遺憾的是,當(dāng)時(shí)的C++語言還沒有引入模板(template)的語法,現(xiàn)在我們可以清楚的看到,模板概念之于STL實(shí)現(xiàn),是何等重要。是時(shí)使然,采用繼承機(jī)制是別無選擇的。盡管如此,Stepanov還是開發(fā)出了一個(gè)龐大的算法庫。與此同時(shí),在與Andrew Koenig(前ISO C++標(biāo)準(zhǔn)化委員會主席)和Bjarne Stroustrup(C++語言的創(chuàng)始人)等頂級大師們的共事過程中,Stepanov開始注意到C/C++語言在實(shí)現(xiàn)其泛型思想方面所具有的潛在優(yōu)勢。就拿C/C++中的指針而言,它的靈活與高效運(yùn)用,使后來的STL在實(shí)現(xiàn)泛型化的同時(shí)更是保持了高效率。另外,在STL中占據(jù)極其重要地位的迭代子概念便是源自于C/C++中原生指針( native pointer)的抽象。

              1988年,Alexander Stepanov開始進(jìn)入惠普的Palo Alto實(shí)驗(yàn)室工作,在隨后的4年中,他從事的是有關(guān)磁盤驅(qū)動器方面的工作。直到1992年,由于參加并主持了實(shí)驗(yàn)室主任Bill Worley所建立的一個(gè)有關(guān)算法的研究項(xiàng)目,才使他重新回到了泛型化算法的研究工作上來。項(xiàng)目自建立之后,參與者從最初的8人逐漸減少,最后只剩下兩個(gè)人--Stepanove本人和Meng Lee。經(jīng)過長時(shí)間的努力,最終,信念與汗水所換來的是一個(gè)包含有大量數(shù)據(jù)結(jié)構(gòu)和算法部件的龐大運(yùn)行庫。這便是現(xiàn)在的STL的雛形(同時(shí)也是STL的一個(gè)實(shí)現(xiàn)版本--HP STL)。 

              1993年,當(dāng)時(shí)在貝爾實(shí)驗(yàn)室的Andrew Koenig看到了Stepanove的研究成果,很是興奮。在他的鼓勵(lì)與幫助下,Stepanove于是年9月的圣何塞為ANSI/ISO C++標(biāo)準(zhǔn)委員會做了一個(gè)相關(guān)演講(題為"The Science of C++ Programming"),向委員們講述了其觀念。然后又于次年3月,在圣迭戈會議上,向委員會提交了一份建議書,以期使STL成為C++標(biāo)準(zhǔn)庫的一部分。盡管這一建議十分龐大,以至于降低了被通過的可能性,但由于其所包含的新思想,投票結(jié)果以壓倒多數(shù)的意見認(rèn)為推遲對該建議的決定。 

              隨后,在眾人的幫助之下,包括Bjarne Stroustrup在內(nèi),Stepanove又對STL進(jìn)行了改進(jìn)。同時(shí)加入了一個(gè)封裝內(nèi)存模式信息的抽象模塊,也就是現(xiàn)在STL中的allocator,它使STL的大部分實(shí)現(xiàn)都可以獨(dú)立于具體的內(nèi)存模式,從而獨(dú)立于具體平臺。在同年夏季的滑鐵盧會議上,委員們以80%贊成,20%反對,最終通過了提案,決定將STL正式納入C++標(biāo)準(zhǔn)化進(jìn)程之中,隨后STL便被放進(jìn)了會議的工作文件中。自此,STL終于成為了C++家族中的重要一員。 

              此后,隨著C++標(biāo)準(zhǔn)的不斷改進(jìn),STL也在不斷地作著相應(yīng)的演化。直至1998年,ANSI/ISO C++標(biāo)準(zhǔn)正式定案,STL始終是C++標(biāo)準(zhǔn)中不可或缺的一大部件。 

              1.3 千絲萬縷的聯(lián)系  1.3.1 STL和C++  1.3.2 STL和C++標(biāo)準(zhǔn)函數(shù)庫1.3.3 STL和GP,GP和OOP   1.4 STL的不同實(shí)現(xiàn)版本  1.4.1 HP STL  1.4.2 P.J. Plauger STLhttp://www.dinkumware.comVisualStudiohashset容器。1.4.3 Rouge Wave STLhttp://www.rougewave.com源代碼C:Program Files\Borland\Cbuilder6\Include\oldstl)。1.4.4 STLporthttp://www.stlport.org免費(fèi)下載VC中的STL要快。比Rouge Wave STL更符合標(biāo)準(zhǔn),也更容易移植。Borland C++ Builder已經(jīng)在其6.0版中加入了對STLport的支持,它使用的STLport就是4.5版的,C++ Builder 6.0同時(shí)還提供了STLport的使用說明。你可以在C++ Builder的Include\Stlport子目錄下找到所有頭文件(比如:C:\Program Files\Borland\Cbuilder6\Include\Stlport)。1.4.5 SGI STLhttp://www.sgi.com新版本是3.3。 


              在你了解了STL的過去之后,一些名詞開始不斷在你的大腦中浮現(xiàn),STL、C++、C++標(biāo)準(zhǔn)函數(shù)庫、泛型程序設(shè)計(jì)、面向?qū)ο蟪绦蛟O(shè)計(jì)……,這些概念意味著什么?他們之間的關(guān)系又是什么?如果你想了解某些細(xì)節(jié),這里也許有你希望得到的答案。 


              沒有C++語言就沒有STL,這么說毫不為過。一般而言,STL作為一個(gè)泛型化的數(shù)據(jù)結(jié)構(gòu)和
              此外,STL對于C++的發(fā)展,尤其是模板機(jī)制,也起到了促進(jìn)作用。比如:模板函數(shù)的偏特化(template function partial specialization),它被用于在特定應(yīng)用場合,為一般模板函數(shù)提供一系列特殊化版本。這一特性是繼STL被ANSI/ISO C++標(biāo)準(zhǔn)委員會通過之后,在Bjarne和Stepanov共同商討之下并由Bjarne向委員會提出建議的,最終該項(xiàng)建議被通過。這使得STL中的一些算法在處理特殊情形時(shí)可以選擇非一般化的方式,從而保證了執(zhí)行的效率。 


              STL是最新的C++標(biāo)準(zhǔn)函數(shù)庫中的一個(gè)子集,這個(gè)龐大的子集占據(jù)了整個(gè)庫的大約80%的分量。而作為在實(shí)現(xiàn)STL過程中扮演關(guān)鍵角色的模板則充斥了幾乎整個(gè)C++標(biāo)準(zhǔn)函數(shù)庫。在這里,我們有必要看一看C++標(biāo)準(zhǔn)函數(shù)庫里包含了哪些內(nèi)容,其中又有哪些是屬于標(biāo)準(zhǔn)模板庫(即STL)的。 

              C++標(biāo)準(zhǔn)函數(shù)庫為C++程序員們提供了一個(gè)可擴(kuò)展的基礎(chǔ)性框架。我們從中可以獲得極大的便利,同時(shí)也可以通過繼承現(xiàn)有類,自己編制符合接口規(guī)范的容器、算法、迭代子等方式對之進(jìn)行擴(kuò)展。它大致包含了如下幾個(gè)組件: 

              C標(biāo)準(zhǔn)函數(shù)庫,基本保持了與原有C語言程序庫的良好兼容,盡管有些微變化。人們總會忍不住留戀過去的美好歲月,如果你曾經(jīng)是一個(gè)C程序員,對這一點(diǎn)一定體會頗深。或許有一點(diǎn)會讓你覺得奇怪,那就是在C++標(biāo)準(zhǔn)庫中存在兩套C的函數(shù)庫,一套是帶有.h擴(kuò)展名的(比如),而另一套則沒有(比如)。它們確實(shí)沒有太大的不同。 

              語言支持(language support)部分,包含了一些標(biāo)準(zhǔn)類型的定義以及其他特性的定義,這些內(nèi)容,被用于標(biāo)準(zhǔn)庫的其他地方或是具體的應(yīng)用程序中。 

              診斷(diagnostics)部分,提供了用于程序診斷和報(bào)錯(cuò)的功能,包含了異常處理(exception handling),斷言(assertions),錯(cuò)誤代碼(error number codes)三種方式。 

              通用工具(general utilities)部分,這部分內(nèi)容為C++標(biāo)準(zhǔn)庫的其他部分提供支持,當(dāng)然你也可以在自己的程序中調(diào)用相應(yīng)功能。比如:動態(tài)內(nèi)存管理工具,日期/時(shí)間處理工具。記住,這里的內(nèi)容也已經(jīng)被泛化了(即采用了模板機(jī)制)。 

              字符串(string)部分,用來代表和處理文本。它提供了足夠豐富的功能。事實(shí)上,文本是一個(gè)string對象,它可以被看作是一個(gè)字符序列,字符類型可能是char,或者wchar_t等等。string可以被轉(zhuǎn)換成char*類型,這樣便可以和以前所寫的C/C++
            代碼和平共處了。因?yàn)槟菚r(shí)侯除了char*,沒有別的。 

              國際化(internationalization)部分,作為OOP特性之一的封裝機(jī)制在這里扮演著消除文化和地域差異的角色,采用locale和facet可以為程序提供眾多國際化支持,包括對各種字符集的支持,日期和時(shí)間的表示,數(shù)值和貨幣的處理等等。畢竟,在中國和在美國,人們表示日期的習(xí)慣是不同的。 

              容器(containers)部分,STL的一個(gè)重要組成部分,涵蓋了許多數(shù)據(jù)結(jié)構(gòu),比如前面曾經(jīng)提到的鏈表,還有:vector(類似于大小可動態(tài)增加的數(shù)組)、queue(隊(duì)列)、stack(堆棧)……。string也可以看作是一個(gè)容器,適用于容器的方法同樣也適用于string。現(xiàn)在你可以輕松的完成數(shù)據(jù)結(jié)構(gòu)課程的家庭作業(yè)了。

              算法(algorithms)部分,STL的一個(gè)重要組成部分,包含了大約70個(gè)通用算法,用于操控各種容器,同時(shí)也可以操控內(nèi)建數(shù)組。比如:find用于在容器中查找等于某個(gè)特定值的元素,for_each用于將某個(gè)函數(shù)應(yīng)用到容器中的各個(gè)元素上,sort用于對容器中的元素排序。所有這些操作都是在保證執(zhí)行效率的前提下進(jìn)行的,所以,如果在你使用了這些算法之后程序變得效率底下,首先一定不要懷疑這些算法本身,仔細(xì)檢查一下程序的其他地方。 

              迭代器(iterators)部分,STL的一個(gè)重要組成部分,如果沒有迭代器的撮合,容器和算法便無法結(jié)合的如此完美。事實(shí)上,每個(gè)容器都有自己的迭代器,只有容器自己才知道如何訪問自己的元素。它有點(diǎn)像指針,算法通過迭代器來定位和操控容器中的元素。 

              數(shù)值(numerics)部分,包含了一些數(shù)學(xué)運(yùn)算功能,提供了復(fù)數(shù)運(yùn)算的支持。 

              輸入/輸出(input/output)部分,就是經(jīng)過模板化了的原有標(biāo)準(zhǔn)庫中的iostream部分,它提供了對C++程序輸入輸出的基本支持。在功能上保持了與原有iostream的兼容,并且增加了異常處理的機(jī)制,并支持國際化(internationalization)。 

              總體上,在C++標(biāo)準(zhǔn)函數(shù)庫中,STL主要包含了容器、算法、迭代器。string也可以算做是STL的一部分。 

              正如前面所提到的,在STL的背后蘊(yùn)含著泛型化程序設(shè)計(jì)(GP)的思想,在這種思想里,大部分基本算法被抽象,被泛化,獨(dú)立于與之對應(yīng)的數(shù)據(jù)結(jié)構(gòu),用于以相同或相近的方式處理各種不同情形。這一思想和面向?qū)ο蟮某绦蛟O(shè)計(jì)思想(OOP)不盡相同,因?yàn)椋贠OP中更注重的是對數(shù)據(jù)的抽象,即所謂抽象數(shù)據(jù)類型(Abstract Data Type),而算法則通常被附屬于數(shù)據(jù)類型之中。幾乎所有的事情都可以被看作類或者對象(即類的實(shí)例),通常,我們所看到的算法被作為成員函數(shù)(member function)包含在類(class)中,類和類則構(gòu)成了錯(cuò)綜復(fù)雜的繼承體系。 

              盡管在象C++這樣的程序設(shè)計(jì)語言中,你還可以用全局函數(shù)來表示算法,但是在類似于Java這樣的純面向?qū)ο蟮恼Z言中,全局函數(shù)已經(jīng)被"勒令禁止"了。因此,用Java來模擬GP思想是頗為困難的。如果你對前述的STL歷史還有印象的話,應(yīng)該記得Alexander Stepanove也曾用基于OOP的語言嘗試過實(shí)現(xiàn)GP思想,但是效果并不好,包括沒有引入模板之前的C++語言。站在巨人的肩膀上,我們可以得出這樣的結(jié)論,在OOP中所體現(xiàn)的思想與GP的思想確實(shí)是相異的。C++并不是一種純面向?qū)ο蟮某绦蛟O(shè)計(jì)語言,它的絕妙之處,就在于既滿足了OOP,又成全了GP。對于后者,模板立下了汗馬功勞。另外,需要指出的是,盡管GP和OOP有諸多不同,但這種不同還不至于到"水火不容"的地步。并且,在實(shí)際運(yùn)用的時(shí)候,兩者的結(jié)合使用往往可以使問題的解決更為有效。作為GP思想實(shí)例的STL本身便是一個(gè)很好的范例,如果沒有繼承,不知道STL會是什么樣子,似乎沒有人做過這樣的試驗(yàn)。
              相信你對STL的感性認(rèn)識應(yīng)該有所提高了,是該做一些實(shí)際的工作了,那么我們首先來了解一下STL的不同實(shí)現(xiàn)版本。ANSI/ISO C++文件中的STL是一個(gè)僅被描述在紙上的標(biāo)準(zhǔn),對于諸多C++編譯器而言,需要有各自實(shí)際的STL,它們或多或少的實(shí)現(xiàn)了標(biāo)準(zhǔn)中所描述的內(nèi)容,這樣才能夠?yàn)槲覀兯谩V杂胁煌膶?shí)現(xiàn)版本,則存在諸多原因,有歷史的原因,也有各自編譯器生產(chǎn)廠商的原因。以下是幾個(gè)常見的STL實(shí)現(xiàn)版本。 

              HP STL是所有其它STL實(shí)現(xiàn)版本的根源。它是STL之父Alexander Stepanov在惠普的Palo Alto實(shí)驗(yàn)室工作時(shí),和Meng Lee共同完成的,是第一個(gè)STL的實(shí)現(xiàn)版本(參見1.2節(jié))。這個(gè)STL是開放源碼的,所以它允許任何人免費(fèi)使用、復(fù)制、修改、發(fā)布和銷售該軟件和相關(guān)文檔,前提是必須在所有相關(guān)文件中加入HP STL的版本信息和授權(quán)信息。現(xiàn)在已經(jīng)很少直接使用這個(gè)版本的STL了。 

              P. J. Plauger STL屬于個(gè)人作品,由P. J. Plauger本人實(shí)現(xiàn),是HP STL的一個(gè)繼承版本,因此在其所有頭文件中都含有HP STL的相關(guān)聲明,同時(shí)還有P. J. Plauger本人的版權(quán)聲明。P. J. Plauger是標(biāo)準(zhǔn)C中stdio庫的早期實(shí)現(xiàn)者,現(xiàn)在是C/C++ User's Journal的主編,與Microsoft保持著良好的關(guān)系。P. J. Plauger STL便是被用于Microsoft的Visual C++中的。在Windows平臺下的同類版本中,其性能不錯(cuò),但是queue組件(隊(duì)列,一種容器)的效率不理想,同時(shí)由于Visual C++對C++語言標(biāo)準(zhǔn)的支持不是很好(至少直到VC6.0為止,還是如此),因此一定程度上影響了P. J. Plauger STL的性能。此外,該版本的源代碼可讀性較差,你可以在VC的Include子目錄下找到所有源文件(比如:C:\Program Files\Microsoft Visual Studio\VC98\Include)。因?yàn)椴皇情_放源碼的(open source),所以這些源代碼是不能修改和銷售的,目前P.J. Plauger STL由Dinkumware公司提供相關(guān)服務(wù). 

              Rouge Wave STL是由Rouge Wave公司實(shí)現(xiàn)的,也是HP STL的一個(gè)繼承版本,除了HP STL的相關(guān)聲明之外,還有Rouge Wave公司的版權(quán)聲明。同時(shí),它也不是開放源碼的,因此無法修改和銷售。該版本被Borland C++ Builder所采用,你可以在C++ Builder的Include子目錄下找到所有頭文件(比如:C:\Program Files\Borland\Cbuilder5\Include)。盡管Rouge Wave STL的性能不是很好,但由于C++ Builder對C++語言標(biāo)準(zhǔn)的支持還算不錯(cuò),使其表現(xiàn)在一定程度上得以改善。此外,其源代碼的可讀性較好。可以從如下網(wǎng)站得到更詳細(xì)的情況介紹: 

              STLport最初源于俄國人Boris Fomitchev的一個(gè)開發(fā)項(xiàng)目,主要用于將SGI STL的基本代碼移植到其他諸如C++Builder或者是Visual C++這樣的主流編譯器上。因?yàn)镾GI STL屬于開放源碼,所以STLport才有權(quán)這樣做。目前STLport的最新版本是4.5。可以從如下網(wǎng)站得到更詳細(xì)的情況介紹: 

              SGI STL是由Silicon Graphics Computer System, Inc公司實(shí)現(xiàn)的,其設(shè)計(jì)者和編寫者包括Alexander Stepanov和Matt Austern,同樣它也是HP STL的一個(gè)繼承版本。它屬于開放源碼,因此你可以修改和銷售它。SGI STL被GCC(linux下的C++編譯器)所采用,你可以在GCC的Include子目錄下找到所有頭文件(比如:C:\cygnus\cygwin-b20\include\g++\include)。由于GCC對C++語言標(biāo)準(zhǔn)的支持很好,SGI STL在linux平臺上的性能相當(dāng)出色。此外,其源代碼的可讀性也很好。



            posted @ 2010-06-02 00:17 幽夢新影 閱讀(479) | 評論 (0)編輯 收藏

            C++類庫:OTL通用的數(shù)據(jù)庫連接類庫

            OTL是一個(gè)純C++的通用數(shù)據(jù)庫連接模板庫,可以支持各種當(dāng)下流行的數(shù)據(jù)庫,如Oracle,Sybase, MySQL, PostgreSQL, EnterpriseDB, SQLite,  MS ACCESS, Firebird等等.它是一個(gè)跨平臺類庫,在MS Windows, Linux/Unix/Mac OS X 都可以使用。

            下載網(wǎng)址:http://otl.sourceforge.net/

            OTL使用簡單, 只要頭文件中包含有: #include "otlv4.h" 就可,實(shí)際上整個(gè)OTL就一個(gè).H的文件,使用起來極為的方便。

            OTL使用方法:

            1、首先指定要連接的數(shù)據(jù)庫類型,OTL用宏定義來指定要連接的數(shù)據(jù)庫類型。OTL會根據(jù)這個(gè)宏定義來初始化數(shù)據(jù)庫連接的環(huán)境。

            相關(guān)的宏定義列表 http://otl.sourceforge.net/otl3_compile.htm

            如: #define OTL_ORA8I 表示連接Oracle 8i 數(shù)據(jù)庫。

             

            2、例子

            #include <iostream>
            using namespace std;
            #include <stdio.h>

            #define OTL_ORA8I // 指定連接的數(shù)據(jù)庫類型
            #include <otlv4.h> // include the OTL 4 header file

            otl_connect db; // 定義數(shù)據(jù)庫實(shí)例

             

            void insert()
            // 添加記錄

            {
             otl_stream o(50, // buffer size
                          "insert into test_tab values(:f1<float>,:f2<char[31]>)",
                             // SQL statement
                          db // connect object
                         );
             char tmp[32];

             for(int i=1;i<=100;++i){
            #if defined(_MSC_VER)
            #if (_MSC_VER >= 1400) // VC++ 8.0 or higher
              sprintf_s(tmp,sizeof(tmp),"Name%d",i);
            #else
              sprintf(tmp,"Name%d",i);
            #endif
            #else
              sprintf(tmp,"Name%d",i);
            #endif
              o<<static_cast<float>(i)<<tmp;
             }
            }

            //查詢記錄

            void select()
            {
             otl_stream i(50, // buffer size
                          "begin "
                           open :cur1 for "
                            select * from test_tab "
                            where f1>=:f<int> and f1<=:f*2; "
                          "end;", // SELECT statement via referenced cursor
                          db, // connect object
                          ":cur1" // referenced cursor placeholder name
                         );
               // create select stream
             
             otl_column_desc* desc;
             int desc_len;

             i<<4; // Before the child SELECT statement can be described,
                   // the PL/SQL master block needs to be executed.
                   // Forcing the execution of the blcok by entering its input
                   // patameter.

             desc=i.describe_select(desc_len);

             for(int n=0;n<desc_len;++n){
              cout<<"========== COLUMN #"<<n+1<<" ==========="<<endl;
              cout<<"name="<<desc[n].name<<endl;
              cout<<"dbtype="<<desc[n].dbtype<<endl;
              cout<<"otl_var_dbtype="<<desc[n].otl_var_dbtype<<endl;
              cout<<"dbsize="<<desc[n].dbsize<<endl;
              cout<<"scale="<<desc[n].scale<<endl;
              cout<<"prec="<<desc[n].prec<<endl;
              cout<<"nullok="<<desc[n].nullok<<endl;
             }
             

            }

            int main()
            {
             otl_connect::otl_initialize(); // 初始化Oracle環(huán)境
             try{

              db.rlogon("scott/tiger"); // 連接Oracle

              otl_cursor::direct_exec
               (
                db,
                "drop table test_tab",
                otl_exception::disabled // disable OTL exceptions
               ); // drop table

              otl_cursor::direct_exec
               (
                db,
                "create table test_tab(f1 number, f2 varchar2(30))"
                );  // create table

              insert(); 
              select();

             }

             catch(otl_exception& p){ // intercept OTL exceptions
              cerr<<p.msg<<endl; // print out error message
              cerr<<p.stm_text<<endl; // print out SQL that caused the error
              cerr<<p.var_info<<endl; // print out the variable that caused the error
             }

             db.logoff(); //斷開數(shù)據(jù)庫連接

             return 0;

            }

            posted @ 2010-06-02 00:05 幽夢新影 閱讀(6334) | 評論 (1)編輯 收藏

            2010年6月1日 #

            boost庫linux編譯安裝

                相對于Windows來,Linux下的boost編譯簡單至極。沒有那么多的可選編譯器,沒有那長的編譯時(shí)間,沒有那么多的硬盤使用量,統(tǒng)一的inlude和lib目錄,你熟悉命令行,不使用IDE,不需要我那么羅嗦的介紹怎么配置EditPlus。

            首先是下載boost,可以在此
            http://sourceforge.net/projects/boost
            尋找一個(gè)合適的版本。比如我下載的是boost_1_33_1.tar.gz,解壓到/opt。

            tar xzvf boost_1_33_1.tar.gz -C/opt

            提醒:做這些事情的時(shí)候你需要有root權(quán)限。

            進(jìn)入boost目錄:

            cd /opt/boost_1_33_1

            首先我們要編譯bjam:

            cd tools/build/jam_src/
            ./build.sh

            很快編譯結(jié)束,默認(rèn)情況下,bjam會被復(fù)制到/usr/local/bin/bjam。

            現(xiàn)在你可以使用bjam編譯boost了。

            cd ../../..
            bjam -sTOOLS=gcc install

            編譯時(shí)間不會如windows那么長久,在我的電腦上編譯了大約40分鐘。你可以在前后使用df命令檢查下磁盤使用,在我的電腦上,編譯boost花費(fèi)了500M的空間。

            使用install會把頭文件復(fù)制到/usr/local/include/boost-1_33_1中,把生成的lib復(fù)制到/usr/local/lib中。這些完成之后,記得要使用ldconfig來更新動態(tài)鏈接庫。

            在測試兩個(gè)例子之前,我們先設(shè)置幾個(gè)環(huán)境變量。

            BOOST_ROOT=/opt/boost_1_33_1
            BOOST_INCLUDE=/usr/local/include/boost-1_33_1
            BOOST_LIB=/usr/local/lib

            為了使其能夠在登錄時(shí)自動導(dǎo)入,你可以寫一個(gè)腳本:
            #!/bin/sh
            #boost settings
            BOOST_ROOT=/opt/boost_1_33_1
            BOOST_INCLUDE=/usr/local/include/boost-1_33_1
            BOOST_LIB=/usr/local/lib
            export BOOST_ROOT BOOST_INCLUDE BOOST_LIB

            將其保存為/etc/profile.d/boost.sh,并使用chmod a+x boost.sh設(shè)置執(zhí)行權(quán)限。

            現(xiàn)在我們可以寫兩段代碼來測試了。

            第一個(gè)測試文件是lex.cpp:

            #include <boost/lexical_cast.hpp>
            #include <iostream>

            int main()
            {
            using boost::lexical_cast;
            int a = lexical_cast<int>("123");
            double b = lexical_cast<double>("123.12");
            std::cout<<a<<std::endl;
            std::cout<<b<<std::endl;
            return 0;
            }

            編譯:
            g++ lex.cpp -I$BOOST_ROOT -o lex

            運(yùn)行:
            ./lex

            輸出:
            123
            123.12

            你可以將$BOOST_ROOT改為$BOOST_INCLUDE,如果你沒有設(shè)置環(huán)境變量,可以改為/opt/boost_1_33_1或者/usr/local/include/boost-1_33_1。

            我們的第二個(gè)例子是re.cpp:

            #include <iostream>
            #include <string>
            #include <boost/regex.hpp>

            int main() {

            std::string s = "who,lives:in-a,pineapple under the sea?";

            boost::regex re(",|:|-|\\s+");
            boost::sregex_token_iterator
            p(s.begin( ), s.end( ), re, -1);
            boost::sregex_token_iterator end;

            while (p != end)
            std::cout << *p++ << '\n';
            }

            編譯:
            g++ re.cpp -I$BOOST_ROOT -lboost_regex-gcc -o re

            運(yùn)行:
            ./re

            輸出:
            who
            lives
            in
            a
            pineapple
            under
            the
            sea?

            這里要使用-l指定了鏈接庫。

            現(xiàn)在boost的基本安裝配置已經(jīng)完成,但是我們可以再改進(jìn)下。

            如果不想每次都指定boost頭文件目錄,可以將其link到/usr/include中:

            ln -s /opt/boost_1_33_1/boost /usr/include/boost

            或者:

            ln -s /usr/local/include/boost-1_33_1/boost /usr/include/boost

            如果你依然嫌boost編譯后占用的空間太大,可以在boost目錄下使用bjam clean:
            cd /opt/boost_1_33_1
            bjam -sTOOLS=gcc clean

            這個(gè)命令會清除編譯時(shí)的中間文件,/usr/local/lib下帶版本號的boost libs,和/usr/local/include下的boost頭文件。但是同時(shí)節(jié)省了幾百M(fèi)的硬盤空間。

            所以如果你使用了clean,記得將BOOST_INCLUDE更為BOOST_ROOT(/opt/boost_1_33_1),將 /usr/include/boost link到/opt/boost_1_33_1/boost,再有就是編譯鏈接時(shí)的boost lib不要帶版本號。

            如果你覺得編譯時(shí)手動鏈接敲那么長的名字比較麻煩,可以使用腳本來自動尋找鏈接:

            #!/usr/bin/python

            import os
            import sys
            import re

            BOOST_ROOT = os.getenv('BOOST_ROOT')
            BOOST_LIB = os.getenv('BOOST_LIB')
            #BOOST_ROOT = '/opt/boost_1_33_1'
            #BOOST_LIB = '/usr/local/lib'

            def getlibs():
            alls = os.listdir(BOOST_LIB)
            libpattern = re.compile(r'^libboost_([^-]+)-gcc')
            libs = {}
            for lib in alls:
            m = libpattern.match(lib)
            if m:
            libs[m.group(1).lower()] = 1
            return libs


            pattern = re.compile(r'^\s*#include\s*<\s*boost/(.+)\.(h|hpp)\s*>')
            libs = getlibs()
            libskeys = libs.keys()
            includes = {}

            ENV = os.environ


            ARGV = sys.argv[1:]
            files = ARGV
            if len(files) == 0:
            sys.exit()

            for f in files:
            if f.lower().endswith('.cpp'):
            fp = open(f, 'r')
            lines = fp.readlines()
            for ln in lines:
            m = pattern.match(ln)
            if m:
            libname = m.group(1).lower()
            if libname in libskeys:
            includes[libname] = 1

            libline = ' '.join(map(lambda lib: '-lboost_'+lib+'-gcc', includes.keys()))


            obj = ARGV[0]
            obj = obj[:len(obj)-4]

            #cmd = 'g++ %s -I%s %s -o %s' % (' '.join(files), BOOST_ROOT, libline, obj)
            cmd = 'g++ %s %s -o %s' % (' '.join(files), libline, obj)

            print cmd
            os.system(cmd)

            將這段代碼寫進(jìn)/usr/local/bin/gccboost,賦予執(zhí)行權(quán)限。

            使用方法:
            gccboost lex.cpp
            gccboost re.cpp

            注意:使用此命令假設(shè)boost頭文件在/usr/include中,如果假設(shè)不成立,請自行修改腳本此行:
            cmd = 'g++ %s %s -o %s' % (' '.join(files), libline, obj)

            為之前的注釋行:
            cmd = 'g++ %s -I%s %s -o %s' % (' '.join(files), BOOST_ROOT, libline, obj)

            如若BOOST_ROOT和BOOST_LIB環(huán)境變量不存在,修改下面兩行代碼:
            BOOST_ROOT = os.getenv('BOOST_ROOT')
            BOOST_LIB = os.getenv('BOOST_LIB')

            為之后注釋行:
            BOOST_ROOT = '/opt/boost_1_33_1'
            BOOST_LIB = '/usr/local/lib'

            另外,gccboost將會自動修改輸出的文件名為*.cpp的文件名(如lex.cpp將輸出lex),如果不需要,請將下面的代碼:
            cmd = 'g++ %s %s -o %s' % (' '.join(files), libline, obj)

            改為:
            cmd = 'g++ %s %s' % (' '.join(files), libline)


            Boost安裝歷程至此基本結(jié)束。

            posted @ 2010-06-01 23:36 幽夢新影 閱讀(7524) | 評論 (0)編輯 收藏

            在Linux系統(tǒng)中安裝和配置ACE(轉(zhuǎn))

            本文詳細(xì)描述了在Linux操作系統(tǒng)上安裝和配置“自適應(yīng)通信環(huán)境(ACE)”的過程和注意事項(xiàng)。
            文中的描述基于RedHat Linux 9操作系統(tǒng)和ACE 5.4版

             

            準(zhǔn)備工作:

             

            1.確保你的Linux操作系統(tǒng)的開發(fā)環(huán)境(gcc、make……)是完整且可用的。
            2.在安裝ACE的過程中,可能需要root權(quán)限,因此確保你擁有root帳戶的密碼。
            3.對于典型的Redhat9系統(tǒng),autoconf程序的版本可能過低,你需要首先升級它。

             

               升級的步驟如下:

             

            1.在控制臺中輸入 autoconf -V,查看你的autoconf程序的版本
            2.如果低于2.57g,你需要到http://www.gnu.org/software/autoconf/去下載最新版本的autoconf程序,目前autoconf程序的版本是2.59
            3.根據(jù)tarball中的安裝說明,編譯并安裝升級autoconf程序
            開始安裝:

             

            經(jīng)過上面的準(zhǔn)備工作,現(xiàn)在你已經(jīng)具備了安裝ACE的條件。下面開始正式的安裝工作。

             

            1.首先確定ACE的安裝路徑,并建立相應(yīng)的目錄。
            這里,我們把ACE安裝在/home/zhc/ace/的目錄下,先執(zhí)行
            mkdir /home/zhc/ace
            2.然后將獲得的ACE源代碼包解至該目錄下。
            tar zxvf ACE-5.4.tar.gz /home/zhc/ace
            3.這時(shí),在ace的目錄下會建立一個(gè)名為ACE_wrappers的子目錄,ACE所有的源代碼、工具、和例子都在這個(gè)目錄中。
            先進(jìn)入該目錄,執(zhí)行
            cd /home/zhc/ace/ACE_wrappers
            然后建立一個(gè)子目錄,用于存放ACE編譯后的目標(biāo)文件
            mkdir build
            cd build
            4.接著,執(zhí)行configure腳本,檢查你的系統(tǒng)環(huán)境,配置并生成適合你的系統(tǒng)的Makefile文件。
            在執(zhí)行configure腳本時(shí),可以加入許多配置選項(xiàng),其詳細(xì)說明可以參看ACE_wrappers目錄下的ACE-INSTALL.html文件,
            對于Redhat9系統(tǒng)來說,不必加任何參數(shù),ACE就可以很好的工作了。
            ../configure
            5.在漫長的檢查和配置之后,如果沒有什么問題,你就可以開始編譯了(如果要讓ace支持ICMP 和對c++庫的支持,需要在config.h頭文件中加上#define ACE_HAS_STANDARD_CPP_LIBRARY 1 和#define ACE_HAS_ICMP_SUPPORT 1)。
            這一步很簡單,執(zhí)行
            make
            然后,泡一杯茶,靜靜的等待就行了。
            6.如果編譯順利的通過(通常都會是這樣的),那么恭喜你,你已經(jīng)成功了一大半了,下面要做的僅僅是以些配置工作而已。
            接下來的工作是安裝,這一步是可選的,目的是把編譯生成的ACE的庫文件、頭文件和手冊頁等安裝到/usr/local的相應(yīng)目錄中去,為了今后使用的方便,我建議你還是進(jìn)行這一步,執(zhí)行
            su
            root's password
            make install
            7.安裝之后,我們就可以進(jìn)行最后的配置工作了。
            首先設(shè)置環(huán)境變量ACE_ROOT,執(zhí)行
            setenv ACE_ROOT /home/zhc/ace/ACE_wrappers
            注意其中的路徑要根據(jù)你的實(shí)際情況而定
            如果你不想每次都手工設(shè)定路徑,可以編輯你的bash的配置文件,執(zhí)行
            cd
            vi .bash_profile
            在其中填加如下兩行
            ACE_ROOT=/home/work/research/ace/ACE_wrappers
            export ACE_ROOT
            8.然后,在$ACE_ROOT/ace目錄中創(chuàng)建一個(gè)名為config.h的文件,在其中包含如下一行
            #include "ace/config-linux.h"
            或者,簡單起見,你也可以直接建立一個(gè)指向config-linux.h的符號鏈接,執(zhí)行
            ln -s config-linux.h config.h
            9.同樣,在$ACE_ROOT/include/makeinclude目錄中創(chuàng)建一個(gè)名為platform_macros.GNU的文件,在其中包含如下一行
            include $(ACE_ROOT)/include/makeinclude/platform_linux.GNU
            或者,簡單起見,你也可以直接建立一個(gè)指向platform_linux.GNU的符號鏈接,執(zhí)行
            ln -s platform_linux.GNU platform_macros.GNU
            10.最后一步,設(shè)置環(huán)境變量LD_LIBRARY_PATH,執(zhí)行
            setenv LD_LIBRARY_PATH $ACE_ROOT/ace:$ACE_ROOT/lib:
            /usr/local/lib:$LD_LIBRARY_PATH
            或者在~/.bash_profile文件中添加如下兩行
            LD_LIBRARY_PATH=$ACE_ROOT/ace:$ACE_ROOT/lib:
            /usr/local/lib:$LD_LIBRARY_PATH
            export LD_LIBRARY_PATH
            安裝完成:

             

            經(jīng)過上面的步驟,ACE已經(jīng)安裝到了您的系統(tǒng)上了,下面我們將編譯并執(zhí)行ACE的例程以驗(yàn)證我們的工作。
            這里我們以《C++網(wǎng)絡(luò)編程,卷一》所帶的例子為例。

             

            1:進(jìn)入例程目錄,執(zhí)行
            cd $(ACE_ROOT)/examples/C++NPv1
            2:編譯例子程序,執(zhí)行
            make
            如果前面的安裝和配置正確,很快就能夠得到編譯后的例子程序。
            3:分別執(zhí)行一個(gè)服務(wù)器程序(隨便哪一個(gè)都行)和一個(gè)客戶端程序(Logging_Client),如果程序運(yùn)行正確,說明前面的安裝和配置成功,下面你就可以開始你的ACE學(xué)習(xí)之旅了

            posted @ 2010-06-01 23:31 幽夢新影 閱讀(482) | 評論 (0)編輯 收藏

            2009年8月28日 #

            開始C++的歷程

            今天正式開始~~

            posted @ 2009-08-28 21:25 幽夢新影 閱讀(221) | 評論 (0)編輯 收藏

            僅列出標(biāo)題  
            天天做夜夜做久久做狠狠| 麻豆成人久久精品二区三区免费| 国产精品一区二区久久精品涩爱| 欧美亚洲另类久久综合| 色综合久久综合中文综合网| 久久强奷乱码老熟女网站| 国产精品久久久久久久久软件 | 日本久久中文字幕| 国产精品久久久久jk制服| 久久精品国产亚洲av高清漫画| 久久久婷婷五月亚洲97号色| 欧美午夜A∨大片久久 | 99久久亚洲综合精品网站| 国产69精品久久久久观看软件| 久久免费视频网站| 久久精品中文闷骚内射| 久久人人爽人人人人片av| 久久男人中文字幕资源站| 亚洲人成网站999久久久综合| 亚洲欧美成人综合久久久| 久久综合视频网站| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 久久免费视频网站| AV无码久久久久不卡网站下载 | 色婷婷噜噜久久国产精品12p| 亚洲国产精品久久久久网站| 国内精品久久久人妻中文字幕| 国产成人精品综合久久久久| 国产91色综合久久免费分享| 久久精品国产亚洲一区二区三区| 综合久久精品色| 亚洲国产综合久久天堂| 亚洲欧美伊人久久综合一区二区 | 久久电影网| 国产精品成人99久久久久91gav | 2021久久精品国产99国产精品| 久久久久无码精品国产| www久久久天天com| jizzjizz国产精品久久| 久久青草国产精品一区| 狠狠人妻久久久久久综合|