C语言基础要素(019):输出ASCII码表

计算机以二进制处理信息,但二进制对人类并不友好。比如说我们规定用二进制值 01000001 表示字母’A’,显然通过键盘输入或屏幕阅读此数据而理解它为字母A,是比较困难的。为了有效的使用信息,先驱者们创建了一种称为ASCII码的交换代码,全称是美国信息交换标准代码。ASCII码使用指定的7位或8位二进制数组合来表示128或256种可能的字符。标准ASCII码也叫基础ASCII码,使用7位二进制数(剩下的1位二进制为0)来表示所有的大写和小写字母,数字0到9、标点符号,以及在美式英语中使用的特殊控制字符。

在这里插入图片描述

其中:

0-31及127:33个控制字符

这些字符在图一第1列标示,它们并没有特定的图形显示,但会依不同的应用程序,而对文本显示有不同的影响。如:

  • 控制符:LF(换行)、CR(回车)、FF(换页)、DEL(删除)、BS(退格)、BEL(响铃)等;
  • 通信专用字符:SOH(文头)、EOT(文尾)、ACK(确认)等;
  • 8(退格)、9(水平制表符)、10(换行符)、13(回车符)等。

32-126:95个可打印字符

  • 32:空格
  • 48-57:0到9十个阿拉伯数字
  • 65-90:26个大写英文字母
  • 97-122:26个小写英文字母
  • 其余为一些标点符号、运算符号等。

128-255:扩展ASCII码

  • 许多基于x86的系统都支持使用扩展ASCII。
  • 扩展ASCII码允许将每个字符的第8位用于确定附加的128个特殊符号字符、外来语字母和图形符号。

字符与编码

基于这些基础知识,我们编写了一个杂乱的程序,但可以帮助我们对字符与编码的关系建立一个雏形。

#include <stdio.h>int main()
{printf("%cello%c%corld!%c", 72, 44, 119, 10);printf("%c%c%c%c%c\n", 48, 49, 50, 51, 52);printf("ASCII\n");return 0;
}

在这里插入图片描述

现在,一定要理解字符'0'数字0的区别,我们所见的、可读的、可书写的“数字”,它们都是数值字符化。数值0的字符化就是字符'0’,它的ASCII编码是48;控制台上输出的 1234,实际是编码为49 50 51 52的几个字符;而数值0的编码就是0,即图中标示为 NULL的值;数值9的编码就是9,是图中以 HT 标示的值。

#include <stdio.h>int main()
{char char9 = '9';       // 字符9,它的ASCII编码为57char char9_2 = 57;      // 字符9的不直观表示,ASCII表中57对应的字符是9int num9 = 9;           // 数值9,它的ASCII编码为9printf("%d %d\n", char9, char9_2);printf("%c %c\n", char9, char9_2);// 字符9与数值9不一样num9 == char9 ? printf("true") : printf("false");printf("\n");// 将字符9转换为数值9int char_to_num1 = (char9 - '0');   // 与字符'0'相减int char_to_num2 = (char9_2 - 48);  // 正好对应相应数值编码num9 == char_to_num1 ? printf("true\n") : printf("false\n");num9 == char_to_num2 ? printf("true\n") : printf("false\n");return 0;
}

在这里插入图片描述

打印ASCII码表

在编写第一版程序前我们先总结一下,标准ASCII码表有128个字符,其中33个是不可显的控制字符,也就是说用printf函数打印后在屏幕上不可见;另外95个是可见字符,即可打印字符;后面的128个字符属于ASCII码扩展表,我们先不予考虑。这里一个潜在的关键点是:编码从0开始并且连续的!这表示我们可以使用循环语句来处理,下面是最简单的第一版。

#include <stdio.h>int main()
{for (int i = 0; i < 128; ++i){printf("%c ", i);}return 0;
}

在这里插入图片描述

在循环体中我们以%c 控制符打印数值对应的ASCII码,从程序运行结果来看,输出有些杂乱,不可显字符被显示为空心框了。也许你在网上查询过ASCII码表的图示,对于控制字符,有些会显示出一些有趣的形状,这与字体有关,不同的字体对应的编码可能是一种有趣的字符,我们可以设置一下控制台的字体,让其显示更加友好,但这不能改变它们是控制字符的本质。

在这里插入图片描述
在这里插入图片描述

此外,还需要注意到这些情况:

  • 我们在循环体中并没有输出换行符,但第一行在输出一些字符后换行了。
  • 我们无法有效的使用它查询信息,比如说字母’X’对应的编码是多少,
  • 在程序运行过程中,你可能会听到“叮咚”一声铃响。

