当前位置: 首页 > news >正文

【再识C进阶2(中)】详细介绍指针的进阶——函数指针数组、回调函数、qsort函数

前言

💓作者简介: 加油,旭杏,目前大二,正在学习C++数据结构等👀
💓作者主页:加油,旭杏的主页👀

⏩本文收录在:再识C进阶的专栏👀

🚚代码仓库:旭日东升 1👀

🌹欢迎大家点赞 👍 收藏 ⭐ 加关注哦!💖💖

学习目标:

       在这一篇博客中,我们要认识并理解函数指针数组的概念,再学会在特定情境下使用函数指针数组;简单认识一下指向函数指针数组的指针;认识一下回调函数,并通过qsort函数来认识一下回调函数。这就是本博客的学习目标。


学习内容:

通过上面的学习目标,我们可以列出要学习的内容:

  1. 认识并理解函数指针数组的概念
  2. 学会在特定情境下使用函数指针数组
  3. 简单认识一下指向函数指针数组的指针
  4. 认识一下回调函数
  5. 通过qsort函数来认识一下回调函数

一、函数指针数组

1.1 函数指针

       我们先来简单回顾一下函数指针,我们在初始C语言中学过指针的初阶,我们认识了整形指针字符指针基本数据类型的指针整形指针是存放整形类型变量的地址字符指针是存放字符类型变量的地址……那么我们来看这个函数指针,明显是一个指针。通过小学学习的找规律进行编写句子,可以轻松地得到:函数指针是存放函数类型变量的地址(可能描述有些不正确)。看下图方便理解:

整形指针:存放整形类型变量的地址,32位平台下是4个字节,在64位平台下是8个字节

字符指针:存放字符类型变量的地址,32位平台下是4个字节,在64位平台下是8个字节

函数指针:存放函数类型变量的地址,32位平台下是4个字节,在64位平台下是8个字节

1.1.1 了解什么是函数的地址? 

啊,读者可能会感觉到有点奇怪!为什么函数也有地址呢?

       因为函数是由一些运行的语句组成的,程序运行的时候就会把函数中的语句调用到内存中去,那么函数代码在内存中开始的那个内存空间的地址就是函数的地址

接下来,让我们用代码来认识一下函数的地址:

void test()
{printf("Hellod,worid!");
}int main()
{printf("%p\n", test);   //函数与数组类似,数组名表示数组首元素的地址,函数名表示函数的地址printf("%p\n", &test);  //&函数名拿到的是函数的地址
}

1.1.2 学习如何使用函数指针? 

       在了解完函数指针是什么,可能大家还不知道什么是函数指针?书接上文,函数的地址要想保存下来,需要怎么保存呢?下面,我们来看代码:

void test()
{printf("Hellod,worid!");
}
//下面pfun1和pfun2哪个有能力存放test函数的地址?
void (*pfun1)();
void *pfun2();

       首先,能够用于存储地址,就要求pfun1或者pfun2是指针,看上面,pfun1先与 * 结合,说明pfun1是指针,去掉指针就是指针所指向的类型是:void (),返回值类型为void。

如何使用函数指针呢?下面来看代码:

void test()
{printf("Hellod,worid!\n");
}int main()
{void (*pf)() = &test; //用pf指针存储函数的地址(*pf)();pf();test();
}

1.2 利用计算器代码来具体介绍函数指针数组

1.2.1 函数指针数组是什么?

       在学习完函数指针后,我们来认识一下函数指针数组是什么?和介绍函数指针一样,函数指针数组的主语是数组,在初始C语言中,我们学过数组的内容中介绍了一些常见的数组类型:整形数组字符数组基本数据类型的数组整形数组是存放一些整形类型的变量字符数组是存放一些字符类型的变量……那么函数指针数组存放的值一些函数指针类型的变量。看下图,方便理解:

整形数组是存放一些整形类型的变量,数组存放的是想同类型的变量;

字符数组是存放一些字符类型的变量;

函数指针数组是存放一些函数指针类型的变量。

 下面用代码来认识一下函数指针数组:

int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int main()
{int (*pf1)(int, int) = &Add;int (*pf2)(int, int) = ⋐int (*pfarr[2])(int, int) = { pf1, pf2 };
}

