指针:这块地方是我的了!

原创 云深之无迹 2025-01-05 11:25

内存里面都是01,这种东西。那如何定位?就是使用指针!

就好像这样

我们又知道,一个文件其实是大量的01构成的。那我们把这堆01翻译成什么文件或者内容全靠我们自己说了算。

指针是一个变量,专门用来存储另一个变量的内存地址。就是一指一个地方,这个地方就有产生一点信息,要用变量来放。

指针具有类型信息(例如 int*、char*、float* 等),表示指向的内存地址存储的数据类型。
也就是说我们说这地址代表的是什么东西,全靠我们自己的定义,也就是类型信息。

我可以说是这些01代表张, 也可以说代表256

指针类型决定了对该地址进行解引用(即访问指针指向的值)时的操作方式。
有了类型信息我们才可以进行操作。
int* ptr;  // 声明一个指向整数类型的指针变量
  1. int* 表示这是一个指向 int 类型变量的指针。
  2. ptr 是指针变量的名字。
*纯粹就是标志。
如何获取?
int a = 10;int* ptr = &a;  // ptr 指向变量 a 的地址

使用 & 操作符可以获得一个变量的地址,然后把这个地址给放到一个指针变量上面。

使用 * 操作符可以访问指针所指向的地址存储的值:

printf("%d\n", *ptr);  // 输出 a 的值:10*ptr = 20;             // 修改指针指向地址的值,a 的值变为 20

我们也可以在地址上面做修改,相当于最底层的操作了

指针未初始化,或指向了被释放的内存,会成为野指针。

int* ptr;  // 未初始化的指针是野指针

好问题,就是你指了一堆01,但是没说这东西到底代表什么

const int a = 10;const int* ptr = &a;*ptr = 20;  // 错误,不能修改常量

指向常量的指针:指针指向的值不可修改

int a = 10, b = 20;int* const ptr = &a;  // 指针本身是常量ptr = &b;             // 错误,不能修改 ptr 的值

常量指针:指针本身的地址不可修改

const就是一个标志,说动不了。就看这个东西在哪里,就修饰什么。

首先是给一片地址起了名字,叫a,然后具体的值是10,它的类型是int,接着我们使用&,取了这个a的地址,存在了在ptr的地址上面,然后也说明了,ptr指的01是int类型,const保护了这个指针。

int* arr = malloc(10 * sizeof(int));  // 动态分配内存free(arr);                            // 释放内存

指向动态分配内存或数组的指针

使用场景:

int* ptr = (int*)malloc(sizeof(int));*ptr = 42;free(ptr);

内存分配

先使用malloc搞了一块内存,就后面括号这么大。然后使用强制类型转换为int类型的指针储存区。接着等号前面是最一开始的起始位置。

int arr[5] = {1, 2, 3, 4, 5};int* ptr = arr;  // 等价于 int* ptr = &arr[0];printf("%d\n", *(ptr + 2));  // 输出 arr[2] 的值:3

数组名可以退化为指针,表示数组的起始地址

