c語言實現(xiàn)庫函數(shù) c語言常用庫函數(shù)含詳細用法

C語言中重新實現(xiàn)庫函數(shù)

fgets是從文件讀取,getchar是從輸入讀取,兩個還是有區(qū)別的

成都創(chuàng)新互聯(lián)專注于網(wǎng)站建設|成都企業(yè)網(wǎng)站維護|優(yōu)化|托管以及網(wǎng)絡推廣,積累了大量的網(wǎng)站設計與制作經(jīng)驗,為許多企業(yè)提供了網(wǎng)站定制設計服務,案例作品覆蓋成都水泥攪拌車等行業(yè)。能根據(jù)企業(yè)所處的行業(yè)與銷售的產(chǎn)品,結(jié)合品牌形象的塑造,量身定制品質(zhì)網(wǎng)站。

懷疑是gets

如果一定要用fgets的話,那么可以先用freopen重定向輸入,來使fgets和getchar的定向相同

以下用gets實現(xiàn)或者被實現(xiàn)

含測試函數(shù)main

#include stdio.h

int str_to_int(char *str)

{

int sign = 1;

int ret = 0;

char *p;

p = str;

if(*p == '+') p++;

else if(*p == '-') sign = -1, p ++;

while(*p)

{

if(*p = '0' *p = '9')

ret = ret * 10 + (*p - '0');

else break;

p ++;

}

return sign * ret;

}

int get_int_with_getchar()

{

char a[20] = {0};

int i = 0;

while(i19)

{

a[i] = getchar();

if(a[i] == '\n')

break;

i ++;

}

a[i] = 0;

return str_to_int(a);

}

int get_int_with_gets()

{

char a[20] = {0};

gets(a);

return str_to_int(a);

}

void gets_with_getchar(char *str)

{

int i = 0;

if(str == NULL) return;

do

{

str[i] = getchar();

}while(str[i++] != '\n' );

}

char *do_strchr(char *src, const char c)

{

char *p = src;

while(*p *p!= c) p ++;

if(*p != c) return NULL;

return p;

}

int do_isdigit(char c)

{

return c = '9' c = '0';

}

int do_isalpha(char c)

{

return (c = 'Z' c = 'A') || (c = 'z' c = 'a');

}

int main()

{

int a;

char test[100] ;

a = get_int_with_getchar();

printf("get_int_with_getchar = %d\n", a);

a = get_int_with_gets();

printf("get_int_with_gets = %d\n", a);

gets_with_getchar(test);

printf("gets_with_getchar = %s\n", test);

printf("do_strchr %s %s\n", do_strchr(test, 'a'), do_strchr(test, 'b'));

printf("do_isdigit = (%d,%d) \n", do_isdigit('1'), do_isdigit('a'));

printf("do_isalpha = (%d,%d) \n", do_isalpha('1'), do_isalpha('a'));

}

C語言:如何自己做庫函數(shù)?

簡單的一個例子,但是基本的程序編譯的思想都在里面了,具體的需要你自己體會和查資料

跟著下面的步驟走

------------------------

第一步:我的電腦-屬性-高級-環(huán)境變量

添加3個變量

1.名字:path 值:vc的安裝目錄里面包含cl.exe的bin目錄路徑+";"+vc目錄下包含mspdb80.dll的目錄的路徑

比如我的是D:\Program Files\Microsoft Visual Studio 8\VC\bin;D:\Program Files\Microsoft Visual Studio 8\Common7\IDE;

2.名字:include 值:vc安裝目錄下的include目錄,主要是為了找到頭文件,我的是D:\Program Files\Microsoft Visual Studio 8\VC\include;D:\Program Files\Microsoft Visual Studio 8\VC\PlatformSDK\Include;

3.名字:lib 值:vc目錄下包含庫文件的目錄路徑

我的是D:\Program Files\Microsoft Visual Studio 8\VC\lib;D:\Program Files\Microsoft Visual Studio 8\VC\PlatformSDK\Lib;

第二步:需要下面兩個文件

//-------------libdemo.c

#include stdio.h

void lib_func()

{

printf("this routine in lib\n");

}

//------------main.c

void lib_func();

