【MySQL篇】 MySQL基础学习

文章目录

  • 前言
  • 基础
    • 数据类型
    • DDL数据库操作
      • 查询数据库
      • 创建数据库
      • 删除数据库
      • 使用数据库
    • DDL表操作
      • 创建表
      • 查询表
      • 修改表
      • 删除
    • DML-增删改
      • 添加数据
      • 更改数据
      • 删除数据
    • DQL-查询
      • 基础查询
      • 条件查询
      • 聚合函数
      • 分组查询
      • 排序查询
      • 分页查询
      • 编写顺序
    • DML-用户及权限
      • 用户管理
      • 权限控制
  • 函数
    • 字符串函数
    • 数值函数
    • 日期函数
    • 流程函数
  • 约束
    • 概述
    • 外键约束
    • 外键更新、删除时的操作
  • 多表查询
    • 概述
    • 连接查询
      • 内连接
      • 外连接
      • 自连接
    • 联合查询
    • 子查询
      • 标量子查询
      • 列子查询
      • 行子查询
      • 表子查询
  • 事务
    • 基础
    • 操作
    • 四大特性
    • 并发事务问题

前言

在做项目的过程中,深深感悟到一个扎实的MySQL基础是多么重要,这篇文章就用来记录一下我对于MySQL的复习,下面开始介绍了。

MySQL 是一款开源免费的数据库。后端程序员的工作离不开与数据库打交道,MySQL因其简单易懂、使用方便的特性深受Java程序员喜爱,这篇文章就让我们一起来了解MySQL。

此外,在这里我非常推荐大家可以去看看鱼皮大佬的MySQL练习网站SQL之母 - SQL自学网站 (yupi.icu),大家可以在上面检验一下自己编写sql语句的水平。

基础

数据类型

数值类型:

在这里插入图片描述

字符串类型:

在这里插入图片描述

日期类型:

在这里插入图片描述

DDL数据库操作

查询数据库

查询所有数据库

show databases;

查询当前数据库

select database();

创建数据库

create database 数据库名;

如果不存在该数据库则创建,存在则创建失败:

create database if not exists 数据库名;

删除数据库

drop database 数据库名;

如果存在该数据库则删除,不存在则删除失败:

drop database if exists 数据库名;

使用数据库

对某个数据库操作(进入该数据库来操作数据库中的表)

use 数据库名;

DDL表操作

记住一定要先进入该数据库,才能对该数据库的表进行操作。

创建表

create table 表名(
    data1 数据类型 可接comment '注释'表示对data注释,
    data2 数据类型,
    data3 数据类型
) comment '注释';

如下表:表示创建一个包含姓名、年龄、爱好这三个字段的用户表

create table user(
	name varchar(20) comment '姓名',
    age int comment '年龄',
    gender varchar(100) comment '爱好'
)comment '用户表';

查询表

1.查询当前库的所有表

show tables;

2.查询表结构

desc 表名;
  1. 查询指定表的建表结构
show create table 表名;

修改表

1.添加字段

alter table 表名 add 字段名 数据类型;

2.修改字段

  • 修改数据类型
alter table 表名 modify 字段名 新数据类型;
  • 修改字段名和字段类型
alter table 表名 change 旧字段名 新字段名 类型;

3.修改表名

alter table 表名 rename to 新表名;

删除

1.删除字段

alter table 表名 drop 字段名;

2.删除表

drop table 表名;

3.删除并重新创建该表

truncate table 表名;

DML-增删改

添加数据

1.给指定字段添加数据

insert into 表名 (字段1、字段2...) values (1,值2...);

2.给全部字段添加数据

insert into 表名 values (1,值2...);

3.批量添加数据

insert into 表名(字段1、字段2...) values (1,值2...),(1,值2...),...;
insert into 表名 values (1,值2...),(1,值2...);

更改数据

这里大致列出基本语法,更新时主要是根据where条件来更新(后面会讲)

1.更新所有数据

update 表名 set 字段1=要更改成的值;

2.更新部分数据

update 表名 set 字段1=要更改成的值 where 字段2=某值;

删除数据

1.删除所有数据

delete from 表名;

2.删除部分数据

delete from 表名 where 字段1=某值;

