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

創(chuàng)新無(wú)極限

Sun Guoqing的Blog

C++博客 首頁(yè) 新隨筆 聯(lián)系 聚合 管理
  3 Posts :: 1 Stories :: 11 Comments :: 0 Trackbacks

打開(kāi)Source Insight來(lái)閱讀EduOS的源代碼,我們?cè)趕tdio.c里找到了printf的實(shí)現(xiàn)代碼.首先看看對(duì)printf的定義:
[code]
int printf (const char *cntrl_string, ...)
[/code]
第一個(gè)參數(shù)cntrl_string是控制字符串,也就是平常我們寫入%d,%f的地方.緊接著后面是一個(gè)變長(zhǎng)參數(shù).

看看函數(shù)頭部的定義:

  [code]int pos = 0, cnt_printed_chars = 0, i;
  unsigned char* chptr;
  va_list ap;[/code]
馬上暈!除了ap我們可以馬上判斷出來(lái)是用來(lái)讀取變長(zhǎng)參數(shù)的,i用于循環(huán)變量.其他變量都不知道是怎么回事.不要著急,我們邊看代碼邊分析.代碼的第一行必然是

[code]va_start (ap, cntrl_string);[/code]
用來(lái)初始化變長(zhǎng)參數(shù).

接下來(lái)是一個(gè)while循環(huán)

[code]while (cntrl_string[pos]) {
...
}[/code]

結(jié)束條件是cntrl_string[pos]為NULL,顯然這個(gè)循環(huán)是用來(lái)遍歷整個(gè)控制字符串的.自然pos就是當(dāng)前遍歷到的位置了.進(jìn)入循環(huán)首先闖入視線的是

[code] if (cntrl_string[pos] == '%') {
      pos++;
      ...
 } [/code]

開(kāi)門見(jiàn)山,上來(lái)就當(dāng)前字符是否辦斷是否%.一猜就知道如果成立pos++馬上取出下一個(gè)字符在d,f,l等等之間進(jìn)行判斷.往下一看,果真不出所料:

