const char * 的终结贴(看完无需其他任何文章,从此不再蛋疼)
我之前也是以为我对const char *ptr 这种形式的写法是掌握了的,真的,不就是说一个指针是不可改变的吗?
那么问题就来了,到底是ptr指针本身不能改变,还是ptr执行的值不能改变呢?
从网上的资料上看,有如下的解释:
Bjarne在他的The C++ Programming Language里面给出过一个助记的方法: 把一个声明从右向左读。
char * const cp; ( * 读成 pointer to ) cp is a const pointer to char --->cp是一个指向字符char的固定指针
const char * ptr; ptr is a pointer to const char; --->ptr是一个指向固定字符char的指针
char const * p; --->无此形式
也就是说,cp和ptr都是指针,cp的值是不可改变的cp指向的内容是可变的;而ptr的值是可以改变的,ptr指向的内容是不可变的
| 指针的值是否可变 | 指向的值是否可变 | |
| cp | n | n |
| ptr | y | n |
下面写几个程序来验证之前所说的内容,并给出一个使用的小函数,期间的警告给出解决办法:
一.先来说const char *ptr
我写如下的程序:

预计写一个检测点分制ip是否合法的函数,我们先说明ptr,之后再改成要的形式:
现在这个函数要验证的就是对一个const char *形式的指针,指针本身的值是否可以改变,只做这一点,问题分析就要一点一点的来,不要有干扰.编译并执行:

这就说明了const char *形式的指针的地址是可以改变的,值的部分我下面验证
我尝试更改ptr指向的字符串的值,如下:

编译的时候出现的错误:

说明这个是从编译的时候就进行了安全性的检查,将错误消灭在萌芽之中,那么我们能否对值进行强行更改而骗过编译器呢?当然是可以的,我做如下的改动

那么虽然有一个警告,但是是可以生成可执行文件的:

无论test我生命为const char *还是 char *,都是提示一样的警告,运行的时候就产生了段错误.这就是说对编译产生的警告一定不要放过,不要以为能够生成可执行文件的程序就是正确的,事实上,只有那种unused的警告可以忽略,其他的都不要放过.
二.再来说char * const ptr
那么这个验证就相对简单了

编译的时候就直接出错,原因也说的很清楚,定指针不能改变自己的值,当然也不能自加;

那么我尝试修改指针指向的值:(这次使用VC调试下)

