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

創新無極限

Sun Guoqing的Blog

C++博客 首頁 新隨筆 聯系 聚合 管理
  3 Posts :: 1 Stories :: 11 Comments :: 0 Trackbacks

打開Source Insight來閱讀EduOS的源代碼,我們在stdio.c里找到了printf的實現代碼.首先看看對printf的定義:
[code]
int printf (const char *cntrl_string, ...)
[/code]
第一個參數cntrl_string是控制字符串,也就是平常我們寫入%d,%f的地方.緊接著后面是一個變長參數.

看看函數頭部的定義:

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

[code]va_start (ap, cntrl_string);[/code]
用來初始化變長參數.

接下來是一個while循環

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

結束條件是cntrl_string[pos]為NULL,顯然這個循環是用來遍歷整個控制字符串的.自然pos就是當前遍歷到的位置了.進入循環首先闖入視線的是

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

開門見山,上來就當前字符是否辦斷是否%.一猜就知道如果成立pos++馬上取出下一個字符在d,f,l等等之間進行判斷.往下一看,果真不出所料:

[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表示僅僅輸出一個字符.因此先通過va_arg進行參數的類型轉換,之后用putchar[1]輸出到屏幕上去.之后是
cnt_printed_chars++,通過這句我們就可以判斷出cnt_printed_chars使用來表示,已經被printf輸出的字符個數的.

再來看看 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++放在了循環內,也證明了剛才提到的他的作用.另外我們也看到了cnptr是用來在處理字符串時的位置指針.到此為止,我們清楚的所有變量的用途,前途變得更加光明了.

接下來:
[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的代碼其實就是比PartI的代碼多一個樣式.在16進制數或八進制前加入0x或是o,等等.因此這里就只分析一下PartI咯.

其實仔細看看PartI的個條case,也就是把參數分發到了更具體的函數用于顯示,然后以返回值的形式返回輸出個數.對于這些函數就不具體分析了.我們先來看看一些善后處理:

先看case的default處理.
[code]default:
 putchar ((unsigned char) cntrl_string[pos]);
 cnt_printed_chars++;[/code]就是直接輸出cntrl_string里%號后面的未知字符.應該是一種容錯設計處理.

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

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

在最后我們有必要談談putChar函數以及基本輸出的基礎函數printChar,先來看看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為主體的.主要是用來應對一些特殊字符,如\n,\r,....這里需要提一下,關于\t的理解.有些人認為\t就是 8個space,有些人則認為,屏幕分為10大列(每個大列8個小列總共80列).一個\t就跳到下一個大列輸出.也就是說不管你現在實在屏幕的第 1,2,3,4,5,6,7位置輸出字符,只要一個\t都在第8個位置開始輸出. VS.NET中就是用的這種理解.因此如果按照這個理解的話,\t的實現可以這樣

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

然后在currentX位置輸出.

接下來看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.通過 y * 160 + x 屏幕(x,y)坐標在顯存中的位置.這里需要知道,一個字符顯示需要兩個字節,一個是ASCII碼,第二個是字符屬性代碼也就是顏色代碼.因此才必須 y * 80 * 2 + x = y * 160 + x.那么ch | (fill_color << 8)也自然就是寫入字符及屬性代碼用的了.每寫一個字符光標位置加1,如果大于屏幕寬度WIDTH就換行.最后通過setVideoCursor設置新的 光標位置.完成了整個printChar過程.

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

評論

# re: 內核printf源代碼分析 2007-11-14 12:53 arden
內核是什么意思?  回復  更多評論
  

# re: 內核printf源代碼分析 2007-11-15 11:32 sunGuoqin
這個是printf的C語言庫函數的內部實現啊,呵呵  回復  更多評論
  

# re: 內核printf源代碼分析 2007-11-15 11:37 <a href=http://minidx.com>minidxer</a>
沒編輯好,讀起來怪怪的  回復  更多評論
  

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产精品一区二区久激情瑜伽| 你懂的视频一区二区| 国产精品美女久久久久aⅴ国产馆| 欧美精品一区二区三区蜜桃| 久久国产乱子精品免费女| 亚洲影院免费观看| 亚洲视频碰碰| 亚洲图片欧洲图片日韩av| 夜夜夜精品看看| 中文精品视频一区二区在线观看| 国产精品99久久久久久白浆小说| 亚洲欧洲一区二区三区久久| 亚洲欧美文学| 久久综合狠狠综合久久综合88| 久久艳片www.17c.com| 免费亚洲一区二区| 亚洲精品一区二区三区蜜桃久| 91久久中文字幕| 亚洲一区二区三区精品在线观看| 篠田优中文在线播放第一区| 久久一日本道色综合久久| 欧美日韩成人在线视频| 国产精品一区二区在线观看不卡 | 一本色道久久综合狠狠躁篇的优点| 亚洲日本在线观看| 欧美一区二区三区免费大片| 免费试看一区| 亚洲免费一在线| 欧美成人午夜激情| 国产一区二区福利| 在线亚洲精品| 美女日韩欧美| 亚洲一区二区三区乱码aⅴ| 久久在线免费观看视频| 国产精品xxxav免费视频| 黄色精品网站| 亚洲欧美日韩成人| 91久久在线| 老司机午夜精品视频在线观看| 国产精品theporn88| 91久久线看在观草草青青| 欧美一区二区三区免费在线看 | 国产精品第2页| 精品91久久久久| 一本色道久久综合亚洲精品不| 久久久久久9| 亚洲一区二区视频| 欧美黄免费看| 性欧美1819性猛交| 亚洲午夜精品国产| 欧美gay视频激情| 篠田优中文在线播放第一区| 欧美午夜理伦三级在线观看| 亚洲国产精品一区制服丝袜 | 欧美国产日韩a欧美在线观看| 国产精品永久在线| 亚洲午夜羞羞片| 亚洲国产精品毛片| 久久亚洲精选| 亚洲高清色综合| 久久久国产精品一区二区三区| 一本色道久久综合亚洲精品不| 欧美精品一区二区三| 亚洲第一天堂无码专区| 欧美一区深夜视频| 亚洲一区二区在线播放| 国产精品久久国产三级国电话系列| 亚洲国产精品va| 欧美成人午夜激情在线| 久久久人人人| 亚洲第一页在线| 欧美成ee人免费视频| 久久精品久久综合| 在线观看中文字幕不卡| 久久久久久亚洲综合影院红桃| 亚洲一区亚洲二区| 国产精品美女视频网站| 亚洲欧美精品一区| 亚洲永久免费观看| 国产精品日韩欧美综合| 亚洲欧美日韩视频一区| 正在播放亚洲一区| 国产欧美视频一区二区三区| 久久久国产一区二区三区| 亚洲一区二区三区精品视频| 欧美午夜一区| 久久久久一区二区三区四区| 欧美亚洲视频| 在线成人性视频| 亚洲毛片一区| 国产精品第一区| 久久亚洲精品一区| 免费欧美在线视频| 亚洲精品一区二区三| 日韩天堂在线观看| 国产精品美女久久久久av超清 | 久久深夜福利| 91久久综合亚洲鲁鲁五月天| 亚洲国产片色| 国产精品红桃| 久久九九有精品国产23| 欧美成人在线免费观看| 亚洲婷婷免费| 久久精品免费观看| 亚洲欧美日韩另类精品一区二区三区 | 欧美国产日韩一区二区在线观看| 黄网站色欧美视频| 欧美国产精品va在线观看| 欧美激情免费在线| 午夜久久tv| 欧美xx69| 久久久777| 欧美性天天影院| 你懂的亚洲视频| 欧美日韩综合网| 免费在线国产精品| 国产精品日产欧美久久久久| 久久久久99精品国产片| 免费成人av在线| 久久国产精品色婷婷| 欧美精品三级| 免费成人av在线| 国产精品乱码人人做人人爱| 久久网站热最新地址| 国产精品h在线观看| 久热精品在线视频| 欧美日韩免费一区二区三区视频| 麻豆av福利av久久av| 欧美激情bt| 裸体一区二区| 国产日韩欧美在线播放| 日韩视频一区二区三区在线播放| 精品福利电影| 亚洲欧美中文日韩v在线观看| 亚洲一区日韩在线| 欧美日本精品| 亚洲国产高清aⅴ视频| 国产日韩欧美在线播放不卡| 亚洲综合精品自拍| 性色av一区二区三区| 欧美金8天国| 欧美电影在线观看完整版| 樱桃成人精品视频在线播放| 欧美在线免费观看视频| 久久青草欧美一区二区三区| 国产中文一区| 久久精品国产77777蜜臀| 久久久www免费人成黑人精品| 国产日产亚洲精品系列| 欧美在线看片| 欧美成人伊人久久综合网| 亚洲激情网站免费观看| 欧美国产日韩一区二区| 一区二区三区你懂的| 先锋a资源在线看亚洲| 国产日韩视频| 久久一二三四| 一区二区免费在线播放| 亚洲免费在线观看视频| 国产欧美一区二区三区国产幕精品| 亚洲欧美日韩中文播放| 久久艳片www.17c.com| 亚洲激情网站免费观看| 欧美午夜精品久久久久久浪潮| 亚洲男人的天堂在线| 久色成人在线| 亚洲视频一区在线| 国产综合亚洲精品一区二| 欧美成人首页| 午夜国产精品影院在线观看| 国产美女精品视频免费观看| 亚洲精品综合| 欧美精品一区二| 亚洲一区二区三区四区在线观看 | 午夜视频在线观看一区二区| 国产一区二区中文| 欧美国产一区在线| 亚洲欧美三级在线| 亚洲激情第一区| 久久riav二区三区| 亚洲日本久久| 国产一区二区三区日韩欧美| 欧美精品一卡| 亚洲一区二区三区精品动漫| 国产偷国产偷亚洲高清97cao| 乱中年女人伦av一区二区| 一区二区不卡在线视频 午夜欧美不卡在 | 午夜精品国产精品大乳美女| 欧美~级网站不卡| 亚洲影院色在线观看免费| 亚洲第一精品福利| 国产亚洲精品激情久久| 欧美私人网站| 欧美精品在线观看| 亚洲欧美精品在线| 亚洲最新视频在线| 在线观看日韩国产| 国产一区二区三区四区hd| 欧美三区在线视频| 欧美成人免费小视频|