C语言 使用char字符实现汉字处理
- 系统:windows 64
- 编译器:gcc version 8.1.0 (x86_64-posix-seh-rev0, Built by MinGW-W64 project)
- 文本编辑器:notepad
- 控制台:Cmder
- 编程语言:C、Python
首先,要想打印汉字,必须考虑到编码问题。在windows下,由于系统使用GBK编码,而GCC解析时使用UTF-8而会导致以下代码运行时出现乱码:
![](https://common.cnblogs.com/images/copycode.gif)
#include <stdio.h> int main()
{
char *str = "你好,世界!"; printf("%s\n", str); return 0;
}
![](https://common.cnblogs.com/images/copycode.gif)
解决方法为:使用“-fexec-charset=gbk”命令
解决了编码问题,我们还需要了解几点:
- char类型本质上是数字,占据一个字节(即八位),可以通过%d打印编码,通过%c打印字符
- 在C语言中,一个汉字占据两个char类型
- 汉字的两个char类型为负数
- 在打印汉字时,它的两个char必须紧跟着
根据这几点,我们可以打印出汉字以及它们的编码:
![](https://common.cnblogs.com/images/copycode.gif)
#include <stdio.h>
#include <string.h> int main()
{
// str为字符指针,指向一个字符字面量,这个字符字面量由'\0'结尾
char *str = "你好,世界!Hello, world!";
// chr为字符指针,指向str所指向的字符字面量的第一个字符的地址,即'你'字符的两个char中的第一个
char *chr = str; printf("%zu %s\n", strlen(str), str);
// 如果遇到'\0',说明字符串结束了
while (*chr != '\0')
{
// 如果chr的编码为负数,则说明遇到了一个汉字
if (*chr < 0)
{
// 打印汉字及汉字的编码
// 注意两个char必须紧紧跟着打印(%c%c),否则会打印出 ??
printf("%c%c: %d%d\n", *chr, *(chr+1), *(chr), *(chr+1));
// chr自增两个字节(因为每个汉字都由两个char组成)
chr += 2;
}
else
{
// 打印英文字符
printf("%c: %d\n", *chr, *chr);
// chr自增一个字节
++chr;
}
} return 0;
}
![](https://common.cnblogs.com/images/copycode.gif)
从上图,我们可以看出,这个字符串占据了25个字节,4个汉字加2个全角符号占据了12个字节,再加上23个英文字符,总共25个字节。我们可以从下图更清晰地看出str的构造:
但是,根据我们在网上查询的结果,汉字‘你’的GBK编码应为:C4E3,但是在这里,却打印出了:-60-29,这是为什么呢?
这里涉及到进制的问题,可能-60-29是十六进制数C4E3的十进制数?
首先,我们先通过Python看看C4E3的二进制数以及十进制数。这好像跟-60-29根本不沾边。
我们先看看下面的代码,导入<limits.h>头文件,看看char类型的取值范围为多少:
![](https://common.cnblogs.com/images/copycode.gif)
#include <stdio.h>
#include <limits.h> int main()
{
printf("[%d ~ %d]\n", CHAR_MIN, CHAR_MAX);
printf("%c%c\n", 0xC4, 0xE3);return 0;
}
![](https://common.cnblogs.com/images/copycode.gif)
我们可以看到:char类型的取值范围为[-128 ~ 127],但是我们却可以打印出汉字”你“。这是为什么呢?明明char的取值范围最多127,而汉字“你”的两个字符分别为:196和227,都超过了这个值。其实这是因为,C语言将这两个数字的二进制数作为负数处理。C中的char类型有1个字节,占8位,而它的最高位为符号位,当它为0时为正,1时则为负。C通过对正数做补码操作得到负数。补码,即对一个二进制数取反,然后再加1。比如,0xC4的二进制数为0b11000100,我们可以看到最高位1,在C中这个数就是负数。我们可以通过对这个二进制数做补码操作,得到0b00111100,即60。所以0b11000100在C中表示为-60。
从以上,我们可以发现,GBK编码中,一个汉字占两字节。因为C中char类型只占一个字节,所以需要使用两个char类型来表示汉字。因为C中char为有符号类型,char的表示范围为[-128 ~ 127],所以在遇到大于127的数字时,会被char表示为负数。
其实,我们还可以使用unsigned char来实现。char默认是有符号的,取值范围为:-128 ~127。而unsigned char的取值范围则为:0~255,那么汉字“你”的编码就会被显示为:196和227。
![](https://common.cnblogs.com/images/copycode.gif)
#include <stdio.h>
#include <string.h> int main()
{
// str为字符指针,指向一个字符字面量,这个字符字面量由'\0'结尾
unsigned char *str = (unsigned char *)"你好,世界!Hello, world!";
// chr为字符指针,指向str所指向的字符字面量的第一个字符的地址,即'你'字符的两个char中的第一个
unsigned char *chr = str; printf("%zu %s\n", strlen(str), str);
// 如果遇到'\0',说明字符串结束了
while (*chr != '\0')
{
// 如果chr的编码大于127,则说明遇到了一个汉字
if (*chr > 127)
{
// 打印汉字及汉字的编码
// 注意两个char必须紧紧跟着打印(%c%c),否则会打印出 ??
printf("%c%c: %d %d\n", *chr, *(chr+1), *(chr), *(chr+1));
// chr自增两个字节(因为每个汉字都由两个char组成)
chr += 2;
}
else
{
// 打印英文字符
printf("%c: %d\n", *chr, *chr);
// chr自增一个字节
++chr;
}
} return 0;
}
![](https://common.cnblogs.com/images/copycode.gif)
C语言 使用char字符实现汉字处理的更多相关文章
- 黑马程序员——C语言基础 char字符 数组
Java培训.Android培训.iOS培训..Net培训.期待与您交流! (以下内容是对黑马苹果入学视频的个人知识点总结) (一)char类型 1)存储细节 ASCII单字节表(双字节GBK\GB2 ...
- Java 语言中一个字符占几个字节?
Java中理论说是一个字符(汉字 字母)占用两个字节. 但是在UTF-8的时候 new String("字").getBytes().length 返回的是3 表示3个字节 作者: ...
- 【转载】C#怎么判断字符是不是汉字
支持并尊重原创!原文地址:http://jingyan.baidu.com/article/2c8c281deb79ed0008252af1.html 判断一个字符是不是汉字通常有三种方法,第1种用 ...
- Atian inputmethod 输入法解决方案 方言与多语言多文字支持 英语汉字汉语阿拉伯文的支持 (au
Atian inputmethod 输入法解决方案 方言与多语言多文字支持 英语汉字汉语阿拉伯文的支持 (au 1.1. Overview概论 支持母语优先的战略性产品,主要是针对不想以及不愿使用普通 ...
- 根据Unicode编码用C#语言把它转换成汉字的代码
rt 根据所具有的Unicode编码用C#语言把它转换成汉字的代码 var s = System.Web.HttpUtility.HtmlDecode(Utf8Str); var o = Newton ...
- Swift3.0语言教程获取字符
Swift3.0语言教程获取字符 Swift3.0语言教程获取字符,在字符串中获取某一下标位置(下标索引)处的字符是很常见的功能,在NSString中使用character(at:)方法实现,其语法形 ...
- 如何利用java把文件中的Unicode字符转换为汉字
有些文件中存在Unicode字符和非Unicode字符,如何利用java快速的把文件中的Unicode字符转换为汉字而不影响文件中的其他字符呢, 我们知道虽然java 在控制台会把Unicode字符直 ...
- Regex 字符是不是汉字
Regex 字符是不是汉字 一. 判断一个字符是不是汉字通常有三种方法: 1.用ASCII码判断 在 ASCII码表中,英文的范围是0-127,而汉字则是大于127 string text = & ...
- C 语言实例 - 查找字符在字符串中出现的次数
C 语言实例 - 查找字符在字符串中出现的次数 C 语言实例 C 语言实例 查找字符在字符串中的起始位置(索引值从 开始). 实例 #include <stdio.h> int main( ...
随机推荐
- JDK环境配置: javac is not recognized as an internal or external command, operable program or batch file
相信大家在配置TestNG的时候,首先都会去确认JDK的安装是否正确,两个命令缺一不可. 打开'cmd' --> 1. 输入'java -version', 返回java home当前路径. j ...
- Scrapy框架安装与使用(基于windows系统)
"人生苦短,我用python".最近了解到一个很好的Spider框架--Scrapy,自己就按着官方文档装了一下,出了些问题,在这里记录一下,免得忘记. Scrapy的安装是基于T ...
- ffmpeg入门篇-滤镜的基本使用
转发自白狼栈:查看原文 滤镜 什么是滤镜?百度百科介绍说"滤镜主要是用来实现图像的各种特殊效果......". 我们最早在ffmpeg是如何转码的一文中了解过滤镜,来回顾下当时的转 ...
- [刘阳Java]_ResourceBundleViewResolver视图解析器_第7讲
ResourceBundleViewResolver是根据proterties文件来找对应的视图来解析"逻辑视图".该properties文件默认是放在classpath路径下的v ...
- SpringBoot 构造器注入、Setter方法注入和Field注入对比
0. 引入 今天在看项目代码的时候发现在依赖注入的时候使用了构造器注入,之前使用过 Field 注入和 Setter 方法注入,对构造器注入不是很了解.经过查阅资料看到,Spring 推荐使用构造器注 ...
- unar命令解压zip文件,解决中文乱码。
unzip解压时,常出现中文乱码.可用unar来代替.
- OpenFaaS实战之一:部署
欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...
- [HNOI2011]XOR和路径 题解
设 \(f(x)\) 表示从 \(x\) 节点走到 \(n\) 的期望.有 $$f(x)=\sum_{{x,y}}\frac{f(y)\oplus w(x,y)}{{\rm deg}(x)}$$ 由于 ...
- shell编程-ssh免交互批量分发公钥脚本
脚本基本原理 1.控制端免交互创建秘钥和公钥: 1 ssh-keygen -t rsa -f /root/.ssh/id_rsa -N "" 2.免交互发送公钥 1 sshpass ...
- 搭建NFS服务
说明:这里是Linux服务综合搭建文章的一部分,本文可以作为单独搭建yum仓库的参考. 注意:这里所有的标题都是根据主要的文章(Linux基础服务搭建综合)的顺序来做的. 如果需要查看相关软件版本和主 ...