Linux系统编程:编译过程以及GDB调试

编译工具链SDK(Software Development Kit)

在windows下编写程序,我们通常会用IDE,比如idea、vs等,这些工具将编译链接什么的全都暗地里解决好了我们只要写程序就行,但很明显,在Linux系统下做不到。
在Linux中,我们使用SDK来完成这些事情,共有两大派系:GCC和Clang。
工作当中一般是公司项目组选择哪个就用哪个。
其中,GCC是用的最多的。
在这里插入图片描述
使用GCC -v来查看版本信息,也可以用来查看自己是否安装了GCC:
在这里插入图片描述
其中Target: x86_64-linux-gnu表示gcc的目标平台架构。
Thread model:posix;表示gcc的线程模型遵从posix标准。
然后最下面一行:gcc version 11.2.0 (Ubuntu 11.2.0-19ubuntu1)表示gcc的版本信息。

代码执行编译过程

在这里插入图片描述
我们编写的.h和.c代码通过预处理后称为以.i结尾的预处理后的文件(此时依然是C语言),然后该文件经过编译成为.s汇编文件(此时就是汇编语言了),汇编文件经过汇编处理后成为目标文件(即.o文件,就是纯机器语言),最后经过OS引导和库函数链接之后称为以*号结尾的可执行文件(这是Linux系统下的文件表示)。

其中从预处理开始到成为目标文件这一段过程称为广义上的编译。

执行编译过程详解

预处理

作用:执行预处理指令
如:#include文件包含,#define M 5宏定义(简单的文本替换),#define SIZE(a)(sizeof(a)/sizeof(a[0]))宏函数等。
接下来我们介绍一些其它的预处理指令。

预处理指令,宏开关:#if [#else] #endif

我们写一个hello world:

  1#include <stdio.h>2 3 int main(){4 #if 05         printf("hello world\n");6 #else 7         printf("hello kitty\n");8 #endif9         return 0;10 }

然后我们使用下面的指令来进行预处理的过程:
在这里插入图片描述

gcc -o是一个命令行选项,它告诉GNU编译器(gcc)将编译器输出的可执行程序文件命名为指定的名称,o是output的意思。
gcc -E是一个命令行选项,它告诉GNU编译器(gcc)只执行预处理器并输出预处理器的结果,而不进行编译、汇编和链接等操作。因此,它可以用来查看程序中的宏定义、条件编译指令等预处理指令的展开结果,以及包含文件的内容等。

gcc -E hello.c -o hello.i是一个命令行选项,它告诉GNU编译器(gcc)只执行预处理器并将预处理器的输出写入到名为hello.i的文件中。这个操作不会将源代码编译成可执行文件,而是将源代码中的宏定义、条件编译指令等预处理指令展开成实际的代码,并将所有包含的头文件内容插入到相应的位置。这个过程可以帮助开发者查看源代码的预处理结果,以便更好地理解程序的工作原理。

接下来我们就可以进hello.i文件进行查看预处理后文件是个什么样子:
在这里插入图片描述
使用vim的vnew命令来左右分屏查看,用大写G命令直接切到.i文件的最后一行进行对比查看,如上图所示。
可以看到在左边.c文件中我们使用的有预处理指令 if 0 ,在预处理过程中会执行该指令,因为为0表示false所以预处理之后if 0后面的代码就不存在了,这种情况就称为宏开关。

与之类似的还有如下预处理指令,比该指令要稍微常用,通常叫条件编译:

#ifdef N [#else] #endif //else是可选的

来试一下这个事情:

  1#include <stdio.h>                                                 2                                                                    3 int main(){                                                       4 #ifdef N                                                            5         printf("hello world\n");                                    6 #else                                                              7         printf("hello kitty\n");                                   8 #endif                                                            9         return 0;                                                  10 }   

再通过上面的预处理指令来查看一下现在的情况:
在这里插入图片描述
因为我们并没有定义N这个东西,所以在预处理过程中依然是只显示了“hello kitty”。

我们定义宏的时候一般不会写死这个N,往往是在预处理的中途进行该宏的定义,使用参数-D即define的意思来对N进行宏定义:
在这里插入图片描述
可以看见此时再查看情况就不同了:
在这里插入图片描述
这是因为我们在预处理时进行了N的宏定义,#ifdef N 条件为真。