int main(int argc, char *argv[])

{

lib_func();

}

第三步:命令

cl /c libdemo.c '編譯libdemo.c 生成libdemo.obj

lib libdemo.obj '生成libdemo.lib 庫文件

cl /c main.c '編譯main.c

link main.obj libdemo.lib '生成main.exe

---------------------------------------------------

呵呵 加點東西,參看4樓的建議,在做個dll,COM組件就不做了,有點復雜了

1。把libdemo.c 稍微改下

//-------------libdemo.c

#include stdio.h

__declspec (dllexport) void lib_func()

{

printf("this routine in lib\n");

}

2。命令

cl /c libdemo.c main.c

link libdemo.obj /dll '生成libdemo.lib

link main.obj libdemo.lib

C語言庫函數(shù)如何編寫?

/***

*printf.c - print formatted

*

* Copyright (c) 1985-1997, Microsoft Corporation. All rights reserved.

*

*Purpose:

* defines printf() - print formatted data

*

*******************************************************************************/

#include

#include

#include

#include

#include

#include

#include

/***

*int printf(format, ...) - print formatted data

*

*Purpose:

* Prints formatted data on stdout using the format string to

* format data and getting as many arguments as called for

* Uses temporary buffering to improve efficiency.

* _output does the real work here

*

*Entry:

* char *format - format string to control data format/number of arguments

* followed by list of arguments, number and type controlled by

* format string

*

*Exit:

* returns number of characters printed

*

*Exceptions:

*

*******************************************************************************/

int __cdecl printf (

const char *format,

...

)

/*

* stdout ''PRINT'', ''F''ormatted

*/

{

va_list arglist;

int buffing;

int retval;

va_start(arglist, format);

_ASSERTE(format != NULL);//斷言宏。如果輸出格式字符串指針為空,則在DEBUG版下斷言,報告錯誤。

_lock_str2(1, stdout);

buffing = _stbuf(stdout);//stdout:指定輸出到屏幕

retval = _output(stdout,format,arglist);

_ftbuf(buffing, stdout);

_unlock_str2(1, stdout);

return(retval);

}

以上為printf()的源代碼

1、從含有可選參數(shù)函數(shù)中獲得可選參數(shù),以及操作這些參數(shù)

typedef char *va_list;

void va_start( va_list arg_ptr, prev_param );

type va_arg( va_list arg_ptr, type );

void va_end( va_list arg_ptr );

假定函數(shù)含有一個必選參數(shù)和多個可選參數(shù),必選參數(shù)聲明為普通數(shù)據(jù)類型,且能通過參數(shù)名來獲得該變量的值。可選參數(shù)通過宏va_start、va_arg和va_end(定義在stdarg.h或varargs.h中)來進行操作,即通過設置指向第一個可選參數(shù)指針、返回當前參數(shù)、在返回參數(shù)后重新設置指針來操作所有的可選參數(shù)。

va_start:為獲取可變數(shù)目參數(shù)的函數(shù)的參數(shù)提供一種便捷手段。設置arg_ptr為指向傳給函數(shù)參數(shù)列表中的第一個可選參數(shù)的指針,且該參數(shù)必須是va_list類型。prev_param是在參數(shù)列表中第一個可選參數(shù)前的必選參數(shù)。

va_arg:返回由arg_ptr所指向的參數(shù)的值,且自增指向下一個參數(shù)的地址。type為當前參數(shù)的類型,用來計算該參數(shù)的長度,確定下一個參數(shù)的起始位置。它可以在函數(shù)中應用多次,直到得到函數(shù)的所有參數(shù)為止,但必須在宏va_start后面調(diào)用。

va_end:在獲取所有的參數(shù)后,設置指針arg_ptr為NULL。

下面舉例說明:

#include

#include

int average( int first, ... );

void main( void )

{

/* Call with 3 integers (-1 is used as terminator). */

printf( "Average is: %d\n", average( 2, 3, 4, -1 ) );

/* Call with 4 integers. */

printf( "Average is: %d\n", average( 5, 7, 9, 11, -1 ) );

/* Call with just -1 terminator. */

printf( "Average is: %d\n", average( -1 ) );

}

