【Linux】掌握库的艺术:我的动静态库封装之旅

1.什么是库

在计算机编程中,库(library)是一个预先编写的代码集合,包含了可以被其他程序调用的函数、类、变量和资源。库的主要目的是为了简化编程过程,提供常用功能的实现,促进代码重用,从而减少开发时间和提高软件的可靠性。

在实践中,我们一定会使用到别人的库,如你在C语言时期一定会调用到c标准库。除了标准库,我们还可以使用第三方库,无论使用那种库都是为了节省我们的时间,不在需要我们自己来造“轮子”。

所有的库都可以从两个方面来认识:

创建者使用者提问:使用者在使用库时,是否能知道该库的源代码呢?回答:在不逆向的情况下,使用者是无法得知库的源代码的,这也就牵扯到了,库的第二个属性隐藏源代码。我们都知道,形成可执行文件的步骤有4步:预处理:头文件展开、去注释、宏替换、条件编译等,生成.i文件。编译:语法分析、语义分析、符号汇总等,检查无误后将代码翻译成汇编指令,生成.s文件,汇编:将汇编指令转化成二进制指令,生成.o文件。链接:将生成的各个.o文件进行链接,生成可执行程序。

而库就是所有.o文件用特定的方式,进行打包,形成的一个文件。

注意库文件需要配合对应的头文件进行使用,头文件不必隐藏

当我们在main.c中使用对应的功能函数:

【Linux】掌握库的艺术:我的动静态库封装之旅删除可执行文件

当有许多不同的源文件去调用这些功能函数时,那要的话,功能函数会被重复的进行预处理、编译、汇编的操作,各自生成.o文件,然后我们的源文件再和这些功能函数一起生成可执行文件。

如此操作会有许多重复的操作,我们完全可以提前让功能函数变成一个个.o文件,再去和源文件进行链接,但是我们还要考虑到,可能会存在非常多的.o文件,为了减轻我们的操作,我们可以对这些.o文件进行打包,这样的文件我们称为库。

如此一来,库的本质就是若干个目标文件的集合,每一个目标文件都包含了由源码编译生成的二进制代码,在保证使用的同时,还有很好的隐藏性

1.2 认识动静态库

动静态库是编程中常用的两种库类型,用于封装和重用代码。它们在链接、加载和使用方面存在显著的差异。

1.2.1 动态库

动态库(或共享库)是在运行时加载的库,通常以 .so(Linux)或 .dll(Windows)文件格式存在。动态库的代码不被复制到可执行文件中,而是在运行时由操作系统加载。

我们先来写一段简单的代码:

代码语言:javascript代码运行次数:0运行复制

#include int main(){    printf("hello world!!!!n");    return 0;}//生成可执行文件mybin

编译成功后,我们使用ldd指令来查看其所链接的动态库

ldd语法:

代码语言:javascript代码运行次数:0运行复制

ldd filename

功能:

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ ldd mybin linux-vdso.so.1 (0x00007ffc9c4db000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fd8df175000)/lib64/ld-linux-x86-64.so.2 (0x00007fd8df3ab000)

libc.so 就是该程序所依赖的动态库,那么我们一个如何识别一个动态库的名字呢?

也就是说,这个动态库的名字就是c,没错这就是c运行库。

1.2.2 静态库

静态库是将一组对象文件(.o 文件)打包成一个库文件(通常为 .a 后缀),在编译时将其链接到最终生成的可执行文件中。链接过程是在编译阶段完成的,库的代码被复制到可执行文件中。

正常情况下,gcc默认都是连接的动态库,如果需要进行静态连接要特别指定。

代码语言:javascript代码运行次数:0运行复制

gcc -o staticBin libTest.c -staticubuntu@VM-20-9-ubuntu:~/libraryTest$ ls -ltotal 904-rw-rw-r-- 1 ubuntu ubuntu     82 Oct 29 11:40 libTest.c-rw-rw-r-- 1 ubuntu ubuntu     74 Oct 29 11:41 makefile-rwxrwxr-x 1 ubuntu ubuntu  15952 Oct 29 11:41 mybin-rwxrwxr-x 1 ubuntu ubuntu 900344 Oct 29 14:49 staticBin

可以发现静态连接的可执行程序所占的空间大小远大于动态连接的。

使用ldd观察是否有依赖的库

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ ldd staticBin not a dynamic executable

发现静态链接生成的可执行程序不依赖其他库文件。

