0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

解析C語(yǔ)言入門基礎(chǔ)之輸入和輸出

電子工程師 ? 來源:編程學(xué)習(xí)總站 ? 作者:寫代碼的牛頓 ? 2021-05-31 14:05 ? 次閱讀

01

標(biāo)準(zhǔn)輸入和標(biāo)準(zhǔn)輸出

C語(yǔ)言里要使用標(biāo)準(zhǔn)輸入和標(biāo)準(zhǔn)輸出必須包含stdio.h頭文件,常用的標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)輸入函數(shù)是printf和scanf,其中printf用來在標(biāo)準(zhǔn)輸出中輸出信息,而函數(shù)scanf則用來從標(biāo)準(zhǔn)輸入中讀取信息。

那么什么是標(biāo)準(zhǔn)輸入和標(biāo)準(zhǔn)輸出呢?

Linux中進(jìn)程通常會(huì)自動(dòng)打開三個(gè)標(biāo)準(zhǔn)文件,即標(biāo)準(zhǔn)輸入文件(stdin)通常對(duì)應(yīng)文件描述符0;

標(biāo)準(zhǔn)輸出文件(stdout)對(duì)應(yīng)文件描述符1和標(biāo)準(zhǔn)錯(cuò)誤輸出文件對(duì)應(yīng)文件描述符2(stderr)。進(jìn)程將從標(biāo)準(zhǔn)輸入文件中讀取輸入數(shù)據(jù),將正常輸出數(shù)據(jù)輸出到標(biāo)準(zhǔn)輸出文件,而將錯(cuò)誤信息送到標(biāo)準(zhǔn)錯(cuò)誤文件中。02

標(biāo)準(zhǔn)輸入函數(shù)

在stdio.h中scanf聲明如下:

/* Read formatted input from stdin. This function is a possible cancellation point and therefore not marked with __THROW. */ extern int scanf (const char *__restrict __format, 。。.) __wur;

使用Mac或Linux的同學(xué),在終端上輸入man scanf回車即可學(xué)習(xí)scanf函數(shù)的用法。我們可以看到注釋上說明,scanf從標(biāo)準(zhǔn)輸入stdin輸入讀取數(shù)據(jù),在glibc中stdin的定義如下:

/*stdio.c*/ FILE *stdin = (FILE *) &_IO_2_1_stdin_; /*libio.h*/ extern struct _IO_FILE_plus _IO_2_1_stdin_; /*libioP.h*/ struct _IO_FILE_plus { FILE file; const struct _IO_jump_t *vtable; };

從以上代碼我們可以知道,最終stdin是一個(gè)FILE文件流指針,我能繼續(xù)追蹤FILE類型為何物。

/* * stdio state variables. * * The following always hold: * * if (_flags&(__SLBF|__SWR)) == (__SLBF|__SWR), * _lbfsize is -_bf._size, else _lbfsize is 0 * if _flags&__SRD, _w is 0 * if _flags&__SWR, _r is 0 * * This ensures that the getc and putc macros (or inline functions) never * try to write or read from a file that is in `read‘ or `write’ mode. * (Moreover, they can, and do, automatically switch from read mode to * write mode, and back, on “r+” and “w+” files.) * * _lbfsize is used only to make the inline line-buffered output stream * code as compact as possible. * * _ub, _up, and _ur are used when ungetc() pushes back more characters * than fit in the current _bf, or when ungetc() pushes back a character * that does not match the previous one in _bf. When this happens, * _ub._base becomes non-nil (i.e., a stream has ungetc() data iff * _ub._base!=NULL) and _up and _ur save the current values of _p and _r. * * NB: see WARNING above before changing the layout of this structure! */ typedef struct __sFILE { unsigned char *_p; /* current position in (some) buffer */ int _r; /* read space left for getc() */ int _w; /* write space left for putc() */ short _flags; /* flags, below; this FILE is free if 0 */ short _file; /* fileno, if Unix descriptor, else -1 */ struct __sbuf _bf; /* the buffer (at least 1 byte, if !NULL) */ int _lbfsize; /* 0 or -_bf._size, for inline putc */ /* operations */ void *_cookie; /* cookie passed to io functions */ int (* _Nullable _close)(void *); int (* _Nullable _read) (void *, char *, int); fpos_t (* _Nullable _seek) (void *, fpos_t, int); int (* _Nullable _write)(void *, const char *, int); /* separate buffer for long sequences of ungetc() */ struct __sbuf _ub; /* ungetc buffer */ struct __sFILEX *_extra; /* additions to FILE to not break ABI */ int _ur; /* saved _r when _r is counting ungetc data */ /* tricks to meet minimum requirements even when malloc() fails */ unsigned char _ubuf[3]; /* guarantee an ungetc() buffer */ unsigned char _nbuf[1]; /* guarantee a getc() buffer */ /* separate buffer for fgetln() when line crosses buffer boundary */ struct __sbuf _lb; /* buffer for fgetln() */ /* Unix stdio files get aligned to block boundaries on fseek() */ int _blksize; /* stat.st_blksize (may be != _bf._size) */ fpos_t _offset; /* current lseek offset (see WARNING) */ } FILE;

