题目就不写了,大概意思就是:尽量用制表符'\t'替换掉字符串中的空格。

同学们需要注意的是,打印一个制表符'\t',其所占长度不是固定的。

这里要理解“制表符”和“制表符终止位”。“制表符”的作用是使得光标移动到下一个“制表符终止位”上。举个例子,假设制表符终止位是4、8、12、16......已经已经输入了10个字符,然后按一下Tab键,那么光标会移动到位置12上,同学们新建一个文本文档试一下就了解了。

这个题目看似简单,但是写一个简单、清晰的程序还是需要花一点脑筋的。

/*
这个程序看似简单,但是写一个简单、清晰的程序还是需要花一点脑筋的。
要去思考:什么情况下可以用一个'\t'代替空格?事实上只有两种情况,那就是:

  1、当一个“非空格字符”和该字符的下一个制表符终止位之间全部都是空格的时候,就可以用
  一个制表符'\t'替换这些空格

  例如:(用 _ 代表一个空格,用 | 表示制表符终止位,注意 | 并不是字符串的一部分)
  _ _ _ a b c _ _ | _ b c d f t q u r | _ d 

  能够用制表符替换的空格只有:(用 * 号 表示用制表符替换掉的空格)
  _ _ _ a b c * * | _ b c d f t q u r | _ d
 2、当两个制表符终止位之间全部都是空格的时候。这些空格就可以用'\t'替换。

  例如:  _ _ _ _ _ a b c | _ _ _ _ _ _ _ _ | a b c

  替换之后:  _ _ _ _ _ a b c | * * * * * * * * | a b c
*/

#include<stdio.h>

#define TABSTOP 8    //在Console中,制表符终止位一般是8,16,24,32......

int main()
{
    ; //已经读到的字符数
    ; //已经读到的空格数
    char c;          //当前读到的字符
    while ((c = getchar()) != EOF)
    {
        if (c != ' ')   //如果c不是一个空格
        {
            ) //如果已经读到 0 个空格
            {
                putchar(c);
                ++total;
            }
            else   //c不是一个空格,但是读到的空格数 不等于 0
            {
                ; --spaces)//输出空格
                    putchar(' ');
                spaces = ;                 //读到的空格数重新置为 0
                ++total;
                putchar(c);//输出当前读到的字符
            }
            if (c == '\n')
            {
                total = ;
                spaces = ;
            }
        }
        else //读了一个空格
        {
            ++total;
            ++spaces;

            int temp = (total-spaces) / TABSTOP;
            int nextLocation = (++temp) * TABSTOP;//下一个 制表符终止位
            if ( total< nextLocation)
                ;
            else//用'\t'填充空格
            {
                ; i < (nextLocation - (total-spaces)); i++)//为了明确输出的是'\t',用*号比较直观。
                    putchar('*');
                //putchar('\t');
                spaces = ;//读到的空格数重新置为 0
            }
        }
    }
    ;

}

如果把一行先读到一个字符数组里面,程序还会简单一些:

 #include<stdio.h>

 #define TABSTOP 8    //在Console中,制表符终止位一般是8,16,24,32......
 #define MAXSIZE 1024 // char数组的最大长度

 int getline(char s[] , int size);

 int main()
 {
     char s [MAXSIZE];
     )
     {
         printf("%s",s);
         ;// 遍历s中的字符
         char temp[MAXSIZE];
         ;//记录temp中的字符的数量

         for (; s[i] != '\0'; ++i)
         {
             ) % TABSTOP ==  && s[i] == ' ')
             {
                 ;
                 while (pre > i - TABSTOP && s[pre] == ' ')
                     --pre;
                 ++pre;
                 ; j < pre; ++j)
                     temp[count++] = s[j];
                 //temp[count++] = '\t';
                 for (int j = pre; j <= i; ++j)
                     temp[count++] = '*';
             }
             ) % TABSTOP ==  && s[i] != ' ')
                 for (int k = i / TABSTOP*TABSTOP; k <= i; ++k)
                     temp[count++] = s[k];
         }
         for (int k = i / TABSTOP*TABSTOP; k < i; ++k)
             temp[count++] = s[k];
         temp[count] = '\0';
         printf("%s",temp);
     }
     ;

 }

 int getline(char s[], int size)
 {
     char c;
     int i;
     ; i < size -  && (c = getchar()) != EOF && c != '\n'; ++i)
         s[i] = c;
     if (c == '\n')
     {
         s[i] = c;
         ++i;
     }
     s[i] = '\0';
     return i;
 }