1.2.2 利用画图对函数指针数组的写法详解:

1.2.3 计算器代码的实现

       在详细介绍分支与循环语句中,我们讲过像这种代码的设计结构,先有一个函数的主题、再有一个菜单、之后根据菜单的内容进行功能的设计,最后进行结束游戏。大致思路就是这,请读者继续跟着我的思路进行设计计算器:

1.2.3.1 计算器代码的主体

int main()
{int input = 0;do {menu();printf("请选择:>");scanf("%d", &input);switch (input){case 1:break;//……default:break;}} while (input);return 0;
}

 1.2.3.2 计算器的菜单

void menu()
{printf("***************************\n");printf("***** 1. Add   2. Sub *****\n");printf("***** 3. Mul   4. Div *****\n");printf("*****      0. exit    *****\n");printf("***************************\n");
}

1.2.3.3 计算器的自定义函数部分 

这个简单的计算器将实现四种计算的功能,分别是:加、减、乘、除

int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}

1.2.3.4 计算器函数主体的选择部分

switch (input){case 1:printf("请输入两个数:>");scanf("%d %d", &x, &y);ret = Add(x, y);printf("结果是:> %d\n", ret);break;case  2:printf("请输入两个数:>");scanf("%d %d", &x, &y);ret = Sub(x, y);printf("结果是:> %d\n", ret);break;case 3:printf("请输入两个数:>");scanf("%d %d", &x, &y);ret = Mul(x, y);printf("结果是:> %d\n", ret);break;case 4:printf("请输入两个数:>");scanf("%d %d", &x, &y);ret = Div(x, y);printf("结果是:> %d\n", ret);break;case 0:printf("退出计算器!\n");break;default:printf("选择错误,请重新选择!\n");break;}

1.2.3.5 这种计算器代码的不足之处

  1. 当计算器的功能逐渐增加的时候,菜单会越来越长,所需要写出的函数也会越来越多,最重要的是switch()语句会越来越长
  2. case语句中的代码存在冗余

1.2.4 对计算器代码进行改进

       这种改进使得在以后对计算器的功能进行升级的时候,会非常方便!因为你只需要将函数指针数组进行修改,以及input的范围进行修改即可。

do {menu();printf("请选择:>");scanf("%d", &input);int (*pfarr[])(int, int) = { NULL, &Add, &Sub, &Mul, &Div };if (0 == input){printf("退出计算器!\n");}else if (input >= 1 && input <= 4){printf("请输入两个数:>");scanf("%d %d", &x, &y);ret = (*pfarr[input])(x, y);printf("结果是:> %d\n", ret);}else{printf("选择错误,请重新选择!\n");}} while (input);

1.3 函数指针数组的用途 

函数指针数组的用途是:转义表

       正确使用函数指针数组的前提条件是,这若干个需要通过函数指针数组保存的函数必须有相同的输入、输出值

       函数指针数组的好处:只要少许行代码,就完成了许多条case语句要做的事,减少了编写代码时工作量,将nStreamType作为数组下标,直接调用函数指针,从代码执行效率上来说,也比case语句高。假如多个函数中均要作如此处理,函数指针数组更能体现出它的优势。

 二、指向函数指针数组的指针(了解)

       我们可以先从简单的入手,我现在有一堆整形变量,现在需要将这些整形变量的地址存储起来,那需要怎么进行存储呢?答用整形指针数组进行存储。如果我现在想拿到这个整形指针数组的地址,需要用什么来接收呢?答用指向整形指针数组的指针来接收

       同理,指向函数指针数组的指针是一个指针指针指向一个数组数组的元素都是函数指针。这就跟俄罗斯套娃一样,一层一层的,我们需要先找到主语,得知类型;继续找主语,得知类型……有点耐心,像剥洋葱一样,慢慢拨开!

下面,我们用代码来认识一下:

void test(const char* str)
{printf("%s\n", str);
}
int main()
{//函数指针pfvoid (*pf)(const char*) = test;//函数指针数组pfarrvoid (*pfarr[10])(const char*);pfarr[0] = test;//指向函数指针数组pfarr的指针ppfarrvoid (*(*ppfarr)[10])(const char*) = &pfarr;return 0;
}

扩展视野(没必要): 函数指针数组指针数组,函数指针数组指针数组指针

三、回调函数(重要)

       回调函数是非常重要的知识点,可以玩出许多高端操作,其依赖于函数指针,有了函数指针我们才能实现回调函数。下面来看一下回调函数的概念:

3.1 回调函数的概念

       回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用于调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用而是在特定的事件或条件发生时由另外一方调用的,用于对该事件或条件进行响应。

       简单来说,就是函数作为参数放在另一个函数的形参中,当另一函数在使用时,会调用作为参数的函数,那么这个作为参数的函数就是回调函数

接下来,我们用计算器来举个例子:

       在1.2.3中实现的计算器代码中,switch()语句中的代码有点冗余,如果我们将这些冗余的部分封装成一个函数,那么这个代码看上去就会好很多。但是冗余的这一部分有一点是不同的,就是所使用的函数,所以我们需要在封装函数的形参中传递功能函数的地址,使封装函数进行调用,这就是回调函数。下面看代码:

void cal(int (*pf)(int, int))
{printf("请输入两个数:>");scanf("%d %d", &x, &y);ret = (*pf)(x, y);printf("结果是:> %d\n", ret);
}

3.2 回调函数的案例:qsort函数 

       先来介绍一下qsort函数qsort函数:其是一个库函数,底层使用的快速排序的方式,对数据进行排序的;这个函数可以直接使用,这个函数可以用来排序任意类型的数据。 

在qsort函数中,有四个参数,我们来认识一下每一个参数所表示的意思:

  • void* base :待排序数组的第一个元素的地址;
  • size_t num:待排序数组的元素个数;
  • size_t size:待排序数组中一个元素的大小;
  • int (*compar) (const void*, const void*) :函数指针——cmp指向了一个函数,这个函数是用来比较两个元素的。

       排序就是有比较组成的,qsort函数可以排序不同类型的数据,但不是所有类型都可以用不等号比较出来,方法是有差异的。比如说,整形可以直接用>比较,而两个结构体的数据可能不能直接用>比较。

       在这个qsort函数中,最难的是第四个形参,这个形参所指向的函数的作用如下图:让两个数进行相减,根据结果与0进行比较判断谁在前,谁在后。

 

介绍一下void* 的指针

  1. void* 是无类型指针,可以接收任意类型的地址;
  2. 不能进行解引用操作;
  3. 不能进行加、减整数的操作。

3.3 qsort函数的使用

//排序整形数组
int int_cmp(const void* e1, const void* e2)
{return (*(int*)e1 - *(int*)e2);
}
int main()
{int arr[10] = { 3,4,5,6,7,8,9,2,1,0 };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), int_cmp);for (int i = 0; i < 10; i++){printf("%d ", arr[i]);}return0;
}
//排序结构体中名字的顺序
struct student {char name[40];int age;
};
int name_cmp(const void* p5, const void* p6)
{return strcmp(((struct student*)p5)->name, ((struct student*)p6)->name);
}
int main()
{struct student s[3] = { {"zhangsan", 23}, {"lisi", 45}, {"wangwu", 78} };int sz2 = 3;qsort(s, sz2, sizeof(s[0]), name_cmp);for (int i = 0; i < 3; i++){printf("%s\n", s[i].name);}return 0;
}
//排序结构体中年龄的大小
struct student {char name[40];int age;
};
int age_cmp(const void* p3, const void* p4)
{return ((struct student*)p3)->age - ((struct student*)p4)->age;
}
int main()
{qsort(s, sz2, sizeof(s[0]), age_cmp);for (int i = 0; i < 3; i++){printf("%d\n", s[i].age);}return 0;
}

