指针和函数

标签:c++


一、基本概念

定义:

也称作“指针变量”,大小为4个字节,内容代表一个内存地址

指针的定义及使用:

  • 定义:

    int *p;//指向的变量类型为整型
    char *pc;
  • 赋值:

    int *p = (int *) 4000;//强制类型转换,让p = 4000
    char ch1 = ‘A’;
    char *pc = &ch1;
    *pc = ‘B’;
    char ch2 = *pc;
    pc = &ch2;

    注:指针定义后就要赋值,防止变成野指针

二、指针的相互赋值

不同类型的指针相互赋值要强制类型转换

int *pn;
char *pc;
char c = 0x65;//e
pn = pc;//编译出错,类型不匹配
pn = &c;//编译出错
pn = (int *) &c;//可以,但是&c只代表一个字节,pn代表四个字节,此时pn会占用&c后面的三个字节,导致错误
int n = *pn;//n中的内容不是0x65

因此,不要用不同类型的指针相互赋值

三、指针的运算

比较大小:

两个同类型的指针p1, p2,如果地址p1<地址p2,则p1<p2,其余同理

相减:

p1 – p2 = (地址1-地址2)/sizeof(T)

T为指针的类型

加减整数类型变量或者常量:

p + n 等价于:地址p + n * sizeof( T )

p - n 等价于:地址p - n * sizeof( T )

自增自减:

同3

下标运算符[ ]:

p[ n ]等价于 *( p + n )

四、空指针

空指针NULL,即为地址0

可以做条件表达式使用

五、指针作为函数参数

  • 传递指针的方式依然是指传递,注意函数是在对指针操作还是对指针所指的内容操作,即函数只能改变指针指向地址中的内容,不能改变指针本身

    #include<iostream>
    using namespace std;
    void Swap(int *p1, int *p2)
    {
    int temp = *p1;
    *p1 = *p2;
    *p2 = temp;
    }
    int main()
    {
    int m = 4, n = 3;
    Swap(&m, &n);
    cout<<m<<','<<n;
    }
    结果:3,4
  • 使用指针作为函数参数时,应先判断指针是否为空,避免野指针对函数造成影响

  • 返回一个指针时,也要注意返回非空指针

六、指针和数组

数组的名字即为一个指针,作为函数形参时,T *pT p[ ]完全等价

七、常量指针和指针常量

常量指针

const T* p

特点:不能通过常量指针修改指向的内容,即内容可以修改,但是要用其他方法

指针常量

T *const p = p0

特点:定义之后指针p就不能指向其他位置,但是可以通过指针对其指向内容修改

八、字符串和指针

普通字符串和指针的关系

字符串常量的类型就是char *,字符数组的类型也是char *

char name[20];
char *pName = name;

string对象和char *指针的关系

  • 可以用char *类型指针对string对象赋值

  • string对象有成员函数:const char *c_str(),返回的是字符串指针

  • 如果s是一个string对象,对其执行 const char *p = s.c_str(),当s的内容改变时,p可能会失效,因为s的位置可能发生改变

    #include<iostream>
    #include<cstring>
    #include<string>
    using namespace std;
    int main()
    {
    string s;
    char str1[20] = "the flower of war";
    char str2[20] = "hhh";
    char *p = str1;
    s = p;
    cout<<"s:"<<s<<endl;
    strcat(str2, s.c_str());
    cout<<"str2:"<<str2<<endl;
    return 0;
    }

常用字符串操作库函数

cstring中声明:

char *strcat(char *dest, const char *src),返回dest

char *strchr(const char *src, int c),寻找csrc中第一次出现的位置,找到则返回位置指针,否则返回NULL

int strcmp(const char *s1, const char *s2),比较大小

char *strcpy(char *dest, const char *src),复制

int strlen(const char *src),求长度

char *strlwr(char *str),字母转小写

char *strupr(char *str),字母转大写

char *strncat(char *dest, const char *src, int n),连接前n个字符到dest

类似有:strncpy, strncmp

cstdlib中声明:

atoi:字符串转整数

atof:字符串转浮点数

itoa:整数转字符串,char *itoa(int value, char *string, int radix)

九、 void指针和内存操作库函数

定义及特点:

指针的类型为void *

  • 可以用任何类型的指针对其初始化
  • 主要用于内存复制
  • sizeof()及各种指针运算不适用于空指针

库函数

cstring中声明

void *memset(void *dest, int ch, int n),将ch复制到destn个字节中

