单测时尽量用fake object

1. 单元测试的难点:外部协作者(external collaborators)的存在

单元测试是软件开发的一个重要部分,它有助于在开发周期的早期发现错误,帮助开发人员增加对生产代码正常工作的信心,同时也有助于改善代码设计。**Go语言从诞生那天起就内置Testing框架(以及测试覆盖率计算工具)**,基于该框架,Gopher们可以非常方便地为自己设计实现的package编写测试代码。

注:《Go语言精进之路》vol2[1]中的第40条到第44条有关于Go包内、包外测试区别、测试代码组织、表驱动测试、管理外部测试数据等内容的系统地讲解,感兴趣的童鞋可以读读。

不过即便如此,在实际开发工作中,大家发现单元测试的覆盖率依旧很低,究其原因,排除那些对测试代码不作要求的组织,剩下的无非就是代码设计不佳,使得代码不易测;或是代码有外部协作者(比如数据库、redis、其他服务等)。代码不易测可以通过重构来改善,但如果代码有外部协作者,我们该如何对代码进行测试呢,这也是各种编程语言实施单元测试的一大共同难点

为此,《xUnit Test Patterns : Refactoring Test Code》[2]一书中提供了**Test Double(测试替身)**的概念专为解决此难题。那么什么是Test Double呢?我们接下来就来简单介绍一下Test Double的概念以及常见的种类。

2. 什么是Test Double?

测试替身是在测试阶段用来替代被测系统依赖的真实组件的对象或程序(如下图),以方便测试,这些真实组件或程序即是外部协作者(external collaborators)。这些外部协作者在测试环境下通常很难获取或与之交互。测试替身可以使开发人员或QA专业人员专注于新的代码而不是代码与环境集成。

d2f7ddfb498f57432ac8e10d662b8d92.png

测试替身是通用术语,指的是不同类型的替换对象或程序。目前xUnit Patterns[3]至少定义了五种类型的Test Doubles:

  • Test stubs

  • Mock objects

  • Test spies

  • Fake objects

  • Dummy objects

这其中最为常用的是Fake objects、stub和mock objects。下面逐一说说这三种test double:

2.1 fake object

fake object最容易理解,它是被测系统SUT(System Under Test)依赖的外部协作者的“替身”,和真实的外部协作者相比,fake object外部行为表现与真实组件几乎是一致的,但更简单也更易于使用,实现更轻量,仅用于满足测试需求即可。

fake object也是Go testing中最为常用的一类fake object。以Go的标准库为例,我们在src/database/sql下面就看到了Go标准库为进行sql包测试而实现的一个database driver:

// $GOROOT/src/database/fakedb_test.govar fdriver driver.Driver = &fakeDriver{}func init() {Register("test", fdriver)
}

我们知道一个真实的sql数据库的代码量可是数以百万计的,这里不可能实现一个生产级的真实SQL数据库,从fakedb_test.go源文件的注释我们也可以看到,这个fakeDriver仅仅是用于testing,它是一个实现了driver.Driver接口的、支持少数几个DDL(create)、DML(insert)和DQL(selet)的toy版的纯内存数据库:

// fakeDriver is a fake database that implements Go's driver.Driver
// interface, just for testing.
//
// It speaks a query language that's semantically similar to but
// syntactically different and simpler than SQL.  The syntax is as
// follows:
//
//  WIPE
//  CREATE|<tablename>|<col>=<type>,<col>=<type>,...
//    where types are: "string", [u]int{8,16,32,64}, "bool"
//  INSERT|<tablename>|col=val,col2=val2,col3=?
//  SELECT|<tablename>|projectcol1,projectcol2|filtercol=?,filtercol2=?
//  SELECT|<tablename>|projectcol1,projectcol2|filtercol=?param1,filtercol2=?param2

与此类似的,Go标准库中还有net/dnsclient_unix_test.go中的fakeDNSServer等。此外,Go标准库中一些以mock做前缀命名的变量、类型等其实质上是fake object。

我们再来看第二种test double: stub。

2.2 stub

stub显然也是一个在测试阶段专用的、用来替代真实外部协作者与SUT进行交互的对象。与fake object稍有不同的是,stub是一个内置了预期值/响应值且可以在多个测试间复用的替身object。