3.4 回调函数的作用

  1. 恰当时间发送通知;
  2. 让代码更加灵活;
  3. 提高运行效率。

学习产出:

  1. 认识并理解函数指针数组的概念
  2. 学会在特定情境下使用函数指针数组
  3. 简单认识一下指向函数指针数组的指针
  4. 认识一下回调函数
  5. 通过qsort函数来认识一下回调函数

相关文章:

【再识C进阶2(中)】详细介绍指针的进阶——函数指针数组、回调函数、qsort函数

前言 &#x1f493;作者简介&#xff1a; 加油&#xff0c;旭杏&#xff0c;目前大二&#xff0c;正在学习C&#xff0c;数据结构等&#x1f440; &#x1f493;作者主页&#xff1a;加油&#xff0c;旭杏的主页&#x1f440; ⏩本文收录在&#xff1a;再识C进阶的专栏&#x1…...

PaddleOCR学习笔记3-通用识别服务

今天优化了下之前的初步识别服务的python代码和html代码。 采用flask paddleocr bootstrap快速搭建OCR识别服务。 代码结构如下&#xff1a; 模板页面代码文件如下&#xff1a; upload.html : <!DOCTYPE html> <html> <meta charset"utf-8"> …...

9.8 校招 实习 内推 面经

绿泡*泡&#xff1a; neituijunsir 交流裙 &#xff0c;内推/实习/校招汇总表格 1、校招 | 长安福特2024校园招聘正式启动 校招 | 长安福特2024校园招聘正式启动 2、2023校招总结--SLAM岗位 - 5 2023校招总结--SLAM岗位 - 5 3、校招&实习 | 格灵深瞳2024秋季校园招聘启…...