void *memcpy(void *dest, void *src, int n),将src的开始n个字节复制到dest

示例

#include<iostream>
using namespace std;
void *myMemcpy(void *dest, const void *src, int n)
{
char *pDest = (char *)dest;//必须先将形参进行强制类型转换
char *pSrc = (char *)src;
for(int i = 0; i < n; ++i)
*(pDest + i) = *(pSrc + i); return dest;
}
int main()
{
char szName[10] = "abcdsedf";
char destName[10];
myMemcpy(destName, szName,5);
cout<<szName<<endl;
for(int i = 0; i < 5; ++i)
cout<<destName[i];
}

十、 函数指针

定义:

类型名 (*指针变量名)(参数类型1,参数类型2,……)

赋值及调用:

  • 可以用一个原型匹配的函数的名字给一个函数指针赋值

  • 调用格式:函数指针名(实参表)

    #include<iostream>
    using namespace std;
    void PrintMin(int a, int b)
    {
    if(a < b)
    cout<<a;
    else
    cout<<b;
    }
    int main()
    {
    int x = 4, y = 1;
    void (*pf)(int ,int) = PrintMin;
    pf(x, y);
    return 0;
    }

应用

qsort()函数

原型:void qsort(void *base, int nelem, unsigned int width, int (*pfCompare)(const void *, const void *))

其中比较函数的原型:int 函数名(const void *elem1, const void *elem2)

注意:

  • 返回值必须是int
  • 两个参数的类型必须都是const void *
  • 假设是对int排序的话,如果是升序,那么就是如果a比b大返回一个正值,小则负值,相等返回0,其他的依次类推
#include<iostream>
#include <cstdlib>
using namespace std; //比较函数可以根据自己的需求写,其作用相当但不限于
//使用sort()时的运算符重载
int myCompare(const void *elem1, const void *elem2)
{
unsigned int *p1, *p2;
p1 = (unsigned int *)elem1;
p2 = (unsigned int *)elem2;
return (*p1 % 10) - (*p2 % 10);
}
const int NUM = 5;
int main()
{
unsigned int a[NUM] = {8,123,11,10,4};
qsort(a, NUM, sizeof(unsigned int), myCompare);
for(int i = 0; i < NUM; ++i)
cout<<a[i]<<endl;
return 0;
}

十一、 指针域动态内存分配

使用方法:

指针类型 *p = new 指针类型(int *p = new int)

指针类型 *p = new 指针类型[空间大小](int *p = new int[N]),分配到N*sizeof(T)的空间

注意

  • 分配空间失败时,系统会抛出异常
  • 动态分配的内存使用结束后要释放,方法为:delete 指针delete [ ] 指针
  • 释放一个指针后,不会使指针的值变为NULL

十二、 二阶指针

指向指针的指针,大小永远为4字节,对应多维数组

十三、指针数组

定义:

T *a[N]

数组中每个元素的类型是T *a的类型是T **

示例:

#include<iostream>
using namespace std;
int main()
{
char **p = NULL;
char *countries[] = {"China","USA","Japan","France"};
p = countries;
for(int i = 0; i < 4; ++i)
cout<<*(p + i)<<endl;
cout<<*((*p) + 1);
}