stub可以理解为一种fake object的特例。

注:fakeDriver在sql_test.go中的不同测试场景中时而是fake object,时而是stub(见sql_test.go中的newTestDBConnector函数)。

Go标准库中的net/http/httptest就是一个提供创建stub的典型的测试辅助包,十分适合对http.Handler进行测试,这样我们无需真正启动一个http server。下面就是基于httptest的一个测试例子:

// 被测对象 client.gopackage mainimport ("bytes""net/http"
)// Function that uses the client to make a request and parse the response
func GetResponse(client *http.Client, url string) (string, error) {req, err := http.NewRequest("GET", url, nil)if err != nil {return "", err}resp, err := client.Do(req)if err != nil {return "", err}defer resp.Body.Close()buf := new(bytes.Buffer)_, err = buf.ReadFrom(resp.Body)if err != nil {return "", err}return buf.String(), nil
}// 测试代码 client_test.gopackage mainimport ("net/http""net/http/httptest""testing"
)func TestClient(t *testing.T) {// Create a new test server with a handler that returns a specific responseserver := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {w.WriteHeader(http.StatusOK)w.Write([]byte(`{"message": "Hello, world!"}`))}))defer server.Close()// Create a new client that uses the test serverclient := server.Client()// Call the function that uses the clientmessage, err := GetResponse(client, server.URL)// Check that the response is correctexpected := `{"message": "Hello, world!"}`if message != expected {t.Errorf("Expected response %q, but got %q", expected, message)}// Check that no errors were returnedif err != nil {t.Errorf("Unexpected error: %v", err)}
}

在这个例子中,我们要测试一个名为GetResponse的函数,该函数通过client向url发送Get请求,并将收到的响应内容读取出来并返回。为了测试这个函数,我们需要“建立”一个与GetResponse进行协作的外部http server,这里我们使用的就是httptest包。我们通过httptest.NewServer建立这个server,该server预置了一个返回特定响应的HTTP handler。我们通过该server得到client和对应的url参数后,将其传给被测目标GetResponse,并将其返回的结果与预期作比较来完成这个测试。注意,我们在测试结束后使用defer server.Close()来关闭测试服务器,以确保该服务器不会在测试结束后继续运行。

httptest还常用来做http.Handler的测试,比如下面这个例子:

// handler.gopackage mainimport ("bytes""io""net/http"
)func AddHelloPrefix(w http.ResponseWriter, r *http.Request) {b, err := io.ReadAll(r.Body)if err != nil {w.WriteHeader(http.StatusBadRequest)return}w.Write(bytes.Join([][]byte{[]byte("hello, "), b}, nil))w.WriteHeader(http.StatusOK)
}// handler_test.gopackage mainimport ("net/http""net/http/httptest""strings""testing"
)func TestHandler(t *testing.T) {r := strings.NewReader("world!")req, err := http.NewRequest("GET", "/test", r)if err != nil {t.Fatal(err)}rr := httptest.NewRecorder()handler := http.HandlerFunc(AddHelloPrefix)handler.ServeHTTP(rr, req)if status := rr.Code; status != http.StatusOK {t.Errorf("handler returned wrong status code: got %v want %v",status, http.StatusOK)}expected := "hello, world!"if rr.Body.String() != expected {t.Errorf("handler returned unexpected body: got %v want %v",rr.Body.String(), expected)}
}

在这个例子中,我们创建一个新的http.Request对象,用于向/test路径发出GET请求。然后我们创建一个新的httptest.ResponseRecorder对象来捕获服务器的响应。 我们定义一个简单的HTTP Handler(被测函数): AddHelloPrefix,该Handler会在请求的内容之前加上"hello, "并返回200 OK状态代码作为响应体。之后,我们在handler上调用ServeHTTP方法,传入httptest.ResponseRecorder和http.Request对象,这会将请求“发送”到处理程序并捕获响应。最后,我们使用标准的Go测试包来检查响应是否具有预期的状态码和正文。