web前段与后端的区别优漫动游

要了解web前后端的区别&#xff0c;首先必须得清楚什么是web前端和web后端。 web前段与后端的区别 首先&#xff1a;web的本意是蜘蛛网和网的意思&#xff0c;在网页设计中我们称为网页的意思。现广泛译作网络、互联网等技术领域。表现为三种形式&#xff0c;即超文本(hyp…...

局域网ntp服务器设置(windows时间同步服务器NetTime)(ubuntu systemd-timesyncd ntp客户端)123端口、ntp校时

文章目录 背景windows如何配置ntp服务器手动配置配置参数AnnounceFlags和Enabled含义 使用软件配置&#xff08;NetTime&#xff09;实操相关疑问&#xff1a;0.nettime.pool.ntp.org是什么&#xff1f; 注意事项请务必检查windows主机123端口是否已被占用&#xff0c;方法请参…...

【个人博客系统网站】我的博客列表页 · 增删改我的博文 · 退出登录 · 博客详情页 · 多线程应用

【JavaEE】进阶 个人博客系统&#xff08;4&#xff09; 文章目录 【JavaEE】进阶 个人博客系统&#xff08;4&#xff09;1. 增加博文1.1 预期效果1.1 约定前后端交互接口1.2 后端代码1.3 前端代码1.4 测试 2. 我的博客列表页2.1 期待效果2.2 显示用户信息以及博客信息2.2.1…...

安全狗陈奋:数据安全需要建立在传统网络安全基础之上

8月22日-23日&#xff0c;由创业邦主办的“2023 DEMO WORLD 企业开放式创新大会”在上海顺利举行。 作为国内云原生安全领导厂商&#xff0c;安全狗受邀出席此次活动。 本次大会以“拥抱开放”为主题&#xff0c;聚焦开放式创新&#xff0c;通过演讲分享、专场对接、需求发布…...

【Redis】深入探索 Redis 的数据类型 —— 哈希表 hash

文章目录 前言一、hash 类型相关命令1.1 HSET 和 HSETNX1.2 HGET 和 HMGET1.3 HKEYS、HVALS 和 HGETALL1.4 HEXISTS 和 HDEL1.5 HLEN1.6 HINCRBY 和 HINCRBYFLOAT1.7 哈希相关命令总结 二、hash 类型内部编码三、hash 类型的应用场景四、原生&#xff0c;序列化&#xff0c;哈希…...

网络安全应急响应典型案例-(DDOS类、僵尸网络类、数据泄露类)

一、DDOS类事件典型案例 DDOS攻击&#xff0c;即分布式拒绝服务攻击&#xff0c;其目的在于使目标电脑的网络或系统资源耗尽&#xff0c;使服务暂时中断或停止&#xff0c;导致其正常用户无法访问。CC攻击使用代理服务器向受害服务器发送大量貌似合法的请求&#xff08;通常…...

【测试开发】Mq消息重复如何测试?