【C++】指针和函数的更多相关文章

  1. 12-返回指针的函数&&指向函数的指针

    前言 接下来我只讲指针的最常见用法,比如这一章的内容----返回指针的函数 与 指向函数的指针   一.返回指针的函数 指针也是C语言中的一种数据类型,因此一个函数的返回值肯定可以是指针类型的. 返回 ...

  2. 我的c++学习(12)指针作为函数参数

    ◆ 引用调用与指针传值调用C++中函数的参数的基本使用方法是传值.为了弥补单纯传值的不足,以引用作为函数的参数,从逻辑上讲引用是别名,在函数中对参数的操作,就是对实参的操作,而在物理上是传实参的地址. ...

  3. C#委托与C语言函数指针及函数指针数组

    C#委托与C语言函数指针及函数指针数组 在使用C#时总会为委托而感到疑惑,但现在总新温习了一遍C语言后,才真正理解的委托. 其实委托就类似于C/C++里的函数指针,在函数传参时传递的是函数指针,在调用 ...

  4. 【C语言】14-返回指针的函数与指向函数的指针

    前言 前面我们花了接近3个章节学习指针,应该都感受到指针的强大了吧.指针可以根据地址直接操作内存中的数据,使用得当的话,不仅能使代码量变少,还能优化内存管理.提升程序性能.关于指针的内容还非常多,比如 ...

  5. C/C++ 一段代码区分数组指针|指针数组|函数指针|函数指针数组

    #include<stdio.h> #include<stdlib.h> #include<windows.h> /* 举列子说明什么是函数指针 */ //以一个加 ...

  6. 【学习笔记】【C语言】返回指针的函数

    函数如果带*的返回的就是指针 char *test(){ } #include <stdio.h> char *test(); /* 返回指针的函数 */ int main() { cha ...

  7. C++ 二维数组(双重指针作为函数参数)

    本文的学习内容参考:http://blog.csdn.net/yunyun1886358/article/details/5659851 http://blog.csdn.net/xudongdong ...

  8. Delphi 函数指针(函数可以当参数)

    首先学习: 指向非对象(一般的)函数/过程的函数指针 Pascal 中的过程类型与C语言中的函数指针相似,为了统一说法,以下称函数指针.函数指针的声明只需要参数列表:如果是函数,再加个返回值.例如声明 ...

  9. C语言中的声明解析规则——数组,指针与函数

    摘要:C语言的申明存在的最大问题是:你无法以一种人们所习惯的自然方式和从左向右阅读一个声明,在引入voliatile和const关键字以后,情况更加糟糕了.由于这些关键字只能出现在声明中,是的声明形式 ...

  10. C语言的本质(12)——指针与函数

    往往,我们一提到指针函数和函数指针的时候,就有很多人弄不懂.下面详细为大家介绍C语言中指针函数和函数指针. 1.指针函数 当一个函数声明其返回值为一个指针时,实际上就是返回一个地址给调用函数,以用于需 ...

随机推荐

  1. java 网络编程基础 TCP/IP协议:服务端ServerSocket;客户端Socket; 采用多线程方式处理网络请求

    1.Java中客户端和服务器端通信的简单实例 Java中能接收其他通信实体连接请求的类是ServerSocket,ServerSocket对象用于监听来自客户端的Socket连接,如果没有连接,它将一 ...

  2. 以太网/ IPV4/IPV6包头,TCP包头格式回顾

    问题:以太网数据包,承载的数据内容大小46~1500字节,是如何来的? 以太网数据包结构  以太网协议规定最小链路层数据包(帧)为64字节,其中以太网首部+尾部共计18字节(源/目的MAC12字节:上 ...

  3. 【LeetCode】1150. Check If a Number Is Majority Element in a Sorted Array 解题报告(C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 字典 二分查找 日期 题目地址:https://lee ...

  4. 【LeetCode】316. Remove Duplicate Letters 解题报告(Python & C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 题目地址:https://leetcode.c ...

  5. LeetCode1238循环码排列

    题目 给你两个整数 n 和 start.你的任务是返回任意 (0,1,2,,...,2n-1) 的排列 p,并且满足: p[0] = start p[i] 和 p[i+1] 的二进制表示形式只有一位不 ...

  6. Abelian Period

    Abelian Period Accepts: 288 Submissions: 984 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 26 ...

  7. 「实用」打造自我感觉非常漂亮的Mac终端

    背景 (今天我是一个美妆博主) 突然发现自己使用的iterm2终端样式有些朴素,为了让她看起来花枝招展的,我决定给她打扮打扮.毕竟每天面对她的时间比对象还多-- 效果对比 因为每个人的喜好都不一样,所 ...

  8. 自我学习与理解:keras框架下的深度学习(三)回归问题

    本文主要是使用keras对其有的波士顿房价数据集做一个回归预测,其代码架构与之前一样(都只是使用多层感知机):数据的预处理.搭建网络框架.编译.循环训练以及测试训练的网络模型.其中除了数据预处理与之前 ...

  9. 使用pynput同时监听鼠标和键盘

    pynput概述 pynput是一个基于python的,能够监听和控制鼠标和键盘的第三方库. pynput主要包括两个类,pynput.mouse和pynput.keyboard,顾名思义,前者可以用 ...

  10. Java初学者作业——使用switch结构实现一个简单的购物计划

    返回本章节 返回作业目录 需求说明: 使用switch结构实现一个购物计划,计划为:星期一.星期三.星期五购买伊利牛奶和面包,星期二.星期四购买苹果和香蕉,星期六.星期日购买啤酒和周黑鸭. 实现思路: ...