如何在Golang中模拟依赖项 介绍gomock和testify的使用技巧

golang中模拟依赖项的核心方法是通过接口抽象和工具辅助实现测试隔离。1. 使用gomock生成mock对象,根据接口定义自动生成模拟实现;2. 在测试中使用gomock控制器设置期望行为,如调用次数和返回值;3. 结合testify的assert或require进行断言,提升测试代码可读性和效率。这种方式确保测试不依赖外部资源,提升速度与稳定性,同时便于控制各种场景并验证代码逻辑正确性。

如何在Golang中模拟依赖项 介绍gomock和testify的使用技巧

在Golang中模拟依赖项,核心在于通过接口抽象和工具辅助来隔离测试单元。gomocktestify这对组合,简直是Go测试领域的黄金搭档。gomock负责生成和管理模拟(mock)对象,让你能精确控制依赖的行为;而testify则提供了一套强大且易用的断言库,让你的测试代码读起来更像自然语言,清晰地表达你期望的结果。它们共同构筑了一个高效、可靠的单元测试环境,确保你的代码在复杂依赖面前依然能够独立接受考验。

如何在Golang中模拟依赖项 介绍gomock和testify的使用技巧

解决方案

模拟依赖的根本,在于将具体的实现细节隐藏在接口之后。当你的服务(或任何一个待测试的单元)依赖于外部资源,比如数据库、HTTP客户端、文件系统,或者其他微服务时,直接在单元测试中调用这些真实依赖往往是不切实际的。它们可能很慢,不稳定,或者根本无法在测试环境中访问。

如何在Golang中模拟依赖项 介绍gomock和testify的使用技巧

gomock的介入,让这一切变得简单。它能根据你定义的接口,自动生成一个“假”的实现——一个模拟对象。这个模拟对象可以被编程,告诉它在接收到特定方法调用时,应该返回什么值,甚至应该被调用多少次。

立即学习“go语言免费学习笔记(深入)”;

基本流程是这样的:

如何在Golang中模拟依赖项 介绍gomock和testify的使用技巧

定义接口: 你的服务不应该直接依赖具体的结构体,而是依赖一个接口。这是Go语言里最重要的一环。例如,一个数据存储服务可能依赖Storage接口。

// storage.gopackage serviceimport "context"type Data struct {    ID   string    Name string}type Storage interface {    Get(ctx context.Context, id string) (*Data, error)    Save(ctx context.Context, data *Data) error}type MyService struct {    store Storage}func NewMyService(s Storage) *MyService {    return &MyService{store: s}}func (s *MyService) GetDataAndProcess(ctx context.Context, id string) (string, error) {    data, err := s.store.Get(ctx, id)    if err != nil {        return "", err    }    // 假设这里有一些业务逻辑处理data    return "Processed: " + data.Name, nil}

生成Mock对象: 使用mockgen工具,根据你的接口生成mock文件。

go install github.com/golang/mock/mockgen@latestmockgen -source=storage.go -destination=mock_storage_test.go -package=service_test

这会生成一个mock_storage_test.go文件,里面包含了MockStorage结构体,它实现了Storage接口。

编写测试: 在你的测试文件中,你可以使用gomock.NewController创建一个控制器,然后用它来实例化你的MockStorage

// service_test.gopackage service_testimport (    "context"    "errors"    "testing"    "github.com/stretchr/testify/assert" // 或者 require    "go-project/service" // 假设你的服务包路径    mock_service "go-project/service/mocks" // mock文件通常放在mocks目录下    "go.uber.org/mock/gomock" // 注意:新版本gomock的导入路径)func TestMyService_GetDataAndProcess(t *testing.T) {    ctrl := gomock.NewController(t)    defer ctrl.Finish() // 确保所有期望都被满足    mockStore := mock_service.NewMockStorage(ctrl)    myService := service.NewMyService(mockStore)    ctx := context.Background()    // 场景1: 成功获取数据    t.Run("should return processed data on success", func(t *testing.T) {        expectedData := &service.Data{ID: "123", Name: "TestItem"}        mockStore.EXPECT().Get(ctx, "123").Return(expectedData, nil).Times(1)        result, err := myService.GetDataAndProcess(ctx, "123")        assert.NoError(t, err)        assert.Equal(t, "Processed: TestItem", result)    })    // 场景2: 获取数据失败    t.Run("should return error when data retrieval fails", func(t *testing.T) {        mockStore.EXPECT().Get(ctx, "456").Return(nil, errors.New("database error")).Times(1)        result, err := myService.GetDataAndProcess(ctx, "456")        assert.Error(t(t), err)        assert.Empty(t, result)        assert.Contains(t, err.Error(), "database error")    })}