int average( int first, ... )

{

int count = 0, sum = 0, i = first;

va_list marker;

va_start( marker, first ); /* Initialize variable arguments. */

while( i != -1 )

{

sum += i;

count++;

i = va_arg( marker, int);

}

va_end( marker ); /* Reset variable arguments. */

return( sum ? (sum / count) : 0 );

}

返回值為:

Average is: 3

Average is: 8

Average is: 0

綜上所述,在printf()函數(shù)中,可以只輸出一個字符串,也可按照一定的形式輸出含有多個可選參數(shù)的字符串信息。因此,首先就要通過這些宏來獲取所有的可選參數(shù)。在上面的源碼可以看出printf()中,只使用了宏at_start,將可選參數(shù)的首地址賦給了arglist。

2、鎖定字符串及輸出字符串到屏幕

#define _lock_str2(i,s) _lock_file2(i,s)

void __cdecl _lock_file2(int, void *);

#define _unlock_str2(i,s) _unlock_file2(i,s)

void __cdecl _unlock_file2(int, void *);

int __cdecl _stbuf(FILE *);

void __cdecl _ftbuf(int, FILE *);

int __cdecl _output(FILE *, const char *, va_list);

在output函數(shù)中,讀取格式字符串中的每一個字符,然后對其進行處理,處理方式根據(jù)每一個字符所代表的意義來進行,如:普通字符直接利用函數(shù)WRITE_CHAR(ch, charsout);輸出到控制臺。

其中的主要部分是對轉(zhuǎn)換說明符(d,c,s,f)的處理,現(xiàn)在將對其中的部分代碼進行詳細說明,這里只說明最基本的轉(zhuǎn)換說明符,對這些須基本的轉(zhuǎn)換說明符進行修飾的修飾符,程序中單獨進行處理。下面是函數(shù)output()(output.c)部分源代碼:

case ST_TYPE:

//表示當前處理的字符的類型為轉(zhuǎn)換說明符。

...

