c语言高低位拷贝_C语言指针详解

1为什么使用指针

假如我们定义了 char a=’A’ ,当需要使用 ‘A’ 时,除了直接调用变量 a ,还可以定义 char *p=&a ,调用 a 的地址,即指向 a 的指针 p ,变量 achar 类型)只占了一个字节,指针本身的大小由可寻址的字长来决定,指针 p 占用 4 个字节。

但如果要引用的是占用内存空间比较大东西,用指针也还是 4 个字节即可。

  • 使用指针型变量在很多时候占用更小的内存空间

变量为了表示数据,指针可以更好的传递数据,举个例子:

第一节课是 1 班语文, 2 班数学,第二节课颠倒过来, 1 班要上数学, 2 班要上语文,那么第一节课下课后需要怎样作调整呢?方案一:课间 1 班学生全都去 2 班, 2 班学生全都来 1 班,当然,走的时候要携带上书本、笔纸、零食……场面一片狼藉;方案二:两位老师课间互换教室。

显然,方案二更好一些,方案二类似使用指针传递地址,方案一将内存中的内容重新“复制”了一份,效率比较低。

  • 在数据传递时,如果数据块较大,可以使用指针传递地址而不是实际数据,即提高传输速度,又节省大量内存。

一个数据缓冲区 char buf[100] ,如果其中 buf[0,3] 为命令号, buf[4,5] 为类型, buf[6~10] 为该类型的数值,类型为 int ,使用如下语句进行赋值:

*(short*)&buf[0]=DataId;
*(short*)&buf[4]=DataType;
*(int*)&buf[6]=DataValue;
  • 数据转换,利用指针的灵活的类型转换,可以用来做数据类型转换,比较常用于通讯缓冲区的填充。
  • 指针的机制比较简单,其功能可以被集中重新实现成更抽象化的引用数据形别
  • 函数指针,形如: #define PMYFUN (void*)(int,int) ,可以用在大量分支处理的实例当中,如某通讯根据不同的命令号执行不同类型的命令,则可以建立一个函数指针数组,进行散转。
  • 在数据结构中,链表、树、图等大量的应用都离不开指针。

2 指针是什么?

操作系统将硬件和软件结合起来,给程序员提供的一种对内存使用的抽象,这种抽象机制使得程序使用的是虚拟存储器,而不是直接操作和使用真实存在的物理存储器。所有的虚拟地址形成的集合就是虚拟地址空间。

内存是一个很大的线性的字节数组,每个字节固定由 8 个二进制位组成,每个字节都有唯一的编号,如下图,这是一个 4G 的内存,他一共有 4x1024x1024x1024 = 4294967296 个字节,那么它的地址范围就是 0 ~ 4294967296 ,十六进制表示就是 0x00000000~0xffffffff ,当程序使用的数据载入内存时,都有自己唯一的一个编号,这个编号就是这个数据的地址。指针就是这样形成的。

f54a2c8a169ea27cd3390fbe6933d816.png
#include <stdio.h>int main(void)
{char ch = 'a';int  num = 97;printf("ch 的地址:%pn",&ch);   //ch 的地址:00BEFDF7printf("num的地址:%pn",&num);  //num的地址:00BEFDF8return 0;
}

373b8f13d3edb84d2f996b2d46aa4e90.png

3d9719ca1405aae8ceca1d0aa07eee31.png

指针不仅可以表示变量的地址,还可以存储各种类型数据的地址,指针变量是用来保存这些地址的变量,与数组类似,依据地址存放的数据类型,指针也分为 int 指针类型, double 指针类型, char 指针类型等等。

综上,指针的实质就是数据在内存中的地址,而指针变量是用来保存这些地址的变量

指针变量 和 指向关系

用来保存 指针 的变量,就是指针变量。如果指针变量p保存了变量 num的地址,则就说:p指向了变量num,也可以说p指向了num所在的内存块,指针变量pp指向了p所在的内存块,以下面为例:

#include <stdio.h>int main(void)
{int num = 97;char ch = 'a';int *p = & num;int **pp = &p;char *p1 = & ch;printf("num 的地址:%pn",&num);   printf("指针p的值:%pn",p);   printf("指针p的地址:%pn",&p);  printf("指针pp的值:%pn",pp); printf("ch 的地址:%pn",&ch);  return 0;
}