在这个例子中,mockStore.EXPECT().Get(ctx, "123").Return(expectedData, nil).Times(1)这行代码就是gomock的魔法所在。它告诉mockStore:当它的Get方法被调用时,如果参数是ctx"123",那么就返回expectedDatanil错误,并且这个调用预期只发生一次。

testify/assert(或者testify/require)则让断言变得异常简洁。assert.NoError(t, err)比手动检查if err != nil要优雅得多,而且能提供更丰富的错误信息。

为什么在Go语言测试中模拟依赖至关重要?

这问题问得好,因为这不单单是技术选择,更是一种测试哲学。我个人觉得,模拟依赖的重要性,远超你想象。它就像给你的测试代码穿上了一层隐形斗篷,让它能够专注于它真正应该关注的核心——也就是你正在测试的那个单元自身的逻辑。

首先,是隔离性。单元测试的精髓就在于“单元”二字。如果你的测试需要连接真实数据库,访问外部API,或者读取文件系统,那它就不是一个纯粹的单元测试了。一旦这些外部依赖出现问题(比如网络抖动、数据库宕机、API限流),你的测试就会莫名其妙地失败,而这失败与你正在测试的代码逻辑本身毫无关系。模拟依赖,能确保你的测试结果只反映被测代码的正确性,而不是外部环境的稳定性。

再来,是速度。真实的网络请求、数据库操作,哪怕是本地的文件读写,都比内存操作慢上几个数量级。想象一下,一个拥有几百上千个测试用例的项目,如果每个测试都要等待外部依赖响应,那整个测试套件跑下来,可能要花上几分钟甚至几十分钟。这对于开发者的反馈循环来说是致命的。模拟依赖,能让你的测试在毫秒级别完成,极大地提升了开发效率和迭代速度。你写完代码,跑个测试,结果立刻就出来了,这种即时反馈简直是福音。

还有,场景控制。真实世界是复杂的,有些边缘情况、错误路径是很难在真实依赖中复现的。比如,数据库连接突然中断,或者外部API返回一个非预期的错误码。通过模拟依赖,你可以轻松地编程让模拟对象在特定条件下返回错误、空数据,或者任何你想要的响应。这样,你就能彻底地测试你的错误处理逻辑和各种异常路径,确保代码的健壮性。这就像在实验室里精确控制变量,去验证某个假设。

最后,是并行开发。在团队协作中,一个模块可能依赖于另一个尚未完成的模块。如果你的测试不模拟依赖,你就得等到所有依赖都开发并部署好才能进行测试。这显然不现实。有了模拟,你可以为尚未完成的依赖定义接口,然后生成模拟对象,这样你的模块就可以独立地进行开发和测试,大大降低了开发流程中的耦合度,提升了并行开发的效率。

所以,在我看来,模拟依赖不只是一种技术手段,它更是构建高效、可靠、可维护测试体系的基石。

如何使用gomock生成和控制模拟对象?

使用gomock来生成和控制模拟对象,说实话,一开始可能会觉得有点绕,但一旦你掌握了它的核心概念,你会发现它真的非常强大,而且用起来相当顺手。它基本上就是把接口编程的优势,在测试层面发挥到了极致。

生成模拟对象

第一步,也是最关键的一步,就是生成模拟对象。这需要用到mockgen这个命令行工具。

安装 mockgen 如果你还没安装,先把它装上。

go install go.uber.org/mock/mockgen@latest

注意,gomock最近将主仓库迁移到了go.uber.org/mock,所以安装时请使用这个新路径。

选择生成方式: mockgen有两种主要的工作模式:

Source 模式: 这是最常用的。你指定一个Go源文件,mockgen会扫描这个文件,找到你指定的接口,然后为这些接口生成模拟实现。

mockgen -source=path/to/your/interface.go -destination=path/to/your/mock_interface_test.go -package=your_test_package_name

例如,如果你的接口在internal/repo/user.go中定义,接口名为UserRepository,你想把mock文件放在internal/repo/mocks/mock_user_repo.go,并且测试包名为repo_test,命令会是:

mockgen -source=internal/repo/user.go -destination=internal/repo/mocks/mock_user_repo.go -package=repo_test

Reflect 模式: 这种模式适用于当你需要为已经编译好的包中的接口生成mock时。你指定包路径和接口名。

mockgen github.com/your/project/some/package InterfaceName > mock_interface.go

这种模式在某些CI/CD环境中可能更方便,因为它不需要原始源文件。但通常,Source模式更直接。

控制模拟对象行为

生成了mock文件后,你就可以在测试代码中像搭积木一样控制模拟对象的行为了。

创建控制器: 每个测试函数或测试套件都需要一个gomock.Controller。它负责管理mock对象的生命周期和验证所有的期望是否被满足。

import "go.uber.org/mock/gomock"func TestSomething(t *testing.T) {    ctrl := gomock.NewController(t)    defer ctrl.Finish() // 这一行至关重要!它会在测试结束时验证所有预期的调用是否都发生了。    // ...}

实例化模拟对象: 使用控制器来实例化你生成的mock对象。

mockUserRepo := mock_repo.NewMockUserRepository(ctrl) // 假设mock文件生成在mock_repo包下

设置期望(Expectations): 这是gomock的核心。你告诉mock对象,当它的某个方法被调用时,应该做什么。

// 预期 GetUserByID 方法被调用,参数是 123,返回一个 User 对象和 nil 错误mockUserRepo.EXPECT().GetUserByID(gomock.Any(), "123").Return(&User{ID: "123", Name: "Alice"}, nil).Times(1)

EXPECT():这是设置期望的入口。GetUserByID(...):调用mock对象上你想要模拟的方法。gomock.Any():这是一个匹配器,表示任何值都可以匹配这个参数。你也可以使用gomock.Eq("some_value")来精确匹配。Return(...):定义这个方法被调用时应该返回什么值。Times(1):定义这个方法预期被调用多少次。你也可以用MinTimes(1)MaxTimes(2)AnyTimes()Do(func(ctx context.Context, id string){ /* ... */ }):如果你需要在方法被调用时执行一些副作用(比如修改某个变量),可以使用DoDoAndReturn(func(...) (ret1, ret2, ...){ /* ... */ }):在执行副作用的同时返回指定的值。

一个典型的场景,模拟一个HTTP客户端:

type HTTPClient interface {    Do(req *http.Request) (*http.Response, error)}// 生成mock_http_client.go// mockgen -source=http_client.go -destination=mock_http_client.go -package=your_test_package// 在测试中mockClient := mock_your_test_package.NewMockHTTPClient(ctrl)req, _ := http.NewRequest("GET", "http://example.com", nil)resp := &http.Response{StatusCode: http.StatusOK, Body: io.NopCloser(bytes.NewBufferString("ok"))}mockClient.EXPECT().Do(gomock.Eq(req)).Return(resp, nil).Times(1)// 你的代码调用 mockClient.Do(req) 时,就会得到 resp 和 nil

gomock的强大之处在于它的灵活性。你可以设置非常精确的匹配规则,比如匹配特定的参数值,或者匹配参数的类型。你也可以定义一系列的调用顺序,确保方法是按照你预期的顺序被调用的。这种精细的控制能力,让你的测试能够覆盖到各种复杂的业务场景和错误路径,而不用担心真实依赖的不可控性。

testify/assert与testify/require在测试断言中的最佳实践是什么?

testify库,特别是它的assertrequire模块,是Go语言测试中我个人觉得不可或缺的工具。它们把Go标准库testing包里那些略显啰嗦的if err != nil { t.Errorf(...) }语句,变得异常简洁和富有表现力。但它们俩虽同根同源,用法上却有着微妙但重要的区别,理解这些区别是实践中至关重要的一环。

assert:继续执行,记录所有失败

assert包中的断言函数,当断言失败时,会记录错误信息,但不会停止当前测试函数的执行。它会继续执行测试函数中后续的代码。

最佳实践场景:

多个独立断言: 当你有一个测试函数,里面包含多个相互独立的断言,并且你希望即便其中一个断言失败,也能看到所有其他断言的结果时,使用assert。这有助于你一次性发现所有问题,而不是每次只看到一个失败。

import "github.com/stretchr/testify/assert"func TestProcessData(t *testing.T) {    result, err := processData(input)    assert.NoError(t, err, "processData should not return an error") // 即使这里失败,也会继续执行    assert.NotNil(t, result, "result should not be nil")    assert.Equal(t, "expected_value", result.Value, "result value should match")    assert.Len(t, result.Items, 3, "result items count should be 3")}

非关键性检查: 对于那些即使失败也不会影响后续逻辑,或者你希望在一次运行中收集尽可能多错误信息的断言,assert是理想选择。

require:立即停止,快速失败

require包中的断言函数,当断言失败时,不仅会记录错误信息,还会立即调用t.FailNow()(或t.Fatal()),终止当前测试函数的执行

最佳实践场景:

前置条件检查: 当一个断言的成功是后续测试逻辑能够继续执行的必要条件时,使用require。例如,如果你的测试依赖于某个初始化操作必须成功,或者某个数据对象必须非空,否则后续的断言就没有意义。

import "github.com/stretchr/testify/require"func TestUserCreation(t *testing.T) {    // 确保用户创建成功,否则后续的获取和更新测试就没有意义    user, err := createUser("test_user")    require.NoError(t, err, "user creation should succeed") // 如果失败,测试立即停止    require.NotNil(t, user, "created user should not be nil")    // 后续的测试逻辑,依赖于user和err为nil    fetchedUser, err := getUser(user.ID)    require.NoError(t, err, "fetching user should succeed")    require.Equal(t, user.Name, fetchedUser.Name, "fetched user name should match")}

避免空指针解引用: 想象一个场景,你断言一个返回对象不为nil,然后立即尝试访问它的字段。如果断言失败但测试继续,很可能导致后续的空指针解引用错误,这会掩盖真正的失败原因。使用require.NotNil可以避免这种情况。

通用最佳实践:

选择正确的断言类型: 这是一个哲学问题,也是一个实用问题。我通常倾向于在测试的“设置”阶段使用require,确保测试环境和前提条件是正确的。在“执行”和“验证”阶段,如果各个断言相对独立,我会使用assert来收集更多信息;如果一个断言失败会导致后续断言完全无意义,那还是会用require明确的错误消息: testify的断言函数都支持传入一个可选的msg参数(以及args)。务必利用这个功能,提供清晰、有用的错误信息。当测试失败时,这些信息能帮你快速定位问题所在。

assert.Equal(t, expected, actual, "结果不符合预期,输入为:%v", input)

使用最具体的断言: testify提供了非常丰富的断言函数,比如EqualNotEqualTrueFalseNilNotNilErrorNoErrorContainsLen等等。尽可能使用最能表达你意图的那个函数,而不是万能的True(t, condition)。这让你的测试代码更具可读性。避免过度断言: 一个测试用例应该只测试一个特定的行为或场景。不要在一个测试函数中塞入过多不相关的断言。如果你的测试函数变得很长,可能意味着它承担了过多的职责,考虑拆分。结构化测试: 结合t.Run来组织你的测试用例,为每个场景提供一个清晰的名称。这样,即使断言失败,你也能一眼看出是哪个具体场景出了问题。

testify的断言,就像是给你的测试代码加上了清晰的旁白。它们不只是检查结果,更是在讲述你的代码应该如何表现的故事。熟练运用assertrequire,能让你的测试既高效又易于维护。

以上就是如何在Golang中模拟依赖项 介绍gomock和testify的使用技巧的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月15日 13:13:18
下一篇 2025年12月15日 13:13:41

相关推荐

  • 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
  • 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
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信