switch (ch) {

//下面對參數(shù)的獲取都是利用宏va_arg( va_list arg_ptr, type );來進行的。

case ''c'': {

//從參數(shù)表中獲取單個字符,輸出到緩沖字符串中,此時,type=int

buffer[0] = (char) get_int_arg(argptr); /* get char to print */

text = buffer;

textlen = 1; /* print just a single character */

}

break;

case ''s'': {

//從參數(shù)表中獲取字符串,輸出到緩沖字符串中,此時,type=char*

int i;

char *p; /* temps */

text = get_ptr_arg(argptr);

...

}

break;

case ''w'': {

//對寬字符進行處理

...

} /* case ''w'' */

break;

...

case ''e'':

case ''f'':

case ''g'': {

//對浮點數(shù)進行操作

...

#if !LONGDOUBLE_IS_DOUBLE

/* do the conversion */

if (flags FL_LONGDOUBLE) {

_cldcvt((LONGDOUBLE*)argptr, text, ch, precision, capexp);

va_arg(argptr, LONGDOUBLE);

//對長雙精度型進行處理,此時,type=long double

}

else

#endif /* !LONGDOUBLE_IS_DOUBLE */

{

//對雙精度型進行處理,此時,type=double

_cfltcvt((DOUBLE*)argptr, text, ch, precision, capexp);

va_arg(argptr, DOUBLE);

}

...

break;

//對整型變量處理

case ''d'':

case ''i'':

...

goto COMMON_INT;

case ''u'':

radix = 10;

goto COMMON_INT;

case ''p'':

...

goto COMMON_INT;

case ''o'':

...

注:對于浮點型double和long double,有相應的轉(zhuǎn)換說明符(%f表示雙精度型,%lf表示長雙精度型),而float卻沒有。其中的原因是,在KRC下,float值用于表達式或用作參數(shù)前,會自動轉(zhuǎn)換成double類型。而ANSI C一般不會自動把float轉(zhuǎn)換成double。有些程序已假定其中的float參數(shù)會被轉(zhuǎn)換成double,為了保護大量這樣的程序,所有printf()函數(shù)的float參數(shù)還是被自動轉(zhuǎn)換成double型。因此,在KRC或ANSI C下,都無需用特定的轉(zhuǎn)換說明符來顯示float型。

綜上所述,轉(zhuǎn)換說明符必須與待打印字符的類型。通常,用戶有種選擇。例如,如要打印一個int類型的值。則只可以使用%d,%x或%o。所有這些說明符都表示要打印一個int類型的值;它們只不過提供了一個數(shù)值的幾種不同表示。類似一,可以用%f、%g和%e來表示double類型的值。但如果轉(zhuǎn)換說明與類型不匹配,將會出現(xiàn)意想不到的結(jié)果。為什么呢?問題就在于C向函數(shù)傳遞信息的方式。

這個失敗的根本細節(jié)與具體實現(xiàn)相關(guān)。它決定了系統(tǒng)中的參數(shù)以何方式傳遞。函數(shù)調(diào)用如下:

float n1;

double n2;

long n3;

long n4;

...

printf("%ld,%ld,%ld,%ld",n1,n2,n3,n4);

這個調(diào)用告訴計算機,要把變量n1,n2,n3和n4的值交給計算機,它把這些變量放進稱作棧(stack)的內(nèi)存區(qū)域中,來完成這一任務。計算機把這些值放進棧中,其根據(jù)是變量的類型而不是轉(zhuǎn)換說明符,比如n1,把8個字節(jié)放入棧中(float被轉(zhuǎn)換成double),類似地,為n2放了8字節(jié),其后給n3和n4各放了4個字節(jié)。接著,控制的對象轉(zhuǎn)移到printf();此函數(shù)從棧中讀數(shù),不過在這一過程中,它是在轉(zhuǎn)換說明符的指導下,讀取數(shù)值的。說明符%ld指定printf()應讀4個字節(jié)(va_arg( va_list arg_ptr, type )中type=long),因此printf()讀入棧中的4個字節(jié),作為它的第一個值。但是這只是n1的前半部分,這個值被看成一個long整數(shù)。下一個說明符%ld讀入4個字節(jié),這正是n1的后半部分,這個值被看成第二個long整數(shù)。類似地,第三、第四次又讀入n2的前后兩部分。因此,盡管我們對n3和n4使用了正確的說明符,printf()仍然會產(chǎn)生錯誤。

C語言 庫函數(shù)的實現(xiàn)???

C語言的庫函數(shù)大多數(shù)是用C語言編的,只有少數(shù)與硬件有直接聯(lián)系的核心部分才用匯編.

所謂庫函數(shù)并不深奧,庫函數(shù)也是由一個一個函數(shù)(子程序)組成的,如同我們自己程序里,有時也寫很多子程序.

如果我們把一些可以反復使用的子程序拿出來,集中到一起,編譯成一個my.obj文件,今后寫新程序時我們不另寫這些子程序,我們只寫子程序的調(diào)用語句,編譯時鏈結(jié)my.obj就可以了.當然,新程序的main()前面要有這些被調(diào)函數(shù)(子程序)的"原型"聲明.

my.obj加上它里面的函數(shù)"原型"聲明,就構(gòu)成了庫函數(shù).函數(shù)"原型"聲明就是這個庫的"頭"文件-- my.h, 這就是靜態(tài)鏈結(jié)庫.

自己也可以寫動態(tài)鏈接庫,動態(tài)鏈接庫包括.DLL,.LIB,.H

自己寫的靜態(tài)鏈結(jié)庫和動態(tài)鏈接庫也可給別人用.就象C語言的庫函數(shù)給你用一樣.當你給別人時,你如果不給你寫的C的函數(shù),別人也"看不到了",別人只能查頭文件,得知函數(shù)名,參數(shù)個數(shù)和類型.

網(wǎng)頁名稱:c語言實現(xiàn)庫函數(shù) c語言常用庫函數(shù)含詳細用法
網(wǎng)站網(wǎng)址:http://www.muchs.cn/article8/doeeiip.html

成都網(wǎng)站建設公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站營銷、電子商務、網(wǎng)站策劃、品牌網(wǎng)站設計、網(wǎng)站收錄外貿(mào)網(wǎng)站建設

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都seo排名網(wǎng)站優(yōu)化