DQL-查询

基础查询

*表示所有字段

1.查询多个字段

select 字段1,字段2,... from 表名;
select * from 表名;

2.设置别名(在我们的sql语句比较复杂时设置别名可以让我们的sql语句更加简洁清晰)

select 字段1 as '别名' from 表名;

3.去除重复记录

select distinct 字段 from 表名;

条件查询

select 字段列表 from 表名 where 条件;

在这里插入图片描述

%表示任意长度的任意字符序列。_表示单个字符。

例如我们要查询一张表中名字中不带李的其他所有人的信息:

select * from 表名 where name not like '%李%';

聚合函数

函数功能
count统计数量
max最大值
min最小值
avg平均值
sum求和

语法:

select 聚合函数(字段) from 表名;

例子如下:

  • 统计user表中未成年的数量(每条数据都有一个id字段用来确定唯一性)
select count(id) from user where age<18;
  • 查询user表中年龄最大的人
select max(age) from user;
  • 查询user表的平均年龄
select avg(age) from user;
  • 统计所有来自南昌的员工的年龄之和
select sum(age) from employee where home='南昌';

注意:null不参与计算。

分组查询

group by是分组查询的关键字,它可以将表中的数据进行分组,再进行查询等操作。换言之,可通俗地理解为:通过group by将原来的表拆分成了几张小表。例如我们可以根据性别将一张大表分成男性表和女性表两个小表

select 字段 from 表名 where 条件 group by 分组字段名 having 分组后的过滤条件;

可以结合以下例子进行理解:

  • 在emp表中,根据性别分组,统计男性员工和女性员工的数量
select gender,count(*) from emp group by gender;
  • 查询年龄小于45的员工,并根据工作地址分组,获取员工数量大于等于3的工作地址。
select workplace,count(*) from emp where age<45 group by workplace having count(*)>=3;

where与having的区别:

  • where在分组前过滤,having在分组后过滤

  • where不能对聚合函数进行判断,having可以

总结:

  • 执行顺序:where>聚合函数>having
  • 分组之后,查询的字段一般为聚合函数和分组字段,查询其他字段无任何意义

排序查询

select 字段 from 表名 order by 字段1 排序方式,字段2 排序方式;

asc升序(默认) desc(降序)

  • 例如:根据创建时间(create_time)降序查询
select * from user order by create_time desc;

注意:如果是多字段排序,第一个字段相同时,才会继续根据第二个字段排序。

分页查询

select 字段 from 表名 limit 起始索引,查询记录数;

注意:*起始索引从0开始,起始索引=(查询页码-1)每页显示记录数。

  • 查询第1页数据,每页10条数据
select * from 表名 limit 0,10;
  • 查询第2页数据,每页10条数据
select * from 表名 limit 10,10;

编写顺序

select 字段列表 from 表名列表 where 条件列表 group by 分组字段列表 having 分组后条件列表 order by 排序字段列表 limit 分页参数;

DML-用户及权限

用户管理

1.查询用户

use mysql;
select * from user;

2.创建用户

create user '用户名'@'主机名' identified by '密码';

3.修改用户密码

alter user '用户名'@'主机名' identified with mysql_native_password by '新密码';

4.删除用户

drop user '用户名'@'主机名';

权限控制

常见权限:

权限意义
all所有权限
select查询数据
insert插入数据
update修改数据
delete删除数据
alter修改表
drop删除数据库、表、视图
create创建数据库、表

1.查询权限

show grants for '用户名'@'主机名';

2.授予权限

grant 权限列表 on 数据库名.表名 to '用户名'@'主机名';

3.撤销权限

revoke 权限列表 on 数据库名.表名 from '用户名'@'主机名';

注意:1.多个权限中用逗号分隔。2.数据库名和表名都可以用*表示,代表所有。

函数

不知道你是否还记得上面我们讲到过聚合函数,这里是其他的函数,具体用法就先不一一介绍了,混个眼熟后续再细说。

字符串函数

函数功能
concat(s1,s2,…,sn)将s1,s2…sn拼接起来
lower(str)将str全转换为小写
upper(str)将str全转换为大写
lpad(str,n,pad)用pad填充str左边,达到n个长度
rpad(str,n,pad)用pad填充str右边,达到n个长度
trim(str)去掉字符串头尾的空格
substring(str,start,len)返回str从start开始len个长度的字符串,索引从1开始