在这个例子中,我们利用net/http/httptest创建了一个测试服务器“替身”,并向其“发送”间接预置信息的请求以测试Go中的HTTP handler。这个过程中其实并没有任何网络通信,也没有http协议打包和解包的过程,我们也不关心http通信,那是Go net/http包的事情,我们只care我们的Handler是否能按逻辑运行。

fake object与stub的优缺点基本一样。多数情况下,大家也无需将这二者划分的很清晰

2.3 mock object

和fake/stub一样,mock object也是一个测试替身。通过上面的例子我们看到fake建立困难(比如创建一个近2千行代码的fakeDriver),但使用简单。而mock object则是一种建立简单,使用简单程度因被测目标与外部协作者交互复杂程度而异的test double,我们看一下下面这个例子:

// db.go 被测目标package main// Define the `Database` interface
type Database interface {Save(data string) errorGet(id int) (string, error)
}// Example functions that use the `Database` interface
func saveData(db Database, data string) error {return db.Save(data)
}func getData(db Database, id int) (string, error) {return db.Get(id)
}// 测试代码package mainimport ("testing""github.com/stretchr/testify/assert""github.com/stretchr/testify/mock"
)// Define a mock struct that implements the `Database` interface
type MockDatabase struct {mock.Mock
}func (m *MockDatabase) Save(data string) error {args := m.Called(data)return args.Error(0)
}func (m *MockDatabase) Get(id int) (string, error) {args := m.Called(id)return args.String(0), args.Error(1)
}func TestSaveData(t *testing.T) {// Create a new mock databasedb := new(MockDatabase)// Expect the `Save` method to be called with "test data"db.On("Save", "test data").Return(nil)// Call the code that uses the databaseerr := saveData(db, "test data")// Assert that the `Save` method was called with the correct argumentdb.AssertCalled(t, "Save", "test data")// Assert that no errors were returnedassert.NoError(t, err)
}func TestGetData(t *testing.T) {// Create a new mock databasedb := new(MockDatabase)// Expect the `Get` method to be called with ID 123 and return "test data"db.On("Get", 123).Return("test data", nil)// Call the code that uses the databasedata, err := getData(db, 123)// Assert that the `Get` method was called with the correct argumentdb.AssertCalled(t, "Get", 123)// Assert that the correct data was returnedassert.Equal(t, "test data", data)// Assert that no errors were returnedassert.NoError(t, err)
}

在这个例子中,被测目标是两个接受Database接口类型参数的函数:saveData和getData。显然在单元测试阶段,我们不能真正为这两个函数传入真实的Database实例去测试。

这里,我们没有使用fake object,而是定义了一个mock object:MockDatabase,该类型实现了Database接口。然后我们定义了两个测试函数,TestSaveData和TestGetData,它们分别使用MockDatabase实例来测试saveData和getData函数。

在每个测试函数中,我们对MockDatabase实例进行设置,包括期待特定参数的方法调用,然后调用使用该数据库的代码(即被测目标函数saveData和getData)。然后我们使用github.com/stretchr/testify中的assert包,对代码的预期行为进行断言。

注:除了上述测试中使用的AssertCalled方法外,MockDatabase结构还提供了其他方法来断言方法被调用的次数、方法被调用的顺序等。请查看github.com/stretchr/testify/mock包的文档,了解更多信息。

3. Test Double有多种,选哪个呢?

从mock object的例子来看,测试代码的核心就是mock object的构建与mock object的方法的参数和返回结果的设置,相较于fake object的简单直接,mock object在使用上较为难于理解。而且对Go语言来说,mock object要与接口类型联合使用,如果被测目标的参数是非接口类型,mock object便“无从下嘴”了。此外,mock object使用难易程度与被测目标与外部协作者的交互复杂度相关。像上面这个例子,建立mock object就比较简单。但对于一些复杂的函数,当存在多个外部协作者且与每个协作者都有多次交互的情况下,建立和设置mock object就将变得困难并更加难于理解。

mock object仅是满足了被测目标对依赖的外部协作者的调用需求,比如设置不同参数传入下的不同返回值,但mock object并未真实处理被测目标传入的参数,这会降低测试的可信度以及开发人员对代码正确性的信心。