本篇文章主要讲述重复消费的原因&#xff0c;以及如何去测试这个场景&#xff0c;最后也会告诉大家&#xff0c;目前互联网项目关于如何避免重复消费的解决方案。 Mq为什么会有重复消费的问题? Mq 常见的缺点之一就是消息重复消费问题&#xff0c;产生这种问题的原因是什么呢…...

C++和C#程序语言的区别

一直学习C++和C#,两者之间的区别总结一下 目录 一、两种语言概述 C++语言 C#语言 二、两种语言对比 2.1运行依赖...

CentOS配置Java环境报错-bash: /usr/local/jdk1.8.0_381/bin/java: 无法执行二进制文件

CentOS配置Java环境后执行java -version时报错&#xff1a; -bash: /usr/local/jdk1.8.0_381/bin/java: 无法执行二进制文件原因是所使用的jdk的版本和Linux内核架构匹配不上 使用以下命令查看Linux架构&#xff1a; [rootlocalhost ~]# cat /proc/version Linux version 3.1…...

MySQL进阶 —— 超详细操作演示!!!(上)

MySQL进阶 —— 超详细操作演示&#xff01;&#xff01;&#xff01;&#xff08;上&#xff09; 一、存储引擎1.1 MySQL 体系结构1.2 存储引擎介绍1.3 存储引擎特点1.4 存储引擎选择 二、索引2.1 索引概述2.2 索引结构2.3 索引分类2.4 索引语法2.5 SQL 性能分析2.6 索引使用2…...

一条爬虫抓取一个小网站所有数据

一条爬虫抓取一个小网站所有数据 ​ 今天闲来无事&#xff0c;写一个爬虫来玩玩。在网上冲浪的时候发现了一个搞笑的段子网&#xff0c;发现里面的内容还是比较有意思的&#xff0c;于是心血来潮&#xff0c;就想着能不能写一个Python程序&#xff0c;抓取几条数据下来看看&am…...

八大排序——快速排序

Hello&#xff0c;大家好&#xff0c;今天分享的八大排序里的快速排序&#xff0c;所谓快速排序是一个叫霍尔的人发明&#xff0c;有很多人可能会觉得为什么不叫霍尔排序&#xff0c;其中原因就是因为它快&#xff0c;快速则体现了它的特点&#xff0c;今天我们就来讲一下快速排…...

【ES】笔记-Class类剖析

Class Class介绍与初体验ES5 通过构造函数实例化对象ES6 通过Class中的constructor实列化对象 Class 静态成员实例对象与函数对象的属性不相通实例对象与函数对象原型上的属性是相通的Class中对于static 标注的对象和方法不属于实列对象&#xff0c;属于类。 ES5构造函数继承Cl…...

数学建模--Seaborn库绘图基础的Python实现

目录 1.绘图数据导入 2. sns.scatterplot绘制散点图 3.sns.barplot绘制条形图 4.sns.lineplot绘制线性图 5.sns.heatmap绘制热力图 6.sns.distplot绘制直方图 7.sns.pairplot绘制散图 8.sns.catplot绘制直方图 9.sns.countplot绘制直方图 10.sns.lmplot绘回归图 1.绘图数…...

lv3 嵌入式开发-2 linux软件包管理

目录 1 软件包管理 1.1流行的软件包管理机制 1.2软件包的类型 1.3软件包的命名 2 在线软件包管理 2.1APT工作原理 2.2更新软件源 2.3APT相关命令 3 离线软件包管理 1 软件包管理 1.1流行的软件包管理机制 Debian Linux首先提出“软件包”的管理机制---Deb软件包 …...

智能小区与无线网络技术

1&#xff0e;1 智能小区 智能小区指的是具有小区智能化系统的小区。所谓小区智能化系统&#xff0c;指的是在 现代计算机网络和通信技术的基础上&#xff0c;将传统的土木建筑技术与计算机技术、自动 控制技术、通信与信息处理技术、多媒体技术等先进技术相结合的自动化和综…...

如何传输文件流给前端

通过链接下载图片&#xff0c;直接http请求然后将文件流返回 注&#xff1a;music.ly是一个下载tiktok视频的免费接口 https://api19-core-c-useast1a.musical.ly/aweme/v1/feed/?aweme_idxxx func (m *FileBiz) DownloadFileV2(ctx *ctrl.Context, fileLink, fileName strin…...

