Sqlite“无法加载 DLL“e_sqlite3”: 找不到指定的模块”解决方法

今天在使用 NuGet 安装 System.Data.SQLite 库后,发现运行就报错:System.DllNotFoundException:"无法加载 DLL"e_sqlite3": 找不到指定的模块。 (异常来自 HRESULT:0x8007007E)。"

百度查了半天,啥都没有,我也是醉了,问AI也没问出个答案,回答全是乱七八糟的,于是我又查看了 System.Data.SQLite 的自述文件和包详细信息,是这样:

啥都没写,这种库也是奇葩,我有点好奇能用在商业项目中么,一点规范都没有,有点像个人开发者的行为。

我又搜了下 e_sqlite3,不是没用就是要钱,what ??

后面我又在 百度 找 System.Data.SQLite 官网,找到了这么一个地址

打开一看,刚好找到了答案,界面如下:

这里的描述文字写着:

System.Data.SQLite is available on nuget.org. Typical usage would involve adding two packages, one for System.Data.SQLite, and one containing a native built of SQLite. For example:

翻译成中文:

System.Data.SQLite 可在 nuget.org 上获取。典型用法通常需要添加两个 NuGet 包:一个用于 System.Data.SQLite,另一个包含 SQLite 的本机构建(例如 System.Data.SQLite.Core)。例如:

看到这里我算是知道了,还需要安装另一个库:SourceGear.sqlite3

这下就好办了,于是我就装了 SourceGear.sqlite3 这个库,如下:

SourceGear.sqlite3 这个库 包详细信息这里有一段描述

This package contains builds of the native SQLite code for various platforms.

These builds have "e_sqlite3" as the base name. The first three numbers in the version number of this package indicate the version of SQLite that was used to build it. The fourth number, if there is one, is incremented as needed for packaging-specific changes.

翻译成中文:

此软件包包含适用于不同平台的原生 SQLite 代码构建版本。这些构建版本的基础名称均为 ​​"e_sqlite3"​​。软件包版本号中的前三位数字表示构建时所使用的 SQLite 版本,若存在第四位数字,则会根据打包相关的修改需求进行递增

好吧,我先搭建下 Sqlite 的环境,让 Winform 能调用 Sqlite 进行增删改查,项目我就不上传了,直接贴一个 SqliteHelper.cs 好了

cs 复制代码
using System;
using System.Data;
using System.Data.SQLite;
using System.Collections.Generic;
using System.Reflection;

public class SqliteHelper
{
    /// <summary>
    /// 数据库连接字符串
    /// </summary>
    public static string ConnectionString { get; set; } = "Data Source=work.db;Version=3;Pooling=True;Journal Mode=WAL;Cache=Shared;Synchronous=NORMAL;BusyTimeout=5000;";

    /// <summary>
    /// 执行 插入、更新、删除 操作
    /// </summary>
    /// <param name="sql">要执行的SQL语句(INSERT/UPDATE/DELETE)</param>
    /// <param name="parameters">SQL参数数组,防止SQL注入</param>
    /// <returns>p1.是否成功,p2.错误信息</returns>
    public static (bool, string) ExecuteSql(string sql, params SQLiteParameter[] parameters)
    {
        if (ConnectionString == null)
            return (false, "请先设置 ConnectionString");

        try
        {
            int line = 0;
            using (var conn = new SQLiteConnection(ConnectionString))
            {
                conn.Open();
                using (var cmd = new SQLiteCommand(sql, conn))
                {
                    if (parameters != null && parameters.Length > 0)
                        cmd.Parameters.AddRange(parameters);
                    // ExecuteNonQuery 返回受影响的行数
                    line = cmd.ExecuteNonQuery();
                }
                conn.Close();
            }
            return (line > 0, null);
        }
        catch (Exception ex)
        {
            Console.WriteLine(ex.ToString());
            return (false, ex.Message);
        }
    }

    /// <summary>
    /// 执行查询SQL
    /// </summary>
    /// <param name="sql">SQL语句</param>
    /// <param name="parameters">SQL参数数组,防止SQL注入</param>
    /// <returns>p1.是否成功,p2.查询的数据,p3.错误信息</returns>
    public static (bool, DataTable, string) Query(string sql, params SQLiteParameter[] parameters)
    {
        if (ConnectionString == null)
            return (false, null, "请先设置 ConnectionString");

        try
        {
            var dataTable = new DataTable();
            using (var conn = new SQLiteConnection(ConnectionString))
            {
                conn.Open();
                using (var cmd = new SQLiteCommand(sql, conn))
                {
                    if (parameters != null && parameters.Length > 0)
                        cmd.Parameters.AddRange(parameters);
                    //使用数据适配器填充 DataTable
                    using (var adapter = new SQLiteDataAdapter(cmd))
                        adapter.Fill(dataTable);
                }
                conn.Close();
            }
            return (true, dataTable, null);
        }
        catch (Exception ex)
        {
            Console.WriteLine(ex.ToString());
            return (false, null, ex.Message);
        }
    }

