青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

今天微薄上看到 http://weibo.com/1401880315/AaNkykg6J#_rnd1379945435634

左耳朵耗子:
現在的程序員,連atoi()都不知道是什么了,沒事,那我改,不叫atoi()改叫StrToInt(),卻發現,好些人連ASCII碼都不知道是怎么一回事,沒事,我教會你。但最終卻發現怎么有這么多人連這樣簡單的程序都寫不好(包括有多年工作經驗的人)。“比技術更恐怖的是有一群不合格的程序員在使用這這些技術”。


好久沒寫這么基本的代碼了,簡單嘗試寫了下, 結果花了半個小時,只寫了一個最基本功能的,通過后面的測試用例還花了不少時間進行調試修改。 
INT StrToInt(const TCHAR* lpszValue)
{
    assert(lpszValue != NULL);

    INT nLen = lstrlen(lpszValue);
    const TCHAR* pEnd = lpszValue + nLen;

    TCHAR* pCurrent = (TCHAR*)lpszValue;
    BOOL bNegative(FALSE);
    if(nLen >= 1)
    {
        if(*pCurrent == _T('+'))
        {
            bNegative = FALSE;
            pCurrent += 1;
        }
        else if(*pCurrent == _T('-'))
        {
            bNegative = TRUE;
            pCurrent += 1;
        }
        else
        {
            bNegative = FALSE;
        }
    }

    INT nBase(10);
    if(pEnd - pCurrent >= 2)
    {
        if(pCurrent[0] == _T('0')
            && ::toupper(pCurrent[1]) == _T('X'))
        {
            pCurrent += 2;
            nBase = 16;
        }
    }

    INT nRet(0);
    INT nValue(0);
    while(pCurrent != pEnd)
    {
        TCHAR ch(*pCurrent);
        if(ch >= _T('0') && ch <= _T('9'))
        {
            nValue = ch - _T('0');
        }
        else if(nBase == 16)
        {
            if(::toupper(ch) >= _T('A') && ::toupper(ch) <= _T('F'))
            {
                nValue = 10 + (::toupper(ch) - _T('A'));
            }
            else
            {
                assert(FALSE);
                break;
            }
        }
        else
        {
            assert(FALSE);
            break;
        }

        nRet += nValue * pow((double)nBase, pEnd - pCurrent - 1);
        pCurrent += 1;
    }

    return bNegative ?  -nRet : nRet;
}

void test
{
    assert(StrToInt(_T("11")) == 11);
    assert(StrToInt(_T("+12")) == 12); 
    assert(StrToInt(_T("-123")) == -123); 
    assert(StrToInt(_T("-0x1CF")) == -0x1CF); 
    assert(StrToInt(_T("-0X123")) == -0x123); 
    assert(StrToInt(_T("0X123")) == 0x123);
}

感慨用慣了Windows API和STL, 對于最基本的字符串處理代碼反而寫不好了。細想一下這個基本的東西確實不好寫, 實際上我是上面只是考慮了10進制和16進制, 沒有考慮其他進制,也沒有考慮小數,非法的字符串或是溢出等情況, 而真正工業級的庫要考慮所有的情況, 另外還要考慮轉換效率等問題。

實際上CRT源碼中有這個函數的實現:
/***
*wcstol, wcstoul(nptr,endptr,ibase) - Convert ascii string to long un/signed
*       int.
*
*Purpose:
*       Convert an ascii string to a long 32-bit value.  The base
*       used for the caculations is supplied by the caller.  The base
*       must be in the range 0, 2-36.  If a base of 0 is supplied, the
*       ascii string must be examined to determine the base of the
*       number:
*           (a) First char = '0', second char = 'x' or 'X',
*               use base 16.
*           (b) First char = '0', use base 8
*           (c) First char in range '1' - '9', use base 10.
*
*       If the 'endptr' value is non-NULL, then wcstol/wcstoul places
*       a pointer to the terminating character in this value.
*       See ANSI standard for details
*
*Entry:
*       nptr == NEAR/FAR pointer to the start of string.
*       endptr == NEAR/FAR pointer to the end of the string.
*       ibase == integer base to use for the calculations.
*
*       string format: [whitespace] [sign] [0] [x] [digits/letters]
*
*Exit:
*       Good return:
*           result
*
*       Overflow return:
*           wcstol -- LONG_MAX or LONG_MIN
*           wcstoul -- ULONG_MAX
*           wcstol/wcstoul -- errno == ERANGE
*
*       No digits or bad base return:
*           0
*           endptr = nptr*
*
*Exceptions:
*       Input parameters are validated. Refer to the validation section of the function.
*
*******************************************************************************/