这样做的好处就是可以进行条件编译,为不同的目标平台生成不同的代码。

与之相同的还有ifndef:

#ifndef [#else] #endif

就是和上面ifdef相反的,即if not def。
常见的用法如下:
在这里插入图片描述
上图意思就是如果没有定义 _FOO_H_的话,那么我就执行下面的语句直到endif结束。

这种方式经常用来作防御式声明,作用是避免重复包含多个头文件,也就是上图中的形式。

(狭义)编译过程

作用:将C语言代码编译成汇编代码。

来验证这个事情,首先准备一个hello.c的简单C代码:

  1 #include <stdio.h>2 3 int main(void){4         printf("hello world\n");5         return 0;6 }

下面是两种编译方式:
第一种是编译我们上面所使用过的那种先被预处理后的文件,将其编译成我们的汇编代码:
在这里插入图片描述

gcc -S是一个命令行选项,它告诉GNU编译器(gcc)只执行编译操作,生成汇编代码而不是可执行文件。编译器会将源代码编译成汇编代码,然后将汇编代码写入到一个文件中。在生成汇编代码后,你可以使用汇编器将其转换为机器码,然后创建可执行文件。

第二种方式是直接根据源代码编译成汇编代码:
在这里插入图片描述
.s就是汇编代码的文件后缀名。

然后来查看汇编代码:
在这里插入图片描述
这就是一个编译过程,有时间也可以去学习一下汇编语言(狗头),然后对照着看一下就能发现底层汇编到底在干什么了。
这只是帮助我们大概了解一下。不懂也没事,不影响接下去的学习。

但上述这两种方式都是狭义上的编译,也就是从.c文件到.s汇编文件的编译。

有一种最常用的针对广义编译的命令,可以直接生成.c对应的.o目标文件:

(广义)编译命令:gcc -c

使用该命令可以直接生成.c对应的.o目标文件:
在这里插入图片描述

链接过程

这个不是很重要,就了解一下即可。
如果想深入了解可以看《程序员的自我修养》,它的英文版叫《Linked & Loader》。
链接做的最朴素的事情就是把调用函数的名字转换成对应的内存地址。
观察过汇编的代码就会发现在函数调用中,都是使用call指令加上函数名的方式,当call指令执行时会去分配一个对应的内存地址给该函数名,而什么时候会去分配呢?就是在链接的时候才会分配,所以链接过程必不可少。

可执行程序

在Linux系统拥有x权限的程序就是可执行程序,怎么执行呢?
只要输入其对应的路径即可,但是如果是在可执行程序所在的当前路径下的话,需要用 ./可执行程序名 来执行。
使用./的形式是为了避免其可能会与系统内部所存在的一些内置命令冲突。

库文件

库文件也就是轮子,是一种公用的工具。
可以认为这是一种特殊的.o文件,是别人写好的我们拿来用的工具。
库文件的文件名后缀为.a或者 .so,这是两种库文件,分别对应着静态库和动态库;
轮子打包到产品中,这是静态库,比如一般的汽车,轮子都是附带着一起被购买来用的。
而轮子在运行的时候才加入到产品中,就是动态库,比如F4方程式赛车,都是现场更换。

特点:

从产品大小来看,静态库比较大,动态库比较小
从部署难度说,静态库容易,动态库难
从升级角度看,静态库难,动态库容易

生成静态库

先来看这么一个例子,文件名为test.c:

  1 #include <stdio.h>2 3 4 int add(int a,int b);5 6 int main(){7         printf("add(3,4) = %d\n",add(3,4));8 }

编译肯定没问题,因为我们没有语法错误,但是链接不了,因为add函数未定义:
在这里插入图片描述
那我们在另外一个文件里写add.c:

  1 int add(int a,int b){2         return a + b;3 }

这个文件一样可以编译,但是链接不了,因为没有main函数:
在这里插入图片描述

但此时我们可以将它们俩合在一起,也就是链接在一起,把两个.o文件链接在一起称为一个可执行程序:
在这里插入图片描述
此时test就可以运行了。