《C程序设计语言》读书笔记----习题1-21的更多相关文章

  1. 《JavaScript高级程序设计》读书笔记--前言

    起因 web编程过程使用javascript时感觉很吃力,效率很低.根本原因在于对javascript整个知识体系不熟,看来需要找些书脑补一下,同时欢迎众网友监督. 大神推荐书籍 看了博客大神们推荐的 ...

  2. 《Javascript高级程序设计》读书笔记之对象创建

    <javascript高级程序设计>读过有两遍了,有些重要内容总是会忘记,写一下读书笔记备忘 创建对象 工厂模式 工厂模式优点:有了封装的概念,解决了创建多个相似对象的问题 缺点:没有解决 ...

  3. 《C程序设计语言》读书笔记----习题1-20

    练习1-20:编写程序detab,将输入中的制表符替换成适当数目的空格,使得空格充满到下一个制表符终止位的地方,.假设制表符终止位的位置时固定的,比如每隔n列就会出现一个终止位. 这里要理解“制表符” ...

  4. JavaScript高级程序设计(读书笔记)(一)

    本笔记汇总了作者认为“JavaScript高级程序设计”这本书的前七章知识重点,仅供参考. 第一章 JavaScript简介 JavaScript发展简史: 1995年,JavaScript诞生 19 ...

  5. 《C++primer》v5 第3章 字符串、向量和数组 读书笔记 习题答案

    本章问题 1.char *p="hello world";与char p[]="hello world"的问题. 简单说前者是一个指向字符串常量的指针,后者是一 ...

  6. 《JavaScript高级程序设计》读书笔记 ---基本包装类型

    为了便于操作基本类型值,ECMAScript 还提供了3 个特殊的引用类型:Boolean.Number 和String.这些类型与本章介绍的其他引用类型相似,但同时也具有与各自的基本类型相应的特殊行 ...

  7. Java核心技术卷一基础知识-第3章-Java的基本程序设计结构-读书笔记

    第3章 Java的基本程序设计结构 本章内容: 一个简单的Java应用程序 字符串 注释 输入输出 数据类型 控制流 变量 大数值 运算符 数组 本章主要讲述程序设计相关的基本概念(如数据类型.分支以 ...

  8. 《实战Java高并发程序设计》读书笔记

    文章目录 第二章 Java并行程序基础 2.1 线程的基本操作 2.1.1 线程中断 2.1.2 等待(wait)和通知(notify) 2.1.3 等待线程结束(join)和谦让(yield) 2. ...

  9. 《JavaScript 高级程序设计》读书笔记

    文章目录 第三章 基本语法 第四章 变量.作用域和内存问题 第五章 应用类型 1. Array 类型 2. RegExp 类型 3. Function 类型 4. String 类型 第六章 面向对象 ...

随机推荐

  1. c++内存流

    1.MemoryStream.h文件内容 ifndef _MEM_STREAM_H_ #define _MEM_STREAM_H_ #include <string> class CMem ...

  2. WebKit框架 浅析

    摘要 WebKit是iOS8之后引入的专门负责处理网页视图的框架,其比UIWebView更加强大,性能也更优. iOS中WebKit框架应用与解析 一.引言 在iOS8之前,在应用中嵌入网页通常需要使 ...

  3. selenium工具简介

    通过selenium百科可知: 组件 Selenium IDE:一个Firefox插件,可以录制用户的基本操作,生成测试用例.随后可以运行这些测试用例在浏览器里回放,可将测试用例转换为其他语言的自动化 ...

  4. linux下环境变量PS1设置

    PS1变量中提示符各项含义:   \d :代表日期,格式为weekday month date,例如:Wed Dec 12 \H :完整的主机名称.例如:hostname是debian.linux \ ...

  5. 1--OC -- HelloWorld

      一.点击Xcode,选择“Create a new Xcode project” 二.左边选择“OS X Application”,右边选择“Command Line Tool”,Next 三.输 ...

  6. javascript 总结学习一

    1 , javascript字符集:javascript采用的是Unicode字符集编码.为什么要采用这个编码呢?原因很简单,16位的Unicode编码可以表示地球人的任何书面语言.这是语言 国际化的 ...

  7. Linux学习 -- 备份与恢复

    备份 Linux系统需要备份的数据 /root/ /home/ /var/spool/mail /etc/ others 备份策略 完全备份 增量备份 差异备份 备份和恢复命令 dump  resto ...

  8. U3D游戏开发基础

    向量: 1. 向量的长度,即向量的模.计算公式为向量各个分量的平方和,然后开平方. 在D3DX库中,方法为:FLOAT  D3DXVec3Length(CONST  D3DXVECTOR3 * pV) ...

  9. linux 命令实现原理

    我们知道有些Linux的命令涉及到一些高效率的算法,在此做出一个积累吧,不是系统的. 1.tail命令打印一个文件的最后num行 2.grep命令从文本中匹配字符串 基于正则表达式的匹配很快. it ...

  10. 一个opencv 博客 大量文章(老版本编写C )

    http://blog.csdn.net/Augusdi/article/category/747412