/* flag values */
#define FL_UNSIGNED   1       /* wcstoul called */
#define FL_NEG        2       /* negative sign found */
#define FL_OVERFLOW   4       /* overflow occured */
#define FL_READDIGIT  8       /* we've read at least one correct digit */


static unsigned long __cdecl wcstoxl (
        _locale_t plocinfo,
        const wchar_t *nptr,
        const wchar_t **endptr,
        int ibase,
        int flags
        )
{
    const wchar_t *p;
    wchar_t c;
    unsigned long number;
    unsigned digval;
    unsigned long maxval;
    _LocaleUpdate _loc_update(plocinfo);


    /* validation section */
    if (endptr != NULL)
    {
        /* store beginning of string in endptr */
        *endptr = nptr;
    }
    _VALIDATE_RETURN(nptr != NULL, EINVAL, 0L);
    _VALIDATE_RETURN(ibase == 0 || (2 <= ibase && ibase <= 36), EINVAL, 0L);

    p = nptr;           /* p is our scanning pointer */
    number = 0;         /* start with zero */

    c = *p++;           /* read char */

    while ( _iswspace_l(c, _loc_update.GetLocaleT()) )
        c = *p++;       /* skip whitespace */

    if (c == '-') {
        flags |= FL_NEG;    /* remember minus sign */
        c = *p++;
    }
    else if (c == '+')
        c = *p++;       /* skip sign */

    if (ibase == 0) {
        /* determine base free-lance, based on first two chars of
           string */
        if (_wchartodigit(c) != 0)
            ibase = 10;
        else if (*p == L'x' || *p == L'X')
            ibase = 16;
        else
            ibase = 8;
    }

    if (ibase == 16) {
        /* we might have 0x in front of number; remove if there */
        if (_wchartodigit(c) == 0 && (*p == L'x' || *p == L'X')) {
            ++p;
            c = *p++;   /* advance past prefix */
        }
    }

    /* if our number exceeds this, we will overflow on multiply */
    maxval = ULONG_MAX / ibase;


    for (;;) {  /* exit in middle of loop */

        /* convert c to value */
        if ( (digval = _wchartodigit(c)) != -1 )
            ;
        else if ( __ascii_iswalpha(c))
            digval = __ascii_towupper(c) - L'A' + 10;
        else
            break;

        if (digval >= (unsigned)ibase)
            break;      /* exit loop if bad digit found */

        /* record the fact we have read one digit */
        flags |= FL_READDIGIT;

        /* we now need to compute number = number * base + digval,
           but we need to know if overflow occured.  This requires
           a tricky pre-check. */

        if (number < maxval || (number == maxval &&
        (unsigned long)digval <= ULONG_MAX % ibase)) {
            /* we won't overflow, go ahead and multiply */
            number = number * ibase + digval;
        }
        else {
            /* we would have overflowed -- set the overflow flag */
            flags |= FL_OVERFLOW;
            if (endptr == NULL) {
                /* no need to keep on parsing if we
                   don't have to return the endptr. */
                break;
            }
        }

        c = *p++;       /* read next digit */
    }

    --p;                /* point to place that stopped scan */

    if (!(flags & FL_READDIGIT)) {
        /* no number there; return 0 and point to beginning of
           string */
        if (endptr)
            /* store beginning of string in endptr later on */
            p = nptr;
        number = 0L;        /* return 0 */
    }
    else if ( (flags & FL_OVERFLOW) ||
          ( !(flags & FL_UNSIGNED) &&
            ( ( (flags & FL_NEG) && (number > -LONG_MIN) ) ||
              ( !(flags & FL_NEG) && (number > LONG_MAX) ) ) ) )
    {
        /* overflow or signed overflow occurred */
        errno = ERANGE;
        if ( flags & FL_UNSIGNED )
            number = ULONG_MAX;
        else if ( flags & FL_NEG )
            number = (unsigned long)(-LONG_MIN);
        else
            number = LONG_MAX;
    }

    if (endptr != NULL)
        /* store pointer to char that stopped the scan */
        *endptr = p;

    if (flags & FL_NEG)
        /* negate result if there was a neg sign */
        number = (unsigned long)(-(long)number);

    return number;          /* done. */
}