此外,如果被测函数的输入输出未发生变化,但内部逻辑发生了变化,比如调用的外部协作者的方法参数、调用次数等,使用mock object的测试代码也需要一并更新维护。

而通过上面的fakeDriver、fakeDNSSever以及httptest应用的例子,我们看到:作为test double,fake object/stub有如下优点:

  • 我们与fake object的交互方式与与真实外部协作者交互的方式相同,这让其显得更简单,更容易使用,也降低了测试的复杂性;

  • fake objet的行为更像真正的协作者,可以给开发人员更多的信心;

  • 当真实协作者更新时,我们不需要更新使用fake object时设置的expection和结果验证条件,因此,使用fake object时,重构代码往往比使用其他test double更容易。

不过fake object也有自己的不足之处,比如:

  • fake object的创建和维护可能很费时,就像上面的fakeDriver,源码有近2k行;

  • fake object可能无法提供与真实组件相同的功能覆盖水平,这与fake object的提供方式有关。

  • fake object的实现需要维护,每当真正的协作者更新时,都必须更新fake object。

综上,测试的主要意义是保证SUT代码的正确性,让开发人员对自己编写的代码更有信心,从这个角度来看,我们在单测时应首选为外部协作者提供fake object以满足测试需要

4. fake object的实现和获取方法

随着技术的进步,fake object的实现和获取日益容易。

我们可以借助类似ChatGPT/copilot的工具快速构建出一个fake object,即便是几百行代码的fake object的实现也很容易。

如果要更高的可信度和更高的功能覆盖水平,我们还可以借助docker来构建“真实版/无阉割版”的fake object。

借助github上开源的testcontainers-go[4]可以更为简便的构建出一个fake object,并且testcontainer提供了常见的外部协作者的封装实现,比如:MySQL、Redis、Postgres等。

以测试redis client为例,我们使用testcontainer建立如下测试代码:

// redis_test.gopackage mainimport ("context""fmt""testing""github.com/go-redis/redis/v8""github.com/testcontainers/testcontainers-go""github.com/testcontainers/testcontainers-go/wait"
)func TestRedisClient(t *testing.T) {// Create a Redis container with a random port and wait for it to startreq := testcontainers.ContainerRequest{Image:        "redis:latest",ExposedPorts: []string{"6379/tcp"},WaitingFor:   wait.ForLog("Ready to accept connections"),}ctx := context.Background()redisC, err := testcontainers.GenericContainer(ctx, testcontainers.GenericContainerRequest{ContainerRequest: req,Started:          true,})if err != nil {t.Fatalf("Failed to start Redis container: %v", err)}defer redisC.Terminate(ctx)// Get the Redis container's host and portredisHost, err := redisC.Host(ctx)if err != nil {t.Fatalf("Failed to get Redis container's host: %v", err)}redisPort, err := redisC.MappedPort(ctx, "6379/tcp")if err != nil {t.Fatalf("Failed to get Redis container's port: %v", err)}// Create a Redis client and perform some operationsclient := redis.NewClient(&redis.Options{Addr: fmt.Sprintf("%s:%s", redisHost, redisPort.Port()),})defer client.Close()err = client.Set(ctx, "key", "value", 0).Err()if err != nil {t.Fatalf("Failed to set key: %v", err)}val, err := client.Get(ctx, "key").Result()if err != nil {t.Fatalf("Failed to get key: %v", err)}if val != "value" {t.Errorf("Expected value %q, but got %q", "value", val)}
}

运行该测试将看到类似如下结果:

$go test
2023/04/15 16:18:20 github.com/testcontainers/testcontainers-go - Connected to docker: Server Version: 20.10.8API Version: 1.41Operating System: Ubuntu 20.04.3 LTSTotal Memory: 10632 MB
2023/04/15 16:18:21 Failed to get image auth for docker.io. Setting empty credentials for the image: docker.io/testcontainers/ryuk:0.3.4. Error is:credentials not found in native keychain2023/04/15 16:19:06 Starting container id: 0d8341b2270e image: docker.io/testcontainers/ryuk:0.3.4
2023/04/15 16:19:10 Waiting for container id 0d8341b2270e image: docker.io/testcontainers/ryuk:0.3.4
2023/04/15 16:19:10 Container is ready id: 0d8341b2270e image: docker.io/testcontainers/ryuk:0.3.4
2023/04/15 16:19:28 Starting container id: 999cf02b5a82 image: redis:latest
2023/04/15 16:19:30 Waiting for container id 999cf02b5a82 image: redis:latest
2023/04/15 16:19:30 Container is ready id: 999cf02b5a82 image: redis:latest
PASS
ok   demo 73.262s

