go语言中file, err := os.OpenFile("zl.txt", os.O_CREATE|os.O_RDWR, 0666)是什么意思

时间: 2023-06-08 11:04:13 浏览: 217
这段代码是在用 Go 语言创建一个文件句柄,文件名为 "zl.txt",标志位包括创建文件和读写权限。其中,os.O_CREATE 表示如果文件不存在就新建,os.O_RDWR 表示读写权限,0666 表示文件的权限为可读可写。同时,该代码也会返回一个 error 类型的值 err,用于判断是否成功创建了该文件的句柄。
相关问题

type Downloader struct { common.FileMetadata // 文件元数据 common.SliceSeq // 需要重传的序号 waitGoroutine sync.WaitGroup // 同步goroutine DownloadDir string // 下载文件保存目录 RetryChannel chan int // 重传channel通道 MaxGtChannel chan struct{} // 限制上传的goroutine的数量通道 StartTime int64 // 下载开始时间 } // DownloadFile 单个文件的下载 func DownloadFile(filename string, downloadDir string) (error){ if !common.IsDir(downloadDir) { fmt.Printf("指定下载路径:%s 不存在\n", downloadDir) return errors.New("指定下载路径不存在") } targetUrl := common.BaseUrl + "download?filename=" + filename req, _ := http.NewRequest("GET", targetUrl, nil) resp, err := (&http.Client{}).Do(req) if err != nil { fmt.Println(err) return err } defer resp.Body.Close() filePath := path.Join(downloadDir, filename) f, err := os.OpenFile(filePath, os.O_WRONLY|os.O_CREATE, 0666) if err != nil { fmt.Printf(err.Error()) return err } defer f.Close() _, err = io.Copy(f, resp.Body) if err != nil { return err } fmt.Printf("%s 文件下载成功,保存路径:%s\n", filename, filePath) return nil }

这段代码是一个 Downloader 结构体和一个 DownloadFile 函数。结构体中包含了文件元数据、需要重传的序号、等待 goroutine 的同步组、下载文件保存目录、重传通道和限制上传 goroutine 数量的通道等字段。DownloadFile 函数是用来下载单个文件的,它会发送一个 GET 请求获取文件内容,并将内容保存到指定的下载目录中。如果下载路径不存在或者下载过程中出现错误,函数会返回相应的错误信息。最后,函数会打印出文件下载成功的消息。

帮我规整以下代码:package main import ( "fmt" "log" "os" "os/exec" "strings" ) const ( apacheConfPath = "/usr/local/lighthouse/softwares/apache/conf/httpd.conf" // 修改Apache配置文件路径 localPort = "8080" ) func main() { // 修改Apache配置文件中的端口号 err := updateApacheConfig() if err != nil { log.Fatalf("Failed to update Apache configuration: %s", err) } // 重新加载Apache配置 err = reloadApache() if err != nil { log.Fatalf("Failed to reload Apache: %s", err) } // 配置防火墙规则 err = configureFirewall() if err != nil { log.Fatalf("Failed to configure firewall: %s", err) } fmt.Printf("Apache端口已修改为本地可见的端口 %s\n", localPort) } func updateApacheConfig() error { // 打开Apache配置文件 apacheConfFile, err := os.OpenFile(apacheConfPath, os.O_RDWR, 0644) if err != nil { return fmt.Errorf("failed to open Apache configuration file: %w", err) } defer apacheConfFile.Close() // 读取Apache配置文件内容 apacheConfBytes, err := os.ReadFile(apacheConfPath) if err != nil { return fmt.Errorf("failed to read Apache configuration file: %w", err) } // 替换端口号 newConfContent := strings.ReplaceAll(string(apacheConfBytes), "Listen 80", "Listen "+localPort) // 回到文件开头 _, err = apacheConfFile.Seek(0, 0) if err != nil { return fmt.Errorf("failed to seek to the beginning of Apache configuration file: %w", err) } // 清空文件内容 err = apacheConfFile.Truncate(0) if err != nil { return fmt.Errorf("failed to truncate Apache configuration file: %w", err) } // 写入新的配置内容 _, err = apacheConfFile.WriteString(newConfContent) if err != nil { return fmt.Errorf("failed to write updated Apache configuration: %w", err) } return nil } func reloadApache() error { cmd := exec.Command("systemctl", "restart", "httpd") // 使用systemctl命令重新启动Apache err := cmd.Run() if err != nil { return fmt.Errorf("failed to reload Apache: %w", err) } return nil } func configureFirewall() error { // 添加防火墙规则允许从本地访问新端口 allowCmd := exec.Command("iptables", "-A", "INPUT", "-p", "tcp", "--dport", localPort, "-j", "ACCEPT") err := allowCmd.Run() if err != nil { return fmt.Errorf("failed to configure firewall to allow local access: %w", err) } // 阻止从外部访问新端口 dropCmd := exec.Command("iptables", "-A", "INPUT", "-p", "tcp", "--dport", localPort, "-j", "DROP") err = dropCmd.Run() if err != nil { return fmt.Errorf("failed to configure firewall to block external access: %w", err) } // 保存防火墙规则 saveCmd := exec.Command("bash", "-c", "iptables-save > /etc/sysconfig/iptables") err = saveCmd.Run() if err != nil { return fmt.Errorf("failed to save firewall rules: %w", err) } return nil }

以下是规整后的代码: ```go package main import ( "fmt" "log" "os" "os/exec" "strings" ) const ( apacheConfPath = "/usr/local/lighthouse/softwares/apache/conf/httpd.conf" // 修改Apache配置文件路径 localPort = "8080" ) func main() { // 修改Apache配置文件中的端口号 err := updateApacheConfig() if err != nil { log.Fatalf("Failed to update Apache configuration: %s", err) } // 重新加载Apache配置 err = reloadApache() if err != nil { log.Fatalf("Failed to reload Apache: %s", err) } // 配置防火墙规则 err = configureFirewall() if err != nil { log.Fatalf("Failed to configure firewall: %s", err) } fmt.Printf("Apache端口已修改为本地可见的端口 %s\n", localPort) } func updateApacheConfig() error { // 打开Apache配置文件 apacheConfFile, err := os.OpenFile(apacheConfPath, os.O_RDWR, 0644) if err != nil { return fmt.Errorf("failed to open Apache configuration file: %w", err) } defer apacheConfFile.Close() // 读取Apache配置文件内容 apacheConfBytes, err := os.ReadFile(apacheConfPath) if err != nil { return fmt.Errorf("failed to read Apache configuration file: %w", err) } // 替换端口号 newConfContent := strings.ReplaceAll(string(apacheConfBytes), "Listen 80", "Listen "+localPort) // 回到文件开头 _, err = apacheConfFile.Seek(0, 0) if err != nil { return fmt.Errorf("failed to seek to the beginning of Apache configuration file: %w", err) } // 清空文件内容 err = apacheConfFile.Truncate(0) if err != nil { return fmt.Errorf("failed to truncate Apache configuration file: %w", err) } // 写入新的配置内容 _, err = apacheConfFile.WriteString(newConfContent) if err != nil { return fmt.Errorf("failed to write updated Apache configuration: %w", err) } return nil } func reloadApache() error { cmd := exec.Command("systemctl", "restart", "httpd") // 使用systemctl命令重新启动Apache err := cmd.Run() if err != nil { return fmt.Errorf("failed to reload Apache: %w", err) } return nil } func configureFirewall() error { // 添加防火墙规则允许从本地访问新端口 allowCmd := exec.Command("iptables", "-A", "INPUT", "-p", "tcp", "--dport", localPort, "-j", "ACCEPT") err := allowCmd.Run() if err != nil { return fmt.Errorf("failed to configure firewall to allow local access: %w", err) } // 阻止从外部访问新端口 dropCmd := exec.Command("iptables", "-A", "INPUT", "-p", "tcp", "--dport", localPort, "-j", "DROP") err = dropCmd.Run() if err != nil { return fmt.Errorf("failed to configure firewall to block external access: %w", err) } // 保存防火墙规则 saveCmd := exec.Command("bash", "-c", "iptables-save > /etc/sysconfig/iptables") err = saveCmd.Run() if err != nil { return fmt.Errorf("failed to save firewall rules: %w", err) } return nil } ``` 希望可以帮到你!如果还有其他问题,请随时提问。
阅读全文

相关推荐

package main /* #include <sys/socket.h> #include <netpacket/packet.h> #include <net/ethernet.h> */ import "C" import ( "fmt" "log" "net" "os" "os/exec" "syscall" "unsafe" ) const ( ProtocolIP = 0x0800 ETH_P_ALL = 0x0003 BufferSize = 65536 ) var Interface string // 网卡接口名称 func main() { // 检查程序是否以root身份运行 if os.Geteuid() != 0 { fmt.Println("Please run the program as root.") // 以root身份重新运行程序 cmd := exec.Command("sudo", os.Args[0]) cmd.Stdout = os.Stdout cmd.Stdin = os.Stdin cmd.Stderr = os.Stderr err := cmd.Run() if err != nil { log.Fatal(err) } os.Exit(0) } // 获取系统的网络接口信息 interfaces, err := net.Interfaces() if err != nil { log.Fatal(err) } // 遍历网络接口,打印接口名称 fmt.Println("Available network interfaces:") for _, iface := range interfaces { fmt.Println(iface.Name) } // 设置要使用的网卡接口(例如:eth0) Interface = "eth0" // 更改为你的网卡接口名 // 创建原始套接字 sockFD, err := syscall.Socket(C.AF_PACKET, syscall.SOCK_RAW, int(htons(ETH_P_ALL))) if err != nil { log.Fatal(err) } defer syscall.Close(sockFD) // 获取网卡接口索引 iface, err := net.InterfaceByName(Interface) if err != nil { log.Fatal(err) } // 绑定原始套接字到网卡接口 sa := C.struct_sockaddr_ll{ sll_family: C.AF_PACKET, sll_protocol: htons(ETH_P_ALL), sll_ifindex: C.int(iface.Index), } if err := syscall.Bind(sockFD, (*syscall.Sockaddr)(unsafe.Pointer(&sa))); err != nil { log.Fatal(err) } // 在一个无限循环中接收数据包 for { buffer := make([]byte, BufferSize) n, _, err := syscall.Recvfrom(sockFD, buffer, 0) if err != nil { log.Fatal(err) } fmt.Printf("Received packet: %s\n", string(buffer[:n])) } } func htons(i uint16) uint16 { return (i<<8)&0xff00 | i>>8 }中无法将 '(*syscall.Sockaddr)(unsafe.Pointer(&sa))' (类型 *syscall.Sockaddr) 用作类型 Sockaddr

package main import ( "flag" "fmt" "os" ) const ( S = 54 // standard size of bmp headers T = 32 // number of bytes needed to hide the text length C = 4 // number of bytes needed to hide a character ) // modify hides an integer to a byte slice func modify(value int, pix []byte, size int) { for i := 0; i < size; i++ { pix[i] = (pix[i] & 0xFC) | byte(value&0x03) value >>= 2 } // TODO: write your code here // replace last 2 bits of pix[i] with the last 2 bits of value // the next iteration repeats with the next 2 bits of value } var ( srcImage string // input image name srcTxt string // input text name destImage string // output doctored image name ) // init sets command line arguments func init() { // DON'T modify this function!!! flag.StringVar(&srcImage, "i", "", "input image name") flag.StringVar(&srcTxt, "t", "", "input text name") flag.StringVar(&destImage, "d", "", "output doctored image name") } func main() { // parse command line arguments flag.Parse() if srcImage == "" || srcTxt == "" || destImage == "" { flag.PrintDefaults() os.Exit(1) } // read input image to a byte slice p p, err := os.ReadFile(srcImage) if err != nil { fmt.Printf("Read image file failed, err = %v\n", err) os.Exit(1) } // read input text to a byte slice t t, err := os.ReadFile(srcTxt) if err != nil { fmt.Printf("Read text file failed, err = %v\n", err) os.Exit(1) } // check if the text is too big if T+len(t)*C > len(p[S:]) { fmt.Println("The text file is too big") os.Exit(1) } // save the text length to p modify(len(t), p[S:S+T], T) // save the content of text to p for i := 0; i < len(t); i++ { offset := S + T + C*i modify(int(t[i]), p[offset:offset+C], C) } // write the modified p to destImage err = os.WriteFile(destImage, p, 0644) if err != nil { fmt.Printf("Write doctored image failed, err = %v\n", err) os.Exit(1) } }这是一个将文字隐藏到图片中的代码,请帮我讲下面这个恢复文字的代码完善package main import ( "flag" "os" "fmt" ) const ( S = 54 // standard size of header T = 32 // number of bytes needed to hide the text length C = 4 // number of bytes needed to hide a character ) func modify(pix []byte, value int, size int){ for i := size-1; i >= 0; i-- { value = int((pix[i] & 0x03) | byte(value&0xFC)) value <<= 2 } } var ( image string // input doctor image name txt string // output text name ) // init sets command line arguments func init() { // DON'T modify this function!!! flag.StringVar(&image, "i", "", "input image name") flag.StringVar(&txt, "t", "", "output text name") } func main() { // parse command line arguments flag.Parse() if image == "" || txt == "" { flag.PrintDefaults() os.Exit(1) } p, err := os.ReadFile(image) if err != nil { fmt.Printf("Read image file failed, err = %v\n", err) os.Exit(1) } modify(p[S:S+T], len(o), T) for i := len(o)-1; i >= 0; i-- { offset := S + T + C*i modify(p[offset:offset+C], int(o[i]) , C) } err = os.WriteFile(txt, o, 0644) if err != nil { fmt.Printf("Write doctored image failed, err = %v\n", err) os.Exit(1) } }

改进以下代码 currentpath = os.path.dirname(os.path.realpath(__file__)) time_date = '{}{}'.format(self.time_date,self.random_char(5)) contents = os.path.join(currentpath, time_date, self.ref.split('/')[-1]) ref = self.ref.split('/')[-1] private_token = self.gl.private_token path = "lib" if ref == "master": if os.path.exists(os.path.join(contents, self.name)): subprocess.call("rm -rf {} ".format(os.path.join(contents, self.name)), shell=True, cwd=contents) time.sleep(3) retcode = start.clone(int(self.project_id), ref, contents, private_token) if retcode == 0: start.clone_frontend(self.get_frontend()[0],self.get_frontend()[1], contents, private_token,self.get_frontend()[2] ) start.clone_abc(self.get_abc()[0], self.get_abc()[1], contents, private_token,"mc_abc") start.clone_model(start.get_clkrst()[0], start.get_clkrst()[1], contents, private_token,"clkrst") start.clone_model(start.get_ara()[0], start.get_ara()[1], contents, private_token,"ara") start.clone_model(start.get_wfl()[0], start.get_wfl()[1], contents, private_token,"wfl") subprocess.call("echo '*.t' >> {}".format(os.path.join(contents, self.name, ".gitignore")),shell=True) code = start.make_lib(os.path.join(contents, self.name)) rel, err = code.communicate() if "make: *** [main] Error 2" in err.decode('utf-8'): print("loading push error log") filename = os.path.join(contents, self.name, "error_make_log") subprocess.call("echo '' > {}".format(filename), shell=True, cwd=contents) start.error_make(filename, rel.decode('utf-8') ) start.error_make(filename, err.decode('utf-8') ) else: print("loading push libs") # start.push_lib(os.path.join(contents, self.name), path, ref)

最新推荐

recommend-type

Go语言中io.Reader和io.Writer的详解与实现

在Go语言中,`io.Reader`和`io.Writer`是两个非常基础且重要的接口,用于处理输入输出(I/O)操作。它们定义在`io`包中,是构建其他复杂I/O操作的基础。 `io.Reader`接口定义了一个单个方法: ```go type Reader ...
recommend-type

Android 出现:java.lang.NoClassDefFoundError...错误解决办法

本文将深入探讨这个问题,特别是在Android环境中如何解决`java.lang.NoClassDefFoundError: android/os/PersistableBundle`这个特定错误。 `PersistableBundle`是Android 5.0(API Level 21)引入的一个新特性,...
recommend-type

若依管理存在任何文件读取漏洞检测系统,渗透测试.zip

若依管理存在任何文件读取漏洞检测系统,渗透测试若一管理系统发生任意文件读取若依管理系统存在任何文件读取免责声明使用本程序请自觉遵守当地法律法规,出现一切后果均与作者无关。本工具旨在帮助企业快速定位漏洞修复漏洞,仅限安全授权测试使用!严格遵守《中华人民共和国网络安全法》,禁止未授权非法攻击站点!由于作者用户欺骗造成的一切后果与关联。毒品用于非法一切用途,非法使用造成的后果由自己承担,与作者无关。食用方法python3 若依管理系统存在任意文件读取.py -u http://xx.xx.xx.xxpython3 若依管理系统存在任意文件读取.py -f url.txt
recommend-type

【java毕业设计】学生社团管理系统源码(完整前后端+说明文档+LW).zip

学生社团的管理系统,是一款功能丰富的实用性网站,网站采用了前台展示后台管理的模式进行开发设计的,系统前台包括了站内新闻展示,社团信息管理以及社团活的参与报名,在线用户注册,系统留言板等实用性功能。 网站的后台是核心,针对系统的前台的功能,学生的社团报名审核以及社团信息的发布等功能进行管理。本系统可以综合成为4个用户权限,普通注册用户,社团团员用户,社团长以及系统管理员。系统管理员主要负责网站的整体信息管理,普通用户可以进行社团活动的浏览以及申社团的加入,社团团员是普通注册用户审核成功后的一个用户权限。经过管理员审核同意,社团团员可以升级成为社团的团长,系统权限划分是本系统的核心功能。 环境说明: 开发语言:Java,jsp JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea 部署容器:tomcat
recommend-type

Python中快速友好的MessagePack序列化库msgspec

资源摘要信息:"msgspec是一个针对Python语言的高效且用户友好的MessagePack序列化库。MessagePack是一种快速的二进制序列化格式,它旨在将结构化数据序列化成二进制格式,这样可以比JSON等文本格式更快且更小。msgspec库充分利用了Python的类型提示(type hints),它支持直接从Python类定义中生成序列化和反序列化的模式。对于开发者来说,这意味着使用msgspec时,可以减少手动编码序列化逻辑的工作量,同时保持代码的清晰和易于维护。 msgspec支持Python 3.8及以上版本,能够处理Python原生类型(如int、float、str和bool)以及更复杂的数据结构,如字典、列表、元组和用户定义的类。它还能处理可选字段和默认值,这在很多场景中都非常有用,尤其是当消息格式可能会随着时间发生变化时。 在msgspec中,开发者可以通过定义类来描述数据结构,并通过类继承自`msgspec.Struct`来实现。这样,类的属性就可以直接映射到消息的字段。在序列化时,对象会被转换为MessagePack格式的字节序列;在反序列化时,字节序列可以被转换回原始对象。除了基本的序列化和反序列化,msgspec还支持运行时消息验证,即可以在反序列化时检查消息是否符合预定义的模式。 msgspec的另一个重要特性是它能够处理空集合。例如,上面的例子中`User`类有一个名为`groups`的属性,它的默认值是一个空列表。这种能力意味着开发者不需要为集合中的每个字段编写额外的逻辑,以处理集合为空的情况。 msgspec的使用非常简单直观。例如,创建一个`User`对象并序列化它的代码片段显示了如何定义一个用户类,实例化该类,并将实例序列化为MessagePack格式。这种简洁性是msgspec库的一个主要优势,它减少了代码的复杂性,同时提供了高性能的序列化能力。 msgspec的设计哲学强调了性能和易用性的平衡。它利用了Python的类型提示来简化模式定义和验证的复杂性,同时提供了优化的内部实现来确保快速的序列化和反序列化过程。这种设计使得msgspec非常适合于那些需要高效、类型安全的消息处理的场景,比如网络通信、数据存储以及服务之间的轻量级消息传递。 总的来说,msgspec为Python开发者提供了一个强大的工具集,用于处理高性能的序列化和反序列化任务,特别是当涉及到复杂的对象和结构时。通过利用类型提示和用户定义的模式,msgspec能够简化代码并提高开发效率,同时通过运行时验证确保了数据的正确性。"
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

STM32 HAL库函数手册精读:最佳实践与案例分析

![STM32 HAL库函数手册精读:最佳实践与案例分析](https://khuenguyencreator.com/wp-content/uploads/2020/07/bai11.jpg) 参考资源链接:[STM32CubeMX与STM32HAL库开发者指南](https://wenku.csdn.net/doc/6401ab9dcce7214c316e8df8?spm=1055.2635.3001.10343) # 1. STM32与HAL库概述 ## 1.1 STM32与HAL库的初识 STM32是一系列广泛使用的ARM Cortex-M微控制器,以其高性能、低功耗、丰富的外设接
recommend-type

如何利用FineReport提供的预览模式来优化报表设计,并确保最终用户获得最佳的交互体验?

针对FineReport预览模式的应用,这本《2020 FCRA报表工程师考试题库与答案详解》详细解读了不同预览模式的使用方法和场景,对于优化报表设计尤为关键。首先,设计报表时,建议利用FineReport的分页预览模式来检查报表的布局和排版是否准确,因为分页预览可以模拟报表在打印时的页面效果。其次,通过填报预览模式,可以帮助开发者验证用户交互和数据收集的准确性,这对于填报类型报表尤为重要。数据分析预览模式则适合于数据可视化报表,可以在这个模式下调整数据展示效果和交互设计,确保数据的易读性和分析的准确性。表单预览模式则更多关注于表单的逻辑和用户体验,可以用于检查表单的流程是否合理,以及数据录入
recommend-type

大学生社团管理系统设计与实现

资源摘要信息:"基于ssm+vue的大学生社团管理系统.zip" 该系统是基于Java语言开发的,使用了ssm框架和vue前端框架,主要面向大学生社团进行管理和运营,具备了丰富的功能和良好的用户体验。 首先,ssm框架是Spring、SpringMVC和MyBatis三个框架的整合,其中Spring是一个全面的企业级框架,可以处理企业的业务逻辑,实现对象的依赖注入和事务管理。SpringMVC是基于Servlet API的MVC框架,可以分离视图和模型,简化Web开发。MyBatis是一个支持定制化SQL、存储过程以及高级映射的持久层框架。 SpringBoot是一种全新的构建和部署应用程序的方式,通过使用SpringBoot,可以简化Spring应用的初始搭建以及开发过程。它使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置。 Vue.js是一个用于创建用户界面的渐进式JavaScript框架,它的核心库只关注视图层,易于上手,同时它的生态系统也十分丰富,提供了大量的工具和库。 系统主要功能包括社团信息管理、社团活动管理、社团成员管理、社团财务管理等。社团信息管理可以查看和编辑社团的基本信息,如社团名称、社团简介等;社团活动管理可以查看和编辑社团的活动信息,如活动时间、活动地点等;社团成员管理可以查看和编辑社团成员的信息,如成员姓名、成员角色等;社团财务管理可以查看和编辑社团的财务信息,如收入、支出等。 此外,该系统还可以通过微信小程序进行访问,微信小程序是一种不需要下载安装即可使用的应用,它实现了应用“触手可及”的梦想,用户扫一扫或者搜一下即可打开应用。同时,它也实现了应用“用完即走”的理念,用户不用关心是否安装太多应用的问题。应用将无处不在,随时可用,但又无需安装卸载。 总的来说,基于ssm+vue的大学生社团管理系统是一款功能丰富、操作简便、使用方便的社团管理工具,非常适合大学生社团的日常管理和运营。
recommend-type

"互动学习:行动中的多样性与论文攻读经历"

多样性她- 事实上SCI NCES你的时间表ECOLEDO C Tora SC和NCESPOUR l’Ingén学习互动,互动学习以行动为中心的强化学习学会互动,互动学习,以行动为中心的强化学习计算机科学博士论文于2021年9月28日在Villeneuve d'Asq公开支持马修·瑟林评审团主席法布里斯·勒菲弗尔阿维尼翁大学教授论文指导奥利维尔·皮耶昆谷歌研究教授:智囊团论文联合主任菲利普·普雷教授,大学。里尔/CRISTAL/因里亚报告员奥利维耶·西格德索邦大学报告员卢多维奇·德诺耶教授,Facebook /索邦大学审查员越南圣迈IMT Atlantic高级讲师邀请弗洛里安·斯特鲁布博士,Deepmind对于那些及时看到自己错误的人...3谢谢你首先,我要感谢我的两位博士生导师Olivier和Philippe。奥利维尔,"站在巨人的肩膀上"这句话对你来说完全有意义了。从科学上讲,你知道在这篇论文的(许多)错误中,你是我可以依