透過這道題確實可以投射出一個程序員的計算機基本功, 大家可以自己嘗試下實現這個函數, 看看自己的計算機基本功。
posted on 2013-09-23 22:39 Richard Wei 閱讀(2765) 評論(0)  編輯 收藏 引用 所屬分類: C++
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            av成人免费在线观看| 午夜久久久久久| 中文久久精品| 在线一区亚洲| 小黄鸭精品aⅴ导航网站入口| 一本一本久久a久久精品综合麻豆| 亚洲国产三级| 一区二区三区欧美在线观看| 亚洲视频你懂的| 欧美在线视频网站| 久久久久久久久久看片| 欧美区日韩区| 亚洲国产综合91精品麻豆| 日韩视频免费大全中文字幕| 日韩午夜激情| 欧美在线不卡视频| 欧美激情性爽国产精品17p| 欧美国产专区| 亚洲最新视频在线| 久久久久久久久久久久久9999| 欧美激情精品久久久久久久变态| 国产精品久久久一区二区| 影音先锋日韩资源| 亚洲伊人一本大道中文字幕| 久久精品一本| 一区二区三区四区蜜桃| 久久久99国产精品免费| 欧美视频久久| 亚洲国产裸拍裸体视频在线观看乱了 | 欧美在线免费观看亚洲| 久久久另类综合| 国产精品视频一区二区三区 | 欧美日韩在线播放三区| 激情久久久久久久久久久久久久久久 | 欧美色图五月天| 一区免费观看| 欧美一区二区三区的| 91久久香蕉国产日韩欧美9色| 亚洲欧美韩国| 欧美极品aⅴ影院| 在线免费不卡视频| 欧美一区二区三区久久精品 | 午夜亚洲影视| 国产精品久久久99| 在线综合视频| 亚洲激情视频网| 美国十次成人| 亚洲国产成人精品久久| 久久免费黄色| 欧美综合77777色婷婷| 国产精品久久久久久久久久直播 | 久久国产精品久久精品国产| 国产精品大全| 亚洲欧美精品suv| 一级成人国产| 国产精品麻豆成人av电影艾秋| 在线综合亚洲欧美在线视频| 国产亚洲亚洲| 国产日韩在线亚洲字幕中文| 亚洲一区二区三区精品动漫| 亚洲黄色毛片| 欧美激情第10页| 日韩视频精品在线观看| 亚洲国产美国国产综合一区二区| 美日韩精品视频| 亚洲国产精品一区二区第一页| 久久综合伊人77777尤物| 午夜久久美女| 激情欧美丁香| 欧美福利视频在线观看| 久久阴道视频| 日韩一级二级三级| 夜夜爽夜夜爽精品视频| 国产精品夫妻自拍| 久久国产欧美| 久久久久网址| 亚洲精品自在在线观看| 亚洲每日更新| 国产欧美日韩一区二区三区| 久久国产天堂福利天堂| 久久久国产精品一区| 亚洲精品你懂的| 一区二区三区高清视频在线观看| 国产精品丝袜91| 欧美va天堂在线| 欧美日韩直播| 久久综合伊人77777| 欧美喷水视频| 久久久精品视频成人| 美国十次成人| 欧美一区二区啪啪| 乱中年女人伦av一区二区| 日韩午夜在线视频| 欧美怡红院视频一区二区三区| 亚洲二区在线视频| 在线亚洲激情| 在线日韩一区二区| 亚洲私人影院| 亚洲日本激情| 久久精品亚洲国产奇米99| 一区二区三区高清视频在线观看| 午夜欧美大尺度福利影院在线看| 亚洲激情在线视频| 午夜欧美电影在线观看| 99视频在线观看一区三区| 欧美一区二区三区四区在线观看| 亚洲免费精品| 久久久久久久国产| 午夜亚洲视频| 欧美日韩国产不卡在线看| 久久久久久久久久久久久久一区| 欧美日韩精品一二三区| 欧美成人精品三级在线观看| 国产伦精品一区二区| 亚洲黄网站黄| 亚洲电影视频在线| 欧美在线观看一区二区| 亚洲女同精品视频| 欧美日韩蜜桃| 亚洲美女在线一区| 亚洲免费av观看| 国内精品久久久久久久影视蜜臀| 久久不射中文字幕| 欧美视频在线观看一区二区| 男人天堂欧美日韩| 国语自产偷拍精品视频偷| 亚洲综合色噜噜狠狠| 亚洲一区二区高清视频| 欧美巨乳波霸| 亚洲开发第一视频在线播放| 亚洲精品1区2区| 欧美成人69av| 亚洲精品国产拍免费91在线| 亚洲欧洲精品一区| 欧美高清视频一区二区三区在线观看| 欧美99在线视频观看| 国产综合婷婷| 久久精品国产欧美激情| 久久蜜桃资源一区二区老牛| 国产一区二区三区网站| 久久久久久久激情视频| 蜜臀91精品一区二区三区| 在线成人黄色| 欧美大片免费观看| 99riav久久精品riav| 亚洲在线电影| 国产亚洲综合性久久久影院| 欧美一区二区三区在线看| 久久久在线视频| 在线成人欧美| 欧美高清视频一区二区三区在线观看 | 亚洲国产网站| 欧美精品一区二区在线播放| 亚洲欧洲一区二区天堂久久| 99精品国产99久久久久久福利| 欧美日韩一区二区视频在线| 亚洲无人区一区| 久久人体大胆视频| 亚洲精品在线电影| 国产精品系列在线播放| 久久蜜桃精品| 日韩西西人体444www| 亚洲图片欧美午夜| 国产丝袜一区二区| 欧美mv日韩mv国产网站app| 一本色道久久| 久久最新视频| 亚洲一区三区视频在线观看| 国产一区视频在线观看免费| 欧美成人国产一区二区| 亚洲一二三区在线观看| 另类图片国产| 亚洲中午字幕| 最新日韩在线| 国产美女扒开尿口久久久| 久久在线免费| 亚洲欧美自拍偷拍| 亚洲乱码精品一二三四区日韩在线| 久久国产色av| 亚洲性图久久| 一区二区亚洲| 国产精品男gay被猛男狂揉视频| 久久精品国产第一区二区三区最新章节 | 日韩视频免费观看高清在线视频| 老司机精品视频一区二区三区| 亚洲国产成人精品女人久久久 | 美女国内精品自产拍在线播放| 99re6这里只有精品视频在线观看| 久久精品国产久精国产思思| 日韩午夜在线观看视频| 狠狠色伊人亚洲综合网站色| 欧美午夜一区二区三区免费大片| 久久视频一区| 欧美一区二区三区久久精品茉莉花| 亚洲精品免费看| 亚洲高清三级视频| 巨乳诱惑日韩免费av| 欧美亚洲自偷自偷| 亚洲一区二区精品视频| 亚洲精品国产精品国自产观看|