数值函数

函数功能
ceil(x)对x向上取整
floor(x)对x向下取整
mod(x,y)返回x除以y取余
rand()返回0~1的随机数
round(x,y)求x四舍五入的值,保留y位小数

日期函数

函数功能
curdate()返回当前日期
curtime()返回当前时间
now()返回当前日期和时间
year(date)获取date的年
month(date)获取date的月
day(date)获取date的日
date_add(date,interval expr type)返回date加上type类型的时间间隔expr后的时间
datediff(date1,date2)返回date1和date2的时间间隔

流程函数

函数功能
if(value,t,f)若value位true,返回t,否则返回f
ifnull(value1,value2)若value不为空,返回value1,否则返回value2
case val1 then res1 when val2 then res2… elae res3 end如果val1为真返回res1,val2为真返回res2,否则返回res3
case expr when val1 then res1 when val2 then res2… elae res3 end如果expr为val1则返回res1,为val2则返回res2,否则返回res3

约束

概述

约束是作用于表中字段上的规则,用于限制存储在表中的数据。其可以保证数据库中数据的正确性、有效性和完整性。

约束描述关键字
非空约束数据不能为nullnot null
唯一约束数据唯一,不能重复unique
主键约束主键是一行数据的唯一表示,要非空且唯一primary key
默认约束若未指定该字段的值,采用默认值default
检查约束保证字段值满足一个条件check
外键约束让两张表的数据之间建立连接,保证数据的一致性以及完整性foreign key

注意:我们可以在创建表的时候添加约束,也可以在修改表的时候添加约束。

实例:

在这里插入图片描述

create table user(
    id int primary key auto_increment comment '主键',
    name varchar(10) not null unique comment '姓名',
    age int check (age>0&&age<=120) comment '年龄',
    status char(1) default '1' comment '状态',
    gender char(1) comment '性别'
)comment '用户表';

外键约束

概述:外键用来让两张表的数据之间建立连接,从而保证数据的一致性和完整性。

在这里插入图片描述

如上图所示,按理来说,当我们删除部门表中的某一个部门时,因为其关联了员工表中的数据,所以不能直接删除。但由于这两张表并未建立外键关联,所以删除部门表数据时,数据就直接被删了。

添加外键的方法有两种:

  • 创建表的时候添加

  • 表结构创建好了再额外添加

alter table 表名 add constraint 外键名称 foreign key(要关联的字段) references 主表(关联的字段);

例如以下表示:把emp表的emp_id与boss表的boss_id添加外键,并取外键名称为guanlian

alter table emp add constraint guanlian foreign key (emp_id) references boss (boss_id);

2.删除外键

alter table 表名 drop foreign key 外键名称;

例如将上面创建的外键删除:

alter table emp drop foreign key guanlian;

外键更新、删除时的操作

行为说明
no action在父表中更新/删除数据时,如果存在外键,不允许操作数据
restrict在父表中更新/删除数据时,如果存在外键,不允许操作数据
cascade在父表中更新/删除数据时,如果存在外键,子表的数据同步改变(父表被删,子表关联的数据也被删)
set null在父表中删除数据时,如果存在外键,设置子表数据为null
set default父表有变更时,子表外键列设置成一个默认值

给外键被更新或删除时设置操作:

alter table 表名 add constraint 外键名称 foreign key(要关联的字段) references 主表(关联的字段) on update 行为 on delete 行为;

多表查询

概述

在项目开发过程中,我们需要设计许多张表,这些表与表之间存在着许多联系,具体分为一对一(例如用户基本信息与用户详情信息)、一对多(例如部门与员工)、多对多(例如学生与课程)

连接查询

内连接

内连接查询的是两张表交集的部分。

  • 隐式内连接
select 字段列表 from 表一,表二 where 条件;
  • 显式内连接
select 字段列表 from 表一 inner join 表二 on 连接条件; -- inner可省略

外连接

  • 左外连接
select 字段列表 from1 left outer join2 on 条件;

查询表1(左表)的所有数据,包含表1表2的交集部分数据。

  • 右外连接
