Android For JNI(三)——C的指针,指针变量,指针常见错误,值传递,引用传递,返回多个值

Android For JNI(三)——C的指针,指针变量,指针常见错误,值传递,引用传递,返回多个值


C中比较难的这一块,大概就是指针了,所以大家还是多翻阅一下资料,当然,如果只是想了解一下,看本篇也就够了,不过我也尽量陈述的很详细

一.指针

要说指针,其实通俗易懂的话来说,他应该是保存内存地址的一个变量,我们来看一下小例子

#include <stdio.h>
#include <stdlib.h>   

main(){
       //int 变量
       int i ;
       i = 5;
       //打印i的值
       printf("%d\n",i);
       //打印i的内存地址
       printf("%#x\n",&i);

      //定义一个指针变量
      //指针的含义是他是用来保存内存地址的
       int * j;
       j= &i;
        //打印指针
       printf("%#x",j);

        //让窗口停留
        int age ;
        scanf("%d",&age);

}

从这段话不难看出,我们定义一个i的话可以打印值和内存地址,但是我们定义一个指针j,也是可以赋值一个指针地址的,不信,我们打印一下

可以看到,内存地址是一样的了,所以我们就可以清晰明了的知道了指针的概念了,指针指向的内存地址上村坊的数值是一个int类型

既然指针可以接收到内存地址,那他同样是可以转换成值的,我们继续来看

//打印指针值
printf("%d",*j); 

输出一个指针int类型的值

就是5 了,*p指向内存地址上的值

二进指针

二进指针的概念,他也是一个指针变量,二级指针存放的地址必须是一个一级指针,也就是j

        //定义一个二进指针
       int ** k;
       //二级指针存放的地址必须是一个一级指针,也就是j
       k = &j; 

       //输出j的内存地址
       printf("%#x\n",&j);
       //打印二级指针‘
       printf("%#x\n",k); 

我们这段运行之后就是k的内存地址和j的是一样的了

指针和指针变量的关系

  • 指针就是地址,地址就是指针
  • 地址就是内存单元的编号
  • 指针变量是存放地址的变量
  • 指针和指针变量是两个不同的概念
  • 我们陈述的时候会把指针变量说成指针,但是含义不一样

二.指针常见错误

我们有时候会碰到一些梗,这都是需要注意到的,我们定义一个指针

        //定义一个指针,在内存中开辟
        int * i;
        //打印i的内存地址
        printf("%#x\n",i); 

没有给这个i赋值,但是可以打印

这里我们就知道,指针已定义就会有一个值了,这个本身不是错误,但是如果你

    *i = 3; 

你就有错误了,因为定义的时候是随机指向一个地址,你现在赋值这个地址为3,就错误了,正常是思路是


        //定义一个指针,在内存中开辟
        int * i;

        i= 5;
        //打印i的内存地址
        printf("%#x\n",i);

        *i = 3; 

这样输出的才是3

所以我们得到的结论就是指针复制之前,不要是桶*i赋值,指针的类型也不能混用

三,值传递和引用传递

这个概念在JAVA中也有,在排序中用到的比较多吧,其实就是定义一个中间变量去转换,我写个小例子大家就简单明了了

#include <stdio.h>
#include <stdlib.h>   

main(){
        //定义两个变量
        int i = 3;
        int j = 5;
        //定义中间变量
        int temp = i;   //temp = 3
        i = j;     // i = 5
        j = temp;  // j = 3 

        //让窗口停留
        int age ;
        scanf("%d",&age);

}

这样就把值给换过来了,不过这样做没什么意义,我们用函数去简化他,我们先来看下用JAVA的概念值传递是一个什么样式

#include <stdio.h>
#include <stdlib.h>   

void function(int i ,int j)
     {
        int temp = i;
        i = j;
        j = temp;
     }

main(){
        //定义两个变量
        int i = 3;
        int j = 5;
        //调用函数
        function(i,j); 

        printf("i的值为:%d\n",i);
        printf("j的值为:%d\n",j); 

        //让窗口停留
        int age ;
        scanf("%d",&age);

}

用JAVA的思想,应该是这样写,我们来运行一下

