Golang使用GORM操作数据库全流程

答案:GORM通过结构体定义模型、自动迁移创建表、提供链式API进行CRUD操作,并支持连接池配置与错误排查。使用GORM需先连接数据库,定义如User等结构体模型,利用AutoMigrate建表,再通过Create、First、Save、Delete等方法实现数据操作,同时可通过标签自定义字段映射,结合Preload处理关联关系,结合事务保证数据一致性。

golang使用gorm操作数据库全流程

在Go语言的开发实践中,操作数据库是绕不开的核心环节。GORM作为Go社区中一个功能强大、设计优雅的ORM(对象关系映射)库,极大地简化了这一过程。它允许开发者通过Go的结构体(struct)来定义数据模型,并以面向对象的方式进行数据库的增删改查,避免了直接编写复杂的SQL语句,显著提升了开发效率和代码的可维护性。

GORM的核心价值在于其将Go语言的类型系统与关系型数据库的表结构无缝连接起来,让我们能够更专注于业务逻辑而非底层数据库操作的细节。

解决方案

使用GORM操作数据库,通常可以概括为以下几个关键步骤,它们共同构成了一个完整的工作流程:

首先,是环境准备和数据库连接。你需要安装GORM及其对应数据库的驱动(例如,MySQL使用

gorm.io/driver/mysql

,PostgreSQL使用

gorm.io/driver/postgres

)。连接数据库时,通过

gorm.Open()

函数传入数据库驱动和连接字符串,获取一个

*gorm.DB

实例。这个实例是后续所有数据库操作的入口。连接字符串的配置是关键,它包含了数据库的地址、端口、用户名、密码以及数据库名等信息。

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