select 字段列表 from1 right outer join2 on 条件;

查询表2(右表)的所有数据,包含表一表二的交集部分数据。

内连接无法查询为null的数据,外连接可以。

自连接

当前表与自身的连接查询,子连接必须使用表的别名

select 字段列表 from1 别名1 join1 别名2 on 条件;

自连接查询,可以是内连接查询,也可以是外连接查询。

例如我们要查询一张员工表,查询员工对应的上级,因为员工和他的上级都在这张员工表中,这时候我们就需要自连接查询了。

联合查询

将多次查询的结果合并起来,形成一个新的查询结果集。

select 字段列表 from 表一 条件
union
select 字段列表 from 表二 条件; -- 去重
select 字段列表 from 表一 条件
union all
select 字段列表 from 表二 条件; -- 不去重

例如:我们要查询出来emp表中年龄(age)大于50岁的员工数据和工资(salary)小于2000的员工数据,不去重

select * from emp where age>50
union all
select * from emp where salary<2000;

注意:union all表示直接合并,union表示去重合并;联合查询的字段列表的列数及字段类型必须保持一致。

子查询

SQl语句中嵌套select语句,也可称为嵌套查询。

select * from 表名 字段1= (select 字段2 from 表名 where 条件); -- 后面的这个select语句就是子查询语句

外部语句可以是insert/update/delete/select

标量子查询

子查询返回的结果是单个值,这种子查询称为标量子查询

例如:查询销售部所有员工的信息,该过程分为两步,1.查询销售部部门的id,2.根据部门id查询员工信息

select * from emp where dept_id=(select id from dept where name='销售部');

该过程中查询销售部部门id的操作就是子查询

列子查询

子查询返回的结果是一列多行

操作符描述
in在指定范围之内,多选一
not in不在指定范围集合内
any子查询返回列表中,有任意一个满足即可
some与any等同,使用some的地方都可以使用any
all子查询返回的所有值都必须满足

例如:查询销售部和市场部所有员工的信息,该过程分为两步:1.查询销售部和市场部的id,2.根据id查询员工

select * from emp where dept_id in (select id from dept where name='销售部' or name='市场部')

行子查询

子查询返回的结果是一行多列

例如:查询张三的薪资以及与张三的薪资相同的员工的信息,该过程分为两步:1.查询张三的薪资,2.查询与张三薪资相同的员工的信息

select * from emp where salary = (select salary from emp where name='张三');

表子查询

子查询返回的结果是多行多列

例如:查询张三和李四的薪资以及与张三和李四的薪资相同的员工的信息,该过程分为两步:1.查询张三和李四的薪资,2.查询与张三和李四薪资相同的员工的信息

select * from emp where salary in (select salary from emp where name='张三' or name='李四')

事务

基础

概述:事务是一组操作的集合,把几个操作作为一个整体,要么所有操作同时成功,要么同时失败。

常见案例如银行转账:该过程分为两个操作,a转账,b收账,这两个操作要么都成功,要么都失败,不存在a转了钱b却没收到钱。

要把一组操作放在一个事务里面,那我们就需要手动开启事务和提交事务,开启事务和提交事务这中间就是我们的事务,如果这中间抛出了异常,那么事务就要手动回滚。值得注意的是,MySQL的事务默认是提交的。

操作

  • 查看/设置事务提交方式
select @@autocommit; -- 查询出来的值是1,表示事务是自动提交的
set @@autocommit=0; -- 系统的事务是自动提交的,这样我们可以将其改为手动提交

0表示自动、1表示手动,默认是自动提交的,我们可以手动开启事务

上述两条语句是通过设置@@autocommit的值来设置事务是自动提交还是手动提交,我们也可以直接通过语句显式地开启事务

  • 开启事务
start transactionbegin ;

2.提交事务

commit;

3.回滚事务

rollback;

注意:如果事务正常执行,那我们需要提交事务,提交事务之后才能看到数据的变化;如果事务执行过程出现异常,我们就需要手动回滚事务。