然而你会发现,他们的值没有变化,这是因为在C中,值传递本身不会对变量产生什么影响,其实在JAVA中这样也不会发生什么改变,左右这里就牵扯出来一个引用传递了,我们写一个JAVA程序来打比方

/**
 * 测试类
 * @author LGL
 *
 */
public class Test {
    //定义两个全局变量
    int i = 3;
    int j = 5;

    public static void main(String[] args) {
        Test test = new Test();
        Function(test);
        System.out.println(test.i);
        System.out.println(test.j);
    }

    private static void Function(Test sTest){
        int temp = sTest.i;
        sTest.i = sTest.j;
        sTest.j = temp;
    }
}

用这个思想就可以把值转换了

这就叫引用传递,但是再C中面向过程的思想是没有对象的,所以传地址也是可以实现的,我们试试看

#include <stdio.h>
#include <stdlib.h>   

void function(int* p ,int* q)
     {
        int temp = *p;
        *p = *q;
        *q = temp;
     }

main(){
        //定义两个变量
        int i = 3;
        int j = 5;
        //调用函数
        function(&i,&j); 

        printf("i的值为:%d\n",i);
        printf("j的值为:%d\n",j); 

        //让窗口停留
        int age ;
        scanf("%d",&age);

}

我们把内存地址传过去之后,进行转换,就可以了,运行如下

值传递和引用传递的概念

  • 值传递:传递一个普通的值
  • 引用传递:传递一个内存地址

其实这些都应该叫值传递,只不过引用传递了一个地址罢了

四.指针返回值

为什么使用指针?

  • 指针是可以直接访问硬件的
  • 快速传递数据
  • 返回一个以上的值
  • 表示复杂的数据结构
  • 方便处理字符串
  • 指针有助于理解面向对象

要是让我们用JAVA去写一个返回值,大家都是用return,很方便,但是再C中,她是可以返回多个值的,我们写个例子

#include <stdio.h>
#include <stdlib.h>   

void function(int* p , int* q)
     {
        //进行操作
        *p = *p*2;
        *q = *q *2;

     }

main(){
        //定义两个变量
        int i = 3;
        int j = 5;
        //调用函数
        function(&i,&j); 

        printf("i的值为:%d\n",i);
        printf("j的值为:%d\n",j); 

        //让窗口停留
        int age ;
        scanf("%d",&age);

}

我们运行

你会发现他变了,这就是C可以返回多个值的概念了,他都可用直接操作了

好的,这篇闲到这里,走到这里,对C已经有一个模糊的印象了,但是还没有摸到门槛,我们还得继续加油才是!

时间: 2024-08-24 11:43:30

Android For JNI(三)——C的指针,指针变量,指针常见错误,值传递,引用传递,返回多个值的相关文章

函数指针与函数指针数组的使用方法

转自:http://blog.csdn.net/feitianxuxue/article/details/7300291 函数指针与函数指针数组的使用方法 函数指针: 函数指针包含函数在内存中的地址.数组名实际上就是数组的第一个元素在内存中的地址,类似地,函数名实际上也是执行这个函数任务的代码在内存中的起始地址. 函数指针可以传递给函数.从函数返回.保存在数组中.赋予另一个函数指针或者调用底层函数. 下面我们用数值算法accumulate讨论下函数指针的用法.accumulate是一种常用的ST

函数何时值传递,何时指针,何时引用传递总结

编程中定义函数,一般三种传递方法,看是简单.想灵活合理选择,还须要大量的编程经验和技巧. 故在这里特意总结一下这三种传递的方法. 根本差别: 函数值传递不改变变量值,假设想改变变量值,须要返回值,然后用变量接收一下这个返回值. 而指针传递和引用传递都能够在函数内改变变量值.不须要通过返回值的形式改变. 应用场合: 当想通过这个函数.改变好几个变量的值,多个变量都通过函数返回值来改变变量值方式费时费力.所以这样的场合就比較适合使用指针和引用. 指针传递须要开内存.假设忘记释放的话,可能导致内存泄露

Android JNI编程(三)——C语言指针的初步认识、指针变量、互换两个数、函数返回多个值

