char *a = "Peter";
char b[] = "Peter";
char *c = new char[];
strcpy_s(c, , "Peter");

这里a指向常量区

b指向栈区

c指向堆区

如果我们有这样一个函数

void show(char *temp)
{
//
//
//
}

我们如何判断根据过来的temp的将这些区分出来呢?

1.我们可以首先将指向常量区的a区分出来

因为它所指向的地方是不可以修改的

判断是否可以修改

a.使用函数IsBadReadPtr

函数原型如下

BOOL WINAPI IsBadWritePtr(
_In_  LPVOID lp,
_In_  UINT_PTR ucb
);

lp:第一个字节的内存块的指针。

ucb:指定的大小,单位为字节的内存块。如果此参数为零,则返回值为零。

MSDN上部分的解释如下

http://msdn.microsoft.com/en-us/library/windows/desktop/aa366716(v=vs.85).aspx

Verifies that the calling process has write access to the specified range of memory.

Important  This function is obsolete and should not be used. Despite its name, it does not guarantee that the pointer is valid or that the memory pointed to is safe to use. For more information, see Remarks on this page.

翻译(自己翻译的):

验证调用的进程是否可以写入指定范围的内存

主要的:这个函数是废弃的,不应该被使用,不管它的名字,它不保证这个指向内存中的指针是合法的和这段内存是可以安全使用的,对于更多的信息,可以查看这页的评论

示例:

bool  isConst(void* pAddress, DWORD dwSize)
{
if (IsBadWritePtr(pAddress, dwSize))
return true;
return false;
} int main()
{
char *a = "Peter";
char b[] = "Peter";
char *c = new char[];
strcpy_s(c, , "Peter");
cout << isConst(a,strlen(a))<< endl;
cout << isConst(b, strlen(a)) << endl;
cout << isConst(c, strlen(a)) << endl; system("pause");
}

结果

b.使用VirtualQuery

函数原型如下

SIZE_T WINAPI VirtualQuery(
_In_opt_  LPCVOID lpAddress,
_Out_     PMEMORY_BASIC_INFORMATION lpBuffer,
_In_      SIZE_T dwLength
);

lpAddress:查询内存的地址。

lpBuffer:指向MEMORY_BASIC_INFORMATION结构的指针,用于接收内存信息。

dwLength:MEMORY_BASIC_INFORMATION结构的大小。

MSDN上部分的解释如下

http://msdn.microsoft.com/en-us/library/windows/desktop/aa366902(v=vs.85).aspx

Retrieves information about a range of pages in the virtual address space of the calling process.

检索对于调用进程的虚拟内存中的页的信息

用于接收的内存信息的MEMORY_BASIC_INFORMATION结构体定义如下

typedef struct _MEMORY_BASIC_INFORMATION {
PVOID  BaseAddress;
PVOID  AllocationBase;
DWORD  AllocationProtect;
SIZE_T RegionSize;
DWORD  State;
DWORD  Protect;
DWORD  Type;
} MEMORY_BASIC_INFORMATION, *PMEMORY_BASIC_INFORMATION;

BaseAddress:保留区域的基地址
AllocationBase:分配的基地址

AllocationProtect:初次保留时所设置的保护属性
RegionSize:区域大小
State:状态(提交、保留或空闲)
Protect: 当前访问保护属性
Type:页面类型

详细请见MSDN

http://msdn.microsoft.com/en-us/library/windows/desktop/aa366775(v=vs.85).aspx
 
示例
bool  isConst(void* pAddress)
{
_MEMORY_BASIC_INFORMATION mi = { };
VirtualQuery(pAddress, &mi, sizeof(mi));
if (mi.Protect == PAGE_READONLY)
{
return true;
}
return false;
} int main()
{
char *a = "Peter";
char b[] = "Peter";
char *c = new char[];
strcpy_s(c, , "Peter");
cout << isConst(a)<< endl;
cout << isConst(b) << endl;
cout << isConst(c) << endl; system("pause");
}

结果

判断是否位于栈上

参考了http://www.cppblog.com/weiym/archive/2012/05/12/174634.html

我们可以在函数内建一个位于栈的对象,然后获得栈空间的初始地址,以及栈的最末尾的地址,就可以判断一个东西是不是为与栈上了

VirtualQuery中用于接收内存信息_MEMORY_BASIC_INFORMATION结构体中有如下成员

BaseAddress:保留区域的基地址
RegionSize:区域大小

代码示例

bool IsObjectOnStack(void* pObject)
{
int nStackValue(); MEMORY_BASIC_INFORMATION mi = { };
DWORD dwRet = VirtualQuery(&nStackValue, &mi, sizeof(mi));
if (dwRet > )
{
return pObject >= mi.BaseAddress
&& (DWORD)pObject < (DWORD)mi.BaseAddress + mi.RegionSize;
} return FALSE;
} int main()
{
char *a = "Peter";
char b[] = "Peter";
char *c = new char[];
strcpy_s(c, , "Peter");
cout << IsObjectOnStack(a) << endl;
cout << IsObjectOnStack(b) << endl;
cout << IsObjectOnStack(c) << endl; system("pause");
}