四大特性

  • 原子性(Atomicity):事务是不可分割的最小操作但愿,要么全部成功,要么全部失败
  • 一致性(Consistency):事务完成时,必须使所有数据都保持一致状态(数据前后的总量不会发生变化)
  • 隔离性(Isolation):数据库系统提供的隔离机制,保证事务在不受外部并发操作影响的独立环境下运行
  • 持久性(Durability):事务一旦提交或回滚,它对数据库中的数据的改变就是永久的

并发事务问题

并发事务:几个事务同时操作数据库时的现象。下面三点表示在并发事务中可能出现的问题。

  • 脏读:一个事务读到另一个事务还没提交的数据(如下图所示,事务B读到了事务A还没有提交的数据)

在这里插入图片描述

  • 不可重复读:一个事务先后读取同一条记录,但两次读取的数据不同(同一条数据先被事务A查询,再被事务B修改或删除,再被事务A查询)

在这里插入图片描述

  • 幻读:一个事务按照条件查询数据时,没有对应的数据行,但是再插入数据时,又发现这行数据已经存在(一条数据被事务A查询时,查不到;接着这条数据在事务B中被插入,事务A再来插入的时候,发现插入不了)

在这里插入图片描述

针对以上问题,我们可以通过设置事务的隔离级别来解决:

在这里插入图片描述

这四个级别从上到下越往下数据越安全,但同时效率也会更低。

下面介绍一些关于事务的隔离级别的SQL语句:

  • 查看事务隔离级别
select @@transaction_isolation
  • 设置事务的隔离级别
set session transaction isolation level 隔离级别; 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

修复网络适配器不工作的14种方法,总有一种适合你

网络适配器是将设备连接到internet或其他计算机的关键硬件组件。如果设备发生故障,你可能会面临连接速度慢的问题,在最坏的情况下,互联网将完全停止工作。 这可能是由于损坏的驱动程序、冲突的设备、配置错误的设置,甚至是硬件故障!但也有可能出现互联网问题,使你认为网…

22.网络游戏逆向分析与漏洞攻防-网络通信数据包分析工具-加载配置文件到分析工具界面

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 如果看不懂、不知道现在做的什么&#xff0c;那就跟着做完看效果 内容参考于&#xff1a;易道云信息技术研究院VIP课 上一个内容&#xff1a;21.配置数据保存…

【java+vue】前后端项目架构详细流程

前端 1.创建vue项目 需要工具&#xff1a;node、Vscode 1.在磁盘上创建文件&#xff08;web&#xff09;&#xff0c;并移到Vscode的工作区 2.进入该文件的终端 3.检测node和vue是否正常&#xff0c;若不显示版本号&#xff0c;则自行下载 4.生成vue 1.执行命令&#xff1a;…

JavaScript中的Set和Map:理解与使用

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

2.JavaWebMySql基础

导语&#xff1a; 一、数据库基本概念 1.什么是数据库 2.关于MySql数据库 二、MySQL的安装与卸载 安装步骤&#xff1a; 卸载步骤&#xff1a; 三、MySQL服务操作 1.服务启动和关闭&#xff1a; 2.登录和退出MySQL&#xff1a; 3.服务自启动&#xff1a; 4.命令行登…

小程序Taro框架 自定义底部Tabbar,处理自定义Tab栏切换卡顿、闪烁

最终效果 最近在用Taro框架开发一个小程序&#xff0c;有一个自定义底部Tabbar的需求&#xff0c;最终效果如下 起步 这页是我第一次接触自定义小程序底部Tabbar&#xff0c;所有第一选择必然是相看官方文档&#xff1a;微信小程序自定义 Tabbar | Taro 文档 &#xff08;如果…

ping多个IP的工具

Ping Tool 项目地址 python开发的IP搜索小工具 ping一个网段所有IP&#xff0c;显示结果查看某个ip地址开放监听的端口配置可保存

IDEA编译安卓源码TVBox

因为电视x受限&#xff0c;无法观看电视直播&#xff0c;为了春晚不受影响&#xff0c;于是网络一顿搜索&#xff0c;试过多个APP&#xff0c;偶尔找到这款开源的TVBox&#xff0c;寒假在家&#xff0c;随便拿来练练手&#xff0c;学习安卓APP的编写&#xff0c;此文做以记录&a…

手拉手RocketMQ基础