目录(?)[+] 一.什么是指针? 简单来说: 指针就是内存地址      内存地址就是指针.来看个小案例 #include<stdio.h> #include<stdlib.h> /** 指针就是内存地址 内存地址就是指针 */ main() { //定义一个int类型的变量i,并且赋值为10 int i=10; //定义一个int类型的一级指针变量p int* p; //把i对应的地址给p变量 p=&i; //指针取值*p:把p变量地址对应的值取出来 printf(&q

Android For JNI(四)——C的数组,指针长度,堆内存和栈内存,malloc,学生管理系统

Android For JNI(四)--C的数组,指针长度,堆内存和栈内存,malloc,学生管理系统 好几天每写JNI了,现在任务也越来越重了,工作的强度有点高,还有好几个系列的博客要等着更新,几本书还嗷嗷待哺的等着我去看,github上的两个散漫的开源,基础入门的视频也在录制,还要学习新的知识, 都是一种挑战,不知道为何,最近懒散了,看来还得再加把劲,今天我们继续延伸一下C的一些小知识 一.数组 C的数组和JAVA也是类似的,我们写一段小程序 #include <stdio.h> #inc

Android系统的智能指针(轻量级指针、强指针和弱指针)的实现原理分析

文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6786239 Android 系统的运行时库层代码是用C++来编写的,用C++来写代码最容易出错的地方就是指针了,一旦使用不当,轻则造成内存泄漏,重则造成系统崩溃.不过系统为 我们提供了智能指针,避免出现上述问题,本文将系统地分析Android系统智能指针(轻量级指针.强指针和弱指针)的实现原理. 在使用C++来编写代码的过程中,指针使用不当造成

Android中强指针和弱指针

因为Android中很多地方代码是用C++编写,为了能够保证C++中指针能够被正确的释放,于是Android引入了其实在C++中已经有的智能指针技术: 智能指针技术的实质就是:记录引用某一个对象的次数,一旦检测到次数为0,这时就自定将此对象所占内存释放. 简单的的智能指针技术因为不能解决对象循环引用的问题:a引用b:b引用a,这样的情况下使用简单的智能指针技术无法解决,故引入强指针和弱指针: 其实完全可以把强弱指针看做c语言中的一个指向对象的地址(为了便于理解),不过区别在于强指针可以操作对象,

OpenCV4Android释疑: 透析Android以JNI调OpenCV的三种方式(让OpenCVManager永不困扰)

前文曾具体探讨了关于OpenCV的使用,原本以为天下已太平.但不断有人反应依旧配不好OpenCV4Android,不能得心应手的在Android上使用OpenCV.大量的精力都浪费在摸索配置上.尤其是OpenCVManager诞生之后.更让人无语.大家第一个反应就是怎样才干不安装OpenCVManager.由于要多安装这个东西对客户来说体验太不好了. 咱家昨夜研究至两点,今早七点起床.最终把头绪理清了. 以下咱家以之前做过的一个基于OpenCV2.3.1.android通过jni调用opencv

C++ Primer 学习笔记_35_面向对象编程(6)--虚函数与多态(三):虚函数表指针(vptr)及虚基类表指针(bptr)、C++对象模型

C++ Primer 学习笔记_35_面向对象编程(6)--虚函数与多态(三):虚函数表指针(vptr)及虚基类表指针(bptr).C++对象模型 一.虚函数表指针(vptr)及虚基类表指针(bptr) C++在布局以及存取时间上主要的额外负担是由virtual引起的,包括: virtual function机制:用以支持一个有效率的"执行期绑定": virtual base class:用以实现多次在继承体系中的基类,有一个单一而被共享的实体. 1.虚函数表指针 C++中,有两种数据

C++函数的三种传递方式为:值传递、指针传递和引用传递

值传递: void fun(int x){ x += 5; //修改的只是y在栈中copy x,x只是y的一个副本,在内存中重新开辟的一块临时空间把y的值 送给了x:这样也增加了程序运行的时间,降低了程序的效率. } void main(void){ int y = 0; fun(y); cout<<\"y = \"<<y<<endl; //y = 0; } 指针传递: void fun(int *x){ *x += 5; //修改的是指针x指向的内