0

0

开发自己的Data Access Application Block[上篇]

php中文网

php中文网

发布时间:2016-06-07 15:48:49

|

1354人浏览过

|

来源于php中文网

原创

经常在网上看到对ORM的讨论沸沸扬扬,我也来凑个热闹,谈谈我写的一个ORM。最近在做一项工作,把我们经常用到的一些业务逻辑抽象出来,写成一个个的Application Block,使之可以运用到不同的Application中,比如Data Access,Messaging,Auditing,Data bind

经常在网上看到对ORM的讨论沸沸扬扬,我也来凑个热闹,谈谈我写的一个ORM。最近在做一项工作,把我们经常用到的一些业务逻辑抽象出来,写成一个个的Application Block,使之可以运用到不同的Application中,比如Data Access,Messaging,Auditing,Data binding等等。现在先做一个Data access application block。由于时间仓促,没有进行什么优化和较多的测试,大家不必深究我所提供的Code ,我只希望为大家的ORM提供另一种想法。

这个application block应达到的目的:

  • 封装所有的Data Access操作。

  • 适合主流的DBMS:SQL Server(2000和2005),Oracle(9i和10g),DB2。

  • 尽量简化Developer的操作和提供最大的灵活性,在Data Retrieval方面,只要指定SQL或者Stored Procedure和相应的参数;在Data Update方面,既可以直接调用SQL和Stored Procedure,还可以把包含多个相互关联Data Table的Dataset通过一次调用实现数据的更新。此外,可以自由地选择使用SQL还是Stored procedure;可以使用Commander builder生成Command或者使用基于Mapped stored procedure生成的Command进行数据更新。

  • 实现泛型编程,使使用该AppBlock的代码能够适合所有的数据库。

  • 实现Transaction。

  • 提供可配置性,包括不同数据库的配置,不同Data Mapping的配置等等。

下面是该AppBlock使用到的Entity:
开发自己的Data Access Application Block[上篇]

  • Database:Abstract Class,封装了绝大部分和具体数据库无关的Data Access操作逻辑。通过两个Mapping:IDbParameterNameMapping和IStoredProcedureNameMapping,实现Dataset和Db的一个映射。比如Dataset中Data table name和Stored procedure name的Mapping,Data table中Field和Stored procedure中参数名的Mapping。这两个Mapping是可以配置的,你只需要实现提供的Interface编写适合你的Mapping provider就可以了。

  • SqlDatabase:封装基于SQL Server 的操作。ADO.NET 2.0在1.0的基础上作了很大的改善,主要的增加的大量的基类,为我们进行泛型编程,编写和具体Db无关的代码变得异常容易。所以我们把大多数Data Access的操作可以封装在Abstract Database类中,SqlDatabase中的内容实际上是很少的。

  • OracleDatabase:封装基于Oracle的操作。

  • IDbParameterNameMapping和IStoredProcedureNameMapping:我想大家都是这样的感受,实现ORM的本质就是实现内存中的数据(主要是Dataset)和数据库的一个映射。在Dataset和数据库中的Table相互Mapping方面,我觉得没有必要采用特殊的Mapping,直接和简单易行的就是Table和Dataset中的Data Table完全匹配(table name 和field name完全匹配)。所以重要的是实现Dataset和Stored procedure的Mapping:Table Name如何与进行Insert,Update,Delete的Stored procedure name匹配,不同Version(original & current)的Field如何与Stored procedure的Parameter name 匹配。而这样一个匹配应该是可配置的,因为每个Application在数据库设计时的命名都有各自的要求,所以我在这里采用的Provider的设计模式。用户可以实现这两个Interface编写适合自己的Mapping provider,通过我提供的Configuration block很容易地完成配置。同时,我写了一个默认的,简单的Mapping:SimpleDbParameterNameMapping和SimpleStoredProcedureNameMapping。

      有一点需要补充的是,要实现上面的Mapping,对Stored Procedure的命名有较高的要求,手工编写的方式已经不能适合我们的要求,所以我们需要一个生成Stored procedure的Generator,这个Generator也使用这两个可配置的Mapping接口。

  • DatabaseFactory: 一个静态的类,根据配置的信息创建你需要的具体的Database对象,实现泛型化编程。

  • DataAccessConfigurationCollection,DataAccessConfigurationElement,DataAccessConfigurationdiv 三个Configuration的Class。

    生活同城信息网系统
    生活同城信息网系统

    fankuan8生活同城信息网系统 v1206采用主流的Asp+Access开发设计,网站美工设计方面更大气,漂亮!网站浏览器兼容性也比较好,网站功能方面的细节方面十分强大。 网站程序的几大特点: 1.全站页面实行了伪静态化,各类型网站服务器的伪静态文件都已近处理好了,无需自己再做伪静态出来。 2.网站前台开始使用了fankuan8独立开发的互助链系统,开始使用时,在网站底部点击链接根据提示马上

    下载