我们看到建立这种真实版的“fake object”的一大不足就是依赖网络下载container image且耗时过长,在单元测试阶段使用还是要谨慎一些。testcontainer更多也会被用在集成测试或冒烟测试上。

一些开源项目,比如etcd,也提供了用于测试的自身简化版的实现(embed)[5]。这一点也值得我们效仿,在团队内部每个服务的开发者如果都能提供一个服务的简化版实现,那么对于该服务调用者来说,它的单测就会变得十分容易。

5. 参考资料

  • 《xUnit Test Patterns : Refactoring Test Code》- https://book.douban.com/subject/1859393/

  • Test Double Patterns - http://xunitpatterns.com/Test%20Double%20Patterns.html

  • The Unit in Unit Testing - https://www.infoq.com/articles/unit-testing-approach/

  • Test Doubles — Fakes, Mocks and Stubs - https://blog.pragmatists.com/test-doubles-fakes-mocks-and-stubs-1a7491dfa3da


“Gopher部落”知识星球[6]旨在打造一个精品Go学习和进阶社群!高品质首发Go技术文章,“三天”首发阅读权,每年两期Go语言发展现状分析,每天提前1小时阅读到新鲜的Gopher日报,网课、技术专栏、图书内容前瞻,六小时内必答保证等满足你关于Go语言生态的所有需求!2023年,Gopher部落将进一步聚焦于如何编写雅、地道、可读、可测试的Go代码,关注代码质量并深入理解Go核心技术,并继续加强与星友的互动。欢迎大家加入!

1016f8ad5009e795d82242bbba524c4c.jpeg18984a99815b34de196d15b3bb4bbf1d.png

28c367bc4852fcff0faa1fbcde762aa1.pngd3bc58550fbdb05b0e9c44b7618175d7.jpeg

著名云主机服务厂商DigitalOcean发布最新的主机计划,入门级Droplet配置升级为:1 core CPU、1G内存、25G高速SSD,价格5$/月。有使用DigitalOcean需求的朋友,可以打开这个链接地址[7]:https://m.do.co/c/bff6eed92687 开启你的DO主机之路。

Gopher Daily(Gopher每日新闻)归档仓库 - https://github.com/bigwhite/gopherdaily

我的联系方式:

  • 微博(暂不可用):https://weibo.com/bigwhite20xx

  • 微博2:https://weibo.com/u/6484441286

  • 博客:tonybai.com

  • github: https://github.com/bigwhite

62f7c0ac8fa0fc9fa09c8dc6f749c66d.jpeg

商务合作方式:撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。

参考资料

[1] 

《Go语言精进之路》vol2: https://book.douban.com/subject/35720729/

[2] 

《xUnit Test Patterns : Refactoring Test Code》: https://book.douban.com/subject/1859393/

[3] 

xUnit Patterns: http://xunitpatterns.com/Test%20Double%20Patterns.html

[4] 

testcontainers-go: https://golang.testcontainers.org/

[5] 

用于测试的自身简化版的实现(embed): https://github.com/etcd-io/etcd/blob/main/tests/integration/embed

[6] 

“Gopher部落”知识星球: https://wx.zsxq.com/dweb2/index/group/51284458844544

[7] 

链接地址: https://m.do.co/c/bff6eed92687

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

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

相关文章

【从零开始玩量化20】BigQuant平台策略代码本地化(与Github同步)

引言 最近发现了个不错的量化平台&#xff0c;BigQuant BigQuant的客服找到我&#xff0c;推荐他们平台给我使用&#xff0c;宣传的是人工智能&#xff0c;里面可以使用类似ChatGPT的聊天机器人&#xff0c;和可视化拖拉拽功能实现策略。 不过&#xff0c;这些都是锦上添花的…