但是目前这个add.o只能我们自己用,我们想把它变成共享的,怎么做?

首先,不管怎么样做,这个add.o文件我们肯定是需要的,源码文件肯定是不公开的,我们选择分享add.o文件。
生成目标文件命令:gcc -c add.c -o add.o

然后第二件事情就是去把这个add.o文件打包成静态库,打包命令是固定的用法,直接用就可以:

ar crsv LibName.a add.o	
//libName是打包后的库文件名,注意lib开头是固定的,中间add名字是可以换的,后缀为.a也是固定的,然后add.o是我们要打包的目标文件名

我们来试一下:
在这里插入图片描述

然后我们将该库文件移动到系统搜索目录下:/usr/lib
来试一下:
在这里插入图片描述
来看一下是否存在了:
在这里插入图片描述
现在我们进行链接就没有之前这么麻烦了,使用-l参数来指定所需的库文件就可以了,-l后面跟库文件名:
在这里插入图片描述
因为是静态库,所以现在哪怕将该库文件删除了,执行刚刚生成的可执行文件一样可以运行,这就是静态库。

小技巧:配置超好用Linux的vim环境

配置这个只需要装一个vimplus即可,具体可以看这个博客链接:
配置vimplus超好用c/c++编辑器
网上资源很多,这个应该搞得定吧哈哈哈。

生成动态库

之前我们已经直到单独链接test.c文件肯定是会出错的,因为main里面并没有定义add函数。
这里我们使用动态库来链接,动态库是在程序运行时才加载到内存中来运行的,动态库和静态库区别如下:
静态库是在链接阶段就链接到数据/代码段中的(这是个静态区),而动态库因为是在程序运行时才加载到内存中所以会被存放在栈空间和堆空间的某个区域,叫共享库映射区(即动态库也叫共享库),即然是共享那就意味着另外的程序事实上也可以使用该库,但此时存在一个问题就是不同的程序在加载时所在的内存地址是不一样的,可众所周知,在程序当中的各个指令的地址肯定都是相对地址不是绝对地址,即位置无关的代码,所以我们在编译动态库的时候必须加上-fpic:

-fpic 是 GCC 编译器的一个选项,表示编译生成与位置无关的代码(Position Independent Code)。这种代码可以被加载到内存的任何位置,并且不影响其正确性。这在编写共享库等时非常有用。
当使用 -fpic 选项编译时,会生成 PIC 代码。而使用 -fPIC 选项时,会生成更严格的 PIC 代码,这意味着生成的代码可以在更多的情况下使用,但可能会稍微降低性能。

所以生成动态库步骤如下:

1、编译生成相关文件
gcc -c add.c -o add.o -fpic2、打包
gcc -shared add.o -o libadd.so(lib前缀和.so后缀都是确定的,动态库名字是可以随意取的)3、移动到系统库目录
sudo cp libadd.so /usr/lib4、链接的时候加上 -ladd
gcc -test.o -o test -ladd

我们可以来试一下将我们的add.c来生成动态库:
在这里插入图片描述
分开编译,链接的时候要一起嗷。
还可以通过ldd命令查看动态库的链接过程:

在这里插入图片描述

ldd只能显示动态链接,无法显示静态链接。

对于动态库,如果将动态库删除,那么该test程序将无法执行:
在这里插入图片描述
如果我们将该文件又复原,会发现该程序又可以正常运行:
在这里插入图片描述
这说明对于动态库来说,在每次执行程序时都会去系统库函数目录下寻找它所需要的库函数,在运行时进行加载以获得运行支持。

此时若我们对add库函数文件进行更新:

  1 int add(int a,int b){2     return a + b + 1;                                                                                                                        3 }

然后重新进行动态库生成:
在这里插入图片描述
从上图可以知道我们并没有进行程序的重新链接,只是更新了对应的库函数就能够实现更新代码后的结果。

所以动态库更新是更容易的相较于静态库而言,这也就是热更新的原理。

但这种热更新会带来一个新的问题,就是万一更新完出了问题,我们怎么回滚到原来的正常运行状态呢?