看到這個(gè)結(jié)構(gòu)體內(nèi)部一大堆成員變量不要慌,我們重點(diǎn)關(guān)注里面的close、read、seek和write函數(shù)指針。我們?cè)谡{(diào)用scanf函數(shù)時(shí)正是通過這幾個(gè)函數(shù)指針間接調(diào)用系統(tǒng)函數(shù)close、read、seek和write實(shí)現(xiàn)標(biāo)準(zhǔn)輸入關(guān)閉、讀取、偏移和寫功能。

int (* _Nullable _close)(void *); int (* _Nullable _read) (void *, char *, int); fpos_t (* _Nullable _seek) (void *, fpos_t, int); int (* _Nullable _write)(void *, const char *, int);

從函數(shù)聲明我們知道scanf返回一個(gè)int型返回值,在調(diào)用時(shí)scanf,返回正整數(shù)表示從標(biāo)準(zhǔn)輸入讀取到的有效數(shù)據(jù)數(shù)量,返回0表示沒有輸入或者輸入不正確,返回負(fù)數(shù)表示發(fā)生了從標(biāo)準(zhǔn)輸入讀取數(shù)據(jù)發(fā)生了錯(cuò)誤。下面我們使用scanf從標(biāo)準(zhǔn)輸入讀取數(shù)據(jù)的代碼。

int num = 0; float f_num = 0; int count = scanf(“%d”, &num); scanf(“%f”, &f_num); scanf_s(“%d”, &num);

在scanf中輸入數(shù)據(jù)并將數(shù)據(jù)保存在變量num和f_num中,調(diào)用scanf輸入數(shù)據(jù)必須要用%,%d表示輸入一個(gè)整數(shù),%f表示輸入一個(gè)單精度浮點(diǎn)數(shù),其他數(shù)據(jù)類型的數(shù)據(jù)參考C語(yǔ)言入門基礎(chǔ)之變量和數(shù)據(jù)類型,count保存scanf輸入數(shù)據(jù)的有效數(shù)。

看到這里可能有人會(huì)有疑問,為什么調(diào)用scanf從標(biāo)準(zhǔn)輸入信息,需要對(duì)變量取地址,為什么要設(shè)計(jì)成這樣?這里就要涉及到后面會(huì)學(xué)到的知識(shí):指針。在C語(yǔ)言里函數(shù)傳參方式有2種,一種是傳值另外一種是傳指針。通過傳值方式形參拷貝實(shí)參,得到一個(gè)實(shí)參副本對(duì)實(shí)參副本進(jìn)行修改不會(huì)影響實(shí)參,而傳指針方式,將會(huì)得到實(shí)參的地址,通過指針解引用可以間接修改實(shí)參的值。

那么回到scanf函數(shù)那里,我們通過對(duì)變量進(jìn)行取址,scanf函數(shù)內(nèi)部有一個(gè)指針,將變量地址值賦給內(nèi)部指針,再將標(biāo)準(zhǔn)輸入的值賦值給實(shí)參,實(shí)參變量因此獲得標(biāo)準(zhǔn)輸入的值。

在代碼片段我們還看到scanf_s這個(gè)函數(shù)(scanf_s不是C標(biāo)準(zhǔn)庫(kù)函數(shù)),由于scanf函數(shù)并不是安全的,在有些編輯器上默認(rèn)禁止使用scanf,如果使用則需要打開一個(gè)宏,而scanf_s是一些廠商提供的scanf函數(shù)安全版本,兩者使用方法一模一樣。

03

標(biāo)準(zhǔn)輸出函數(shù)

在stdio.h中printf函數(shù)聲明如下:

/* Write formatted output to stdout. This function is a possible cancellation point and therefore not marked with __THROW. */ extern int printf (const char *__restrict __format, 。。.);

看到這里是不是很熟悉?printf函數(shù)的返回值也是int型,調(diào)用printf函數(shù)將會(huì)返回輸出字符個(gè)數(shù),出錯(cuò)則返回一個(gè)負(fù)數(shù)。

同樣在Linux/Mac平臺(tái)的終端上輸入man printf函數(shù)可以查看函數(shù)的詳細(xì)使用方法(任何C標(biāo)準(zhǔn)函數(shù)都可以在Linux/Mac平臺(tái)上輸入man+函數(shù)名的方式查看函數(shù)使用方法)。下面是我們使用printf函數(shù)在標(biāo)準(zhǔn)輸出中輸出數(shù)據(jù)的代碼。

int output_count = printf(“num = %d ”, num); printf(“output_count = %d ”, output_count); output_count = printf(“f_num = %f ”, f_num); printf(“output_count = %d ”, output_count);

在代碼片段里我們看到一個(gè) 字符,在C語(yǔ)言里這是一個(gè)換行符??吹竭@里是不是又有疑問了,為什么printf函數(shù)輸出變量值時(shí)不需要對(duì)變量取地址?這就回到前面我們說過的問題了,在C語(yǔ)言里傳值,形參是實(shí)參的副本,形參修改了不會(huì)影響到實(shí)參。而printf函數(shù)只是在標(biāo)準(zhǔn)輸出中輸出信息,不會(huì)修改實(shí)參的值,因此使用傳值方式。

那么標(biāo)準(zhǔn)輸出是什么呢?從print函數(shù)聲明代碼注釋上看,標(biāo)準(zhǔn)輸出正是stdou,我們繼續(xù)在glibc中繼續(xù)追蹤stdout到底是什么?在stdout.c中我們看到stdout和stderr定義如下:

FILE *stdout = (FILE *) &_IO_2_1_stdout_; FILE *stderr = (FILE *) &_IO_2_1_stderr_;

我們發(fā)現(xiàn)stdout、stderr和stdin的定義一模一樣都是一個(gè)FILE類型指針,那么使用方式就和stdin一樣了,區(qū)別則在于stdin和文件描述符0綁定,stdout和文件描述符1綁定,stderr和文件描述符2綁定。

04

結(jié)語(yǔ)

后面講解C語(yǔ)言知識(shí)時(shí)我會(huì)穿插有Linux相關(guān)知識(shí),講解C語(yǔ)言不能僅僅停留在語(yǔ)法層面。據(jù)我的觀察,很多人學(xué)習(xí)了C語(yǔ)言語(yǔ)法后很迷茫,不知道C語(yǔ)言能做什么,根本原因就是你沒有了解某個(gè)平臺(tái)的系統(tǒng)編程API。Linux是一個(gè)開源操作系統(tǒng),結(jié)合Linux學(xué)習(xí)C語(yǔ)言將會(huì)更加有趣,在Linux上進(jìn)行C語(yǔ)言開發(fā)絕對(duì)是最佳選擇。

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11207

    瀏覽量

    208717
  • C語(yǔ)言
    +關(guān)注

    關(guān)注

    180

    文章

    7594

    瀏覽量

    135858
  • File
    +關(guān)注

    關(guān)注

    0

    文章

    19

    瀏覽量

    14315
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4277

    瀏覽量

    62323
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4722

    瀏覽量

    68231

原文標(biāo)題:C語(yǔ)言入門基礎(chǔ)之輸入和輸出

文章出處:【微信號(hào):AndroidPush,微信公眾號(hào):Android編程精選】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    FPGA編程語(yǔ)言入門教程

    FPGA(現(xiàn)場(chǎng)可編程邏輯門陣列)的編程涉及特定的硬件描述語(yǔ)言(HDL),其中Verilog和VHDL是最常用的兩種。以下是一個(gè)FPGA編程語(yǔ)言(以Verilog為例)的入門教程: 一、Verilog
    的頭像 發(fā)表于 10-25 09:21 ?120次閱讀

    TMS320LF240x DSP的C語(yǔ)言和匯編代碼快速入門

    電子發(fā)燒友網(wǎng)站提供《TMS320LF240x DSP的C語(yǔ)言和匯編代碼快速入門.pdf》資料免費(fèi)下載
    發(fā)表于 10-18 10:14 ?0次下載
    TMS320LF240x DSP的<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>和匯編代碼快速<b class='flag-5'>入門</b>

    模擬DAC38RF8x輸入/輸出緩沖器信息的快速入門方法

    電子發(fā)燒友網(wǎng)站提供《模擬DAC38RF8x輸入/輸出緩沖器信息的快速入門方法.pdf》資料免費(fèi)下載
    發(fā)表于 10-09 11:21 ?0次下載
    模擬DAC38RF8x<b class='flag-5'>輸入</b>/<b class='flag-5'>輸出</b>緩沖器信息的快速<b class='flag-5'>入門</b>方法

    C語(yǔ)言中的輸入&amp;輸出

    int printf(const char *format, ...) 函數(shù)把輸出寫入到標(biāo)準(zhǔn)輸出流 stdout ,并根據(jù)提供的格式產(chǎn)生輸出。
    發(fā)表于 02-28 11:17 ?451次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>中的<b class='flag-5'>輸入</b>&amp;<b class='flag-5'>輸出</b>

    plc編程語(yǔ)言c語(yǔ)言的聯(lián)系 c語(yǔ)言和PLC有什么區(qū)別

    PLC編程語(yǔ)言C語(yǔ)言的聯(lián)系 PLC(可編程邏輯控制器)是一種針對(duì)自動(dòng)化控制系統(tǒng)的特殊計(jì)算機(jī)。PLC編程語(yǔ)言是為了控制和管理自動(dòng)化生產(chǎn)過程中的各種設(shè)備而設(shè)計(jì)的。與
    的頭像 發(fā)表于 02-05 14:21 ?3746次閱讀

    c語(yǔ)言,c++,java,python區(qū)別

    操作系統(tǒng)、嵌入式系統(tǒng)等對(duì)性能要求較高的場(chǎng)景。C語(yǔ)言的語(yǔ)法相對(duì)簡(jiǎn)單,學(xué)習(xí)曲線較平緩,也是學(xué)習(xí)其他高級(jí)語(yǔ)言入門語(yǔ)言。
    的頭像 發(fā)表于 02-05 14:11 ?2117次閱讀

    vb語(yǔ)言c++語(yǔ)言的區(qū)別

    Microsoft開發(fā)的一種面向?qū)ο蟮氖录?qū)動(dòng)編程語(yǔ)言。它的設(shè)計(jì)目標(biāo)是簡(jiǎn)化編程過程,讓初學(xué)者也能快速上手。與相比,C++語(yǔ)言是一種通用的、面向?qū)ο蟮木幊?/div>
    的頭像 發(fā)表于 02-01 10:20 ?1949次閱讀

    C語(yǔ)言構(gòu)建高效的嵌入式程序

    嵌入式工程師在編寫C語(yǔ)言程序時(shí),需要注重效率和清晰的思路。本文將通過解析經(jīng)典問題“猴子選大王”來展示如何用C語(yǔ)言思維方式構(gòu)建高效、清晰的程序
    的頭像 發(fā)表于 12-21 09:27 ?584次閱讀

    c語(yǔ)言gets函數(shù)可以輸入數(shù)字嗎

    C語(yǔ)言中的gets函數(shù)是用來讀取字符串的,而不是用來讀取數(shù)字的。它會(huì)讀取輸入的字符直到遇到換行符或者文件結(jié)束符。因此,如果你嘗試使用gets函數(shù)來讀取數(shù)字,是無法準(zhǔn)確獲取數(shù)字本身的。 首先,讓我們
    的頭像 發(fā)表于 11-24 10:00 ?1374次閱讀

    scanf在C語(yǔ)言中的作用

    scanf在C語(yǔ)言中的作用? scanf是C語(yǔ)言中的輸入函數(shù),用于從標(biāo)準(zhǔn)輸入設(shè)備(如鍵盤)讀取數(shù)
    的頭像 發(fā)表于 11-23 14:13 ?1246次閱讀

    c語(yǔ)言程序設(shè)計(jì)概述

    電子發(fā)燒友網(wǎng)站提供《開源硬件-開發(fā)者智慧c語(yǔ)言程序設(shè)計(jì)(帶書簽).pdf》資料免費(fèi)下載
    發(fā)表于 11-20 10:25 ?1次下載
    <b class='flag-5'>c</b><b class='flag-5'>語(yǔ)言</b>程序設(shè)計(jì)概述

    C語(yǔ)言編程入門教程

    電子發(fā)燒友網(wǎng)站提供《C語(yǔ)言編程入門教程.rar》資料免費(fèi)下載
    發(fā)表于 11-20 10:23 ?8次下載
    <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>編程<b class='flag-5'>入門</b>教程

    51單片機(jī)C語(yǔ)言編程入門學(xué)習(xí)資料

    電子發(fā)燒友網(wǎng)站提供《51單片機(jī)C語(yǔ)言編程入門學(xué)習(xí)資料.pdf》資料免費(fèi)下載
    發(fā)表于 11-18 11:04 ?14次下載
    51單片機(jī)<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>編程<b class='flag-5'>入門</b>學(xué)習(xí)資料

    C語(yǔ)言進(jìn)階嵌入式系統(tǒng)高級(jí)C語(yǔ)言編程

    電子發(fā)燒友網(wǎng)站提供《C語(yǔ)言進(jìn)階嵌入式系統(tǒng)高級(jí)C語(yǔ)言編程.rar》資料免費(fèi)下載
    發(fā)表于 11-18 10:32 ?1次下載
    <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>進(jìn)階<b class='flag-5'>之</b>嵌入式系統(tǒng)高級(jí)<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>編程

    C語(yǔ)言映射表在串口數(shù)據(jù)解析中的應(yīng)用

    C語(yǔ)言映射表在串口數(shù)據(jù)解析中的應(yīng)用
    的頭像 發(fā)表于 11-13 09:33 ?439次閱讀