消息中间件的对比 消息中间件 ActiveMQ RabbitMQ RocketMQ kafka 开发语言 java erlang java scala 单击吞吐量 万级 万级 10万级 10万级 时效性 ms us ms ms 可用性 高(主从架构) 高(主从架构) 非常高(主从架构) 非常高(主从架构) 消息中间件: activ…

分享软件项目实施方案模板

本项目在实施过程中将遵守做到以下几个方面&#xff1a; 与建设单位共同完成整个系统软件、网络等设计,负责系统的开发、测试、调试、人员培训、系统的试运行和交付&#xff0c;并保证系统质量。负责系统的维护、应用软件的升级和更新。提出对系统硬件设备的相关技术要求。在项…

JDBC的学习记录

JDBC就是使用java语言操作关系型数据库的一套API。 JDBC&#xff08;Java Database Connectivity&#xff09;是Java语言中用于连接和操作数据库的一种标准接口。它提供了一组方法和类&#xff0c;使Java程序能够与各种不同类型的关系型数据库进行交互。 JDBC的主要功能包括建…

PythonWeb

例题一 from flask import Flask app Flask(__name__) app.route(/index) def index():return f<h1>这是首页&#xff01;</h1> def second():return f<h1>这是第二页&#xff01;</h1> if __name__ __name__:app.run(host"0.0.0.0",port…

图腾柱PFC工作原理:一张图

视屏链接&#xff1a; PFC工作原理

SpringCloud--Sentinel使用

一、快速开始 Sentinel 的使用可以分为两个部分&#xff1a; 核心库&#xff08;Java 客户端&#xff09;&#xff1a;不依赖任何框架/库&#xff0c;能够运行于 Java 8 及以上的版本的运行时环境&#xff0c;同时对 Dubbo / Spring Cloud 等框架也有较好的支持。控制台&…

OpenFeign相关配置(超时,重试,HttpClient5,日志)

1.超时控制 默认OpenFeign客户端等待60秒钟&#xff0c;但是服务端处理超过规定时间会导致Feign客户端返回报错。 yml文件对应配置: connectTimeout&#xff1a;连接超时时间 readTimeout:请求处理超时时间 1.yml配置 server:port: 80spring:application:name: cloud-consu…

F - Earn to Advance

解题思路 由于对于一点不知道后面得花费&#xff0c;所以无法决策当前是否要停下赚钱或要停下多久考虑一点&#xff0c;可以由其左上方的所有点到达所以从往前推&#xff0c;得出到的总花费然后考虑从之后不赚钱直接到最终所用次数和剩余钱若存在&#xff0c;在后面点赚钱更优…

面试经典150题 -- 图的广度优先遍历 (总结)

总的链接 面试经典 150 题 - 学习计划 - 力扣&#xff08;LeetCode&#xff09;全球极客挚爱的技术成长平台 909 . 蛇梯棋 链接 : . - 力扣&#xff08;LeetCode&#xff09; 题意 &#xff1a; 直接bfs就好了 &#xff0c; 题意难以理解 : class Solution:def snakesA…

mockjs学习

1.前言 最近面试发现之前团队协同合作的项目没有mock数据难以向面试官直接展示&#xff0c;所以迟到得来速学一下mockjs。 参考视频&#xff1a;mockJs 妈妈再也不用担心我没有后端接口啦_哔哩哔哩_bilibili 一开始查阅了一些资料&#xff0c;先是看了下EasyMock&#xff0c…

分享几个Google Chrome谷歌浏览器历史版本下载网站

使用selenium模块的时候&#xff0c;从官网下载的谷歌浏览器版本太高&#xff0c;驱动不支持&#xff0c;所以需要使用历史的谷歌浏览器版本 &#xff0c;这里备份一下以防找不到了。 驱动下载地址&#xff1a;https://registry.npmmirror.com/binary.html?pathchromedriver 文…

Windows C++ 实现远程虚拟打印机(远程共享打印机)

编译错误已经修改完后的工程修改后的下载地址 https://download.csdn.net/download/2403_83063732/88928550 1、下载clawpdf&#xff08;0.8.7版本&#xff09; https://github.com/clawsoftware/clawPDF 2、打开clawpdf工程开始编译C#工程&#xff0c;出现如下错误&#xf…