[code]switch (cntrl_string[pos]) {
    case 'c':
...
    case 's':
...
    case 'i':
...
    case 'd':
...
    case 'u':
...[/code]

用上switch-case了. 快速瀏覽一下下面的代碼.

首先看看case 'c'的部分

[code]case 'c':
 putchar (va_arg (ap, unsigned char));
 cnt_printed_chars++;
 break;[/code]

%c表示僅僅輸出一個(gè)字符.因此先通過(guò)va_arg進(jìn)行參數(shù)的類型轉(zhuǎn)換,之后用putchar[1]輸出到屏幕上去.之后是
cnt_printed_chars++,通過(guò)這句我們就可以判斷出cnt_printed_chars使用來(lái)表示,已經(jīng)被printf輸出的字符個(gè)數(shù)的.

再來(lái)看看 case 's':
[code]      case 's':
 chptr = va_arg (ap, unsigned char*);
 i = 0;
 while (chptr [i]) {
   cnt_printed_chars++;
   putchar (chptr [i++]);
 }
 break;[/code]和case 'c',同出一轍.cnt_printed_chars++放在了循環(huán)內(nèi),也證明了剛才提到的他的作用.另外我們也看到了cnptr是用來(lái)在處理字符串時(shí)的位置指針.到此為止,我們清楚的所有變量的用途,前途變得更加光明了.

接下來(lái):
[code]// PartI
      case 'i':
      case 'd':
 cnt_printed_chars += printInt (va_arg (ap, int));
 break;
      case 'u':
 cnt_printed_chars += printUnsignedInt (va_arg (ap, unsigned int));
 break;
      case 'x':
 cnt_printed_chars += printHexa (va_arg (ap, unsigned int), 'x');
 break;
      case 'X':
 cnt_printed_chars += printHexa (va_arg (ap, unsigned int), 'X');
 break;
      case 'o':
 cnt_printed_chars += printOctal (va_arg (ap, unsigned int));
 break;
// Part II
 case 'p':
 putchar ('0');
 putchar ('x');
 cnt_printed_chars += 2; /* of '0x' */
 cnt_printed_chars += printHexa (va_arg (ap, unsigned int), 'x');
 break;
      case '#':
 pos++;
 switch (cntrl_string[pos]) {
 case 'x':
   putchar ('0');
   putchar ('x');
   cnt_printed_chars += 2; /* of '0x' */
   cnt_printed_chars += printHexa (va_arg (ap, unsigned int), 'x');
   break;
 case 'X':
   putchar ('0');
   putchar ('X');
   cnt_printed_chars += 2; /* of '0X' */
   cnt_printed_chars += printHexa (va_arg (ap, unsigned int), 'X');
   break;
 case 'o':
   putchar ('0');
   cnt_printed_chars++;
   cnt_printed_chars += printOctal (va_arg (ap, unsigned int));
   break;[/code]
注意觀察一下,PartII的代碼其實(shí)就是比PartI的代碼多一個(gè)樣式.在16進(jìn)制數(shù)或八進(jìn)制前加入0x或是o,等等.因此這里就只分析一下PartI咯.

其實(shí)仔細(xì)看看PartI的個(gè)條case,也就是把參數(shù)分發(fā)到了更具體的函數(shù)用于顯示,然后以返回值的形式返回輸出個(gè)數(shù).對(duì)于這些函數(shù)就不具體分析了.我們先來(lái)看看一些善后處理:

先看case的default處理.
[code]default:
 putchar ((unsigned char) cntrl_string[pos]);
 cnt_printed_chars++;[/code]就是直接輸出cntrl_string里%號(hào)后面的未知字符.應(yīng)該是一種容錯(cuò)設(shè)計(jì)處理.

再看看if (cntrl_string[pos] == '%')的else部分
     
[code]else {
      putchar ((unsigned char) cntrl_string[pos]);
      cnt_printed_chars++;
      pos++;
 }[/code]
如果不是%開(kāi)頭的,那么直接輸出這個(gè)字符.

最后函數(shù)返回前
  [code]va_end (ap);
  return cnt_printed_chars;[/code]va_end處理變長(zhǎng)參數(shù)的善后工作.并返回輸出的字符個(gè)數(shù).

在最后我們有必要談?wù)刾utChar函數(shù)以及基本輸出的基礎(chǔ)函數(shù)printChar,先來(lái)看看putChar

[code]int putchar (int c) {
  switch ((unsigned char) c) {
  case '\n' :
    newLine ();
    break;
  case '\r' :
    carriageReturn ();
    break;
  case '\f' :
    clearScreen ();
    break;
  case '\t' :
    printChar (32); printChar (32); /* 32 = space */
    printChar (32); printChar (32);
    printChar (32); printChar (32);
    printChar (32); printChar (32);
    break;
  case '\b':
    backspace ();
    break;
  case '\a':
    beep ();
    break;
  default :
    printChar ((unsigned char) c);
  }
  return c;
}[/code]
通 覽一下,也是switch-case為主體的.主要是用來(lái)應(yīng)對(duì)一些特殊字符,如\n,\r,....這里需要提一下,關(guān)于\t的理解.有些人認(rèn)為\t就是 8個(gè)space,有些人則認(rèn)為,屏幕分為10大列(每個(gè)大列8個(gè)小列總共80列).一個(gè)\t就跳到下一個(gè)大列輸出.也就是說(shuō)不管你現(xiàn)在實(shí)在屏幕的第 1,2,3,4,5,6,7位置輸出字符,只要一個(gè)\t都在第8個(gè)位置開(kāi)始輸出. VS.NET中就是用的這種理解.因此如果按照這個(gè)理解的話,\t的實(shí)現(xiàn)可以這樣

[code]int currentX = ((currentX % 10) + 1) * 8;[/code]

然后在currentX位置輸出.

接下來(lái)看printChar也就是輸出部分最低層的操作咯

[code]void printChar (const byte ch) {
  *(word *)(VIDEO + y * 160 + x * 2) = ch | (fill_color << 8);
  x++;
  if (x >= WIDTH)
    newLine ();
  setVideoCursor (y, x);
}[/code] 這里VIDEO表示顯存地址也就是0xB8000.通過(guò) y * 160 + x 屏幕(x,y)坐標(biāo)在顯存中的位置.這里需要知道,一個(gè)字符顯示需要兩個(gè)字節(jié),一個(gè)是ASCII碼,第二個(gè)是字符屬性代碼也就是顏色代碼.因此才必須 y * 80 * 2 + x = y * 160 + x.那么ch | (fill_color << 8)也自然就是寫入字符及屬性代碼用的了.每寫一個(gè)字符光標(biāo)位置加1,如果大于屏幕寬度WIDTH就換行.最后通過(guò)setVideoCursor設(shè)置新的 光標(biāo)位置.完成了整個(gè)printChar過(guò)程.

posted on 2007-11-14 12:27 sunGuoqin 閱讀(1450) 評(píng)論(3)  編輯 收藏 引用 所屬分類: c語(yǔ)言

評(píng)論

# re: 內(nèi)核printf源代碼分析 2007-11-14 12:53 arden
內(nèi)核是什么意思?  回復(fù)  更多評(píng)論
  

# re: 內(nèi)核printf源代碼分析 2007-11-15 11:32 sunGuoqin
這個(gè)是printf的C語(yǔ)言庫(kù)函數(shù)的內(nèi)部實(shí)現(xiàn)啊,呵呵  回復(fù)  更多評(píng)論
  

# re: 內(nèi)核printf源代碼分析 2007-11-15 11:37 <a href=http://minidx.com>minidxer</a>
沒(méi)編輯好,讀起來(lái)怪怪的  回復(fù)  更多評(píng)論
  

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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| 久久综合中文| 亚欧成人在线| 亚洲午夜在线观看| 韩国美女久久| 国产欧美综合一区二区三区| 欧美人与禽猛交乱配| 欧美成人免费视频| 欧美高清一区| 欧美伦理在线观看| 欧美精品色综合| 欧美一区二区视频97| 亚洲在线成人精品| 亚洲伊人一本大道中文字幕| 日韩午夜黄色| 日韩一区二区精品| 一本色道久久88精品综合| 亚洲精品国久久99热| 蜜桃精品久久久久久久免费影院| 久久精品一区二区三区不卡| 欧美在线黄色| 久久人人看视频| 美国十次成人| 欧美国产日韩一区二区| 亚洲第一精品夜夜躁人人躁| 午夜精品一区二区三区电影天堂 | 久久久天天操| 欧美va天堂va视频va在线| 欧美v国产在线一区二区三区| 亚洲一区在线观看视频| 亚洲欧美国产一区二区三区| 午夜久久久久久| 久久蜜桃av一区精品变态类天堂| 老牛影视一区二区三区| 欧美精彩视频一区二区三区| 欧美日韩国产在线一区| 欧美99在线视频观看| 欧美成人资源网| 欧美成人一区二免费视频软件| 久久视频这里只有精品| 欧美在线短视频| 欧美成人精品影院| 国产精品第十页| 精品二区视频| 一区二区三区日韩欧美| 久久成人免费视频| 你懂的一区二区| 亚洲一区二区免费视频| 久久精品国产综合精品| 欧美日本国产一区| 韩国精品一区二区三区| 正在播放亚洲一区| 欧美专区一区二区三区| 亚洲国内自拍| 久久国产精品一区二区| 欧美日韩另类字幕中文| 精品成人一区| 亚洲在线观看免费视频| 午夜精品一区二区三区在线播放| 欧美国产精品| 久久久国产精品亚洲一区| 欧美日韩精品一本二本三本| 一区二区三区自拍| 亚洲欧美日本视频在线观看| 中文在线资源观看网站视频免费不卡| 久久精品综合网| 国产精品99久久久久久久久| 另类天堂视频在线观看| 国产亚洲在线| 在线精品高清中文字幕| 久久精品国产999大香线蕉| 亚洲精品国产精品久久清纯直播| 久久婷婷综合激情| 好看的日韩av电影| 欧美一区二区视频在线观看| 99在线|亚洲一区二区| 欧美电影免费网站| 在线视频成人| 麻豆精品国产91久久久久久| 午夜精品一区二区三区电影天堂| 欧美日韩一区二区免费在线观看 | 欧美激情欧美激情在线五月| 亚洲国产视频一区二区| 老司机免费视频一区二区| 午夜精品视频在线观看| 国产欧美一区二区精品秋霞影院 | 国产精品免费看久久久香蕉| 亚洲黄页视频免费观看| 久久久蜜桃精品| 久久精品2019中文字幕| 国模吧视频一区| 久久久欧美精品| 久久久久久免费| 亚洲第一页在线| 欧美激情无毛| 久久精品99国产精品日本| 羞羞视频在线观看欧美| 亚洲激情第一页| 亚洲免费一在线| 亚洲精品美女在线| 欧美一级视频| 亚洲天堂网在线观看| 久久精品一区二区| 亚洲欧美视频一区| 免费亚洲一区二区| 久久久久久91香蕉国产| 国产精品video| 亚洲成人在线视频播放| 国产日韩精品一区| 一区二区三区四区五区精品视频| 韩日精品中文字幕| 亚洲欧美国产77777| 夜夜爽夜夜爽精品视频| 久久野战av| 久久久久女教师免费一区| 欧美三区免费完整视频在线观看| 麻豆av一区二区三区久久| 国产毛片久久| 亚洲天堂成人| 亚洲性夜色噜噜噜7777| 欧美黄色一级视频| 老妇喷水一区二区三区| 国产精品自在线| 亚洲视频www| 亚洲一区二区精品| 欧美精品一区在线播放| 欧美国产日本| 亚洲高清毛片| 久久尤物视频| 欧美成人黄色小视频| 红桃视频成人| 久久久www免费人成黑人精品| 久久国产加勒比精品无码| 国产精品一区三区| 亚洲制服丝袜在线| 欧美制服丝袜| 国产一区亚洲| 久久久久国产一区二区三区四区| 久久精品国产一区二区电影 | 亚洲欧美综合网| 国产精品女同互慰在线看| 亚洲天堂男人| 久久丁香综合五月国产三级网站| 国产欧美日韩亚洲一区二区三区| 亚洲主播在线观看| 久久久久久久久久看片| 伊人精品在线| 欧美精品18videos性欧美| 亚洲精品乱码久久久久久日本蜜臀 | 亚洲欧美日韩视频一区| 国产精品久线观看视频| 亚洲午夜在线视频| 久久精品亚洲精品国产欧美kt∨| 国产啪精品视频| 久久深夜福利免费观看| 亚洲高清在线播放| 亚洲永久在线观看| 国产一区二区三区久久精品| 久久久久国产一区二区| 亚洲国产天堂久久综合| 亚洲网站在线观看| 国产亚洲欧美一区二区| 老司机午夜精品视频| 日韩视频在线观看一区二区| 亚洲永久免费av| 一区二区亚洲精品| 欧美日韩高清一区| 欧美中文字幕不卡| 亚洲精品小视频| 久久久久久久性| 99re8这里有精品热视频免费| 国产乱码精品一区二区三区av| 久久久之久亚州精品露出| 亚洲人精品午夜| 国产美女精品一区二区三区| 女人天堂亚洲aⅴ在线观看| 亚洲无亚洲人成网站77777 | 欧美激情视频一区二区三区在线播放| 国产精品99久久久久久白浆小说| 国产婷婷成人久久av免费高清 | 亚洲综合国产激情另类一区| 欧美成人国产一区二区| 午夜精品视频| 99视频一区| 激情综合中文娱乐网| 欧美日韩成人综合在线一区二区| 欧美一区二视频在线免费观看| 亚洲人线精品午夜| 久久影院午夜论| 欧美一区亚洲二区| 一本色道88久久加勒比精品| 在线播放中文一区| 国产视频久久| 国产精品激情电影| 欧美精品久久久久久| 久热这里只精品99re8久| 性色av一区二区怡红| 亚洲色图制服丝袜| 99精品国产在热久久婷婷|