为了使大家清楚地看出这个Application block所有的操作,我把所有的操作封装在一个IDatabase的interface中,不过需要注意的是,我采用的是基于Abstract class的编程,而不是基于Interface的编程,相信大家对这两种方式讨论得已经碰倒的太多了,孰优孰劣我就不想对说了。这个IDatabase 接口,只是展示所有Operation之用,并没有在我的代码中用到。

开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]namespace Artech.ApplicationBlock.DataAccess
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]{
开发自己的Data Access Application Block[上篇]    
public interface IDatabase
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]    
开发自己的Data Access Application Block[上篇]{
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
Fill a System.Data.DataSet with retrieved data.#region Fill a System.Data.DataSet with retrieved data.
开发自己的Data Access Application Block[上篇]        
void FillDataSet(DataSet dataInfo, string commandText, IDictionarystringobject> parameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
void FillDataSet(DataSet dataInfo, string tableName, string commandText, IDictionarystringobject> parameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
void FillDataSet(DataSet dataInfo, string tableName, CommandType commandType, string commandText, IDictionarystringobject> parameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
#endregion

开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
Save the changed data which is stored in a dataset into database.#region Save the changed data which is stored in a dataset into database.
开发自己的Data Access Application Block[上篇]        
void UpdateData(DataSet dataInfo);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
void UpdateData(DataTable table);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
void UpdateData(DataTable table, string insertCommandText, string updateCommandText, string deleteCommandText,
开发自己的Data Access Application Block[上篇]           Dictionary
stringobject> insertParameters, Dictionarystringobject> updateParameters, Dictionarystringobject> deleteParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]                
void UpdateData(DataTable table, CommandType commandType, string insertCommandText, string updateCommandText, string deleteCommandText,
开发自己的Data Access Application Block[上篇]            Dictionary
stringobject> insertParameters, Dictionarystringobject> updateParameters, Dictionarystringobject> deleteParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
void UpdateData(DataTable table, DbCommand insertCommand, DbCommand updateCommand, DbCommand deleteCommand);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
#endregion

开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
Execute a command and return the affect row count.#region Execute a command and return the affect row count.
开发自己的Data Access Application Block[上篇]        
int ExecuteNonQuery(CommandType commandType, string commandText, Dictionarystringobject> inputParameters, Dictionarystringobject> outputParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
int ExecuteNonQuery(CommandType commandType, string commandText, Dictionarystringobject> inputParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
int ExecuteNonQuery( string commandText, Dictionarystringobject> inputParameters, Dictionarystringobject> outputParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
int ExecuteNonQuery( string commandText, Dictionarystringobject> inputParameters);
开发自己的Data Access Application Block[上篇]        
#endregion

开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
Execute a command and return the data in the form of data reader.#region Execute a command and return the data in the form of data reader.
开发自己的Data Access Application Block[上篇]        DbDataReader ExecuteReader(CommandType commandType, 
string commandText, Dictionarystringobject> inputParameters, Dictionarystringobject> outputParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        DbDataReader ExecuteReader(CommandType commandType, 
string commandText, Dictionarystringobject> inputParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        DbDataReader ExecuteReader(
string commandText, Dictionarystringobject> inputParameters, Dictionarystringobject> outputParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        DbDataReader ExecuteReader(
string commandText, Dictionarystringobject> inputParameters);
开发自己的Data Access Application Block[上篇]        
#endregion

开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
Execute a command and return a scalar value.#region Execute a command and return a scalar value.
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
object ExecuteScalar(CommandType commandType, string commandText, Dictionarystringobject> inputParameters, Dictionarystringobject> outputParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
object ExecuteScalar(CommandType commandType, string commandText, Dictionarystringobject> inputParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
object ExecuteScalar(string commandText, Dictionarystringobject> inputParameters, Dictionarystringobject> outputParameters);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]        
object ExecuteScalar(string commandText, Dictionarystringobject> inputParameters);
开发自己的Data Access Application Block[上篇]        
#endregion

开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
Transaction based operation#region Transaction based operation
开发自己的Data Access Application Block[上篇]        
void BeginTransaction();
开发自己的Data Access Application Block[上篇]        
void Commit();
开发自己的Data Access Application Block[上篇]        
void RollBack();
开发自己的Data Access Application Block[上篇]        
#endregion

开发自己的Data Access Application Block[上篇]    }

开发自己的Data Access Application Block[上篇]}

开发自己的Data Access Application Block[上篇]

这个列表和大部分ORM没有什么太大的区别,大家已经司空见惯,实现起来也不会有什么太大的困难。对于大部分操作,我不会做详细的介绍。接下来我们来简要地看看这样一个AppBlock是如何实现的。
1. Data Mapping

我们首先来看看Data Mapping:实现Dataset中Table name和Stored Procedure Name的Mapping,以及Dataset 中的Field 和Stored procedure的参Parameter name的Mapping。

IDbParameterNameMapping

开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]using System;
开发自己的Data Access Application Block[上篇]
using System.Collections.Generic;
开发自己的Data Access Application Block[上篇]
using System.Text;
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]
namespace Artech.ApplicationBlock.DataMapping
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]{
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]    
/**//// 


开发自己的Data Access Application Block[上篇]    
/// IStoredProcedureNameMapping defines the mapping between the data table name and the name of stored procedures to perform insertion, modification and deletion operation.
开发自己的Data Access Application Block[上篇]    
/// 
开发自己的Data Access Application Block[上篇]    public interface IStoredProcedureNameMapping
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]    
开发自己的Data Access Application Block[上篇]{
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
/**//// 
开发自己的Data Access Application Block[上篇]        
/// Get the name of stored procedure to perform seletion operation.
开发自己的Data Access Application Block[上篇]        
/// 
开发自己的Data Access Application Block[上篇]        
/// The name of the database table.
开发自己的Data Access Application Block[上篇]        
/// The name of stored procedure to perform seletion operation

开发自己的Data Access Application Block[上篇]        string GetSelectStoredProcedureName(string tableName);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
/**//// 
开发自己的Data Access Application Block[上篇]        
/// Get the name of stored procedure to perform insert operation.
开发自己的Data Access Application Block[上篇]        
/// 
开发自己的Data Access Application Block[上篇]        
/// The name of the database table.
开发自己的Data Access Application Block[上篇]        
/// The name of stored procedure to perform insertion operation

开发自己的Data Access Application Block[上篇]        string GetInsertStoredProcedureName(string tableName);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
/**//// 
开发自己的Data Access Application Block[上篇]        
/// Get the name of stored procedure to perform modification operation.
开发自己的Data Access Application Block[上篇]        
/// 
开发自己的Data Access Application Block[上篇]        
/// The name of the database table.
开发自己的Data Access Application Block[上篇]        
/// The name of stored procedure to perform modification operation

开发自己的Data Access Application Block[上篇]        string GetModifyStoredProcedureName(string tableName);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
/**//// 
开发自己的Data Access Application Block[上篇]        
/// Get the name of stored procedure to perform deletion operation.
开发自己的Data Access Application Block[上篇]        
/// 
开发自己的Data Access Application Block[上篇]        
/// The name of the database table.
开发自己的Data Access Application Block[上篇]        
/// The name of stored procedure to perform deletion operation

开发自己的Data Access Application Block[上篇]        string GetDeleteStoredProcedureName(string tableName);
开发自己的Data Access Application Block[上篇]    }

开发自己的Data Access Application Block[上篇]}

开发自己的Data Access Application Block[上篇]

IDbParameterNameMapping

开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]using System;
开发自己的Data Access Application Block[上篇]
using System.Collections.Generic;
开发自己的Data Access Application Block[上篇]
using System.Text;
开发自己的Data Access Application Block[上篇]
using System.Data;
开发自己的Data Access Application Block[上篇]
namespace Artech.ApplicationBlock.DataMapping
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]{
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]    
/**//// 


开发自己的Data Access Application Block[上篇]    
/// IDbParameterNameMapping define the defult mapping between the source column name and the parameter name of the corresponding stored procedure.
开发自己的Data Access Application Block[上篇]    
/// 
开发自己的Data Access Application Block[上篇]    public interface IDbParameterNameMapping
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]    
开发自己的Data Access Application Block[上篇]{
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
/**//// 
开发自己的Data Access Application Block[上篇]        
/// Get the source column name based on the parameter name of the related stored procedure.
开发自己的Data Access Application Block[上篇]        
/// 
开发自己的Data Access Application Block[上篇]        
/// The parameter name of the corresponding stored procedure.
开发自己的Data Access Application Block[上篇]        
/// The source column name corresponding to the parameter name.

开发自己的Data Access Application Block[上篇]        string GetSourceCoulmnName(string patameterName);
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]        
/**//// 
开发自己的Data Access Application Block[上篇]        
/// Get the source parameter name based on the source column name.
开发自己的Data Access Application Block[上篇]        
/// 
开发自己的Data Access Application Block[上篇]        
/// The source column name corresponding to the parameter name.
开发自己的Data Access Application Block[上篇]        
/// The data row version of the source solumn conressponding to the parameter.
开发自己的Data Access Application Block[上篇]        
/// The parameter name of the corresponding stored procedure.

开发自己的Data Access Application Block[上篇]        string GetParameterName(string columnName, DataRowVersion rowVersion);
开发自己的Data Access Application Block[上篇]    }

开发自己的Data Access Application Block[上篇]}

