Fork me on GitHub

c++之指针

一、指针的基本概念

指针的作用:可以通过指针间接访问内存。

  • 内存编号是从0开始记录的,一般用十六进制数字表示。
  • 可以利用指针变量保存地址。

二、指针变量的定义和使用

指针变量定义语法:数据类型 *变量名

    int a = 10;
    //声明指针变量
    int* p;
    //指针指向a的地址
    p = &a;
    //利用*解引用,找到指针指向内存中的数据
    cout << *p << endl;

三、指针所占内存空间

在32位操作系统下,无论指针数据类型是什么数据类型,都是占4个字节,而在64位操作系统中占8个字节。

四、空指针和野指针

1.空指针:指针变量指向内存中编号为0的空间。

用途:初始化指针变量。

注意:空指针指向的内存是不能够访问的。

int* p = NULL;
//*p;内存0-255之间的内存编号是系统占用的,不可以访问

2.野指针:指针变量指向非法的内存空间

//在程序中,避免出现野指针
int* p = (int*)0x1100;

五、const修饰指针

const修饰的指针有三种情况:

  • const修饰指针--常量指针
  • const修饰常量--指针常量
  • const修饰指针,又修饰常量
    int a = 10;
    int b = 10;
    //常量指针,指针指向的值不可以改,指针的指向可以改
    //即*p=20是非法的,p=&b是合法的
    const int* p = &a;
    //指针常量,指针的指向不可以改,指向的值可以改
    //即*p2=20是合法的,p2=&b是不合法的
    int* const p2 = &a;
    //指针指向的值和指向的值都不可更改
    //即* p3 = 20是不合法的,p3 = &b是不合法的
    const int* const p3 = &a;

六、指针配合数组使用

1.通过指针可以访问到数组中的值

int main() {
    int arr[] = { 1,2,3,4,5,6 };
    //数组名就是数组的首个元素的地址
    int* p = arr;
    //利用指针访问数组中的第一个元素
    //解引用取出4个字节中的数据
    cout << *p << endl;
    //因为p是一个整型指针,执行p++之后,指针指向的就是下一个元素
    p++;
    //利用指针访问数组中的第二个元素
    cout << *p << endl;
}

2.通过指针来遍历数组

    //利用指针遍历数组:
    int* p2 = arr;
    for (int i = 0; i < (sizeof(arr) / sizeof(arr[0])); i++) {
        cout << *p2 << endl;
        p2++;
    }

七、指针和函数

主要区别是值传递还是引用传递

1.值传递:改变形参的值不影响传递进来的实参

#include <iostream>
using namespace std;

void swap(int num1, int num2) {
    cout << "交换之前num1的值:" << num1 << endl;
    cout << "交换之前num2的值:" << num2 << endl;
    int tmp = num1;
    num1 = num2;
    num2 = tmp;
    cout << "交换之后num1的值:" << num1 << endl;
    cout << "交换之后num2的值:" << num2 << endl;
}

int main() 
{
    int a = 1;
    int b = 2;
    cout << "实参未传入之前a的值:" << a << endl;
    cout << "实参未传入之前b的值:" << b << endl;
    swap(a, b);
    cout << "实参传入之后a的值:" << a << endl;
    cout << "实参传入之后b的值:" << b << endl;
    system("pause");
    return 0;
}

输出:

2.引用传递:改变形参的值会影响到实参的值;

#include <iostream>
using namespace std;

void swap(int* num1, int* num2) {
    cout << "交换之前num1的值:" << *num1 << endl;
    cout << "交换之前num2的值:" << *num2 << endl;
    int tmp = *num1;
    *num1 = *num2;
    *num2 = tmp;
    cout << "交换之后num1的值:" << *num1 << endl;
    cout << "交换之后num2的值:" << *num2 << endl;
}


int main() 
{
    int a = 1;
    int b = 2;
    int* p1 = &a;
    int* p2 = &b;
    cout << "实参未传入之前a的值:" << a << endl;
    cout << "实参未传入之前b的值:" << b << endl;
    swap(p1, p2);
    cout << "实参传入之后a的值:" << a << endl;
    cout << "实参传入之后b的值:" << b << endl;
    system("pause");
    return 0;
}

输出:

 

 八、指针函数和数组

封装一个函数,利用指针对数组进行冒泡排序。

#include <iostream>
using namespace std;

void bubbleSort(int* p,int length) {
    //如果在这里计算数组的长度,输出为1,因为传入进来的是数组的首地址,里面只存放1个元素
    //int len = sizeof(arr) / sizeof(arr[0]);
    //cout << len << endl;
    //arr是一个指针,指向数组的第一个元素,在数组中用下标访问元素的值时,相当于是将指针指向该元素的内存空间
    //我的理解是c++会自己定位到那,并取得值
    for (int i = length - 1; i >= 0; i--) {
        for (int j = i - 1; j >= 0; j--) {
            if (p[i] < p[j]){
                int tmp = p[i];
                p[i] = p[j];
                p[j] = tmp;
            }
        }
    }
}
void printArr(int* arr, int length) {
    for (int i = 0; i < length; i++) {
        cout << arr[i] << endl;
    }
}

int main() {
    int arr[] = { 1,4,2,7,6,3,5 };
    //注意要在函数外先计算数组的长度
    //如果在函数里面计算,由于数组传进去的是首元素的地址,则长度永远是1
    int length = sizeof(arr) / sizeof(arr[0]);
    bubbleSort(arr, length);
    printArr(arr, length);

    system("pause");
    return 0;
}

输出:

 

传入数组是引用传递,因此在里面改变了值会影响到原来数组arr的值,所以最后可以直接对arr进行打印输出。 

posted @ 2019-12-22 16:39  西西嘛呦  阅读(459)  评论(0编辑  收藏  举报