zoukankan      html  css  js  c++  java
  • 读书笔记之:C/C++程序员实用大全—C/C++最佳编程指南

    这本书中列出了31章共1500个知识点,带有很多的例子。本书适合对C/C++有一些了解的程序员进行查缺补漏。因为这里边对知识点的讲解比较凌乱不是很条理,前面的内容有时就用到了后面内容,如果你不是了解的话,读起来就比较困难了。

    本书还有一个很大的特点是很多地方是针对Dos 平台的,我是使用的linux,所以有很多地方是不适合的。即便是对于windows 平台,有些地方可能也不适用了。因为这本书还是比较老的。有些不需要的地方可以跳过。关于C++的介绍,如果你是刚刚学C++话还是要先读《C++ Primer》,那本书比较条理,可以让你理清关系。了解各个知识点。

    书最后的关于Windows编程的部分我基本没有看,因为相对来说知识比较旧了,并且我是Linux平台的。

     第1章 C语言入门

    1. 许多C编译器把%i格式符等价于%d。最好使用%d,因为%i是一个比较陈旧的格式符,以后的编译器可能不支持它。

    %ld可以来显示长整型

    %e 科学计数法显示浮点数

    %g 根据实际情况自行选择使用%f还是%e来显示浮点数

    显示数值正负号:在格式符中紧跟%后添加一个+号即可。如%+d,%+f

    %05d 其中5用来表示显示数值时所采用的最少的字符数目,0表示使用0填充,

    %8.3f表示共至少8位,其中小数点后保留3位

    %-5d 左对齐

    2. 为了引导printf在八进制或十六进制前添加合适的前缀,可紧跟格式符号%后置入#。如:

    printf("value is %#x\n",value);

    3. 判断printf已经显示的字符数目

    当程序作复杂的屏幕格式化时,有时需要知道printf已经显示的字符数目,可以使用%n

    如:printf("abd%n 123456%n\n",&a,&b);

    执行完之后a=3,b=10

    printf的返回值是已经输出的字符总数

    4.ANSI设备驱动器

     

    第2章 宏与常量

    1. 预定义宏

    __FILE__ __LINE__ __DATE__ __TIME__

    __STDC__用来判断是否是ANSI C编译器

    2.改变与处理器行计数

    利用#line预处理来改变行号

    如:#line 100 "FILENAME.c" //表示当前行为100行,下面如果出现__LINE__的话,是从现在开始计数的

     

    第3章 字符串

    1. C中常用的字符处理函数

    #include <ctype.h>

    int isalnum(int c);

    int isalpha(int c);

    int iscntrl(int c);

    int isascii(int c);

    #include <ctype.h>

    int isdigit(int c);

    int isodigit(int c);

    int isxdigit(int c);

    #include <ctype.h>

    int isgraph(int c);

    int ispunct(int c);

    int isprint(char c);

    #include <ctype.h>

    int islower(int c);

    int isupper(int c);

    #include <ctype.h>

    int isblank(int c);

    int isspace(int c);

    ing iswhite(int c);

    #include <ctype.h>

    int tolower(int c);

    int toupper(int c);

     

    2. 常用的字符串处理函数

    #include <string.h>

    char* strcat(char*dest,const char*src);

    char* strncat(char*dest,const char*src,size_t n);

    #include <string.h>

    int strcmp(const char *s1,const char *s2);

    int strncmp(const char *s1,const char *s2,size_t n);

    #include <string.h>

    size_t strlen(const char*s);

    #include <string.h>

    char*strstr(const char*src,cosnt char*sub);

    char*strtok(char* str,const char*set);

    #include <string.h>

    char *strchr(const char*s,int c);

    char *strrchr(const char*s,int c);

    #include <stdlib.h>

    double strtod(const char *nptr, char **endptr);

    float strtof(const char *nptr, char **endptr);

    long double strtold(const char *nptr, char **endptr);

    #include <stdlib.h>

    double atof(const char*str);

    int atoi(const char*str);

    long atol(const char*str);

    long long atoll(const char*str);

    #include <string.h>

    char* strcpy(char*dest,const char*src);

    char* strncpy(char*dest,const char*src,size_t n);

    #include <string.h>

    int strcoll(const char*s1,const char*s2);

    size_t strxfrm(char *dest,const char*src,size_t len);

    #include <string.h>

    size_t strspn(const char*s,cosnt char*set);

    size_t strcspn(const char*s,cosnt char*set);

    char* strpbrk(const char*s,const char*set);

     

    第4章 函数

    1. 在C语言中,如果用户自定义的函数名称和库函数的名称是一样的,那么编译器一般会选择使用用户定义的函数。

    堆栈的基本目的就是用来支持函数调用的。函数调用的时候,编译器会将返回地址和参数压入堆栈。

    程序员将计算机用来压入和弹出堆栈所花费的时间成为函数的开销。

    2. volatile关键字

    3. 与堆栈相关:调用结构和基指针

    4. 支持参数个数可变的函数

    为了支持参数可变,在C中是使用宏va_arg,va_end和va_start(在头文件stdarg.h中定义)引导程序创建自己的支持参数个数可变的函数。实质上是宏每次从堆栈中取得参数,一直到程序获取最后一个参数。使用这些宏获取参数时,必须知道每个参数的类型。在printf这个最典型的不定参数个数函数中使用标识符(如%d,%s,%f)来匹配参数类型。

    可以使用类似printf中使用的格式符将参数类型传递给函数:

    result=add_values("%d %d %d",1,2,3);

     

    第5章 键盘操作

    1.

    #include <stdio.h>

    int fgetc(FILE *stream);

    char *fgets(char *s, int size, FILE *stream);

    int getc(FILE *stream);

    int getchar(void);

    char *gets(char *s);

    int ungetc(int c, FILE *stream);

    上面几个函数的关系:

    fgetc()是从stream中读取下一个字符串,然后返回,

    getc和fgetc是等价的,只不过它是用宏实现的。

    getchar()等价于getc(stdin)

     

    gets()是从stdin读入多个字符到缓冲s中,知道遇到行终结符,并且在s中使用'\0'代替终结符

    第6章 数学

    1. 获得浮点数的尾数和指数

    可以使用frexp函数来获得浮点数的指数和尾数

    #include <math.h>

    double frexp(double value,int *exponent);

    其中返回值是位数,第二个参数是指数

     

    类型float大小为4字节,即32位,内存中的存储方式如下:

    高地址<-------------------------------------->低地址

    | 符号位 | 指数 | 尾数 |

    | 1 bit | 8 bit | 23 bit |

    31<------>30<--------->22<---------------------->0

     

    类型double大小为8字节,即64位,内存布局如下:

    高地址<---------------------------------------->低地址

    | 符号位 | 指数 | 尾数 |

    | 1 bit | 11 bit | 52 bit |

    63<------>62<------------>51<--------------------->0

     

    2. 计算X*2E的结果

    可以使用ldexp函数来计算x*2e

    #include <math.h>

    double ldexp(double value,int exponent);

    3. 在文件stdlib.h提供了宏min和max

    4. 把浮点值分解成整数和小数部分

    使用modf函数来把浮点数分解成整数和小数部分。

    #include <math.h>

    double modf(double value,double *integer_part);

    5. #include <math.h>

    double pow(double value,double power);

    double pow10(int power);

     

    double log(double value);

    double log10(double value);

    6.生成随机数

    C中提供的生成随机数函数:rand和random,它们两个都返回整型随机数。

    #include <stdlib.h>

    int rand(void);

    int random(int ceiling);

    rand返回一个范围在0和RAND_MAX之间的随机整数。第二个函数random返回一个范围0与ceiling之间的随机数,ceiling规定了随机数的最大值,主调函数把它传递给random函数。

     

    第7章 文件、目录和磁盘

    1. FILE结构

    C中进行文件流操作的FILE是一个结构体,一般使用FILE*指针。

    通常是使用fopen函数返回一个FILE*指针。

    大体的FILE的结构格式如下:

    2. 使用低级和高级文件I/O

    其实高级I/O就是我们常用的基于流的文件功能。而所谓的低级文件I/O就是一些系统调用。上面所列出的函数都是一些系统调用,不同的系统的实现是不一样的。这儿其实就是一个"库函数与系统调用关系"的问题。

    3、已知文件流指针,获取对应的文件描述符

    int fileno(FILE *stream);

    主要应用:如果程序先前用fopen打开了一个文件,但是又想对其进行锁定操作,

    例如 int fcntl(int fildes, int command, struct flock *flock_structure)(注意:command为F_GETLK, F_SETLK, F_SETLKW中其中一者)

    或者 int lockf(int fd, int cmd, off_t len),就得先用fileno得到对应的文件描述符后再进行fcntl/lockf操作。

    4、已知文件描述符,获取对应的文件流指针

    FILE *fdopen(int fildes, const char *type);

    将一个文件流关联到一个打开的文件描述符

    fildes可以是open,dup, dup2, creat, pipe, socket等系统调用返回的结果。

    type指定打开的方式,同fopen的"r","w","a"等等。

    fdopen的打开方式受制于fildes的打开方式,例如:open时用O_RDONLY,那么fdopen就只能用"r"方式。

    主要应用:当你不得不只能打开文件号,但却想用fprintf,fscanf等流操作进行数据读写时,就再用fdopen一次便可。

    5. 重命名文件

    C中给出了函数rename可以来进行文件的重命名或移动

    #include <stdio.h>

    int rename(const char *old, const char *new);

    6. 删除文件

    C库函数中给出了remove函数来删除文件

    #include <stdio.h>

    int remove(const char *pathname);

    同时在unix操作系统中还提供了函数来删除文件

    #include <unistd.h>

    int unlink(const char *path);

    7. 判断程序如何访问文件

    C函数access检查指定文件是否存在,以及用户是否能按要求打开文件。

    这个一般是操作系统提供实现方式。

    8.将缓冲写入磁盘

    C中提供函数fflush将缓存中的数据写入到磁盘中去。

    #include <stdio.h>

    int fflush(FILE *stream);

    9. 获取文件流的文件句柄

    #include <stdio.h>

    int fileno(FILE *stream);

    10. 临时文件C接口

    如果在C程序中碰到文本处理,极有可能会涉及到临时文件的处理。

    在不同情境下,可能对临时文件接口有不同的需求。比如有时只是将临时文件作为临时的数据存储空间,无需与其它进程共享,这时可以直接使用tmpfile()函数。tmpfile()返回的是一个FILE *句柄,由tmpfile()创建的临时文件没有实际的文件名,这样处理的好处是,当程序结束时,文件不需要程序手动删除。

    但如果你需要与其它进程共享临时文件,或是需要以临时文件的文件名作为参数调用其它程序以传递数据,tmpfile()就不能胜任了,这时就需要用到mkstemp()。mkstemp()通过传入的模板字符串生成一个不存在的文件名,同时创建该文件,将文件句柄做为函数返回值返回,而文件名可以从被修改了的模板字符串中得到。

    以上两个函数算是比较现代的了,如果对于历史感兴趣,可以看看下面三个函数。

    mktemp()函数用于生成一个不存在的文件名。这个函数现在已经不提倡使用,在POSIX.1-2008中甚至已经将其删除了。从glibc mktemp(3)中可以了解到,不提倡使用mktemp()的主要原因是mktemp()只是生成一个当前不存在的文件名,而没有直接创建该文件。而如果需要创建这个文件,那么在mktemp()和creat()之间可能这个临时文件被其它进程创建,从而导致文件创建失败。

    tmpnam()函数有与mktemp()一样的问题,同时,当传入参数为NULL时,由于需要用到static变量,所以不是线程安全的。在POSIX.1-2008中已经不建议使用。

    tempnam()与tmpnam()相似,只是参数更多,可控性更高。同样,在POSIX.1-2008中已不建议使用。

    11. 每次读写一个字

    C中提供了函数getw和putw来读写一个字

    #include <stdio.h>

    int getw(FILE *stream);

    int putw(int w, FILE *stream);

     

    12. 读写结构体

    低级的read和write函数

    高级fread和fwrite函数都可以进行结构体读写

    13. 把文件句柄和文件流联系起来

    可以使用fdopen函数来晶文件句柄和文件流联系起来:

    #include <stdio.h>

    FILE *fdopen(int fd, const char *mode);

    14. 目录操作

    打开目录:opendir

    读取目录:readdir

    重置目录列表:rewinddir

     

    15. 按行读取和写文本

    使用函数fgets和fputs

     

    16. 为什么不能使用fgets和fputs来复制二进制文件

    因为在fgets函数在读取文本时,fgets把CTRL+Z(ASCII值为26)当作文件结尾。因为二进制文件中很可能多次出现数值26,fgets将在第一次遇到26时就结束拷贝过程。所以要想可靠的拷贝二进制文件的话,必须使用C 的低级I/O函数。

    17. 判断文件结尾

    当fgets函数遇到文件结尾时,它返回NULL,同样当fgetc遇到文件结尾时返回EOF。所以在程序进行某些特定操作之前,很有必要判断文件指针是否处于文件尾端。一般使用函数feof。

    #include <stdio.h>

    int feof(FILE *stream);

    18. 获取文件句柄信息

    #include <sys/types.h>

    #include <sys/stat.h>

    #include <unistd.h>

    int stat(const char *path, struct stat *buf);

    int fstat(int fd, struct stat *buf);

    int lstat(const char *path, struct stat *buf);

    stat结构体

    struct stat {

    dev_t st_dev; /* ID of device containing file */

    ino_t st_ino; /* inode number */

    mode_t st_mode; /* protection */

    nlink_t st_nlink; /* number of hard links */

    uid_t st_uid; /* user ID of owner */

    gid_t st_gid; /* group ID of owner */

    dev_t st_rdev; /* device ID (if special file) */

    off_t st_size; /* total size, in bytes */

    blksize_t st_blksize; /* blocksize for file system I/O */

    blkcnt_t st_blocks; /* number of 512B blocks allocated */

    time_t st_atime; /* time of last access */

    time_t st_mtime; /* time of last modification */

    time_t st_ctime; /* time of last status change */

    };

     

     

  • 相关阅读:
    【2021-04-15】台上一分钟,背后十年功
    【2021-04-14】买了人生中的第一个车位
    计算机网络实验部分
    乘积最大子数组
    计算各个位数不同的数字个数
    周总结
    最长回文子序列
    最长回文串
    回文子串
    拼图
  • 原文地址:https://www.cnblogs.com/xkfz007/p/2425612.html
Copyright © 2011-2022 走看看