VC解释的很清除,就是说内存中的地址不能为write,也就是说ptr不仅仅是地址不能改变,连指向的值也是不能改变的.这下也基本明白了为什么那么多windows程序都会出现这个提示框了,对const理解的不深入啊,杯具~~,使用memcpy也是一致的,就不在截图了
三.好了,那么使用今天总结的知识来做一个使用的检测IP是否为点分值的小程序:
在socket编程中经常会用到点分制ip和unsigned long形式的ip,那么判断其有效性是提高程序健壮性的必要条件,下面只对点分制的进行判断,unsigned long形式的是系统函数生成的,inet_addr,所以无需我来担心.
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h> int check_ip(const char *buf)
{ int i;
int nLen;
const char *ip = buf;
char *tmp;
nLen = strlen(ip);
//first : check length
if(nLen > 16 || nLen < 7)
{
printf("the string is too long(%s)/n", ip);
return -1;
}
//second : check whether a string only has number and '.'
for (i = 0; i < nLen; i++)
{
printf("11/n");
if (ip[i] != '.' && isdigit(ip[i]) == 0) //it is not '.' or number
{
printf("here is not right(%c)(%d)/n", ip[i], ip[i]);
return -1;
}
} printf("number is ok (%s)/n", ip); i = 0;
//third, check . do
{
tmp = strchr(ip, '.');
printf("current(%s)/n", tmp); if(tmp != NULL)
{
tmp++;
if(*tmp == '.')
{
printf("format error/n");
return -1;
}
i++;
}
ip = tmp;
//sleep(1);
}while(tmp != NULL); printf("%d/n", i);
if(i == 3)
{
printf("it's a string type ip/n");
return 0;
}
else
{
printf("it's not a string type ip(%s)/n", ip);
return -1;
} } int main(int argc, char *argv[])
{
int nRet; if(argc < 2)
printf("enter a cmd like : ./test_ip 192.168.1.1"); printf("(%s)/n", argv[1]);
nRet = check_ip(argv[1]);
printf(">>>>>>>(%d)/n", nRet); return 0;
} 点分制的几个安全检测如下: 1.长度是在[7,16] 2.都是有数字和'.'组成 2.不能用连续的二个'.' 3.只能有3个'.'
满足这4个条件的字符串就可以认为是点分制的IP地址.
reference:http://blog.csdn.net/scarlettsp/article/details/5949415
const char * 的终结贴(看完无需其他任何文章,从此不再蛋疼)的更多相关文章
- 【图解】你还在为 TCP 重传、滑动窗口、流量控制、拥塞控制发愁吗?看完图解就不愁了
每日一句英语学习,每天进步一点点: 前言 前一篇「硬不硬你说了算!近 40 张图解被问千百遍的 TCP 三次握手和四次挥手面试题」得到了很多读者的认可,在此特别感谢你们的认可,大家都暖暖的. 来了,今 ...
- 盘点国内程序员不常用的热门iOS第三方库:看完,还敢自称”精通iOS开发”吗?【转载】
综合github上各个项目的关注度与具体使用情况,涵盖功能,UI,数据库,自动化测试,编程工具等类型,看完,还敢自称”精通iOS开发”吗? https://github.com/syedhali/EZ ...
- 【转】深入理解const char*p,char const*p,char *const p,const char **p,char const**p,char *const*p,char**const p
一.可能的组合: (1)const char*p (2)char const*p (3)char *const p(4)const char **p (5)char const**p (6)char ...
- c语言检测文件是否存在int __cdecl access(const char *, int);
最近写代码,遇到很多地方需要判断文件是否存在的.网上的方法也是千奇百怪,“百家争鸣”. fopen方式打开的比较多见,也有其他各种方式判断文件是否存在的,由于其他方法与本文无关,所以不打算提及. 笔者 ...
- C语言中strcpy(char *strDest, const char *strScr)字符串复制库函数的理解与分析
1.原版的strcpy()函数原型 char * strcpy( char *strDest, const char *strSrc ) { assert( (strDest != NULL) &am ...
- C语言中为什么不能把char**赋给const char**
这是我在知乎回答的一个问题. 这个问题是C中的一个深坑,首先说结论: char ** 和 const char ** 是两个不相容(incompatible)的类型,能够理解为不能直接赋值 在C11的 ...
- QStringLiteral(源代码里有一个通过构造函数产生的从const char*到QString的隐式转换,QStringLiteral字符串可以放在代码的任何地方,编译期直接生成utf16字符串,速度很快,体积变大)
原作者: Olivier Goffart 点击打开链接http://woboq.com/blog/qstringliteral.html 译者: zzjin 点击打开链接http://www.tuic ...
- c++ const char *[] or char [][]
]={"hello","world"}; ]={"hello"};char (*ch3)[6]= ch1; std::cout<< ...
- const char *p;和char * const p的区别
const char *p; const修饰*p,所以*p是一个常量,不可修改. char* const p; const修饰p,所以指针p是一个常量,不可修改. #include< ...
随机推荐
- js 如何判断数据是数据还是对象
如果用typeof测试,数组和对象都是显示的Object, 测试方式:var mycars=new Array();mycars[0]="Saab";mycars[1]=" ...
- java反射知识
java反射机制是在运行状态中,对于任意一个类(class文件),都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称 ...
- javaweb 中的乱码问题
一.post 方式 首先我们看下面一段代码,在该HTML中我们指定的编码为“UTF-8”,如图所示. 在该代码中,我们将表单数据提交给ParamServlet 处理 servlet 会将接收到的数据打 ...
- SQL Server自动化运维系列 - 监控磁盘剩余空间及SQL Server错误日志(Power Shell)
需求描述 在我们的生产环境中,大部分情况下需要有自己的运维体制,包括自己健康状态的检测等.如果发生异常,需要提前预警的,通知形式一般为发邮件告知. 在所有的自检流程中最基础的一个就是磁盘剩余空间检测. ...
- 学习笔记_Java_day12_设计模式MVC(13).JavaWeb的三层框架(14)
MVC 1. 什么是MVC MVC模式(Model-View-Controller)是软件工程中的一种软件架构模式,把软件系统分为三个基本部分:模型(Model).视图(View)和控制器(Contr ...
- OC - 12.NSURLRequest与NSURLConnection
##NSURLRequest NSURLRequest封装了一次网络请求所需要的数据,主要封装了以下信息: 请求路径(URL) 请求方法(GET或POST) 请求头 请求体 超时参数 NSURLReq ...
- autorelease 的基本使用
5-autorelease 的基本使用 0,引入 Person *p = [Persom new];[p release]; [p run]; [p run]; // 希望不立即释放,待 run执行完 ...
- phaser源码解析(一) Phaser.Utils类下shuffle方法
/** * #一个 基于 费雪耶茨排列 洗牌方法 * A standard Fisher-Yates Array shuffle implementation. * @method Phaser.Ut ...
- Mybatis 学习历程
MyBatis是一个支持普通SQL查询,存储过程和高级映射的优秀持久层框架. MyBatis消除了几乎所有的JDBC代码和参数的手工设置以及对结果集的检索封装. MyBatis可以使用简单的XML或注 ...
- struts 文件上传下载
上传 1.编写上传action类 UploadAction.java package jxf.b_upload; import java.io.File; import java.io.IOExcep ...