这个时候一个非常好的做法是将/usr/lib文件下面我们链接时指定的文件名libadd.so文件改成软链接,让其动态的根据需要指向我们想要进行动态链接的一个真正的libadd.so:
在这里插入图片描述
所以我们可以给每一次生成的库函数分别标上一个版本号ID,然后想用哪个版本的时候就让我们的libadd.so软链接指向它即可实现回滚的功能:
在这里插入图片描述
上图是我们想要更换版本了,就删除原来的软链接,再新建一个同名软链接指向我们的新版本库函数文件即可。

gcc的其它选项

-D 编译时进行宏定义

当使用gcc编译时使用该参数,就相当于是在代码中进行宏定义:
在这里插入图片描述

-I 编译时增加搜索路径

编译时增加搜索路径:
在这里插入图片描述
上面第一次编译找不到"test.h",因为这么写默认是在当前目录下进行寻找该头文件,如果不加-I也可以手动修改代码,把路径补全即可:
在这里插入图片描述
像上面这样也是没问题的。

-O 进行编译优化

注意该参数后面要跟数字,比如-O0,-O1等等。
该参数将进行编译优化,编译器作者会在底层进行修改顺序和内存位置,这样做结果不会改变(编译器会保证这一点)但指令数量变少了且执行速度变快,代价就是你写的代码在进行单步调试时会很麻烦,因为结构和一开始写的不一样了。

注意-O后面跟的数字越大,则意味着优化的深度越深,C和汇编的对应关系对应不上的风险大大增加(-O0表示不开优化)。

工作中用-O0和-O1都没问题,但发布产品的时候都使用-O1.

-Wall 让编译器给出编译警告

在Linux系统中,出现警告一定要重视,这往往是最后出现问题的根源。
在这里插入图片描述

GDB调试(必会)

在这里插入图片描述
GDB使用前提:
首先不能开编译优化,其次要补充调试信息,为什么要补充调试信息?
之前我们聊到汇编的时候说过,在汇编代码中变量什么的是没有名字的,但调试的时候肯定是要知道变量名字的呀,所以必须要把调试信息补充上,这一点通过-g参数可以实现。

这里我们写一个C语言文件来进行学习:

  1 #include <stdio.h>2 3 void func(int i){4 5     printf("I am func , i = %d\n",i);6 }7 8 int main(){9 10     int j = 10;11 12     int arr[3] = {1,2,3};13 14     int *p;15     arr[2] = 4;16     p = arr;17 18     fun(j);19 20     for(int i=0;i < 5; i++){21         puts("hello world");22     }23     return 0;                                                                                                                                24 }

在这里插入图片描述
现在编译完成,启动gdb进行调试:
在这里插入图片描述
使用gdb加文件名的方式启动调试,如上图所示,这就表示进入了gdb命令行的意思。

list或l命令打印源码语句

注意list命令后面可以跟数字参数表示从第几行开始展示,每次展示十行:
在这里插入图片描述
除了可以跟行号,list命令后面还可以跟文件名冒号行号,这可以在多文件gdb调试中用来切换所要调试的文件:
在这里插入图片描述
也可以后面跟函数名,就可以定位到函数名位置:
在这里插入图片描述

run或者r运行程序

在这里插入图片描述

break或者b打断点

可以选择在第几行打,也可以选择在哪个函数位置打:
在这里插入图片描述
现在我们再运行程序就会运行到我们所打的断点位置停下:
在这里插入图片描述

continue或者c继续运行程序

在这里插入图片描述

step/s 单步调试无法跳过函数调用

我们来使用step,发现会下一步会进入到puts函数调用:
在这里插入图片描述

finish直接让程序运行到该函数运行完为止

在这里插入图片描述

next/n 单步调试会将下次函数调用跳过

在这里插入图片描述
可以看见直接跳过了puts函数的深层调用,就可以清晰的看见循环过程。

info break / i b查看断点信息

在这里插入图片描述
从上图可以看到,两个断点都分别命中了一次。

delete num 删除指定断点,num是对应的断点信息

在这里插入图片描述

如果delete后面不跟参数,那就是默认删除所有断点。

ignore [bpNum] [ignoreTimes]忽略某断点多少次