void increment(int* p) {    (*p)++;}
int main() { int a = 10; increment(&a); printf("%d\n", a); // 输出 11 return 0;}

通过指针实现参数的引用传递,允许在函数内部修改外部变量的值,因为指针就是在一块内存区域做操作。

int add(int a, int b) { return a + b; }int (*func_ptr)(int, int) = add;printf("%d\n", func_ptr(3, 4));  // 输出 7

函数的地址也可以存储在指针中,用于回调函数或动态调用

有一些注意事项:

if (ptr != NULL) {    *ptr = 10;}

使用指针前,检查是否为 NULL

int arr[5] = {1, 2, 3, 4, 5};int* ptr = arr;ptr = ptr + 1;  // 移动到下一个元素printf("%d\n", *ptr);  // 输出 2

指针支持加减运算,但必须确保指针不越界

因为我们就是在内存区域里面做操作,你定义在里面的数据类型是有固定的大小的,当然可以靠加减地址在定位。

整点高级的:

函数参数可以是指针指针参数可以指向任何类型的变量(包括基本数据类型、结构体、数组、函数等)。

使用指针作为函数参数允许函数直接操作传递给它的变量,而无需返回任何值。

好理解吧?就是给了这块内存一个机器,直接在上面操作了,不需要复制一个新的地方在操作。

当函数参数是指向基本数据类型的指针时,函数可以修改传递给它的变量的值。这通常用于模拟“传值传引用”的机制,即通过指针将数据传递给函数,从而使函数能够直接修改原始数据。

#include 
void modifyValue(int* ptr) { *ptr = 20; // 修改指针指向的值}
int main() { int x = 10; printf("Before: %d\n", x); modifyValue(&x); // 传递x的地址给函数 printf("After: %d\n", x); // 输出20,说明x的值被修改 return 0;}

在这个例子中,modifyValue 函数接收一个指向 int 类型的指针(int* ptr)。通过解引用该指针(*ptr),函数修改了传递给它的变量 x 的值。

你x=10是一个值,也有一块地址

我指针来了,直接在这里进行处理,变成了20

接下来看个大戏!

这种函数

传了个结构体进来

指针也可以指向结构体,这允许函数操作结构体的成员,而不需要复制整个结构体。通过指针传递结构体,能够节省内存和提高效率,尤其是在结构体较大时。

是这样的,一个函数的参数在调用的时候,是需要把参数复制一遍的,如果这个参数本身就复合类型,那确实很大,如果就是传个地址进来就很小了。

#include 
struct Person { char name[50]; int age;};
void updateAge(struct Person* p) { p->age = 30; // 修改结构体指针指向的成员}
int main() { struct Person person = {"Alice", 25}; printf("Before: %s, Age: %d\n", person.name, person.age); updateAge(&person); // 传递结构体的地址给函数 printf("After: %s, Age: %d\n", person.name, person.age); // 输出修改后的年龄 return 0;}

在这个例子中,updateAge 函数接收一个指向 struct Person 的指针,并通过 p->age 修改结构体的 age 成员。-> 运算符用于通过指针访问结构体的成员。

#include 
void modifyArray(int* arr, int size) { for (int i = 0; i < size; i++) { arr[i] = arr[i] * 2; // 修改数组中的元素 }}
int main() { int arr[] = {1, 2, 3, 4, 5}; int size = sizeof(arr) / sizeof(arr[0]); printf("Before modification:\n"); for (int i = 0; i < size; i++) { printf("%d ", arr[i]); } printf("\n");
modifyArray(arr, size); // 传递数组的指针 printf("After modification:\n"); for (int i = 0; i < size; i++) { printf("%d ", arr[i]); // 数组元素被修改 } printf("\n"); return 0;}

在C语言中,数组名本身就是指向数组首元素的指针,因此可以将数组的指针作为函数参数传递。

通过数组指针,函数可以访问和修改数组中的元素。

modifyArray 函数接收一个指向整数数组的指针(int* arr)。通过该指针,函数可以访问并修改数组的每个元素。这个也好理解,因为数组就是一块连续的内存,在上面做加减移动当然可以。

#include 
void greet() { printf("Hello, world!\n");}
void farewell() { printf("Goodbye, world!\n");}
void execute(void (*func)()) { func(); // 调用通过函数指针传递的函数}
int main() { execute(greet); // 传递greet函数的指针 execute(farewell); // 传递farewell函数的指针 return 0;}

指针也可以指向函数,这允许将函数作为参数传递,甚至在运行时决定调用哪个函数。

这个眼熟不?

你看看这是什么?这就是回调

execute 函数接收一个指向无返回值且无参数的函数的指针,并根据传递的函数指针调用相应的函数。

那上面这些优点就是:

  1. 修改原始数据:使用指针可以让函数修改传入参数的值,而不仅仅是操作参数的副本。
  2. 提高效率:传递指针而不是大块数据(如大数组或结构体)可以避免复制数据,从而提高程序效率。
聊聊引用传递怎么样?
在C语言中,引用传递并不像C++那样有直接的引用类型(&)来实现,但是我们可以通过使用指针来模拟引用传递的效果。
指针作为函数参数时,允许函数操作传递给它的变量的内存地址,从而实现对原始数据的修改。

引用传递意味着传递的是变量本身的地址而不是变量的副本。这样,在函数内部修改参数的值会直接影响到调用该函数时传递的变量。这与值传递(按值传递)不同,值传递会复制参数的值,函数内部修改参数不会影响外部变量。

虽然C语言没有直接的引用类型,但可以通过传递指针来模拟引用传递。通过传递指向变量的指针,函数可以修改变量的值,而不是仅操作其副本。

#include 
// 通过指针模拟引用传递void modifyValue(int* ptr) { *ptr = 20; // 修改指针指向的值,相当于修改原始变量}
int main() { int x = 10; printf("Before: %d\n", x); // 输出 10 modifyValue(&x); // 传递 x 的地址 printf("After: %d\n", x); // 输出 20,说明 x 的值被修改 return 0;}

modifyValue 函数接受一个指向 int 类型的指针。

我们在调用 modifyValue 时传递了 x 的地址(&x),函数内部通过解引用该指针(*ptr)修改了 x 的值。因此,x 的值从 10 被修改成了 20,这就是通过指针实现的引用传递。其实还是上面例子更进一步的解读啦。

肯定这样做是有优点的!

  1. 修改原始数据:使用引用传递,函数可以直接修改传递给它的变量的值,而不仅仅是副本。这样可以在函数中执行更复杂的操作。

  2. 节省内存:传递指针而不是整个数据(如大结构体或大数组)可以减少内存的使用和数据的复制,尤其在处理大对象时,传递指针效率较高

  3. 提高性能:通过引用传递,避免了复制数据的开销,特别是在处理大量数据时,性能上有显著提升。

其实还有很多内容,但是不想写了,不过可以再研究一个内容!

指针是用于存储变量的内存地址的变量,这个内存地址就是变量在地址中的第一个位置吗?

内存地址并不表示“变量的第一个位置”,而是变量实际存储的位置

这个是答案,你说对了吗?

你先要知道什么是内存地址!

  1. 内存地址是指计算机内存中某个特定位置的地址,它是一个数字,表示存储数据的位置。

  2. 变量在内存中的存储方式取决于它的数据类型。例如,一个 int 类型的变量通常占用 4 字节(32 位系统上),而一个 char 类型的变量通常占用 1 字节。


当你声明一个变量时,操作系统为该变量分配一定的内存空间,变量的内存地址就是该空间的起始地址。

int a = 10;int* p = &a;
a 是一个 int 类型的变量,它存储在内存的某个位置(比如假设 a 被分配在地址 0x100)。
&a 返回的是 a 变量在内存中的地址,也就是指向 a 的内存地址,比如 0x100。
p 是一个指针变量,存储了 a 的内存地址(即 p = &a)。
但是我上面的问题其实是对了一半,至于为什么这样说,继续看!
  1. 对于简单类型(如 intchar 等),变量在内存中的位置是固定的。它在内存中占据从某个起始地址开始的一段连续空间。指针存储的是这段空间的起始地址。

  2. 对于数组等复杂类型,数组的内存地址是数组第一个元素的内存地址,但数组的所有元素是连续存储的。所以,数组的内存地址实际上就是数组第一个元素的地址。


int arr[3] = {1, 2, 3};int* p = arr;  // arr 是数组名,p 存储的是 arr[0] 的内存地址
数组 arr 中的元素是连续存储的。
数组 arr[0]、arr[1]、arr[2] 在内存中是顺序排列的,arr 本身是数组首元素 arr[0] 的地址。
p = arr 将指针 p 指向了数组 arr 的第一个元素 arr[0] 的地址。
指针存储的是变量或数组等数据结构的内存地址
内存地址表示变量或数组的起始位置
对于基本数据类型,内存地址是该变量在内存中的存储位置;而对于数组、结构体等数据结构,指针指向的是该数据结构的起始位置,元素或成员通常是连续存储的。
我觉得你一定学懂了!

再看看这个

这两行代码是 C 语言中的 函数指针类型定义(typedef)。

先看第一个函数:

这行代码定义了一个函数指针类型,该指针可以指向一个不接受任何参数并且没有返回值的函数。

  1. void:表示该函数没有返回值。

  2. (*func_ptr_t):这是函数指针的声明方式。它表示 func_ptr_t 是一个指向函数的指针。

  3. (void):表示该函数没有参数。即指向的函数不接受任何参数。

func_ptr_t 是一个新的类型名,它代表了一个指向无返回值、无参数的函数的指针类型。
使用 func_ptr_t 时,可以声明一个指向此类型函数的指针,并将其指向一个实际的符合此类型签名的函数。
void myFunction(void) {    // 函数实现}
int main() { func_ptr_t ptr = myFunction; // 使用 typedef 定义的指针类型 ptr(); // 调用 myFunction return 0;}

再看一个

这行代码定义了另一个函数指针类型,该指针可以指向一个接受一个 uint8_t 类型参数并且没有返回值的函数。
void:表示该函数没有返回值。
(*func_ptr_arg1_t):表示 func_ptr_arg1_t 是一个指向函数的指针。
(uint8_t u8Param):表示该函数接受一个类型为 uint8_t 的参数(通常 uint8_t 是无符号 8 位整数,即 unsigned char)。
func_ptr_arg1_t 是一个新的类型名,它代表了一个指向接受一个 uint8_t 类型参数且无返回值的函数的指针类型。
使用 func_ptr_arg1_t 时,可以声明一个指向此类型函数的指针,并将其指向一个符合此类型签名的函数。
void myFunctionWithParam(uint8_t param) {    // 函数实现    printf("Received parameter: %d\n", param);}
int main() { func_ptr_arg1_t ptr = myFunctionWithParam; // 使用 typedef 定义的指针类型 ptr(10); // 调用 myFunctionWithParam 并传递参数 10 return 0;}

那知道了这个东西,有什么用呢?

函数指针常被用作回调函数,让用户可以指定某些行为。在事件驱动编程中,程序在特定事件发生时调用用户定义的函数,而用户函数的地址通过函数指针传递。

#include 
void onEventCallback(void) { printf("Event triggered!\n");}
void registerCallback(func_ptr_t callback) { callback(); // 调用传递进来的函数}
int main() { registerCallback(onEventCallback); // 注册回调 return 0;}
这里 func_ptr_t 被用来注册一个回调函数,当事件发生时触发调用。可以解耦调用者和被调用者,提高灵活性。
使用函数指针作为参数或回调,可以极大地提高模块化设计的灵活性。设计通用的库函数时,可以将实现的具体细节委托给用户提供的函数。
#include 
void defaultHandler(uint8_t param) { printf("Default handler: %d\n", param);}
void runWithCustomHandler(func_ptr_arg1_t handler, uint8_t param) { if (handler) { handler(param); // 调用用户提供的处理函数 } else { defaultHandler(param); // 使用默认处理函数 }}
int main() { runWithCustomHandler(NULL, 5); // 使用默认处理函数 runWithCustomHandler(defaultHandler, 10); // 使用用户提供的函数 return 0;}

这里通过函数指针允许用户自定义处理逻辑。

void (*callback)(void);void (*handler)(uint8_t);

如果每次使用函数指针都显式声明。当函数指针的使用场景多时,会显得繁琐且不清晰。

通过 typedef 起别名,简化函数指针的声明和使用,让代码更易读。

评论
  • 日前,商务部等部门办公厅印发《手机、平板、智能手表(手环)购新补贴实施方案》明确,个人消费者购买手机、平板、智能手表(手环)3类数码产品(单件销售价格不超过6000元),可享受购新补贴。每人每类可补贴1件,每件补贴比例为减去生产、流通环节及移动运营商所有优惠后最终销售价格的15%,每件最高不超过500元。目前,京东已经做好了承接手机、平板等数码产品国补优惠的落地准备工作,未来随着各省市关于手机、平板等品类的国补开启,京东将第一时间率先上线,满足消费者的换新升级需求。为保障国补的真实有效发放,基于
    华尔街科技眼 2025-01-17 10:44 211浏览
  • 在物联网(IoT)短距无线通信生态系统中,低功耗蓝牙(BLE)数据透传是一种无需任何网络或基础设施即可完成双向通信的技术。其主要通过简单操作串口的方式进行无线数据传输,最高能满足2Mbps的数据传输速率,可轻松实现设备之间的快速数据同步和实时交互,例如传输传感器数据、低采样率音频/图像与控制指令等。低功耗蓝牙(BLE)数据透传解决方案组网图具体而言,BLE透传技术是一种采用蓝牙通信协议在设备之间实现数据透明传输的技术,设备在通信时会互相验证身份和安全密钥,具有较高的安全性。在不对MCU传输数据进
    华普微HOPERF 2025-01-21 14:20 58浏览
  • 随着消费者对汽车驾乘体验的要求不断攀升,汽车照明系统作为确保道路安全、提升驾驶体验以及实现车辆与环境交互的重要组成,日益受到业界的高度重视。近日,2024 DVN(上海)国际汽车照明研讨会圆满落幕。作为照明与传感创新的全球领导者,艾迈斯欧司朗受邀参与主题演讲,并现场展示了其多项前沿技术。本届研讨会汇聚来自全球各地400余名汽车、照明、光源及Tier 2供应商的专业人士及专家共聚一堂。在研讨会第一环节中,艾迈斯欧司朗系统解决方案工程副总裁 Joachim Reill以深厚的专业素养,主持该环节多位
    艾迈斯欧司朗 2025-01-16 20:51 166浏览
  • 高速先生成员--黄刚这不马上就要过年了嘛,高速先生就不打算给大家上难度了,整一篇简单但很实用的文章给大伙瞧瞧好了。相信这个标题一出来,尤其对于PCB设计工程师来说,心就立马凉了半截。他们辛辛苦苦进行PCB的过孔设计,高速先生居然说设计多大的过孔他们不关心!另外估计这时候就跳出很多“挑刺”的粉丝了哈,因为翻看很多以往的文章,高速先生都表达了过孔孔径对高速性能的影响是很大的哦!咋滴,今天居然说孔径不关心了?别,别急哈,听高速先生在这篇文章中娓娓道来。首先还是要对各位设计工程师的设计表示肯定,毕竟像我
    一博科技 2025-01-21 16:17 60浏览
  •  光伏及击穿,都可视之为 复合的逆过程,但是,复合、光伏与击穿,不单是进程的方向相反,偏置状态也不一样,复合的工况,是正偏,光伏是零偏,击穿与漂移则是反偏,光伏的能源是外来的,而击穿消耗的是结区自身和电源的能量,漂移的载流子是 客席载流子,须借外延层才能引入,客席载流子 不受反偏PN结的空乏区阻碍,能漂不能漂,只取决于反偏PN结是否处于外延层的「射程」范围,而穿通的成因,则是因耗尽层的过度扩张,致使跟 端子、外延层或其他空乏区 碰触,当耗尽层融通,耐压 (反向阻断能力) 即告彻底丧失,
    MrCU204 2025-01-17 11:30 160浏览
  • 80,000人到访的国际大展上,艾迈斯欧司朗有哪些亮点?感未来,光无限。近日,在慕尼黑electronica 2024现场,ams OSRAM通过多款创新DEMO展示,以及数场前瞻洞察分享,全面展示自身融合传感器、发射器及集成电路技术,精准捕捉并呈现环境信息的卓越能力。同时,ams OSRAM通过展会期间与客户、用户等行业人士,以及媒体朋友的深度交流,向业界传达其以光电技术为笔、以创新为墨,书写智能未来的深度思考。electronica 2024electronica 2024构建了一个高度国际
    艾迈斯欧司朗 2025-01-16 20:45 227浏览
  • 2024年是很平淡的一年,能保住饭碗就是万幸了,公司业绩不好,跳槽又不敢跳,还有一个原因就是老板对我们这些员工还是很好的,碍于人情也不能在公司困难时去雪上加霜。在工作其间遇到的大问题没有,小问题还是有不少,这里就举一两个来说一下。第一个就是,先看下下面的这个封装,你能猜出它的引脚间距是多少吗?这种排线座比较常规的是0.6mm间距(即排线是0.3mm间距)的,而这个规格也是我们用得最多的,所以我们按惯性思维来看的话,就会认为这个座子就是0.6mm间距的,这样往往就不会去细看规格书了,所以这次的运气
    wuliangu 2025-01-21 00:15 94浏览
  • 本文介绍瑞芯微开发板/主板Android配置APK默认开启性能模式方法,开启性能模式后,APK的CPU使用优先级会有所提高。触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。源码修改修改源码根目录下文件device/rockchip/rk3562/package_performance.xml并添加以下内容,注意"+"号为添加内容,"com.tencent.mm"为AP
    Industio_触觉智能 2025-01-17 14:09 130浏览
  • 现在为止,我们已经完成了Purple Pi OH主板的串口调试和部分配件的连接,接下来,让我们趁热打铁,完成剩余配件的连接!注:配件连接前请断开主板所有供电,避免敏感电路损坏!1.1 耳机接口主板有一路OTMP 标准四节耳机座J6,具备进行音频输出及录音功能,接入耳机后声音将优先从耳机输出,如下图所示:1.21.2 相机接口MIPI CSI 接口如上图所示,支持OV5648 和OV8858 摄像头模组。接入摄像头模组后,使用系统相机软件打开相机拍照和录像,如下图所示:1.3 以太网接口主板有一路
    Industio_触觉智能 2025-01-20 11:04 125浏览
  •  万万没想到!科幻电影中的人形机器人,正在一步步走进我们人类的日常生活中来了。1月17日,乐聚将第100台全尺寸人形机器人交付北汽越野车,再次吹响了人形机器人疯狂进厂打工的号角。无独有尔,银河通用机器人作为一家成立不到两年时间的创业公司,在短短一年多时间内推出革命性的第一代产品Galbot G1,这是一款轮式、双臂、身体可折叠的人形机器人,得到了美团战投、经纬创投、IDG资本等众多投资方的认可。作为一家成立仅仅只有两年多时间的企业,智元机器人也把机器人从梦想带进了现实。2024年8月1
    刘旷 2025-01-21 11:15 53浏览
  • 数字隔离芯片是一种实现电气隔离功能的集成电路,在工业自动化、汽车电子、光伏储能与电力通信等领域的电气系统中发挥着至关重要的作用。其不仅可令高、低压系统之间相互独立,提高低压系统的抗干扰能力,同时还可确保高、低压系统之间的安全交互,使系统稳定工作,并避免操作者遭受来自高压系统的电击伤害。典型数字隔离芯片的简化原理图值得一提的是,数字隔离芯片历经多年发展,其应用范围已十分广泛,凡涉及到在高、低压系统之间进行信号传输的场景中基本都需要应用到此种芯片。那么,电气工程师在进行电路设计时到底该如何评估选择一
    华普微HOPERF 2025-01-20 16:50 38浏览
  • Ubuntu20.04默认情况下为root账号自动登录,本文介绍如何取消root账号自动登录,改为通过输入账号密码登录,使用触觉智能EVB3568鸿蒙开发板演示,搭载瑞芯微RK3568,四核A55处理器,主频2.0Ghz,1T算力NPU;支持OpenHarmony5.0及Linux、Android等操作系统,接口丰富,开发评估快人一步!添加新账号1、使用adduser命令来添加新用户,用户名以industio为例,系统会提示设置密码以及其他信息,您可以根据需要填写或跳过,命令如下:root@id
    Industio_触觉智能 2025-01-17 14:14 91浏览
  • 电竞鼠标应用环境与客户需求电竞行业近年来发展迅速,「鼠标延迟」已成为决定游戏体验与比赛结果的关键因素。从技术角度来看,传统鼠标的延迟大约为20毫秒,入门级电竞鼠标通常为5毫秒,而高阶电竞鼠标的延迟可降低至仅2毫秒。这些差异看似微小,但在竞技激烈的游戏中,尤其在对反应和速度要求极高的场景中,每一毫秒的优化都可能带来致胜的优势。电竞比赛的普及促使玩家更加渴望降低鼠标延迟以提升竞技表现。他们希望通过精确的测试,了解不同操作系统与设定对延迟的具体影响,并寻求最佳配置方案来获得竞技优势。这样的需求推动市场
    百佳泰测试实验室 2025-01-16 15:45 328浏览
  • 嘿,咱来聊聊RISC-V MCU技术哈。 这RISC-V MCU技术呢,简单来说就是基于一个叫RISC-V的指令集架构做出的微控制器技术。RISC-V这个啊,2010年的时候,是加州大学伯克利分校的研究团队弄出来的,目的就是想搞个新的、开放的指令集架构,能跟上现代计算的需要。到了2015年,专门成立了个RISC-V基金会,让这个架构更标准,也更好地推广开了。这几年啊,这个RISC-V的生态系统发展得可快了,好多公司和机构都加入了RISC-V International,还推出了不少RISC-V
    丙丁先生 2025-01-21 12:10 63浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