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-03 09:03:11

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

C++中的值传递,引用传递及指针传递

C++的值传递,引用传递,指针传递这些概念一直是困扰C++程序员,我一直也没有一个能说清他们之间的关系通过一个简单的实例,或者说浅显的说法,找了一些相关资料,和自已的个人感受,简单介绍一下. c++中传递参数的方式有三种:传参数的值(称为值传递,简称传值),传参数的地址(称为地址传递,简称为传址),和引用传递(简称为传引用),相应的函数也就是传值调用,传址调用和传引用调用 函数定义时参数表中的参数称为形式参数,简称形参,函数调用时参数表中的参数称为实际参数,简称实参,实参和形参之间的数据传递称为

jni使用基础(五)之ndk常见错误

    常见错误: 1.方法名与c语言实现的签名不对应.        异常:UnsatisfiedLinkError   2.如果在执行ndk-build时,出现make:***  Error 1      Error的id为1,C代码的源文件出现了问题.       从上往下解决异常.   3.android.mk文件配置错误    mak:*** No rule to make target    查看 编译的源文件是否存在 并且可读   4.一运行程序,程序界面无法显示.而在logcat

《指针的编程艺术(第二版)》一第二章 指针与变量2.1 一颗星星:表示一把钥匙

第二章 指针与变量 指针的编程艺术(第二版)2-1 一颗星星:表示一把钥匙 2-2 两颗星星:表示两把钥匙 2-3 三颗星星:表示三把钥匙 2-4 练习 如果要间接访问某一变量值,我们可使用指针来完成. 2.1 一颗星星:表示一把钥匙 指针的编程艺术(第二版)本章将介绍指针变量(pointer variable)与一般变量的关系.首先介绍的是,如何定义一个指针变量,如下所示. int x=100; int *ptr = &x; 上述语句表示下列3种含义. 1.ptr是一个指针变量. 2.ptr变

java及C++中传值传递、引用传递和指针方式的理解_java

java的值传递理解: 代码1: public class Test { /** * @param args */ public static void main(String[] args) { StringBuffer buffer= new StringBuffer("colin"); SChange(buffer); System.out.println( buffer); } public static void SChange (StringBuffer str) { st

php数组函数序列之prev() - 移动数组内部指针到上一个元素的位置,并返回该元素值_php技巧

prev() 定义和用法 prev() 函数把指向当前元素的指针移动到上一个元素的位置,并返回该元素值. 如果内部指针已经超过数组的第一个元素之前,函数返回 false. 语法 prev(array)参数 描述 array 必需.规定要使用的数组. 说明 prev() 和 next() 的行为类似,不过它将内部指针倒回一位而不是前移一位. 注意:如果数组包含空的单元,或者单元的值是 0 则该函数碰到这些单元也返回 FALSE.要正确遍历可能含有空单元或者单元值为 0 的数组,请参见 each()

Android For JNI(五)——C语言多级指针,结构体,联合体,枚举,自定义类型

Android For JNI(五)--C语言多级指针,结构体,联合体,枚举,自定义类型 我们的C已经渐渐的步入正轨了,基础过去之后,就是我们的NDK和JNI实战了 一.多级指针 指针的概念我们在前面也讲了许多,也提到了多级指针的概念,那具体我们怎么去操作呢 #include <stdio.h> #include <stdlib.h> main(){ int i = 3; //一级 int* p = &i; //二级,保存一级指针 int** w = &p; //三

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

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

Android在JNI中使用ByteBuffer的方法

  Android在JNI中使用ByteBuffer的方法 本文实例讲述了Android在JNI中使用ByteBuffer的方法.分享给大家供大家参考.具体如下: 一.ByteBuffer 定义 在NIO中,数据的读写操作始终是与缓冲区相关联的(读取时信道(SocketChannel)将数据读入缓冲区,写入时首先要将发送的数据按顺序填入缓冲区) 缓冲区是定长的,基本上它只是一个列表,它的所有元素都是基本数据类型.ByteBuffer是最常用的缓冲区,它提供了读写其他数据类型的方法,且信道的读写方

Android For JNI(六)——交叉编译,NDK概述以及文件结构,编写自己的第一个JNI工程

Android For JNI(六)--交叉编译,NDK概述以及文件结构,编写自己的第一个JNI工程 终于回到我们的 android了,我们先要配置这个NDK的环境,但是之前,我们还要了解一下基本的术语 一.交叉编译 在一个平台下编译出另一个平台可以执行的二进制程序 CPU平台:ARM X86 MIPS(指令集) 系统平台:windows linux mac 原理:模拟另一个平台的特性去编译程序 源代码-->预编译-->编译-->链接-->可执行程序 工具链:一个工具使用自动调用下