c0c3ea4edbe5e3739c4b79ae6945fbfd.png
  • int型的num值为97占4个字节,内存地址为:0113F924char 型的ch('a')值为97占1个字节,内存地址为:0113F91B

b12d53230586caf6e91045025b2a979f.png

82d9d74ea6e40777dc6e7928e8a3b02e.png
  • num的地址为:0113F924num的值为 97 ,指针 p 指向 num 的内存块,指针 p 地址为:0113F90Cp的内存保存的值就是num的地址0113F924

896b29822c295ee98c9048ad930828db.png
  • 指针变量 pp 指向 指针 p,指针 pp 内存值为 指针 p 的地址:0113F90C,形成了只想指针的指针。

9a5abcff64789918c0275d09d5f04d37.png

定义指针变量

C语言中,定义变量时,在变量名 前 写一个 * 星号,这个变量就变成了对应变量类型的指针变量。必要时要加( ) 来避免优先级的问题。

引申:C语言中,定义变量时,在定义的最前面写上typedef ,那么这个变量名就成了一种类型,即这个类型的同义词。

int a ; //int类型变量 a
int *a ; //int* 变量a
int arr[3]; //arr是包含3个int元素的数组
int (* arr )[3]; //arr是一个指向包含3个int元素的数组的指针变量int* p_int; //指向int类型变量的指针 
double* p_double; //指向idouble类型变量的指针 
struct Student *p_struct; //结构体类型的指针
int(*p_func)(int,int); //指向返回类型为int,有2个int形参的函数的指针 
int(*p_arr)[3]; //指向含有3个int元素的数组的指针 
int** p_pointer; //指向 一个整形变量指针的指针

取地址

既然有了指针变量,那就得让他保存其它变量的地址,使用& 运算符取得一个变量的地址。

int add(int a , int b)
{return a + b;
}int main(void)
{int num = 97;float score = 10.00F;int arr[3] = {1,2,3};int* p_num = &num;float* p_score = &score;int (*p_arr)[3] = &arr;           int (*fp_add)(int ,int )  = add;  //p_add是指向函数add的函数指针return 0;
}

特殊的情况,他们并不一定需要使用&取地址

  • 数组名的值就是这个数组的第一个元素的地址。
  • 函数名的值就是这个函数的地址。
  • 字符串字面值常量作为右值时,就是这个字符串对应的字符数组的名称,也就是这个字符串在内存中的地址。
int add(int a , int b){return a + b;
}
int main(void)
{int arr[3] = {1,2,3};int* p_first = arr;int (*fp_add)(int ,int )  =  add;const char* msg = "Hello world";return 0;
}

解地址

对一个指针解地址,就可以取到这个内存数据,解地址 的写法,就是在指针的前面加一个 * 号。

解指针的实质是:从指针指向的内存块中取出这个内存数据。

int main(void)
{int age = 19;int*p_age = &age;*p_age  = 20;  //通过指针修改指向的内存数据printf("age = %d",*p_age);   //通过指针读取指向的内存数据printf("age = %d",age);return 0;
}

空指针

空指针在概念上不同于未初始化的指针。空指针可以确保不指向任何对象或函数;而未初始化的指针则可能指向任何地方。空指针不是野指针。

在C语言中,我们让指针变量赋值为NULL表示一个空指针,而C语言中,NULL实质是 ((void*)0) , 在C++中,NULL实质是0。

#ifdef __cplusplus#define NULL    0
#else    #define NULL    ((void *)0)
#endif

void*类型指针

void是一种特殊的指针类型,可以用来存放任意对象的地址。一个void指针存放着一个地址,这一点和其他指针类似。不同的是,我们对它到底储存的是什么对象的地址并不了解。

double a=2.3;
int b=5;
void *p=&a;
cout<<p<<endl;   //输出了a的地址p=&b;
cout<<p<<endl;   //输出了b的地址//cout<<*p<<endl;这一行不可以执行,void*指针只可以储存变量地址,不可以直接操作它指向的对象

由于void是空类型,只保存了指针的值,而丢失了类型信息,我们不知道他指向的数据是什么类型的,只指定这个数据在内存中的起始地址,如果想要完整的提取指向的数据,程序员就必须对这个指针做出正确的类型转换,然后再解指针。