开发自己的Data Access Application Block[上篇]

这两个Mapping主要用在通过Dataset跟新数据库的场景,利用IDbParameterNameMapping,我们通过Dataset中各个Table name获得对它进行Insert,Update,Delete操作的Stored procedure的name。利用IDbParameterNameMapping,我们可以为Stored procedure的Parameter指定对应的Source field.  

注:GetParameterName方法实际上是不需要的,我把使用在另一个AppBlock中。

接下来我们来写两个实现了上面连个Interface的默认的mapping:SimpleStoredProcedureNameMapping和SimpleDbParameterNameMapping。他实际上实现了这样的Mapping:比如Table name为T_ABC_DEF(我经常用的命名方式:以T开头代表Table,名称大写并一下划线连接),那么对应的Stored procedure name分别为:sp_abc_def_s(Select), sp_abc_def_i(Insert), sp_abc_def_u(Update), sp_abc_def_d(delete)。如果Field name为ABC_123,那么对于Original version的Parameter name为o_abc_123(o代表Original),Current version的Parameter name为p_abc_123(p代表一般意义的Parameter)。

开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]using System;
开发自己的Data Access Application Block[上篇]
using System.Collections.Generic;
开发自己的Data Access Application Block[上篇]
using System.Text;
开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]
namespace Artech.ApplicationBlock.DataMapping
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]
开发自己的Data Access Application Block[上篇]{
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]    
/**//// 