微软“封杀”了推特,马斯克怒了:我要起诉微软!

文&#xff5c;鱼羊 发自 凹非寺源&#xff5c;量子位 OpenAI还没撕完&#xff0c;马斯克又跟微软杠上了&#xff0c;甚至直接在推特上放话&#xff1a; 我要告微软&#xff01; 这又是发生了甚么&#xff1f; 原因无它&#xff1a;微软刚刚“封杀”了推特。公告显示&#xff0…

新 Bing 惨遭微软“脑叶切除”,引大量网友不满!

整理 | 禾木木 责编 |梦依丹 出品 | CSDN&#xff08;ID&#xff1a;CSDNnews&#xff09; 在新版 Bing 引入 ChatGPT 爆火之后&#xff0c;Bing 就开始了各种“作妖”秀&#xff0c;翻车离谱事件是一件接着一件。有不少用户表示在和 Bing 的交流过程中&#xff0c;发现…

成为钢铁侠!只需一块RTX3090,微软开源贾维斯(J.A.R.V.I.S.)人工智能AI助理系统

梦想照进现实&#xff0c;微软果然不愧是微软&#xff0c;开源了贾维斯(J.A.R.V.I.S.)人工智能助理系统&#xff0c;贾维斯(jarvis)全称为Just A Rather Very Intelligent System&#xff08;只是一个相当聪明的人工智能系统&#xff09;&#xff0c;它可以帮助钢铁侠托尼斯塔克…

【JAVA】让 ChatGPT 来描述 IOC

前言 又迎来了一年一度的金三银四&#xff0c;虽然说今年的大环境不好&#xff0c;但是招聘还是在火热进行中。 面试过 Java 工程师的小伙伴都知道&#xff0c;Spring 中的 IOC 是面试高频题&#xff0c;面试官上来就问&#xff0c;知道什么是 IOC 吗&#xff0c;IOC 是如何初…

【工具】VScode|Linux 中怎么调试 Python 项目比较方便?又名 VScode 怎么调试 Python 项目(兼容环境Ubuntu18.04)

使用过 Anaconda、Jupyter、Pycharm、VScode、VS2022、pdb 这几个 IDE 去编写 python 项目或者维护 python 环境&#xff0c;各有各的优缺点&#xff0c;但 VScode yyds&#xff01; 可能会被网上说得天花乱坠的 Python 配置项吓退&#xff0c;会被 VScode 各种插件介绍吓退&a…

日本僧人问道弘法寺当家师

时间过得真快&#xff0c;六名日本禅僧在弘法寺为期10天的体验生活已进行到第三天&#xff0c;晚上10&#xff1a;00&#xff0c;方丈印顺大和尚依然等候在丈室&#xff0c;与前二日不同的是&#xff0c;弘法寺的当家师智空法师也来到了方丈室。 六位日本僧人行礼完…

佛教基础知识

佛教基础知识 1、《佛教常识》&#xff08;1&#xff09;佛陀和佛教的创立&#xff08;2&#xff09;佛法的基本内容、书籍&#xff08;3&#xff09;僧团和佛的弟子&#xff08;4&#xff09;佛教在印度的发展、衰亡及复兴&#xff08;5&#xff09;佛教在中国的发展、演变 2、…

【ChatGPT与网络安全攻击】AI密码破解器可在60秒内攻破50%以上普通密码

研究表明&#xff0c;ChatGPT等功能强大AI工具已经被用于网络攻击者实施犯罪活动&#xff0c;例如开发恶意软件和生成钓鱼邮件等。如果人们的密码从数据库泄露或被破坏&#xff0c;那么网络攻击者采用AI密码破解器猜出密码是概率几乎是100%&#xff0c;其中50%以上会在60秒内被…

chatgpt赋能Python-pythonmd5解密

Python MD5解密原理及应用 MD5是一种广泛使用的哈希算法&#xff0c;被用于加密敏感数据。MD5算法使用不可逆的方法将任何长度的数据转换为固定长度的哈希值&#xff0c;并且只能通过暴力破解的方式破解加密后的敏感数据。尽管MD5算法被广泛采用&#xff0c;但历史上已发现其存…

chatgpt赋能python:Python怎么破解Windows软件?