我们先把程序改进一些,让输出变得美观一些,然后再来解释这些现象,下面是我们改进后的第2版。

#include <stdio.h>int main()
{printf("DEC\t HEX\t CHR\n");printf("--------------------\n");for (int i = 0; i < 128; i += 1){printf("%d\t %x\t %c\n", i, i, i);}return 0;
}

在这里插入图片描述

看一下运行结果图,感觉还不错。这一版中,我们首先使用printf函数打印了一个表头,该表格有三列,分别用于显示当前数值的十进制,十六进制(用格式控制符%x指示)以及字符表示。我们使用水平制表符’\t’控制表格的列间距,'\t’后面多出一个空格,只是为了使代码看起来更清晰。在循环体中,由于printf函数使用了三个控制字符,因此需要将当前数值传值三次。在CHR列,有些控制字符在当前字体下仍然没有对应的显示字符。同时还要注意第10行与第11行之间多了一个空行,这是为什么呢?

查阅一下ASCII码表,我们会发现数值10代表换行控制符。当以%c格式符打印数值10时,相当于执行了一次换行操作,而在循环体中使用printf输出字符时又追了一个\n,因此中间多出了一个空行。同时ASCII码值为7的控制字符代表响铃,其转义字符用’\a’表示,这就是你能听到“叮咚”一声响的原因。你可以使用下面两行响铃代码测试一下。

#include <stdio.h>int main()
{// 第1次响铃printf("\a");// 相当于延时一定时间(糟糕的延时代码),否则两次响铃可能只会听到一次。for (int i = 0; i < 999999999; i++); // 注意:这里是空语句printf("here");// 第二次响铃printf("%c", 7);return 0;
}

现在回到主线任务,这一版的程序已经相当友好了,只是假设控制台是一张纸的话,这样的输出有些太浪费了,毕竟右边空白了那么多,我们完全可以按两列输出。

#include <stdio.h>int main()
{printf("DEC\t HEX\t CHR\t\t DEC\t HEX\t CHR\n");printf("----------------------------------------------------\n");for (int i = 0; i < 128; i += 2){printf("%d\t %x\t %c\t\t ", i, i, i);printf("%d\t %x\t %c\n", i + 1, i + 1, i + 1);}return 0;
}

在这里插入图片描述

看一下输出效果。编码10(LF)代表换行,程序进行换行后,导致下一个输出(编码11)错乱,这是美中不足的地方。关键的一点是:换行后,编码11并没有从下一行的开头输出,而是直接跳转到下一行的当前位置。如果把键盘想像为打字机,它的当前位置就是将要书写的字符位置,再把控制台想像为纸张,在Windows下换行的意思就是让纸向上移动一行,但打字机的书写针头位置并没有改变,这就是为什么11会在这里输出的原因。在Windows下,让打字机回到纸张左边的命令是“回车”,即ASCII编码值为13(CR)的值。所以,你想要跳转到下一行开头输出时,那么对应的控制命令就是回车换行(CRLF)。但在有些系统下,可能仅仅一下LF命令就能完成同样的操作。

这篇文章我是用 Microsoft Visual Code 编写的,我们可以在右下角的状态栏上看到它控制文本文件的换行命令是CRLF

在这里插入图片描述

现在我们尝试将其按三列输出,看看还会有什么问题。

#include <stdio.h>int main()
{printf( "DEC\tHEX\tCHR\t   DEC\tHEX\tCHR\t   DEC\tHEX\tCHR\n" );printf( "----------------------------------""---------------------------------\n" );for ( int i = 0; i < 128; i += 3 ){printf( "%d\t%x\t%c\t   ", i, i, i );printf( "%d\t%x\t%c\t   ", i+1, i+1, i+1 );printf( "%d\t%x\t%c\n",    i+2, i+2, i+2 );}printf( "\n" );return 0;
}

在这里插入图片描述

这次的输出前半部分太凌乱了!

  • 数值10/11/14并没有与列起始处对齐。
  • 数值12消失了!
  • 13居然在14的后面输出!

先来解决10没有对齐到列的问题。数值9代表的是控制字符'\t',这使得对于9的输出,相当于在后面插入一个制表符,即当程序按"%c"格式输出它时,相当于增加了水平间距,这导致后面10的输出没有对齐。为了解决这个问题,我们在按"%c"输出9时,把9替换为32(空格),相当于消除的制表符自身的影响。同理,编码10会引起换行操作,我们也将它替换空格字符。对于消失的12,它是控制符换页的意思,我们也将它替换为空格即可。

最终的成果