开发自己的Data Access Application Block[上篇]    
/// IStoredProcedureNameMapping defines the mapping between the data table name and the name of stored procedures to perform insertion, modification and deletion operation.
开发自己的Data Access Application Block[上篇]    
/// 
开发自己的Data Access Application Block[上篇]    public class SimpleStoredProcedureNameMapping:IStoredProcedureNameMapping
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]    
开发自己的Data Access Application Block[上篇]{
开发自己的Data Access Application Block[上篇]开发自己的Data Access Application Block[上篇]

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

0

2026.02.28

Golang 工程化架构设计:可维护与可演进系统构建
Golang 工程化架构设计:可维护与可演进系统构建

Go语言工程化架构设计专注于构建高可维护性、可演进的企业级系统。本专题深入探讨Go项目的目录结构设计、模块划分、依赖管理等核心架构原则,涵盖微服务架构、领域驱动设计(DDD)在Go中的实践应用。通过实战案例解析接口抽象、错误处理、配置管理、日志监控等关键工程化技术,帮助开发者掌握构建稳定、可扩展Go应用的最佳实践方法。

2

2026.02.28

Golang 性能分析与运行时机制:构建高性能程序
Golang 性能分析与运行时机制:构建高性能程序

Go语言以其高效的并发模型和优异的性能表现广泛应用于高并发、高性能场景。其运行时机制包括 Goroutine 调度、内存管理、垃圾回收等方面,深入理解这些机制有助于编写更高效稳定的程序。本专题将系统讲解 Golang 的性能分析工具使用、常见性能瓶颈定位及优化策略,并结合实际案例剖析 Go 程序的运行时行为,帮助开发者掌握构建高性能应用的关键技能。

0

2026.02.28

Golang 并发编程模型与工程实践:从语言特性到系统性能
Golang 并发编程模型与工程实践:从语言特性到系统性能

本专题系统讲解 Golang 并发编程模型,从语言级特性出发,深入理解 goroutine、channel 与调度机制。结合工程实践,分析并发设计模式、性能瓶颈与资源控制策略,帮助将并发能力有效转化为稳定、可扩展的系统性能优势。

12

2026.02.27

Golang 高级特性与最佳实践:提升代码艺术
Golang 高级特性与最佳实践:提升代码艺术

本专题深入剖析 Golang 的高级特性与工程级最佳实践,涵盖并发模型、内存管理、接口设计与错误处理策略。通过真实场景与代码对比,引导从“可运行”走向“高质量”,帮助构建高性能、可扩展、易维护的优雅 Go 代码体系。

16

2026.02.27

Golang 测试与调试专题:确保代码可靠性
Golang 测试与调试专题:确保代码可靠性

本专题聚焦 Golang 的测试与调试体系,系统讲解单元测试、表驱动测试、基准测试与覆盖率分析方法,并深入剖析调试工具与常见问题定位思路。通过实践示例,引导建立可验证、可回归的工程习惯,从而持续提升代码可靠性与可维护性。

1

2026.02.27

漫蛙app官网链接入口
漫蛙app官网链接入口

漫蛙App官网提供多条稳定入口,包括 https://manwa.me、https

94

2026.02.27

deepseek在线提问
deepseek在线提问

本合集汇总了DeepSeek在线提问技巧与免登录使用入口,助你快速上手AI对话、写作、分析等功能。阅读专题下面的文章了解更多详细内容。

6

2026.02.27

AO3官网直接进入
AO3官网直接进入

AO3官网最新入口合集,汇总2026年可用官方及镜像链接,助你快速稳定访问Archive of Our Own平台。阅读专题下面的文章了解更多详细内容。

174

2026.02.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Node.js 教程
Node.js 教程

共57课时 | 12.3万人学习

Rust 教程
Rust 教程

共28课时 | 6.4万人学习

Vue 教程
Vue 教程

共42课时 | 9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号