如果当我们遇到了循环次数很大的这种情况使用单步调试的话手都会按断,所以ignore可以帮我们忽略掉循环次数。
bpNum是断点编号,ignoreTimes是忽略次数。

在GDB当中去查看监视

print 或者 p打印数据

在这里插入图片描述

事实上print可以打印任意的合法表达式:
在这里插入图片描述
有了print命令我们就可以结合next命令实现一步一打印的效果,但是这样太繁琐了,所以我们这里有个新的命令用来代替这种繁琐的事情。

display 自动打印

display后面跟一个表达式,表示需要开启自动打印的内容:
在这里插入图片描述
那开启了怎么关呢?首先我们需要先打印一下display的信息。

info display 展示display信息

在这里插入图片描述

undisplay dNo 停止自动打印

dNo表示需要停止的内容编号:
在这里插入图片描述
可以看到后面就不再有自动打印了。

在GDB中去查看内存

在gdb中查看内存借助x指令:
在这里插入图片描述
看完上图大致能得出几个结论。
首先查看内存的命令形式:x/FMT address.
所谓的FMT就是有三个部分组成的:
第一个部分是repeat count是个数字,第二个是个format letter字母格式,第三个是个大小字母。

即第一个部分数字表示看多少个单位;
第二个字母格式表示用什么进制格式来查看,o表示八进制,x表示十六进制,d表示十进制,u表示无符号十进制。
第三个大小则表示每个单位的大小,b表示字节,h表示两个字节,w表示四个字节,g表示八个字节。

查看数组arr的内存信息:
在这里插入图片描述
这表示看3个单位的arr数组的内存信息,然后每个单位的内存内容用二进制来表示,每段二进制用四个字节来展示。
正好展示的内容就是数组arr的内容{1,2,4}

在GDB中检查崩溃的程序

core文件:程序中的“黑匣子”

