11深入理解C指针之---指针和常量
该系列文章源于《深入理解C指针》的阅读与理解,由于本人的见识和知识的欠缺可能有误,还望大家批评指教。
指针作为C语言的左膀右臂,使用方便,修改容易,引用数据快速都是很有前景的应用。C语言中常量是值不能改变的量,由于在某些应用中,想要阻止没有授权的数据的修改时,就需要将指针和常量结合起来。
先看以下代码:
#include <stdio.h> int main(int argc, char **argv)
{
int iVar1 = ;
int *ptrVar1 = &iVar1;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1); iVar1++;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1); (*ptrVar1)++;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1); return ;
}
代码结果:
iVar1 value: and iVar1 address: 0x5b483c1c
*ptrVar1 value: ptrVar1 value: 0x5b483c1c and ptrVar1 address: 0x5b483c10
iVar1 value: and iVar1 address: 0x5b483c1c
*ptrVar1 value: ptrVar1 value: 0x5b483c1c and ptrVar1 address: 0x5b483c10
iVar1 value: and iVar1 address: 0x5b483c1c
*ptrVar1 value: ptrVar1 value: 0x5b483c1c and ptrVar1 address: 0x5b483c10
通过以上代码不难发现,将指针指向变量后,无论通过指针或变量都可以轻松修改变量。有时只是想利用指针的便利,而又不想承担指针可以修改变量的风险是,与const结合无疑是最好的。
一、const修饰指针定义:通过指针引用的方式修改变量将会被禁止
1、特征:
1)、提供一种特殊的方式来操纵数据
2)、指针的值可以改变
3)、不能通过指针改变指向变量的值
2、两种形式,两种形式完全一样的:
1)、int const *ptrVar1;
2)、const int *ptrVar1;
将代码1的第6行代码改为如下:
int const *ptrVar1 = &iVar1;
代码编译错误如下:
test11_01.c::: 错误:令只读位置‘*ptrVar1’自增
(*ptrVar1)++;
^~
只有将第14行如下代码注释掉,才能通过编译:
(*ptrVar1)++;
再看如下代码段:
#include <stdio.h> int main(int argc, char **argv)
{
int iVar1 = ;
int iVar2 = ;
int const *ptrVar1 = &iVar1;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1); iVar1++;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1); ptrVar1 = &iVar2;
printf("iVar2 value: %d and iVar2 address: %#x\n", iVar2, &iVar2);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1); return ;
}
结果如下:
iVar1 value: and iVar1 address: 0xf15c983c
*ptrVar1 value: ptrVar1 value: 0xf15c983c and ptrVar1 address: 0xf15c9830
iVar1 value: and iVar1 address: 0xf15c983c
*ptrVar1 value: ptrVar1 value: 0xf15c983c and ptrVar1 address: 0xf15c9830
iVar2 value: and iVar2 address: 0xf15c9838
*ptrVar1 value: ptrVar1 value: 0xf15c9838 and ptrVar1 address: 0xf15c9830
根据定义,容易知道由于const修饰的是*ptrVar1,是指针的引用,因此无法通过指针引用修改变量的值,但是依旧可以通过变量赋值修改变量,而且指针变量可以指向其他变量如iVar2等。
二、const修饰指针定义:可以通过指针引用的方式修改变量,禁止指针指向其他变量
1、特征:
1)、提供一种特殊的方式来操纵数据
2)、指针的值不可以改变
3)、通过指针改变指向变量的值
2、应用形式:
1)、int * const ptrVar1;
将代码2的第6行代码改为如下:
6 int * const ptrVar1 = &iVar1;
代码编译错误如下:
test11_01.c:15:10: 错误:向只读变量‘ptrVar1’赋值
ptrVar1 = &iVar2;
只有将第15行如下代码注释掉或改为(*ptrVar1)++,才能通过编译:
ptrVar1 = &iVar2;
请看如下具体代码段:
#include <stdio.h> int main(int argc, char **argv)
{
int iVar1 = ;
int *const ptrVar1 = &iVar1;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1); iVar1++;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1); //ptrVar1 = &iVar2;
(*ptrVar1)++;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1); return ;
}
根据定义,容易知道由于const修饰的是ptrVar1,是指针变量自身,指针的值不能改变。因此可以通过指针引用修改变量的值,也可以通过变量赋值修改变量,但是指针变量不可以指向其他变量如iVar2等。
三、const修饰指针定义,同时修改指针变量:通过指针引用的方式修改变量将会被禁止,指针也不能指向其他变量
1、特征:
1)、提供一种特殊的方式来操纵数据
2)、指针的值不可以改变
3)、不可通过指针改变指向变量的值
2、应用形式:
1)、int const * const ptrVar1;
代码如下:
#include <stdio.h>
int main(int argc, char **argv)
{
int iVar1 = ;
int iVar2 = ;
int const *const ptrVar1 = &iVar1;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1);
iVar1++;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1);
>> ptrVar1 = &iVar2;
//(*ptrVar1)++;
printf("iVar1 value: %d and iVar1 address: %#x\n", iVar1, &iVar1);
printf("*ptrVar1 value: %d ptrVar1 value: %#x and ptrVar1 address: %#x\n", *ptrVar1, ptrVar1, &ptrVar1);
return ;
}
保留第15行代码,编译错误如下:
test11_01.c::: 错误:向只读变量‘ptrVar1’赋值
ptrVar1 = &iVar2;
^
保留第16行代码,编译错误如下:
test11_01.c::: 错误:令只读位置‘*ptrVar1’自增
(*ptrVar1)++;
^~
由此可见,通过两个const的修饰,使得无论是指针变量的值,还是指针变量的解引用值都无法改变,可以有效提高数据的安全性。
11深入理解C指针之---指针和常量的更多相关文章
- 00深入理解C指针之--- 指针之外
该系列文章源于<深入理解C指针>的阅读与理解,由于本人的见识和知识的欠缺可能有误,还望大家批评指教. C语言从诞生之初就非常善于和硬件打交道,经过这么多年的发展之后,其灵活性和超强的特征是 ...
- 10深入理解C指针之---指针运算和比较
该系列文章源于<深入理解C指针>的阅读与理解,由于本人的见识和知识的欠缺可能有误,还望大家批评指教. 指针运算有很多种,主要有指针的声明*.指针的解引*.指针指向(*).或->.指针 ...
- 从四个属性的角度来理解C语言的指针也许会更好理解
文章会在文末更新! 关于指针是什么,很多教材已经作出了定义,大多数都会定义为"存放变量内存地址的变量".从这句话中我觉得除了让我知道这个定义有11个字以外,其他就没什么用了.我个人 ...
- 02深入理解C指针之---指针类型和值
该系列文章源于<深入理解C指针>的阅读与理解,由于本人的见识和知识的欠缺可能有误,还望大家批评指教. 1.指针的类型: 可以在声明指针时,指定指针的类型,例如: (1)void *x 声 ...
- 06深入理解C指针之---指针操作和比较
该系列文章源于<深入理解C指针>的阅读与理解,由于本人的见识和知识的欠缺可能有误,还望大家批评指教. 指针作为一种特殊类型的变量,必须遵守C语言中变量先声明后使用的原则.本节内容中指针的操 ...
- 理解C语言中指针的声明以及复杂声明的语法
昨天刚把<C程序设计语言>中"指针与数组"章节读完,最终把心中的疑惑彻底解开了.如今记录下我对指针声明的理解.顺便说下怎样在C语言中创建复杂声明以及读懂复杂声明. 本文 ...
- C语言指针的理解以及指针的指针的理解
指针指向的是内存地址编号,内存地址编号指向的是对应的内容. 我们需要一个变量,来储存内存地址编号,这个变量的值是一个内存地址编号,但是我们可以通过修改变量的值,来不断的改变内存地址编号. 但是,我们如 ...
- 这样子来理解C语言中指针的指针
友情提示:阅读本文前,请先参考我的之前的文章<从四个属性的角度来理解C语言的指针也许会更好理解>,若已阅读,请继续往下看. 我从4个属性的角度来总结了C语言中的指针概念.对于C语言的一个指 ...
- 理解git 中的HEAD指针&branch指针
理解git 中的HEAD指针&branch指针 Yooye关注 2019.02.28 10:44:32字数 492阅读 668 HEAD指针 使用git checkout 来移动HEAD指针, ...
- 深入理解C语言-函数指针
函数指针在C++中有着重要的应用,函数的函数名其本质就是代表一个地址,这个地址叫做函数入口,得到这个地址就可以对这个函数进行各种操作. 函数类型基础 函数三要素: 名称.参数.返回值 C语言中的函数有 ...
随机推荐
- c#—OpenFileDialog
OpenFileDialog是什么? OpenFileDialog是一个类,实例化此类可以设置弹出一个文件对话框.比如:我们发邮件时需要上传附件的时候,就会弹出一个让我们选择文件的对话框, 我们可以根 ...
- 【NOIP提高A组模拟2018.8.14】 区间
区间加:差分数组修改 O(n)扫描,负数位置单调不减 #include<iostream> #include<cstring> #include<cstdio> # ...
- Struts2和SpringMVC简单配置以及区别总结
Struts2: struts 2 是一个基于MVC(mode-view-con)设计模式的Web应用框架,是由Struts1和WebWork两个经典框架发展而来的. 工作流程: 1客户端浏览器发出H ...
- Linux基础学习-NFS网络文件系统实时文件共享
NFS网络文件系统 如果大家觉得Samba服务程序的配置太麻烦了,那么你共享文件的主机都是Linux系统,那么推荐大家在客户端部署nfs服务来共享文件.nfs(网络文件系统)服务可以将远程Linux系 ...
- GoogleTest 之路3-Mocking Framework
当你写一个原型或者测试的时候,依赖整个object 是不可行和明智的.一个 mock object和 real object 有同样的接口(所以它可以像同一个使用),但是让你在运行时进行指定它应该如何 ...
- 《linux设备驱动开发详解》笔记——8阻塞与非阻塞IO
8.1 阻塞与非阻塞IO 8.1.0 概述 阻塞:访问设备时,若不能获取资源,则进程挂起,进入睡眠状态:也就是进入等待队列 非阻塞:不能获取资源时,不睡眠,要么退出.要么一直查询:直接退出且无资源时, ...
- IOC容器和Bean的配置实例
实验1: <!--实验1:通过IOC容器创建对象,并为属性赋值 --> <!-- 需要由IOC容器创建对象的全类名 --> <!-- 为了便于从IOC容器中获取book对 ...
- 天气API接口的使用
最近项目中使用到了天气预报的功能,需要从网上获取天气数据,然后显示在公司系统的页面上. 在这里和大家分享下我的做法,希望能对大家有所帮助,如果有错误,欢迎大家指正. 先给大家看看效果: 下面开始进行讲 ...
- build.xml: 21: Class not found: javac1.8
在eclipse里运用ant时经常碰到class not found的错误提示,从而编译失败,其实是eclipse中本身的ant版本太老造成该的,但我今天安装的ant是1.8.4,感觉已经很新了,但编 ...
- dubbo与zk注册中心如何对接,如何做到服务自动发现
先看下consumer端发起调用时的链路流程: +---------------------------+ +---------------------------+ +--------------- ...