    /// <summary>
    /// 执行查询SQL
    /// </summary>
    /// <typeparam name="T">结果集列表中对象的类型,需提供无参构造函数</typeparam>
    /// <param name="sql">SQL语句</param>
    /// <param name="parameters">SQL参数数组,防止SQL注入</param>
    /// <returns>p1.是否成功,p2.查询的数据,p3.错误信息</returns>
    public static (bool, List<T>, string) Query<T>(string sql, params SQLiteParameter[] parameters) where T : new()
    {
        if (ConnectionString == null)
            return (false, null, "请先设置 ConnectionString");

        List<T> list = new List<T>();
        try
        {
            using (var conn = new SQLiteConnection(ConnectionString))
            {
                conn.Open();
                using (var cmd = new SQLiteCommand(sql, conn))
                {
                    if (parameters != null && parameters.Length > 0)
                        cmd.Parameters.AddRange(parameters);
                    
                    using (var reader = cmd.ExecuteReader())
                    {
                        // 获取一次性获取列的元数据,用于反射匹配属性
                        var schemaTable = reader.GetSchemaTable();
                        // 建立列名到序号的映射
                        var ordinals = new Dictionary<string, int>();
                        foreach (DataRow col in schemaTable.Rows)
                        {
                            string colName = col["ColumnName"].ToString();
                            ordinals[colName] = (int)col["ColumnOrdinal"];
                        }
                        // 获取泛型类型的所有可写属性
                        Type objType = typeof(T);
                        PropertyInfo[] properties = objType.GetProperties();
                        while (reader.Read())
                        {
                            T obj = new T();
                            foreach (PropertyInfo prop in properties)
                            {
                                if (!prop.CanWrite) continue;
                                if (ordinals.ContainsKey(prop.Name))
                                {
                                    object value = reader.GetValue(ordinals[prop.Name]);
                                    if (value != DBNull.Value)
                                    {
                                        // 转换为属性的实际类型后赋值
                                        Type targetType = Nullable.GetUnderlyingType(prop.PropertyType) ?? prop.PropertyType;
                                        object safeValue = Convert.ChangeType(value, targetType);
                                        prop.SetValue(obj, safeValue);
                                    }
                                }
                            }
                            list.Add(obj);
                        }
                        reader.Close();
                    }
                }
                conn.Close();
            }
            return (true, list, null);
        }
        catch (Exception ex)
        {
            Console.WriteLine(ex.ToString());
            return (false, null, ex.Message);
        }
    }
}

运行项目,你会发现项目依然报错:

翻译成中文:此软件包不支持 Any CPU 版本

也就是说,生成的时候,不能使用 Any CPU

点击项目的 配置管理器

在 "活动解决方案平台" 这里展开下拉框,点击新建

选择一个平台,复制设置我选择空(Any CPU 这项没测试过),然后点击确定。关闭这个窗体。

然后在 运行这里就能看到 x64 的选项了

选择 x64,现在 Sqite 能正常的运行了

如果你不想使用 x64 ,想删除这个选项,也很简单,同样是点击配置管理器

找到活动解决方案平台

点击编辑

然后选中 x64,然后点击移除就行了

end

相关推荐
Mahir089 小时前
Redis 与 MySQL 数据同步:一致性保证的完整解决方案
数据库·redis·mysql·缓存·面试·数据一致性
2301_769340679 小时前
如何在 Vuetify 中可靠捕获 Chip 关闭事件(包括键盘触发).txt
jvm·数据库·python
AC赳赳老秦9 小时前
供应链专员提效:OpenClaw自动跟踪物流信息、更新库存数据,异常自动提醒
java·大数据·服务器·数据库·人工智能·自动化·openclaw
灵犀学长10 小时前
基于 Spring ThreadPoolTaskScheduler + CronTrigger 实现的动态定时任务调度系统
java·数据库·spring
北秋,10 小时前
PostgreSQL(Postgres)数据库基础用法 + 数字型 + 字符型 完整联合注入实战
数据库·postgresql·开源
m0_5967490911 小时前
JavaScript中手动实现一个new操作符的底层逻辑
jvm·数据库·python
多加点辣也没关系11 小时前
Redis 的安装(详细教程)
数据库·redis·缓存
数据库小学妹12 小时前
数据库连接池避坑指南:告别“连接超时”与“资源耗尽”,让系统跑得更快!
数据库·redis·sql·mysql·缓存·dba
dishugj12 小时前
HANA 数据库备份与恢复
数据库·oracle
前进的李工12 小时前
EXPLAIN输出格式全解析:JSON、TREE与可视化
开发语言·数据库·mysql·性能优化·explain