core文件存储了当程序崩溃时整个程序运行的内存的堆栈情况。
来写段会崩溃的代码试一下,使用quit命令可以退出gdb:

  1 int main(){2     3     int* p = 0 ;4     //这里没有给指针具体的地址5     //就进行解引用的操作是肯定报错的                                                                                                         6     *p=1;7 8 }

然后我们去编译运行这段代码:
在这里插入图片描述
可以看见段发生错误然后core dumped,表示core文件已经存储起来了。
但是我们这里现在看不到,因为在Linux中OS会限制这个core file的一个大小,因为往往这个文件都会非常非常大,我们可以用ulimit -a来查看当前系统中的所有被限制了的内容:
在这里插入图片描述
可以看到core file size 大小默认为0,当然看不到了,所以我们将其设置一下改为无限大:
在这里插入图片描述
注意这个大小的设置是临时的只能影响到当前终端,哪怕换个终端那么这个设置也就失效了,重新登录自然也是会失效的。
如果进行该操作之后发现还是无法生成core文件的话,就进行下面的操作:
在这里插入图片描述
注意上面是切换成了root用户才进行的该操作嗷,那么现在我们再执行报错的程序来看是否存在core文件了(注意要切换回普通用户再执行嗷):
在这里插入图片描述

可以看见此时的core文件已经出现。
这个时候就可以使用下面的命令进行gdb的查看:
在这里插入图片描述
可以清楚的看见是在第六行*p=1这一行出错了。

同时我们也可以使用bt命令去查看堆栈的情况:
在这里插入图片描述

GDB打印命令行参数

我们经常会按如下方式启动程序:

  1 #include <stdio.h>2 3 int main(int argc,char* argv[]){4                                                                                                                                              5     printf("argc = %d\n",argc);6     7     for(int i=0;i<argc; i++){8         puts(argv[i]);9     }10 }

编译运行如下:
在这里插入图片描述
在第一次调用中,因为没有其它的命令行参数,所以命令行参数只有./args(也就是它自己)。
在第二次调用中我们设置了how are you 三个参数,所以打印如上图所示,这就是打印命令行参数。

用gdb启动也是类似:
在这里插入图片描述
除了可以用set去设置命令行参数,也可以用show args去显示命令行参数信息:
在这里插入图片描述

总结

本文简述了程序编译链接运行的流程,然后学习了GDB调试,后面这个GDB调试非常重要,必须要很熟嗷。

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

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

相关文章

前端axios发送请求,在请求头添加参数

1.在封装接口传参时&#xff0c;定义形参&#xff0c;params是正常传参&#xff0c;name则是我想要在请求头传参 export function getCurlList (params, name) {return request({url: ********,method: get,params,name}) } 2.接口调用 const res await getCurlList(params,…

ubuntu离线编译安装cmake 3.22.5(could not fonud OPENSSL) and cmake-versinon查不到版本问题

1、首先去cmake官网下载压缩包,例如: cmake-3.22.5.tar.gz 2、拉到ubuntu进行解压: tar -zxcf cmake-3.22.5.tar.gz 3、cd 进入目录 cd cmake-3.22.5 4、执行configure可执行文件 ./configure 如果在编译过程中出现报错:Could NOT findOpenSSL,原因可能是缺少ssl库 按…

从城市吉祥物进化到虚拟人IP需要哪些步骤?

在2023年成都全国科普日主场活动中&#xff0c;推出了全国首个科普数字形象大使“科普熊猫”&#xff0c;科普熊猫作为成都科普吉祥物&#xff0c;是如何进化为虚拟人IP&#xff0c;通过动作捕捉、AR等技术&#xff0c;活灵活现地出现在大众眼前的&#xff1f; 以广州虚拟动力虚…

【Acwing187】导弹防御系统(LIS+剪枝+贪心+dfs+迭代加深)

题目描述 看本文需要准备的知识 1.最长上升子序列&#xff08;lis&#xff09;的算法思想和算法模板 2.acwing1010拦截导弹&#xff08;lis贪心&#xff09;题解 本题题解&#xff0c;需要知道这种贪心算法 3.简单了解dfs暴力搜索、剪枝、搜索树等概念 思路讲解 dfs求最…

代数——第3章——向量空间

第三章 向量空间(Vector Spaces) fmmer mit den einfachsten Beispielen anfangen. (始终从最简单的例子开始。) ------------------------------David Hilbert 3.1 (R^n)的子空间 我们的向量空间的基础模型(本章主题)是n 维实向量空间 的子空间。我们将在本节讨论它。…

【Qt】顶层窗口和普通窗口区别以及用法

区别 在Qt项目开发中&#xff0c;经常会用到窗体控件用于显示及数据操作和其他交互等。 但&#xff0c;窗体分为顶层窗口&#xff08;Top-level Window&#xff09;和普通窗口&#xff08;Regular Window&#xff09;。 他们之间是有区别的&#xff0c;包括在项目实际中的用法…

实现动态表单的一种思路 | 京东云技术团队

一、动态表单是什么 区别于传统表单前后端配合联调的开发实现方式&#xff0c;动态表单通过一种基于元数据管理的配置化方法来实现表单的动态生成&#xff0c;并能根据配置自由增改删指定字段。实现特定需求的自助化。 图1.1 传统表单前后台协作模式 图1.2 动态表单前后台协作…

CY7C68013A芯片与FPGA

文章目录 环境软件环境其它工具 USB基础USB2.0设备组成USB设备模型USB设备分层USB Host Controller 主机控制器分类 USB HostUSB2.0 数据帧USB传输事务传输类型 芯片 cypress CY7C68013开发包安装FX3 固件程序设计步骤 驱动程序设计计算机上层应用软件USB2.0 FPGAUSB基础资料官…

单目标应用:墨西哥蝾螈优化算法(Mexican Axolotl Optimization,MAO)求解微电网优化MATLAB

一、微网系统运行优化模型 微电网优化模型介绍&#xff1a; 微电网多目标优化调度模型简介_IT猿手的博客-CSDN博客 二、墨西哥蝾螈优化算法MAO 墨西哥蝾螈优化算法&#xff08;Mexican Axolotl Optimization&#xff0c;MAO&#xff09;由Yenny Villuendas-Rey 1等人于2021…

Clion中使用C/C++开发stm32程序

前言 从刚开始学习阶段&#xff0c;一直是用的keil5开发stm32程序&#xff0c;自从看到稚晖君推荐的CLion开发嵌入式程序后&#xff0c;这次尝试在CLion上开发stm32程序。 1、配置CLion用于STM32开发的环境 这里我就不详细写了&#xff0c;没必要重新写&#xff0c;网上教程很多…

安卓-APP启动优化技术方案汇总

应用有三种启动状态&#xff1a;冷启动、温启动或热启动。每种状态都会影响应用向用户显示所需的时间。在冷启动中&#xff0c;应用从头开始启动。在另外两种状态中&#xff0c;系统需要将后台运行的应用带入前台。 我们建议您始终在假定冷启动的基础上进行优化。这样做也可以…

金融信创黄金三年:小程序生态+跨端技术框架构建

小程序应用场景生态的发展&#xff0c;受益于开源技术的发展&#xff0c;以及响应快速开发的实际业务需求&#xff0c;一些跨端框架如&#xff1a;Electron、wxPython、FinClip、Tauri、Flutter等发展也非常迅速&#xff0c;小程序生态跨端技术框架&#xff0c;不仅能满足自有超…

【C刷题】day4

一、选择题 1、设变量已正确定义&#xff0c;以下不能统计出一行中输入字符个数&#xff08;不包含回车符&#xff09;的程序段是&#xff08; &#xff09; A: n0;while(chgetchar()!\n)n; B: n0;while(getchar()!\n)n; C: for(n0;getchar()!\n…

利用SoapUI工具生成Java WebService客户端代码

一. 下载安装软件 安装SoapUI 5.4.0-EB&#xff1b;下载axis-1_4&#xff0c;下载后解压至个人目录下即可。 注&#xff1a;axis-1_4下载地址&#xff08;https://archive.apache.org/dist/ws/axis/1_4/axis-bin-1_4.zip&#xff09; 二. 创建SOAP Project 点击File–>New…

Vue-2.1scoped样式冲突

默认情况&#xff1a;写在组件中的样式会全局生效->因此很容易造成多个组件之间的样式冲突问题 1.全局样式&#xff1a;默认组件中的样式会作用到全局 2.局部样式&#xff1a;可以给组件加上scoped属性&#xff0c;可以让样式只作用于当前组件 <style scoped> <…

保姆级微服务部署教程

大家好&#xff0c;我是鱼皮。 项目上线是每位学编程同学必须掌握的基本技能。之前我已经给大家分享过很多种上线单体项目的方法了&#xff0c;今天再出一期微服务项目的部署教程&#xff0c;用一种最简单的方法&#xff0c;带大家轻松部署微服务项目。 开始之前&#xff0c;…

Mall脚手架总结(二) —— SpringData操作Elasticsearch

前言 万字长文带你弄清楚SpringData中的Elasticsearch操作以及在脚手架里接口的结构关系&#xff01;经过前面鉴证授权的整合&#xff0c;荔枝开始熟悉项目的学习的方法了&#xff0c;虽然脚手架中的内容比较简单&#xff0c;但是把边角的知识点全部扫到还是比较花时间的尤其是…

【轻松玩转MacOS】故障排除篇

引言 在使用 MacOS 时&#xff0c;遇到故障是在所难免的。不要担心&#xff0c;这篇文章将为您提供一些常见的故障排除步骤&#xff0c;并介绍如何联系苹果的支持团队寻求帮助。让我们一起来看看吧&#xff01; 一、常见的故障排除步骤 1.1 网络连接问题 如果你发现你的Mac…

Redis(三)

文章目录 一、单节点Redis的问题&#xff08;一&#xff09;数据丢失&#xff08;二&#xff09;并发能力问题&#xff08;三&#xff09;存储能力问题&#xff08;四&#xff09;故障恢复问题 二、Redis持久化&#xff08;一&#xff09;RDB1、RDB是什么2、rdb配置3、手动触发…

面试题:MySQL 中 InnoDB 的索引结构以及使用 B+ 树实现索引的原因

文章目录 概述表空间索引结构为什么使用 B 树实现索引&#xff1f;总结 概述 在 MySQL 的众多存储引擎中&#xff0c;InnoDB 是最常用的存储引擎&#xff0c;也是 MySQL 现阶段唯一免费支持事务机制的存储引擎。在本文中&#xff0c;我们以 InnoDB 为例&#xff0c;介绍 MySQL…