数组和指针

  • 同类型指针变量可以相互赋值,数组不行,只能一个一个元素的赋值或拷贝
  • 数组在内存中是连续存放的,开辟一块连续的内存空间。数组是根据数组的下进行访问的。指针很灵活,它可以指向任意类型的数据。指针的类型说明了它所指向地址空间的内存。
  • 数组所占存储空间的内存:sizeof(数组名) 数组的大小:sizeof(数组名)/sizeof(数据类型),在32位平台下,无论指针的类型是什么,sizeof(指针名)都是 4 ,在 64 位平台下,无论指针的类型是什么,sizeof(指针名)都是 8 。
  • 数组名作为右值的时候,就是第一个元素的地址
int main(void)
{int arr[5] = {1,2,3,4,5};int *p_first = arr;printf("%d",*p_first);  //1return 0;
}
  • 指向数组元素的指针 支持 递增 递减 运算。p= p+1意思是,让p指向原来指向的内存块的下一个相邻的相同类型的内存块。在数组中相邻内存就是相邻下标元素。

函数与指针

函数的参数和指针

C语言中,实参传递给形参,是按值传递的,也就是说,函数中的形参是实参的拷贝份,形参和实参只是在值上面一样,而不是同一个内存数据对象。这就意味着:这种数据传递是单向的,即从调用者传递给被调函数,而被调函数无法修改传递的参数达到回传的效果。

void change(int a)
{a++;      //在函数中改变的只是这个函数的局部变量a,而随着函数执行结束,a被销毁。age还是原来的age,纹丝不动。
}
int main(void)
{int age = 60;change(age);printf("age = %d",age);   // age = 60return 0;
}

有时候我们可以使用函数的返回值来回传数据,在简单的情况下是可以的,但是如果返回值有其它用途(例如返回函数的执行状态量),或者要回传的数据不止一个,返回值就解决不了了。

传递变量的指针可以轻松解决上述问题。

void change(int* pa)
{(*pa)++;   //因为传递的是age的地址,因此pa指向内存数据age。当在函数中对指针pa解地址时,//会直接去内存中找到age这个数据,然后把它增1。
}
int main(void)
{int age = 160;change(&age);printf("age = %d",age);   // age = 61return 0;
}

比如指针的一个常见的使用例子:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>void swap(int *,int *);
int main()
{int a=5,b=10;printf("a=%d,b=%dn",a,b);swap(&a,&b);printf("a=%d,b=%dn",a,b);return 0;
}
void swap(int *pa,int *pb)
{int t=*pa;*pa=*pb;*pb=t;
}

在以上的例子中,swap函数的两个形参pa和pb可以接收两个整型变量的地址,并通过间接访问的方式修改了它指向变量的值。在main函数中调用swap时,提供的实参分别为&a,&b,这样就实现了pa=&a,pb=&b的赋值过程,这样在swap函数中就通过*pa修改了 a 的值,通过*pb修改了 b 的值。因此,如果需要在被调函数中修改主调函数中变量的值,就需要经过以下几个步骤:

  • 定义函数的形参必须为指针类型,以接收主调函数中传来的变量的地址;
  • 调用函数时实参为变量的地址;
  • 在被调函数中使用*间接访问形参指向的内存空间,实现修改主调函数中变量值的功能。

指针作为函数的形参的另一个典型应用是当函数有多个返回值的情形。比如,需要在一个函数中统计一个数组的最大值、最小值和平均值。当然你可以编写三个函数分别完成统计三个值的功能。但比较啰嗦,如:

int GetMax(int a[],int n)
{int max=a[0],i;for(i=1;i<n;i++){if(max<a[i]) max=a[i];}return max;
}
int GetMin(int a[],int n)
{int min=a[0],i;for(i=1;i<n;i++){if(min>a[i]) min=a[i];}return min;
}
double GetAvg(int a[],int n)
{double avg=0;int i;for(i=0;i<n;i++){avg+=a[i];}return avg/n;
}

其实我们完全可以在一个函数中完成这个功能,由于函数只能有一个返回值,可以返回平均值,最大值和最小值可以通过指针类型的形参来进行实现:

double Stat(int a[],int n,int *pmax,int *pmin)
{double avg=a[0];int i;*pmax=*pmin=a[0];for(i=1;i<n;i++){avg+=a[i];if(*pmax<a[i]) *pmax=a[i];if(*pmin>a[i]) *pmin=a[i];}return avg/n;
}

函数的指针