Python怎么破解Windows软件&#xff1f; 作为一名有10年Python编程经验的工程师&#xff0c;我想分享一些破解Windows软件的经验。Python是一种高级编程语言&#xff0c;可以用于许多不同的应用程序&#xff0c;包括软件破解。 什么是软件破解&#xff1f; 软件破解是指绕过…

讯飞星火认知大模型与ChatGPT的对比分析

引言&#xff1a; 人工智能是当今科技领域的热门话题&#xff0c;自然语言处理是人工智能的重要分支。自然语言处理的目标是让计算机能够理解和生成自然语言&#xff0c;实现人机交互和智能服务。近年来&#xff0c;随着深度学习的发展&#xff0c;自然语言处理领域出现了许多创…

【Unity开发小技巧】UnityWebGL移动端和电脑端调起输入法,中文输入处理

目录 一.TextMesh Pro中文显示问题 1.PC端和移动端中文显示异常乱码&#xff08;解决方案&#xff09; 1.制作TextMesh Pro字体 方式一 2.制作TextMesh Pro字体 方式二 3.通用字体资源 2.web端中文不能输入窗口模式&#xff08;解决方案&#xff09; 二.移动端Inputfile调…

分享一个利用ChatGPT为世界上任何城市建立旅行路线(带链接)的工具 GPTravel Advisor

GPTravel Advisor - 在几秒钟内创建世界上任何城市的旅行路线 网址链接&#xff1a;https://gpt-travel-advisor.vercel.app/ GIthub&#xff1a;https://github.com/dabit3/gpt-travel-advisor ChatGPT中文论坛&#xff1a;https://gptocean.com/

ChatGPT 新版 API 推出 語音轉換文字模型 Whisper

OpenAI 宣布釋出新 ChatGPT API&#xff0c;允許第三方開發人員通過 API 將 ChatGPT 整合到他們的網站、應用程式及產品中。同時發表開源的&#xff0c;讓用户用以轉錄或翻譯音訊。 OpenAI 表示&#xff0c;新版的 ChatGPT API 不僅可用於創建人工智能聊天界面&#xff0c;更可…

怎么玩chatgpt?如何利用ChatGPT来编写PRD?

很多人对于chatgpt不知道怎么玩&#xff1f;其实对于一个产品经理来说&#xff0c;他可以这样玩&#xff01;在产品开发过程中&#xff0c;产品需求文档&#xff08;PRD&#xff09;是一个非常重要的文档&#xff0c;它描述了产品的功能、特性和目标用户等信息。编写PRD需要耗费…

我群 300+人已熟练使用的 ChatGPT Prompt 技巧

这是吴恩达联合 OpenAI 官方录制的 ChatGPT Prompt 免费视频课&#xff0c;最后一个总结&#xff0c;视频&#xff1a; https://learn.deeplearning.ai/chatgpt-prompt-eng/lesson/2/guidelines 在这一节中&#xff0c;我会分享两个技巧&#xff1a;Transforming 和 Expanding …

利用python进行数据分析~基金分析

利用python进行基金数据分析 背景说明分析过程1.获取所有种类基金数据1.1导入相关包1.2通过天天基金网接口获取基金数据1.2.1获取网页信息1.2.2将数据转化成二维表并写入本地磁盘&#xff08;dataframe&#xff09; 1.3数据概览1.3.1查看前几行数据1.3.2查看各类型基金分布及可…

Tushare+Talib基金指标分析

本文介绍python语言下的两个第三方库&#xff0c;Tushare&#xff08;获取股票和基金数据&#xff09;和Ta-Lib&#xff08;用于数据指标分析&#xff09;&#xff0c;及其相关使用案例。 一、安装 Tushare安装 # 方式1&#xff1a;pip install tushare# 如果安装网络超时可尝…

chatgpt赋能python:Python实现人机对话

Python实现人机对话 人机对话是指人类和机器之间的交互过程&#xff0c;其中人类作为用户&#xff0c;通过语音或文字与机器进行交互。Python作为一门流行的编程语言&#xff0c;可以用来实现人机对话系统。本文介绍如何使用Python实现人机对话。 Python语言的优势 Python是…