package mainimport (    "gorm.io/driver/sqlite" // 示例使用SQLite,方便演示    "gorm.io/gorm"    "log")var DB *gorm.DBfunc init() {    var err error    // 实际项目中会连接MySQL/PostgreSQL等    DB, err = gorm.Open(sqlite.Open("gorm.db"), &gorm.Config{})    if err != nil {        log.Fatalf("无法连接到数据库: %v", err)    }    log.Println("数据库连接成功!")}func main() {    // 后续操作将在这里进行}

接着,是定义数据模型。在GORM中,数据模型就是普通的Go结构体。GORM会根据结构体的字段名、类型以及可选的GORM标签来推断数据库表的结构、字段类型和约束。通常,我们会在结构体中嵌入

gorm.Model

,它会自动提供

ID

CreatedAt

UpdatedAt

DeletedAt

这些常用字段,用于主键、创建时间、更新时间和软删除。

type User struct {    gorm.Model    Name    string `gorm:"type:varchar(100);not null;uniqueIndex"` // 姓名,非空,唯一索引    Email   string `gorm:"type:varchar(255);unique"`               // 邮箱,唯一    Age     int    `gorm:"default:18"`                             // 年龄,默认18    Address string `gorm:"size:255"`                               // 地址}

然后,执行数据库迁移。定义好模型后,你需要让GORM根据这些模型创建或更新数据库表结构。这通过

DB.AutoMigrate()

方法实现。它会检查数据库中是否存在对应的表,如果不存在则创建;如果表已存在,它会尝试添加新的字段,但不会删除或修改现有字段,这是一个相对安全的做法。

func init() {    // ... 数据库连接代码 ...    err = DB.AutoMigrate(&User{}) // 自动迁移User模型    if err != nil {        log.Fatalf("数据库迁移失败: %v", err)    }    log.Println("数据库迁移完成。")}

最后,是核心的CRUD操作。GORM提供了一系列直观的方法来执行数据的创建(Create)、读取(Retrieve)、更新(Update)和删除(Delete)。

创建:使用

DB.Create(&user)

来插入一条新记录。读取

DB.First(&user, id)

根据主键获取单条记录;

DB.Find(&users)

获取多条记录;

DB.Where("age > ?", 20).Find(&users)

进行条件查询。更新

DB.Save(&user)

保存一个已存在且被修改的记录;

DB.Model(&User{}).Where("id = ?", 1).Update("name", "New Name")

更新特定字段;

DB.Model(&User{}).Where("id = ?", 1).Updates(map[string]interface{}{"name": "New Name", "age": 30})

更新多个字段。删除

DB.Delete(&user, id)

进行软删除(如果模型包含

gorm.DeletedAt

字段),或

DB.Unscoped().Delete(&user, id)

进行硬删除。

// 创建用户func createUser(user *User) {    result := DB.Create(user)    if result.Error != nil {        log.Printf("创建用户失败: %v", result.Error)        return    }    log.Printf("用户创建成功,ID: %d", user.ID)}// 查询用户func getUserByID(id uint) *User {    var user User    result := DB.First(&user, id) // 根据ID查找    if result.Error != nil {        log.Printf("查询用户ID %d 失败: %v", id, result.Error)        return nil    }    log.Printf("查询到用户: %+v", user)    return &user}// 更新用户func updateUser(user *User) {    result := DB.Save(user) // 保存所有更改    if result.Error != nil {        log.Printf("更新用户失败: %v", result.Error)        return    }    log.Printf("用户更新成功,ID: %d", user.ID)}// 删除用户 (软删除)func deleteUser(id uint) {    result := DB.Delete(&User{}, id)    if result.Error != nil {        log.Printf("删除用户ID %d 失败: %v", id, result.Error)        return    }    log.Printf("用户ID %d 已被软删除", id)}

Golang GORM数据库连接失败?常见错误排查与最佳实践

在使用GORM进行数据库连接时,开发者常常会遇到一些连接上的问题,比如“dial tcp: lookup db_host: no such host”或“access denied for user”。这些错误通常指向几个核心问题:连接字符串配置不当、数据库服务未运行或网络不通、以及权限不足。

首先,检查连接字符串。这是最常见的“坑”。确保数据库地址(

host

)、端口(

port

)、用户名(

user

)、密码(

password

)和数据库名(

dbname

)都准确无误。例如,MySQL的连接字符串格式通常是

user:password@tcp(host:port)/dbname?charset=utf8mb4&parseTime=True&loc=Local

。特别要注意

parseTime=True

,它能让GORM正确解析数据库中的时间类型字段到Go的

time.Time

。如果少了,日期时间字段可能会报错。

loc=Local

也很重要,它确保时间处理符合本地时区。我个人就曾因为忘记

parseTime=True

,在开发环境一切正常,部署到生产环境后却发现时间字段无法解析,花了很长时间才定位到问题。

其次,确认数据库服务状态和网络连通性。数据库服务是否已经启动?防火墙是否阻止了Go应用与数据库端口的通信?尝试从Go应用运行的机器上,使用命令行工具(如

mysql -h host -P port -u user -p

psql -h host -p port -U user -d dbname

)手动连接数据库,这能有效验证网络和数据库服务的可用性。如果命令行都连不上,那问题肯定不在Go代码本身。

再者,数据库用户权限。连接数据库的用户是否拥有对目标数据库的读写权限?有时,为了安全,数据库管理员可能会限制用户的权限,导致应用无法执行某些操作。检查数据库用户对应的权限配置,确保其至少拥有

SELECT

,

INSERT

,

UPDATE

,

DELETE

等基本权限。

最后,驱动与GORM版本兼容性。虽然不常见,但偶尔也会遇到GORM版本与特定数据库驱动版本不兼容的情况。查阅GORM官方文档或驱动的GitHub仓库,确认你使用的版本是否支持。当遇到一些难以理解的错误时,尝试更新或降级GORM和驱动版本,有时能奇迹般地解决问题。

最佳实践方面,建议将数据库连接字符串等敏感信息,通过环境变量或配置文件进行管理,而不是硬编码在代码中。这不仅提高了安全性,也使得在不同部署环境(开发、测试、生产)之间切换配置变得更加灵活。同时,使用连接池也是必不可少的。GORM默认会使用连接池,但你可以通过

DB.SetMaxIdleConns()

DB.SetMaxOpenConns()

DB.SetConnMaxLifetime()

等方法进行调优,以适应应用的并发需求,避免连接频繁创建和销毁带来的性能开销。

// 优化连接池设置DB.SetMaxIdleConns(10)           // 设置空闲连接池中的最大连接数DB.SetMaxOpenConns(100)          // 设置数据库的最大打开连接数DB.SetConnMaxLifetime(time.Hour) // 设置连接可重用的最长时间

GORM模型设计:如何通过Go Struct优雅映射数据库表结构?

GORM模型的设计是使用GORM的关键,它直接影响到我们与数据库交互的便捷性和效率。Go结构体与数据库表的映射并非简单的一对一,GORM提供了丰富的标签(tag)和约定来精细化控制这一过程。

基础映射:默认情况下,GORM会将结构体字段名转换为蛇形命名(snake_case)作为数据库的列名。例如,

CreatedAt

会映射到

created_at

。结构体字段的类型也会被GORM映射到合适的数据库类型。例如,

string

通常映射到

VARCHAR

int

映射到

int

gorm.Model

的妙用:这是我个人非常推崇的一个设计。嵌入

gorm.Model

结构体,可以自动为你的模型添加

ID

(主键,

uint

)、

CreatedAt

(

time.Time

)、

UpdatedAt

(

time.Time

) 和

DeletedAt

(

gorm.DeletedAt

,用于软删除) 四个字段。这大大减少了重复代码,并使得软删除功能开箱即用。当

DeletedAt

字段不为空时,GORM的查询默认会排除这些“已删除”的记录,只有使用

Unscoped()

方法才能查询到。

自定义字段名与类型:如果你需要自定义数据库列名或类型,可以使用GORM标签。

gorm:"column:your_column_name;type:jsonb;size:255;not null;unique;default:value"

是常用的格式。

column:

指定数据库列名。

type:

指定数据库字段类型,如

VARCHAR(100)

TEXT

JSONB

等。这在需要使用数据库特有类型时非常有用。

size:

指定字段长度,通常用于字符串类型。

not null

:指定字段非空。

unique

:创建唯一索引。

default:

指定字段的默认值。

index

uniqueIndex

:创建普通索引或唯一索引。

type Product struct {    ID        uint           `gorm:"primaryKey"` // 明确指定主键    Code      string         `gorm:"unique;not null;index:idx_code"` // 唯一且非空,并创建名为idx_code的索引    Price     uint           `gorm:"default:0"`    CreatedAt time.Time    UpdatedAt time.Time    DeletedAt gorm.DeletedAt `gorm:"index"` // 软删除,并为DeletedAt字段创建索引    Details   string         `gorm:"type:jsonb"` // 示例:使用JSONB类型存储复杂数据}

关联关系:GORM对模型间的关联关系(一对一、一对多、多对多)提供了很好的支持。通过在结构体中定义关联字段,GORM能够自动处理外键和关联数据的加载。

一对一/一对多:在子模型中添加一个父模型的ID字段,并可以添加父模型结构体。多对多:通常需要一个中间表来维护关系。GORM可以通过

many2many

标签或手动定义中间表模型来处理。

// User 和 CreditCard 是一对一关系type CreditCard struct {    gorm.Model    Number string    UserID uint // 外键}// User 和 Order 是一对多关系type Order struct {    gorm.Model    OrderSN string    UserID  uint // 外键    User    User // 关联模型,用于Preload或Joins}// User 和 Role 是多对多关系type Role struct {    gorm.Model    Name  string    Users []User `gorm:"many2many:user_roles;"` // 中间表名为 user_roles}

在实际开发中,合理设计模型不仅能让代码更清晰,还能有效利用数据库索引,提升查询性能。例如,为经常用于查询条件的字段添加索引(

index

uniqueIndex

),能显著加速查询。对于复杂的数据结构,考虑使用JSONB等类型存储,避免过度范式化带来的join开销。当然,这需要权衡查询的灵活性和存储的效率。

GORM CRUD实战:掌握数据增删改查的核心操作技巧

GORM提供的CRUD操作方法,在日常开发中是使用频率最高的。掌握它们的细微之处,能帮助我们写出更健壮、更高效的代码。

创建 (Create)

DB.Create(&user)

是最直接的创建方式。GORM会自动填充

CreatedAt

UpdatedAt

字段,并在成功后将数据库生成的主键

ID

回填到结构体中。

newUser := User{Name: "Alice", Email: "alice@example.com", Age: 25}result := DB.Create(&newUser)if result.Error != nil {    log.Printf("创建用户失败: %v", result.Error)} else {    log.Printf("创建用户成功,ID: %d", newUser.ID)}

批量创建

DB.Create(&[]User{{Name: "Bob"}, {Name: "Charlie"}})

可以一次性插入多条记录,GORM会优化为单次SQL插入,效率更高。

读取 (Retrieve)

按主键查询

DB.First(&user, 1)

。如果找不到记录,

result.Error

会是

gorm.ErrRecordNotFound

条件查询

DB.Where("age > ?", 20).Find(&users)

Where

子句支持多种操作符,如

=


IN

LIKE

等。链式查询:GORM的查询方法可以链式调用,非常灵活。

DB.Where("name LIKE ?", "%o%").Order("age desc").Limit(10).Offset(0).Find(&users)

选择特定字段

DB.Select("name", "email").First(&user)

,只查询

name

email

字段,减少网络传输和内存开销。预加载关联数据

DB.Preload("Orders").First(&user, 1)

。这会在查询

user

的同时,额外执行一次查询来加载该用户的所有

Orders

,避免N+1查询问题。

// 查询年龄大于20,按年龄降序排列的前5个用户var activeUsers []UserDB.Where("age > ?", 20).Order("age desc").Limit(5).Find(&activeUsers)log.Printf("查询到的活跃用户: %+v", activeUsers)// 查找第一个匹配条件的用户var firstUser UserDB.Where("name = ?", "Alice").First(&firstUser)log.Printf("查找到的Alice: %+v", firstUser)

更新 (Update)

保存所有字段

DB.Save(&user)

。如果

user

结构体有主键,GORM会更新所有字段(包括零值),否则会创建新记录。更新单个字段

DB.Model(&User{}).Where("id = ?", 1).Update("name", "New Name")

。这只会更新

name

字段,其他字段不受影响。更新多个字段

DB.Model(&User{}).Where("id = ?", 1).Updates(map[string]interface{}{"name": "New Name", "age": 30})

DB.Model(&User{}).Where("id = ?", 1).Updates(User{Name: "New Name", Age: 30})

零值更新:默认情况下,GORM在

Updates

方法中会忽略零值字段(例如,

int

类型的0,

string

类型的空字符串)。如果需要更新零值,可以使用

DB.Model(&User{}).Where("id = ?", 1).UpdateColumn("age", 0)

DB.Model(&User{}).Where("id = ?", 1).Select("age").Updates(User{Age: 0})

。这在某些业务场景下非常重要,比如将一个计数器清零。

// 更新用户年龄if userToUpdate := getUserByID(1); userToUpdate != nil {    userToUpdate.Age = 26    updateUser(userToUpdate) // Save会更新所有字段}// 仅更新邮箱字段DB.Model(&User{}).Where("id = ?", 2).Update("email", "bob.new@example.com")

删除 (Delete)

软删除:如果模型包含

gorm.DeletedAt

字段,

DB.Delete(&user, id)

会执行软删除,即设置

DeletedAt

字段为当前时间戳,而不是真正从数据库中删除记录。这是GORM的默认行为,非常有用,可以防止数据误删。硬删除:如果需要真正从数据库中删除记录,可以使用

DB.Unscoped().Delete(&user, id)

批量删除

DB.Where("age > ?", 60).Delete(&User{})

可以批量删除符合条件的记录。

// 软删除ID为3的用户deleteUser(3)// 硬删除ID为4的用户 (假设存在且需要彻底移除)DB.Unscoped().Delete(&User{}, 4)log.Printf("用户ID 4 已被硬删除")

在实际应用中,灵活运用这些CRUD方法,并结合事务(

DB.Transaction(func(tx *gorm.DB) error { ... })

)来保证数据的一致性,是构建稳健数据库操作逻辑的关键。同时,对于复杂的查询,GORM也支持执行原生SQL,作为ORM的补充。这提供了一个很好的平衡点,让我们在享受ORM便利性的同时,也能处理那些ORM难以表达的复杂场景。

以上就是Golang使用GORM操作数据库全流程的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
BeautifulSoup:从包含嵌套标签的HTML元素中高效提取文本内容
上一篇 2026年5月10日 10:38:06
C++ 框架的安全性漏洞如何影响应用程序?
下一篇 2026年5月10日 10:38:07

相关推荐

  • Go mgo 教程:高效存储扁平化 Go 嵌套结构体

    本教程旨在解决使用 `mgo` 库将 Go 语言中的嵌套结构体存储到 MongoDB 时,默认行为导致文档结构出现嵌套的问题。我们将深入探讨如何利用 `bson` 包提供的 `inline` 标签,将嵌入式结构体的字段提升到父级文档中,从而实现扁平化的 MongoDB 文档结构,提升数据存储的直观性…

    2026年5月10日
    000
  • PHP如何实现动态图表_PHP动态图表生成的方法与代码实例

    PHP通过结合前端图表库实现动态图表生成,常用方法包括:1. 使用Chart.js与Ajax获取PHP输出的JSON数据绘制柱状图;2. 利用Google Charts在前端嵌入PHP生成的JSON数据展示折线图;3. 通过ECharts调用PHP接口返回的数据渲染交互式饼图。核心是PHP处理数据并…

    2026年5月10日
    000
  • Python中如何实现Bellman-Ford算法?

    bellman-ford算法在python中可通过多次放松操作实现,用于求解最短路径并检测负权环。1)初始化距离数组,设源点距离为0。2)进行|v|-1次放松操作。3)检测负权环,若存在则抛出异常。该算法在金融网络中应用广泛,但处理大规模图时性能较慢,可考虑优化和并行化。 在Python中实现Bel…

    2026年5月10日
    100
  • 什么是币安人生?如何买入、卖出币安人生操作步骤教程

    币安人生指通过币安平台参与理财项目实现数字资产增值。首先登录账户,进入【资金】-【理财】页面,选择活期或定期产品并点击【申购】,输入金额前需重点关注预期年化收益、计息规则等条款;赎回时进入对应产品详情页,点击【赎回】并输入数量,注意不同产品规则差异及可能的收益损失,确认后资产将退回现货账户。 欧易官…

    2026年5月10日
    000
  • 从 Django 视图传递变量到模板中的 JavaScript 脚本

    在 Django Web 开发中,经常需要在前端 JavaScript 代码中使用后端 Python 代码中的数据。例如,你可能需要根据数据库中的数据动态生成图表,或者根据用户的角色显示不同的界面元素。直接在 JavaScript 中使用 Django 模板变量可能会导致安全问题,并且不够优雅。Dj…

    2026年5月10日
    000
  • 如何使用Golang实现错误处理_使用error类型和自定义错误

    Go错误处理显式依赖error接口,通过errors.New、fmt.Errorf(支持%w包装)和自定义结构体实现;用==、errors.Is、errors.As判断错误,支持错误链与类型提取。 Go 语言的错误处理强调显式判断和传递,不依赖异常机制。核心是使用内置的 error 接口类型,并可通…

    2026年5月10日
    000
  • HTML5网页如何实现拖拽功能 HTML5网页拖放API的详细解析

    首先设置元素draggable=”true”并监听dragstart事件,通过dataTransfer传递数据;然后为目标区域绑定dragover、dragenter和drop事件,其中dragover需调用preventDefault()以允许投放;最后在drop事件中获取…

    2026年5月10日
    000
  • Node.js http.createServer 常见陷阱与正确响应处理

    本文深入探讨了Node.js中使用`http.createServer`时常见的配置错误和响应处理问题。我们将详细讲解如何正确地将请求监听器函数传递给服务器实例,并强调在构建HTTP响应时,确保内容类型(Content-Type)与实际发送的数据(如HTML或JSON)保持一致的重要性,避免发送冲突…

    2026年5月10日
    000
  • Go语言中类型无关函数的实现:接口的应用

    在go语言中,与haskell等语言的hindley-milner类型系统不同,无法直接使用类型变量。go通过空接口`interface{}`来模拟类型无关的函数行为,允许函数处理任何类型的数据,从而实现类似泛型的功能,例如在实现`map`等高阶函数时。这种方式在go引入泛型之前是处理多态性的主要手…

    2026年5月10日
    100
  • Go语言中指针操作符*与取地址符&的全面解析

    本文深入探讨Go语言中*和&这两个核心操作符的作用。&用于获取变量的内存地址,生成一个指向该变量的指针;而*则用于声明指针类型、对指针进行解引用以访问其指向的值,以及通过指针间接修改变量的值。理解它们对于掌握Go的内存管理和数据传递机制至关重要,尤其是在函数参数传递和结构体操作中。 …

    2026年5月10日
    000
  • Chrome性能面板中XHR Ready State Change请求来源如何查看?

    Chrome性能面板:追踪XHR Ready State Change请求来源 Chrome开发者工具的性能面板火焰图中,有时会出现与XHR Ready State Change相关的任务,但缺少对应的请求信息。 以下步骤将帮助您定位这些请求的来源: 打开Chrome浏览器并访问目标网页。按下F12…

    用户投稿 2026年5月10日
    000
  • Electron 渲染进程安全集成 Node.js fs 模块指南

    本教程旨在指导开发者如何在 Electron 渲染进程中安全地使用 Node.js 的 fs 模块,避免启用 nodeIntegration: true 和 contextIsolation: false 等不安全的配置。通过利用 Electron 的 IPC(进程间通信)机制和预加载脚本(prel…

    2026年5月10日
    100
  • GLTF模型加载纹理缺失:从源头排查与解决指南

    在使用GLTFLoader加载3D模型时,若遇到纹理缺失问题,首要且关键的排查步骤是验证GLTF模型本身的完整性。本教程将指导您如何通过在线工具检查模型纹理,区分模型源文件问题与代码加载问题,并提供相应的解决方案,确保您的3D对象能正确显示纹理。 理解GLTF与纹理加载机制 gltf(gl tran…

    2026年5月10日
    000
  • PowerShell 调用 PHP 网页功能及结果处理

    本教程详细阐述了如何利用 PowerShell 的 Invoke-WebRequest cmdlet 外部调用 PHP 网页,并有效处理其返回结果。内容涵盖了基本的网页请求发送、HTTP 状态码的检查、网页内容的获取以及健壮的异常处理机制,旨在帮助用户实现与远程网页的自动化交互和数据处理。 使用 P…

    2026年5月10日
    000
  • GolangWeb项目路由优化与请求调度实践

    模块化路由设计提升Golang Web系统可维护性与性能。通过gin等框架按业务拆分路由组,实现清晰结构(如SetupUserRoutes管理用户路由);利用中间件分层调度,全局日志、局部权限校验(如AuthMiddleware作用于/api组)提升复用与安全;优化路由匹配,采用静态路径、减少嵌套、…

    2026年5月10日
    000
  • C++的consteval和constinit是什么_C++20中真正的编译期常量初始化

    consteval 强制函数在编译期求值,如 consteval int square(int n) 只能接受编译期常量参数;constinit 确保变量以常量初始化,如 constinit static int x = 42 避免动态初始化,用于解决静态初始化顺序问题。两者分别强化了编译期计算和初…

    2026年5月10日
    000
  • Blazor JS Interop 调用 Geolocation API 教程

    在 Blazor 中调用 Geolocation API 需通过 JS Interop:JavaScript 封装 navigator.geolocation 为 Promise 函数 getLocation,C# 使用 IJSRuntime.InvokeAsync 调用并匹配字段名,同时处理权限拒…

    2026年5月10日
    000
  • 如何精确获取多组单选按钮的最终选中值

    本教程旨在解决前端开发中,如何高效且准确地获取多组单选按钮(如产品变体选项)的最终选中值。我们将探讨在“添加到购物车”等操作触发时,避免中间选择状态干扰,仅捕获用户最终确认选项的最佳实践,并通过JavaScript代码示例详细演示其实现方法,确保数据一致性与用户体验。 场景描述与挑战 在电子商务网站…

    2026年5月10日
    000
  • 如何处理图像EXIF方向并转换为Base64,避免数据丢失

    本教程旨在解决图像EXIF方向信息在转换为Base64编码过程中丢失的问题。通过结合使用piexif库提取并移除EXIF方向数据,以及Jimp库对图像进行实际旋转,我们可以确保生成的Base64图像在视觉上保持正确的方向,从而满足API调用等需求,避免因EXIF元数据丢失而导致的显示错误。 在处理图…

    2026年5月10日
    000
  • html文档中含有java怎么运行_html含java运行方法【教程】

    现代浏览器不支持Java Applet,推荐通过JavaScript调用Java后端服务或使用WebAssembly运行Java代码。 如果您在HTML文档中嵌入了Java代码,但发现无法正常运行,这通常是因为现代浏览器不再支持Java小程序(Applet)或相关插件。以下是几种实现HTML中Jav…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信