使用file查看:

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ file staticBin staticBin: ELF 64-bit LSB executable, x86-64, version 1 (GNU/Linux), statically linked, BuildID[sha1]=d4a39b68ac04f9fd8b37bd8c17ce42a8b27ad8c2, for GNU/Linux 3.2.0, not stripped

发现是:statically linked

如果你是centos用户,可能会遇到静态库未安装的情况,只需要搜索一下安装方法即可。

2.封装动静态库2.1 封装动态库

先写好需要封装的代码:

add.h:

代码语言:javascript代码运行次数:0运行复制

#pragma onceint add(int a,int b);

sub.h:

代码语言:javascript代码运行次数:0运行复制

#pragma onceint sub(int a,int b);

add.c:

代码语言:javascript代码运行次数:0运行复制

#include "add.h"int add(int a,int b){    return a+b;}

sub.c:

代码语言:javascript代码运行次数:0运行复制

#include "sub.h"int sub(int a,int b){    return a-b;}

dynLibTest.c:

代码语言:javascript代码运行次数:0运行复制

#include #include "add.h"#include "sub.h"int main(){    int a = 100;    int b = 200;    printf("%d + %d = %dn",a,b,add(a,b));    printf("%d - %d = %dn",a,b,sub(a,b));    return 0;}

然后我们还需要知道位置无关码的概念(Position-Independent Code,PIC)

位置无关码(Position-Independent Code,PIC)是一种编译代码的方式,使得生成的代码可以在内存的任意位置执行,而不需要修改代码中的地址。这种特性在动态链接库和共享库中非常重要,因为它们可以被多个进程共享,并在加载时被放置到不同的内存地址。

为了实现位置无关,编译器在生成代码时使用相对地址而不是绝对地址。例如,在访问全局变量时,编译器不会生成直接访问变量的绝对地址的代码,而是使用相对于当前指令位置的偏移量。这样,无论代码被加载到哪个地址,访问都可以通过相对计算来实现。

位置无关码是一种重要的编程技术,尤其在动态链接和共享库中具有广泛的应用。它提供了灵活性和内存使用效率,同时也增强了程序的安全性。

如果我要实现位置无关码可以在gcc后面加上-fPIC选项

代码语言:javascript代码运行次数:0运行复制

gcc -fPIC -c add.cgcc -fPIC -c sub.c
【Linux】掌握库的艺术:我的动静态库封装之旅生成目标文件

我们也知道不加-fPIC一样可以生成.o文件。但是它们之间还是很有区别的。

位置无关码对于gcc:

fPIC作用于编译阶段,告诉编译器于位置无关的代码,此时产生的代码中没有绝对地址,全部都使用相对地址,从而代码可以被加载到内存的任意位置可以正确的执行。这正是共享库被加载时,在内存的位置不是不是固定的。如果不加-fPIC选项,则加载.so文件的代码时,代码段引用的数据对象需要重定位,这个重定位会修改代码段地内容,这就会造成每一个使用这个.so文件代码段的内核里都会生成这个.so文件代码的拷贝,并且每一个拷贝都不一样,这就是使得内存的消耗变大。为此我们总是会用-fPIC来生成.so文件,但是不会用点-fPIC来生成.a静态文件。我们当然可以不用-fPIC来生成.so文件,只是这样的话.so文件必须要在加载到用户的地址空间时重定向所有表目。使用**-shard**将文件打包为动态库代码语言:javascript代码运行次数:0运行复制

gcc -shared -o libmyc.so add.o sub.o

通过readelf -S 来查看库的信息,如偏移量offset

【Linux】掌握库的艺术:我的动静态库封装之旅查看库信息

2.1.1 组织头文件和库文件

为了方便我们后续的操作,我会写一个makefile来组织头文件和库文件。

代码语言:javascript代码运行次数:0运行复制

.PHONY:allall:libmyc.solibmyc.so:add.o sub.ogcc -shared add.o sub.o -o libmyc.soadd.o:add.cgcc -fPIC -c add.c -o add.osub.o:sub.cgcc -fPIC -c sub.c -o sub.o.PHONY:outputoutput:mkdir -p mylib/libmkdir -p mylib/includecp -rf *.h mylib/includecp -rf *.so mylib/lib.PHONY:cleanclean:rm -rf *.o mylib *.a *.so output
【Linux】掌握库的艺术:我的动静态库封装之旅makefile

2.1.2 动态库的使用

当我们把当前目录的头文件和目标文件删除后

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ rm -rf *.o *.h