#include <stdio.h>int main()
{int i;int c1, c2;printf("DEC\tHEX\tCHR\t   DEC\tHEX\tCHR\t   DEC\tHEX\tCHR\n");printf("---------------------------------""----------------------------------\n");for (i = 0; i < 128; i += 3){if (i == 9 || i == 12 || i == 27){c1 = 32;c2 = 32;}else{c1 = i;c2 = i + 1;}// 当前行从编码i开始,连续输出3个编码,i, i+1, i+2// if语句将影响输出格式的控制字符i转换为了空格printf("%d\t0x%x\t%c\t   ", i, i, c1);printf("%d\t0x%x\t%c\t   ", i + 1, i + 1, c2);if ((i + 2) < 128)printf("%d\t0x%x\t%c\n",    i + 2, i + 2, i + 2);}return 0;
}

在这里插入图片描述

练习

  1. 编写程序将一个小写字母转换为相应的大写字母。
  2. 将控制台重置回之前的设置。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/76511.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

鸿蒙定位开发服务

引言 鸿蒙操作系统&#xff08;HarmonyOS&#xff09;作为面向万物互联时代的分布式操作系统&#xff0c;其定位服务&#xff08;Location Kit&#xff09;为开发者提供了多场景、高精度的位置能力支持。本文将从技术原理、开发流程到实战案例&#xff0c;全面解析鸿蒙定位服务…

rknn_convert的使用方法

rknn_convert是RKNN-Toolkit2提供的一套常用模型转换工具&#xff0c;通过封装上述API接口&#xff0c;用户只需编辑模型对应的yml配置文件&#xff0c;就可以通过指令转换模型。以下是如何使用rknn_convert工具的示例命令以及支持的指令参数&#xff1a; python -m rknn.api.…

解决 axios get请求瞎转义问题

在Vue.js项目中&#xff0c;axios 是一个常用的HTTP客户端库&#xff0c;用于发送HTTP请求。qs 是一个用于处理查询字符串的库&#xff0c;通常与 axios 结合使用&#xff0c;特别是在处理POST请求时&#xff0c;将对象序列化为URL编码的字符串。 1. 安装 axios 和 qs 首先&a…

【XTerminal】【树莓派】Linux系统下的函数调用编程

目录 一、XTerminal下的Linux系统调用编程 1.1理解进程和线程的概念并在Linux系统下完成相应操作 (1) 进程 (2)线程 (3) 进程 vs 线程 (4)Linux 下的实践操作 1.2Linux的“虚拟内存管理”和stm32正式物理内存&#xff08;内存映射&#xff09;的区别 (1)Linux虚拟内存管…

torch 拆分子张量 分割张量

目录 unbind拆分子张量 1. 沿着第n个维度拆分&#xff08;即按“批次”拆分&#xff09; split分割张量 常用用法&#xff1a; 总结&#xff1a; unbind拆分子张量 import torchquaternions torch.tensor([[1, 2, 3, 4], [5, 6, 7, 8]]) result torch.unbind(quaternio…

【Linux】内核驱动学习笔记(二)

7、framebuffer驱动详解 7.1、什么是framebuffer (1)裸机中如何操作LCD (2)OS下操作LCD的难点 (3)framebuffer帧缓冲&#xff08;简称fb&#xff09;是linux内核中虚拟出的一个设备 (4)framebuffer向应用层提供一个统一标准接口的显示设备 (5)从驱动来看&#xff0c;fb是一个…

用 Docker Compose 与 Nginx 反向代理部署 Vikunja 待办事项管理系统

在高效管理日常任务和项目的过程中&#xff0c;开源待办事项工具 Vikunja 以其简洁、直观的设计和多视图支持受到越来越多用户的青睐。本文将详细介绍如何使用 Docker Compose 快速部署 Vikunja&#xff0c;并通过 Nginx 反向代理实现 HTTPS 访问&#xff0c;从而确保服务安全稳…

使用Python快速接入DeepSeek API的步骤指南

使用Python快速接入DeepSeek API的步骤指南 1. 前期准备 注册DeepSeek账号 访问DeepSeek官网注册账号 完成邮箱验证等认证流程 获取API密钥 登录后进入控制台 → API管理 创建新的API Key并妥善保存 安装必要库 pip install requests # 可选&#xff1a;处理复杂场景 pip…

Redis 主要能够用来做什么

Redis&#xff08;Remote Dictionary Server&#xff09;是一种基于内存的键值存储数据库&#xff0c;它的性能极高&#xff0c;广泛应用于各种高并发场景。以下是 Redis 常见的用途&#xff1a; 1. 缓存&#xff08;Cache&#xff09; 作用&#xff1a;存储热点数据&#xf…

印度股票实时数据API接口选型指南:iTick.org如何成为开发者优选

在全球金融数字化浪潮中&#xff0c;印度股票市场因其高速增长潜力备受关注。对于量化交易开发者、金融科技公司而言&#xff0c;稳定可靠的股票报价API接口是获取市场数据的核心基础设施。本文将深度对比主流印度股票API&#xff0c;并揭示iTick在数据服务领域的独特优势。 一…

24.多路转接-poll

poll也是一种linux中的多路转接的方案 解决select的fd有上限的问题解决select每次调用都要重新设置关心的fd poll函数接口 poll, ppoll - wait for some event on a file descriptor#include <poll.h>int poll(struct pollfd *fds, nfds_t nfds, int timeout);DESCRIP…

Linux 基础入门操作 前言 linux操作指令介绍

1 linux 目录介绍 Linux 文件系统采用层次化的目录结构&#xff0c;所有目录都从根目录 / 开始 1.1 核心目录 / (根目录) 整个文件系统的起点、包含所有其他目录和文件 /bin (基本命令二进制文件) 存放系统最基本的shell命令&#xff1a;如 ls, cp, mv, rm, cat 等&#…

Chrome开发者工具实战:调试三剑客

在前端开发的世界里&#xff0c;Chrome开发者工具就是我们的瑞士军刀&#xff0c;它集成了各种强大的功能&#xff0c;帮助我们快速定位和解决代码中的问题。今天&#xff0c;就让我们一起来看看如何使用Chrome开发者工具中的“调试三剑客”&#xff1a;断点调试、调用栈跟踪和…

函数柯里化(Currying)介绍(一种将接受多个参数的函数转换为一系列接受单一参数的函数的技术)

文章目录 柯里化的特点示例普通函数柯里化实现使用Lodash进行柯里化 应用场景总结 函数柯里化&#xff08;Currying&#xff09;是一种将接受多个参数的函数转换为一系列接受单一参数的函数的技术。换句话说&#xff0c;柯里化将一个多参数函数转化为一系列嵌套的单参数函数。 …

torch.nn中的非线性激活介绍合集——Pytorch中的非线性激活

1、nn.ELU 基本语法&#xff1a; class torch.nn.ELU(alpha1.0, inplaceFalse)按元素应用 Exponential Linear Unit &#xff08;ELU&#xff09; 函数。 论文中描述的方法&#xff1a;通过指数线性单元 &#xff08;ELU&#xff09; 进行快速准确的深度网络学习。 ELU 定义为…

Databend Cloud Dashboard 全新升级:直击痛点,释放数据价值

自 Databend Cloud 上线以来&#xff0c;我们一直致力于为用户提供高效的数据处理与可视化体验。早期&#xff0c;我们在工作区的“图表”区域推出了轻量级可视化功能&#xff0c;支持积分卡、饼图、柱状图和折线图四种展示方式。这些功能简单易用&#xff0c;基本满足了用户对…

Android Fresco 框架扩展模块源码深度剖析(四)

Android Fresco 框架扩展模块源码深度剖析 一、引言 在 Android 开发领域&#xff0c;图片处理一直是一个重要且具有挑战性的任务。Fresco 作为 Facebook 开源的强大图片加载框架&#xff0c;在图片的加载、缓存和显示等方面已经提供了非常完善的功能。然而&#xff0c;为了满…

蓝桥杯最后十天冲刺 day 2 双指针的思想

双指针思想介绍 双指针&#xff08;Two Pointers&#xff09;是一种在数组或链表等线性结构中常用的算法技巧&#xff0c;通过使用两个指针&#xff08;索引或引用&#xff09;以不同的速度或方向遍历数据结构&#xff0c;从而高效解决问题。双指针通常用于优化暴力解法&#…

Axure 使用笔记

1.Axure如何制作页面弹窗 https://blog.csdn.net/SDTechnology/article/details/143948691 2.axure 怎么点击按钮打开新页面 &#xff08;1&#xff09;新建交互 &#xff08;2&#xff09;单击是触发 &#xff08;3&#xff09;选择打开链接 &#xff08;4&#xff09;选择…

STM32实现一个简单电灯

新建工程的步骤 建立工程文件夹&#xff0c;Keil中新建工程&#xff0c;选择型号工程文件夹里建立Start、Library、User等文件夹&#xff0c;复制固件库里面的文件到工程文件夹工程里对应建立Start、Library、User等同名称的分组&#xff0c;然后将文件夹内的文件添加到工程分组…