一个函数总是占用一段连续的内存区域,函数名在表达式中有时也会被转换为该函数所在内存区域的首地址。我们可以把函数的这个首地址赋予一个指针变量,使指针变量指向函数所在的内存区域,然后通过指针变量就可以找到并调用该函数。这种指针就是函数指针。

函数指针的定义形式为:

returnType (*pointerName)(param list);

returnType 为函数返回值类型,pointerNmae 为指针名称,param list 为函数参数列表。参数列表中可以同时给出参数的类型和名称,也可以只给出参数的类型,省略参数的名称,这一点和函数原型非常类似。

用指针来实现对函数的调用:

#include <stdio.h>
//返回两个数中较大的一个
int max(int a, int b)
{return a>b ? a : b;
}
int main()
{int x, y, maxval;//定义函数指针int (*pmax)(int, int) = max;  //也可以写作int (*pmax)(int a, int b)printf("Input two numbers:");scanf("%d %d", &x, &y);maxval = (*pmax)(x, y);printf("Max value: %dn", maxval);return 0;
}

结构体和指针

结构体指针有特殊的语法: -> 符号

如果p是一个结构体指针,则可以使用 p ->【成员】 的方法访问结构体的成员

typedef struct
{char name[31];int age;float score;
}Student;int main(void)
{Student stu = {"Bob" , 19, 98.0};Student*ps = &stu;ps->age = 20;ps->score = 99.0;printf("name:%s age:%d
",ps->name,ps->age);return 0;
}

const 和 指针

  • 指向常量的指针,值不能改变,指向可改变
  • 常指针值能改变,指向不可改变
  • 指向常量的常指针,都不能改变
#include <stdio.h>int main()
{// 1 可改变指针const int a = 10;int *p = &a;*p = 1000;printf("*p = %dn", *p);// 2 可改变指针const b = 10;int *pb = &b;pb = p;printf("*pb = %dn", *pb);// 3const c = 10;int * const pc = &c;*pc = 1000;//pc = pb;不能改变//4const d = 10;const * int const pd = &d;//*pd = 1000; 不能改变printf("n");return 0;
}

深拷贝和浅拷贝

如果2个程序单元(例如2个函数)是通过拷贝 他们所共享的数据的 指针来工作的,这就是浅拷贝,因为真正要访问的数据并没有被拷贝。如果被访问的数据被拷贝了,在每个单元中都有自己的一份,对目标数据的操作相互 不受影响,则叫做深拷贝。

#include <iostream>
using namespace std;class CopyDemo
{
public:CopyDemo(int pa,char *cstr)  //构造函数,两个参数{this->a = pa;this->str = new char[1024]; //指针数组,动态的用new在堆上分配存储空间strcpy(this->str,cstr);    //拷贝过来}//没写,C++会自动帮忙写一个复制构造函数,浅拷贝只复制指针,如下注释部分//CopyDemo(CopyDemo& obj)  //{//   this->a = obj.a;//  this->str = obj.str; //这里是浅复制会出问题,要深复制//}CopyDemo(CopyDemo& obj)  //一般数据成员有指针要自己写复制构造函数,如下{this->a = obj.a;// this->str = obj.str; //这里是浅复制会出问题,要深复制this->str = new char[1024];//应该这样写if(str != 0)strcpy(this->str,obj.str); //如果成功,把内容复制过来}~CopyDemo()  //析构函数{delete str;}public:int a;  //定义一个整型的数据成员char *str; //字符串指针
};int main()
{CopyDemo A(100,"hello!!!");CopyDemo B = A;  //复制构造函数,把A的10和hello!!!复制给Bcout <<"A:"<< A.a << "," <<A.str << endl;//输出A:100,hello!!!cout <<"B:"<< B.a << "," <<B.str << endl;//输出B:100,hello!!!//修改后,发现A,B都被改变,原因就是浅复制,A,B指针指向同一地方,修改后都改变B.a = 80;B.str[0] = 'k';cout <<"A:"<< A.a << "," <<A.str << endl;//输出A:100,kello!!!cout <<"B:"<< B.a << "," <<B.str << endl;//输出B:80,kello!!!return 0;
}

根据上面实例可以看到,浅复制仅复制对象本身(其中包括是指针的成员),这样不同被复制对象的成员中的对应非空指针会指向同一对象,被成员指针引用的对象成为共享的,无法直接通过指针成员安全地删除(因为若直接删除,另外对象中的指针就会无效,形成所谓的野指针,而访问无效指针是危险的;

除非这些指针有引用计数或者其它手段确保被指对象的所有权);而深复制在浅复制的基础上,连同指针指向的对象也一起复制,代价比较高,但是相对容易管理。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/555094.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MybatisPlus中@TableField注解的使用详解

MybatisPlus中TableField注解的使用详解 实现 官方文档说明&#xff1a; com.baomidou.mybatisplus.annotations.TableField TableField注解新增属性 update 预处理 set 字段自定义注入 (讲解&#xff1a;比如我们使用mybatisplus自带的insert()方法向数据库插入数据时&…

dataframe记录数_大数据系列之Spark SQL、DataFrame和RDD数据统计与可视化

Spark大数据分析中涉及到RDD、Data Frame和SparkSQL的操作&#xff0c;本文简要介绍三种方式在数据统计中的算子使用。1、在IPython Notebook运行Python Spark程序IPython Notebook具备交互式界面&#xff0c;可以在Web界面输入Python命令后立刻看到结果&#xff0c;还可将数据…

bug的生命周期、bug状态转换图

当我们发现一个bug的时候&#xff0c;应该怎么理清他们之间的关系呢&#xff1f;一个bug 从open到close的所有状态 都是我们测试人员需要注意的。 一、bug的状态 新建&#xff08;New&#xff09; 新发现的bug&#xff0c;未经评审决定是否指派给开发人员进行修改。 确认&…

wps居中对齐不在中间_WPS文字快捷键总结(Windows版本)--值得收藏

WPS Office是一款国产的办公软件套装&#xff0c;有WPS文字、WPS表格和WPS演示三个板块&#xff0c;可以实现办公软件最常见的文字、表格、演示等多种功能&#xff0c;支持阅读和输出PDF文件&#xff0c;全面兼容Microsoft Office97-2010格式。想要熟练地使用WPS办公软件&#…

bugzilla使用规范分享

bugzilla使用规范分享 1.new/confirmed 测试人员将Bug提交给任务分发人员&#xff08;研发模块负责人&#xff09;&#xff0c; 此时Bug状态为new/confirmed&#xff0c;开始Bug的生命周期&#xff0c;如果测试人员知道具体负责的研发人员&#xff0c;也可以直接指定&#x…

编程语言_如何正确地学习编程语言

首先&#xff0c;当前学生和职场人学习编程已经成为了一个大的趋势&#xff0c;掌握编程语言不仅能够提升自身获取信息的能力&#xff0c;同时也能够拓展自身的能力边界&#xff0c;这一点在工业互联网时代会有更加明显的体现。编程语言本身并不难&#xff0c;但是要想形成自己…

测试游戏帧率电脑温度的软件,游戏中显示帧数和温度方法_游戏画面中实时显示FPS帧数温度技巧...

相信很多游戏玩家平时都是在用电脑玩游戏&#xff0c;而且大家也喜欢看一看自己在在游戏画面中实时的FPS帧数和温度信息。但是很网友对这个游戏画面中如何实时显示FPS帧数、频率、硬件温度不太清楚&#xff0c;下面智能手机网分享一下具体的操作方法&#xff0c;以便大家在玩游…

12面魔方公式图解法_一位建筑工程师:多年渴望就是学会魔方还原,只按这七步就可以!...

本人性别男&#xff0c;年龄47岁&#xff0c;一位建筑工程师&#xff0c;性格开朗&#xff0c;喜欢学习&#xff0c;2013年在网上搜索记忆关键词&#xff0c;从此开始了学习超级记忆和思维导图之路&#xff01;也因此&#xff0c;接触了魔方&#xff01;初学魔方&#xff0c;我…

navicat运行db文件_使用 YAML 文件配置 Jenkins 流水线

本文转载自&#xff1a;Jenkins 中文社区这也是一种自定义流水线 DSL 的方法几年前&#xff0c;我们的 CTO 写了一篇关于 使用 Jenkins 和 Docker 为 Ruby On Rails 应用提供持续集成服务 的文章。这些年&#xff0c;我们一直使用这个 CI 流水线解决方案&#xff0c;直到我们最…

Mybatis-Plus实现逻辑删除

数据库中的数据删除会分为两种&#xff1a;物理删除 和 逻辑删除 物理删除 物理删除就是我们删除数据库中的一条数据时&#xff0c;数据会真的被删除 逻辑删除 逻辑删除指的是我们删除一条数据时&#xff0c;数据不会在数据库中消息&#xff0c;逻辑删除是我们现在开发中经…

完美国际单机修改服务器端,完美国际改国内版单机一键服务端

最喜欢的一款网游&#xff0c;曾经出来的时候可以说是划时代的&#xff0c;3D游戏&#xff0c;空战&#xff0c;大地图无缝对接等。从比较早的113版本&#xff0c;经典的六职业136版本&#xff0c;到现在的155版本。都保存的有。玩官服就不说啦&#xff0c;从13年接触服务端到现…

开发 数组里面的字典_Redis字典结构与rehash解读

关注公众号&#xff1a;后端技术漫谈&#xff0c;技术之路不迷路~字典是一种用于保存键值对的抽象数据结构&#xff0c;也被称为查找表、映射或关联表。在字典中&#xff0c;一个键(key)可以和一个值(value)进行关联&#xff0c;这些关联的键和值就称之为键值对。抽象数据结构&…

MyBatisPlus 学习笔记_MP的AR模式

狂神说 MyBatisPlus 学习笔记 一、快速入门 文档&#xff1a;https://mp.baomidou.com/ 使用第三方组件&#xff1a; 导入对应依赖研究依赖如何配置代码如何编写提高扩展技术能力 步骤&#xff1a; 1、创建数据库 mybatis_plus 2、创建user表 DROP TABLE IF EXISTS user;CREATE…

ajax调用java程序,从微信小程序到鸿蒙JS开发-JS调用Java

除轻量级智能穿戴设备&#xff0c;现鸿蒙支持的手机、汽车、TV、手表、平板等属于富鸿蒙&#xff0c;在JS语言的项目中也有Java模块&#xff0c;并提供了JS跨语言调用Java方法的技术。现需要实现查看商品评论时&#xff0c;统计出长评、中评和短评的比例&#xff0c;这里将评论…

文本删除空行_010 Editor for mac(文本和十六进制编辑器)

为大家带来最新版本的010 Editor for mac&#xff0c;这是一款专业的文本和十六进制编辑器&#xff0c;新版本的010 editor mac版包含了语法突出显示、更多字符集支持、添加了删除行和删除空行命令等新功能&#xff0c;另外修复了各种错误&#xff0c;功能更加全面。010editor …

Mybatis-Plus之四种lambda方式LambdaQueryWrapper,QueryWrapper<实体>().lambda(),LambdaQueryChainWrapper<实体>

Mybatis-Plus之四种lambda方式 lambda四种表达形式 前言 使用了lambda表达式 可以通过方法引用的方式来使用实体字段名的操作&#xff0c;避免直接写数据库表字段名时的错写名字&#xff1b; 一、LambdaQueryWrapper<> /*** lambda 条件构造器* 生成的sql语句 SELECT…

sql怎么修改服务器角色,sql角色服务器的设置

sql角色服务器的设置 内容精选换一换如果您需要对华为云上购买的DDM资源&#xff0c;为企业中的员工设置不同的访问权限&#xff0c;为达到不同员工之间的权限隔离&#xff0c;您可以使用统一身份认证服务(Identity and Access Management&#xff0c;简称IAM)进行精细的权限管…

MyBatis-Plus——字段类型处理器TypeHandler

字段类型处理器&#xff08;TypeHandler&#xff09; 1&#xff0c;准备工作 &#xff08;1&#xff09;MyBatis 中的 TypeHandler 类型处理器用于 JavaType 与 JdbcType 之间的转换&#xff0c;假设我们用户表中有一个联系方式字段&#xff0c;类型为字符串&#xff1a; &am…

额外参数_Pytorch获取模型参数情况的方法

分享人工智能技术干货&#xff0c;专注深度学习与计算机视觉领域&#xff01;相较于Tensorflow&#xff0c;Pytorch一开始就是以动态图构建神经网络图的&#xff0c;其获取模型参数的方法也比较容易&#xff0c;既可以根据其内建接口自己写代码获取模型参数情况&#xff0c;也可…

Mybatis-Plus之逻辑删除

概念 什么是逻辑删除 逻辑删除:假删除。将对应数据中代表是否被删除字段状态修改为“被删除状态”,之后在数据库中仍旧能看到此条数据记录。 数据库实现思路:插入数据时,标记为未删除状态;查询、修改时,只获取未删除状态的数据进行操作;删除时则更新删除状态为已删除…