Spring Security OAuth2 远程命令执行漏洞

文章目录 一、搭建环境二、漏洞验证三、准备payload四、执行payload五、变形payload 一、搭建环境 cd vulhub/spring/CVE-2016-4977/ docker-compose up -d 二、漏洞验证 访问 http://192.168.10.171:8080/oauth/authorize?response_type${233*233}&client_idacme&s…...

Python之并发编程介绍

一、并发编程介绍 1.1、串行、并行与并发的区别 串行(serial)&#xff1a;一个CPU上&#xff0c;按顺序完成多个任务并行(parallelism)&#xff1a;指的是任务数小于等于cpu核数&#xff0c;即任务真的是一起执行的并发(concurrency)&#xff1a;一个CPU采用时间片管理方式&am…...

GO语言网络编程(并发编程)并发介绍,Goroutine

GO语言网络编程&#xff08;并发编程&#xff09;并发介绍&#xff0c;Goroutine 1、并发介绍 进程和线程 A. 进程是程序在操作系统中的一次执行过程&#xff0c;系统进行资源分配和调度的一个独立单位。 B. 线程是进程的一个执行实体,是CPU调度和分派的基本单位,它是比进程更…...

英语连词总结

前言 总结一些常用的英语连词&#xff0c;以下用法只是我希望我自己这么用。分类我可能分的不好&#xff0c;慢慢积累&#xff0c;慢慢改进。 1&#xff09;表递进: firstly、secondly、thirdly、finally、af first、at the beginning、in the end、to begin with&#xff0…...

LeetCode 92. Reverse Linked List II【链表,头插法】中等

本文属于「征服LeetCode」系列文章之一&#xff0c;这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁&#xff0c;本系列将至少持续到刷完所有无锁题之日为止&#xff1b;由于LeetCode还在不断地创建新题&#xff0c;本系列的终止日期可能是永远。在这一系列刷题文章…...

【图论】Floyd

算法提高课笔记&#xff09; 文章目录 例题牛的旅行题意思路代码 排序题意思路代码 观光之旅题意思路代码 例题 牛的旅行 原题链接 农民John的农场里有很多牧区&#xff0c;有的路径连接一些特定的牧区。 一片所有连通的牧区称为一个牧场。 但是就目前而言&#xff0c;你…...

SpringCloudAlibaba Gateway(三)-整合Sentinel功能路由维度、API维度进行流控

Gateway整合Sentinel ​ 前面使用过Sentinel组件对服务提供者、服务消费者进行流控、限流等操作。除此之外&#xff0c;Sentinel还支持对Gateway、Zuul等主流网关进行限流。 ​ 自sentinel1.6.0版开始&#xff0c;Sentinel提供了Gateway的适配模块&#xff0c;能针对路由(rou…...

【笔试强训选择题】Day38.习题(错题)解析

作者简介&#xff1a;大家好&#xff0c;我是未央&#xff1b; 博客首页&#xff1a;未央.303 系列专栏&#xff1a;笔试强训选择题 每日一句&#xff1a;人的一生&#xff0c;可以有所作为的时机只有一次&#xff0c;那就是现在&#xff01;&#xff01; 文章目录 前言一、Day…...

DAY08_MyBatisPlus——入门案例标准数据层开发CRUD-Lombok-分页功能DQL编程控制DML编程控制乐观锁快速开发-代码生成器

目录 一 MyBatisPlus简介1. 入门案例问题导入1.1 SpringBoot整合MyBatisPlus入门程序①&#xff1a;创建新模块&#xff0c;选择Spring初始化&#xff0c;并配置模块相关基础信息②&#xff1a;选择当前模块需要使用的技术集&#xff08;仅保留JDBC&#xff09;③&#xff1a;手…...

分光棱镜BS、PB、NPBS的区别

BS&#xff08;分光棱镜&#xff09;&#xff1a;对入射偏振敏感&#xff0c;线偏振角度会影响分光比。若入射的是自然光或圆偏振光&#xff0c;则按50&#xff1a;50分光。分束的时候只管分能量&#xff0c;理想器件下出射的两路光偏振态还是原来的样子&#xff0c;实际工艺缺…...