此时如果我们直接使用gcc来编译会发生什么呢?

没错,肯定会报错说,找不到头文件啦。

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ gcc dynLibTest.c dynLibTest.c:2:10: fatal error: add.h: No such file or directory    2 | #include "add.h"      |          ^~~~~~~compilation terminated.

为了让编译器能够找到我们的头文件,需要我们加上-I 头文件所在路径

代码语言:javascript代码运行次数:0运行复制

gcc -I./mylib/include dynLibTest.c

现在的错误就是,找不到目标函数了,因为我们没有给编译器指明在哪里,它肯定就找不到了。

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ gcc -I./mylib/include dynLibTest.c/usr/bin/ld: /tmp/ccOkC8RY.o: in function `main':dynLibTest.c:(.text+0x25): undefined reference to `add'/usr/bin/ld: dynLibTest.c:(.text+0x52): undefined reference to `sub'collect2: error: ld returned 1 exit status

所有我们还需要加上`-L库文件所在路径 l库名

代码语言:javascript代码运行次数:0运行复制

gcc -I./mylib/include -L./mylib/lib dynLibTest.c -lmyc

‘-I’:指定头文件搜索路径。—L:指定库文件搜索路径。-l:指明需要链接库文件路径下的哪一个库。

然后我们就可以得到一个可执行文件了,但是如果你去执行它就会发现,它居然不能执行!!!

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ ./a.out ./a.out: error while loading shared libraries: libmyc.so: cannot open shared object file: No such file or directory

为什么会这样?

算了,先用ldd看看它依赖的动态库的吧。

【Linux】掌握库的艺术:我的动静态库封装之旅ldd

居然没有连到,岂有此理,辛苦打了一连串指令竟然完全没作用。

其实不是这样啦,因为是动态库,在执行时我们仍然需要知道动态库的位置在哪。

对于动态库,编译时会搜索库的路径,运行时也会搜索库的路径。

为了解决这个问题,我们有4种解决方案:

直接将库进行安装(拷贝)到系统库当中。(最傻瓜操作)将不在系统默认库搜索路径下的库路径,添加到LD_LIBRARY_PATH3.使用idconfig指令。

拷贝到系统目录

天工AI 天工AI

昆仑万维推出的国内首款融入大语言模型的AI对话问答、AI搜索引擎,知识从这里开始。

天工AI 400 查看详情 天工AI 代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ sudo cp mylib/lib/libmyc.so /lib/x86_64-linux-gnu/ubuntu@VM-20-9-ubuntu:~/libraryTest$ ldd a.out linux-vdso.so.1 (0x00007fffc4f54000)libmyc.so => /lib/x86_64-linux-gnu/libmyc.so (0x00007f35c8f9c000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f35c8d73000)/lib64/ld-linux-x86-64.so.2 (0x00007f35c8fae000)

这样虽然简单,但是由于我们自己书写的库,大概率是不成熟的,这样会污染系统库目录。

更改环境变量

代码语言:javascript代码运行次数:0运行复制

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:home/ubuntu/libraryTest/mylib/lib

ldd查看:

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:home/ubuntu/libraryTest/mylib/libubuntu@VM-20-9-ubuntu:~/libraryTest$ ldd a.outlinux-vdso.so.1 (0x00007fff8fdee000)libmyc.so (0x00007f85aaf99000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f85aad6a000)/lib64/ld-linux-x86-64.so.2 (0x00007f85aafa5000)

该方法为临时方法,在系统重新登入时就失效了。

使用Idconfig指令

/etc/ld.so.conf.d/目录下的文件用来指定动态库搜索路径。这些文件被包含在/etc/ld.so.conf文件中,ldconfig命令会在默认搜寻目录(/lib和/usr/lib)以及动态库配置文件/etc/ld.so.conf内所列的目录下,搜索可共享的动态链接库,并创建出动态装入程序(ld.so)所需的连接和缓存文件。

.conf文件用来存储各种文件的路径,我们只要把我们自己写的第三方库的路径存放进去,程序运行时就会去里面找了。

代码语言:javascript代码运行次数:0运行复制

echo /home/ubuntu/libraryTest/mylib/lib > libmyc.conf

然后我们需要将,.conf文件拷贝到/etc/ld.so.conf.d/下。

【Linux】掌握库的艺术:我的动静态库封装之旅ldconfig

代码语言:javascript代码运行次数:0运行复制

sudo cp libmyc.conf /etc/ld.so.conf.d/

此时如果我们直接ldd是无法找到的,我们来需要更新一下。

代码语言:javascript代码运行次数:0运行复制

sudo ldconfig

然后就可以了

【Linux】掌握库的艺术:我的动静态库封装之旅在这里插入图片描述

那么动态库就先到这里,下面我们开始静态库的讲解。

2.2 封装静态库

静态库的操作会比动态库的更为简单。

代码的话,依然是上面的那些代码。

代码语言:javascript代码运行次数:0运行复制

add.c sub.c add.h sub.h staLibTest.c

staLibTest.c里面的代码和dynLibTest.c

首先我们需要把所有的.c源文件都编译为目标文件。(此时不在需要位置无关码)

代码语言:javascript代码运行次数:0运行复制

gcc -c add.cgcc -c sub.c

我们可以直接将主程序和这些.o文件进行编译。

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ gcc -o staticBin staLibTest.c add.o sub.oubuntu@VM-20-9-ubuntu:~/libraryTest$ ./staticBin 100 + 200 = 300100 - 200 = -100

正常编译,不过目标文件少的情况下还好,如果有很多的目标文件,我们就有点吃力了。为此我们可以打一个包。

使用ar指令将目标文件打包为静态库:

语法:

代码语言:javascript代码运行次数:0运行复制

ar [选项] [库名] [依赖文件]

主要功能:

r:插入文件。如果目标文件已经存在,则替换它。c:创建一个新的库,如果库文件不存在。s:创建索引,以加快链接过程。t:列出库中包含的文件。x:从库中提取指定的文件。d:从库中删除指定的文件。v:查看库的信息举例:将add.osub.o打包为静态库代码语言:javascript代码运行次数:0运行复制

ar -rc libmyc.a add.o sub.o

利用-t -v选项来查看静态库的文件以及信息。

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ ar -tv libmyc.arw-r--r-- 0/0   1224 Jan  1 08:00 1970 add.orw-r--r-- 0/0   1224 Jan  1 08:00 1970 sub.o

和动态库时一样,我们现在利用makefile来完成这些操作

代码语言:javascript代码运行次数:0运行复制

libmyc.a:add.o sub.oar -rc libmyc.a add.o sub.oadd.o:add.cgcc -o add.o -c add.csub.o:sub.cgcc -o sub.o -c sub.c.PHONY:outputoutput:mkdir -p mylibs/libmkdir -p mylibs/includecp -rf *.h mylibs/includecp -rf *.a mylibs/lib.PHONY:cleanclean:rm -rf *.o mylibs libmyc.a
【Linux】掌握库的艺术:我的动静态库封装之旅makefile

2.2.1 使用静态库‘-I’:指定头文件搜索路径。—L:指定库文件搜索路径。-l:指明需要链接库文件路径下的哪一个库。

在动态库的篇章中,我们就已经了解了指定库的路径了。

在静态库也同理。

代码语言:javascript代码运行次数:0运行复制

gcc staLibTest.c -I./mybins/include -L./mylibs/lib -lmyc

编译完后就可以直接运行了。

代码语言:javascript代码运行次数:0运行复制

ubuntu@VM-20-9-ubuntu:~/libraryTest$ gcc staLibTest.c -I./mybins/include -L./mylibs/lib -lmyc -o testbin -staticubuntu@VM-20-9-ubuntu:~/libraryTest$ ./testbin 100 + 200 = 300100 - 200 = -100

同样的,如果你觉得没错都显示指定库连接很麻烦,可以把目标文件拷贝到系统库当中。

3. 动静态库小知识3.1 gcc对动静态的优先级

如果我们同时提供动态库和静态库,gcc会默认使用动态库。如果我们非要静态链接,必须使用static指定。如果我们只提供静态库,那我们的程序只能对该库进行静态链接,但是程序不一定整体是静态链接的。如果我们只提供动态库,默认只能动态连接,非要静态链接的话会报错。

3.2 动静态库的区别

特性

静态库

动态库

文件后缀

.a

.so (Linux), .dll (Windows)

链接方式

编译时链接

运行时链接

可执行文件大小

较大(包含所有库代码)

较小(只包含引用)

外部依赖

需要在运行时提供

更新方式

需要重新编译所有依赖的程序

只需替换库文件

性能

加载速度快

加载速度相对较慢

共享性

不支持多个进程共享

支持多个进程共享

4.总结

动静态库各有优缺点,选择使用哪种库通常取决于具体的项目需求、资源限制和开发环境。静态库适用于对依赖性和更新不敏感的应用,而动态库则更灵活,适合需要频繁更新和共享代码的场景。在实际开发中,合理选择和使用这两种库能够提高代码的复用性和维护性。

以上就是【Linux】掌握库的艺术:我的动静态库封装之旅的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/727506.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月25日 10:08:19
下一篇 2025年11月25日 10:09:52

相关推荐

  • CSS mask属性无法获取图片:为什么我的图片不见了?

    CSS mask属性无法获取图片 在使用CSS mask属性时,可能会遇到无法获取指定照片的情况。这个问题通常表现为: 网络面板中没有请求图片:尽管CSS代码中指定了图片地址,但网络面板中却找不到图片的请求记录。 问题原因: 此问题的可能原因是浏览器的兼容性问题。某些较旧版本的浏览器可能不支持CSS…

    2025年12月24日
    900
  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 为什么设置 `overflow: hidden` 会导致 `inline-block` 元素错位?

    overflow 导致 inline-block 元素错位解析 当多个 inline-block 元素并列排列时,可能会出现错位显示的问题。这通常是由于其中一个元素设置了 overflow 属性引起的。 问题现象 在不设置 overflow 属性时,元素按预期显示在同一水平线上: 不设置 overf…

    2025年12月24日 好文分享
    400
  • 网页使用本地字体:为什么 CSS 代码中明明指定了“荆南麦圆体”,页面却仍然显示“微软雅黑”?

    网页中使用本地字体 本文将解答如何将本地安装字体应用到网页中,避免使用 src 属性直接引入字体文件。 问题: 想要在网页上使用已安装的“荆南麦圆体”字体,但 css 代码中将其置于第一位的“font-family”属性,页面仍显示“微软雅黑”字体。 立即学习“前端免费学习笔记(深入)”; 答案: …

    2025年12月24日
    000
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • 为什么我的特定 DIV 在 Edge 浏览器中无法显示?

    特定 DIV 无法显示:用户代理样式表的困扰 当你在 Edge 浏览器中打开项目中的某个 div 时,却发现它无法正常显示,仔细检查样式后,发现是由用户代理样式表中的 display none 引起的。但你疑问的是,为什么会出现这样的样式表,而且只针对特定的 div? 背后的原因 用户代理样式表是由…

    2025年12月24日
    200
  • CSS元素设置em和transition后,为何载入页面无放大效果?

    css元素设置em和transition后,为何载入无放大效果 很多开发者在设置了em和transition后,却发现元素载入页面时无放大效果。本文将解答这一问题。 原问题:在视频演示中,将元素设置如下,载入页面会有放大效果。然而,在个人尝试中,并未出现该效果。这是由于macos和windows系统…

    2025年12月24日
    200
  • inline-block元素错位了,是为什么?

    inline-block元素错位背后的原因 inline-block元素是一种特殊类型的块级元素,它可以与其他元素行内排列。但是,在某些情况下,inline-block元素可能会出现错位显示的问题。 错位的原因 当inline-block元素设置了overflow:hidden属性时,它会影响元素的…

    2025年12月24日
    000
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 为什么使用 inline-block 元素时会错位?

    inline-block 元素错位成因剖析 在使用 inline-block 元素时,可能会遇到它们错位显示的问题。如代码 demo 所示,当设置了 overflow 属性时,a 标签就会错位下沉,而未设置时却不会。 问题根源: overflow:hidden 属性影响了 inline-block …

    2025年12月24日
    000
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 为什么我的 CSS 元素放大效果无法正常生效?

    css 设置元素放大效果的疑问解答 原提问者在尝试给元素添加 10em 字体大小和过渡效果后,未能在进入页面时看到放大效果。探究发现,原提问者将 CSS 代码直接写在页面中,导致放大效果无法触发。 解决办法如下: 将 CSS 样式写在一个单独的文件中,并使用 标签引入该样式文件。这个操作与原提问者观…

    2025年12月24日
    000
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 为什么我的 em 和 transition 设置后元素没有放大?

    元素设置 em 和 transition 后不放大 一个 youtube 视频中展示了设置 em 和 transition 的元素在页面加载后会放大,但同样的代码在提问者电脑上没有达到预期效果。 可能原因: 问题在于 css 代码的位置。在视频中,css 被放置在单独的文件中并通过 link 标签引…

    2025年12月24日
    100

发表回复

登录后才能评论
关注微信