运行结果

解释:nStackValue是一个位于栈上的对象

我们针对它使用 VirtualQuery获得相关的内存信息mi

mi.BaseAddress是栈的初始地址

(DWORD)mi.BaseAddress + mi.RegionSize是栈的最末尾的地址

我们只要判断地址是不是在这二者之间,就可以判断是不是位于栈上了

C++判断char*的指向的更多相关文章

  1. Java 中判断char 是否为空格 和空

    //判断是否char是否为空import java.util.*; public class test{ public static void main(String[] args){ String ...

  2. JAVA中判断char是否是中文的几种方法

    1.方法一 char c = 'a'; if((c >= 0x4e00)&&(c <= 0x9fbb)) { System.out.println("是中文&qu ...

  3. 你好,C++(15)四两拨千斤——3.9 指向内存位置的指针

    3.9  指向内存位置的指针 一天,两个变量在街上遇到了: “老兄,你家住哪儿啊?改天找你玩儿去.” “哦,我家在静态存储区的0x0049A024号,你家呢?” “我家在动态存储区的0x0022FF0 ...

  4. JS 判断数据类型的三种方法

    说到数据类型,我们先理一下JavaScript中常见的几种数据类型: 基本类型:string,number,boolean 特殊类型:undefined,null 引用类型:Object,Functi ...

  5. C语言中 指向函数的指针 简介

    引子:在学习CPrimerPlus的第十四章的14.13节中,遇到了如下三行文字,是有关指向函数的指针的,把我搞晕了. char * fump(); //返回指向char的指针的函数 char (* ...

  6. 彻底搞定char/wchar_t/unicode

    彻底搞定char/wchar_t!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! (2013-07-17 10:18:28) 转载▼     从char/wchar_t到TCHAR(1) ...

  7. char类型输出地址

    问题描述: 当输出char的地址时,发现输出的是一个字符: char ch = 'a'; cout<<&ch<<endl;//a @ 因为cout得到一个char类型的 ...

  8. NDK(14)Native的char*和Java的String相互转换

    转自: http://www.cnblogs.com/canphp/archive/2012/11/13/2768937.html 首先确保C/C++源文件的字符编码是UTF-8与JAVA的class ...

  9. char指针

    1.在C语言中,没有字符串类型,因此使用char指针表示字符串. 2.那么问题来了,使用char* 表示字符串,到哪里是结尾呢?因此需要一个特殊的字符作为哨兵,类似迭代器中的end(),这个哨兵就是' ...

随机推荐

  1. 时间戳转为C#格式时间

    经常发现很多地方使用一个时间戳表示时间.比如: 1370838759 表示 2013年6月10日 12:32:39. 我们就需要一个工具,方便地转换这种时间格式 什么是时间戳? 时间戳, 又叫Unix ...

  2. quartz 任务高度的动态修改

    package com.example.demo.controller; import org.quartz.*;import org.quartz.impl.StdSchedulerFactory; ...

  3. android targetSdkVersion>=26收不到广播的处理

    背景:GP新政策,要求Google Player上架应用的targetSdkVersion>=26. 一. 为啥GP要求targetSdkVersion>=26? 1 targetSdkV ...

  4. Python学习过程笔记整理(二)

    程序三大结构 -顺序 -分支 -循环 分支 -语法: if 条件表达式: 语句 ... -双向分支 if 条件表达式: 语句 ... else: 语句 ... -多路分支 if 条件表达式: 语句 . ...

  5. InsertionSort 直接插入排序(java)

    排序思想: 相当于一堆数字,一开始先取出2个数排序,2个数排好序之后,再从一堆数字里面取一个数排序,直到结束伪代码: INSERTION_SORT(A) for j = 2 to A.length k ...

  6. xml解析数据信息并实现DBManager操作mysql

      先前一直都是用的直接用加载驱动 然后创建连接进行操作数据 如果我的数据库换了 那么要修改的地方也比较多 不利于维护 所以就想到了将所有配置连接信息都用xml封装起来 以至于我每次都只要修改一下我的 ...

  7. 32bit 天堂2 windows 2000 server架设教程

    服务器的软件要求: Microsoft Windows 2000 Server or advanced Server license 100 Microsoft SQL Server 2000 Ent ...

  8. 2018NOIP爆0记第一弹

    初赛篇 选择即王道 迪杰斯特拉那道题的A选项自己yy一下觉得甚是不妥,就没选 就和30分完美选择题擦肩而过. 填空最后一题不太会搞,就跳过了,最后蒙了个512上去...其实还有点接近的... 5分 然 ...

  9. 使用yum安装文件时提示安装文件重复问题2:nodejs-10.15.3-1nodesource.x86_64: [Errno 256] No more mirrors to try.

    原因:yum命令缓存问题 解决办法: sudo yum clean all

  10. Netty源码分析第1章(Netty启动流程)---->第5节: 绑定端口

    Netty源码分析第一章:Netty启动步骤 第五节:绑定端口 上一小节我们学习了channel注册在selector的步骤, 仅仅做了注册但并没有监听事件, 事件是如何监听的呢? 我们继续跟第一小节 ...