
本教程详细介绍了如何将包含分层信息的扁平json字符串(例如“clothes - pants - jeans”)导入mysql数据库,并构建出具有父子关系的层级结构。通过php脚本解析json、拆分字符串并巧妙管理父级id,确保数据正确地存储为可查询的树状结构,解决在创建层级关系时常见的父级id关联错误。
在数据处理和存储中,我们经常会遇到需要将某种格式的扁平数据转换为具有层级关系的结构。本教程将指导您如何使用PHP将包含以特定分隔符表示的层级信息的JSON数据导入MySQL数据库,并构建一个可查询的树状结构。
假设我们有一个JSON文件,其中包含产品类别信息,但这些信息是以扁平的、通过分隔符连接的字符串形式存在的:
[
  {"productCategory":"Clothes - Pants - Jeans"},
  {"productCategory":"Clothes - Pants - Chinos"},
  {"productCategory":"Electronics - Laptops - Gaming"}
]我们的目标是将这些类别导入到MySQL数据库中,并形成一个具有父子关系的层级结构。期望的数据库表结构如下:
| taxonomy_id | taxonomy_name | taxonomy_parent | taxonomy_type | 
|---|---|---|---|
| 1 | Clothes | 0 | Category | 
| 2 | Pants | 1 | Category | 
| 3 | Jeans | 2 | Category | 
| 4 | Chinos | 2 | Category | 
| 5 | Electronics | 0 | Category | 
| 6 | Laptops | 5 | Category | 
| 7 | Gaming | 6 | Category | 
其中,taxonomy_id 是主键,taxonomy_name 是类别名称,taxonomy_parent 指向其父类别的 taxonomy_id(根类别为0),taxonomy_type 标识类别类型。
处理这种数据转换的核心思路是:
最初的尝试可能遇到的问题是,在处理子层级时,父级ID的跟踪逻辑可能出现偏差,导致某些子类别的 taxonomy_parent 字段未能正确关联到其父级。
我们将使用PHP来完成这一任务。为了简化数据库操作,我们假设存在一个 Insert_Taxonomy 类,它封装了与数据库交互的方法:
完整解决方案代码:
<?php
// 引入数据库操作类,例如 Insert_Taxonomy
include_once('../../../products/categories.inc.php');
// 读取JSON文件内容
$string = file_get_contents("category.json");
// 将JSON字符串解码为PHP关联数组
$json_decode = json_decode($string, true);
// 实例化数据库操作对象
$taxonomy = new Insert_Taxonomy();
// 遍历JSON数据中的每个产品类别条目
foreach ( $json_decode as $keys ) {
    $category_full_path = $keys['productCategory'];
    // 使用 '-' 分隔符将完整路径拆分成各个层级名称
    $split_categories = explode( '-', $category_full_path );
    // 移除每个层级名称前后的空白字符
    $trimmed_categories = array_map('trim', $split_categories);
    // 初始化 previous_value,用于跟踪当前层级的父级名称。
    // 对于第一个层级(根类别),其父级名称为0。
    $previous_value = 0; 
    // 遍历当前产品类别路径中的每个层级名称
    foreach ( $trimmed_categories as $key => $current_category_name ) {
        // 1. 检查当前类别名称是否已存在于数据库中
        $exists_tax_name = $taxonomy->Exists_Taxonomy_Name($current_category_name);
        // 2. 获取当前类别的父级ID
        // $previous_value 存储的是父级“名称”(或0),需要通过 Get_Taxonomy_Id 转换为实际的ID
        $parent_id = $taxonomy->Get_Taxonomy_Id($previous_value);
        // 3. 如果当前类别不存在,则插入新记录
        if ( empty( $exists_tax_name ) ) {
            $args = array($current_category_name, $parent_id, 'category');
            $taxonomy->create_taxonomy($args);
        }
        // 4. 更新 previous_value 为当前类别名称,以便在下一次循环中作为子类别的父级
        // 这一步至关重要,确保了正确的父子关系链
        $previous_value = $current_category_name;
    }
}
?>文件读取与解码:
$string = file_get_contents("category.json");
$json_decode = json_decode($string, true);这部分负责读取JSON文件并将其转换为PHP数组,true 参数确保解码为关联数组。
实例化数据库操作类:
$taxonomy = new Insert_Taxonomy();
在循环外部实例化 Insert_Taxonomy 对象,避免在每次迭代中重复创建,提高效率。
拆分与清理:
$split_categories = explode( '-', $category_full_path );
$trimmed_categories = array_map('trim', $split_categories);explode 函数将字符串按 '-' 分隔符拆分为数组。array_map('trim', ...) 用于移除每个子字符串两端的空白,确保数据干净。
父级ID跟踪 ($previous_value):
$previous_value = 0; // 初始化为0,表示根类别的父级 // ... $parent_id = $taxonomy->Get_Taxonomy_Id($previous_value); // ... $previous_value = $current_category_name; // 更新为当前类别名称
这是解决层级关系的关键。$previous_value 变量在每次内部循环中存储 前一个 已经处理过的类别名称(或0),这个名称随后被用来通过 Get_Taxonomy_Id 方法获取其对应的 taxonomy_id,作为当前类别的 taxonomy_parent。
操作顺序的重要性:
$exists_tax_name = $taxonomy->Exists_Taxonomy_Name($current_category_name); // 1. 检查是否存在
$parent_id = $taxonomy->Get_Taxonomy_Id($previous_value); // 2. 获取父ID
if ( empty( $exists_tax_name ) ) { // 3. 如果不存在则插入
    $args = array($current_category_name, $parent_id, 'category');
    $taxonomy->create_taxonomy($args);
}
$previous_value = $current_category_name; // 4. 更新previous_value正确的执行顺序是:
为了使上述代码能够运行,Insert_Taxonomy 类可能看起来像这样(仅为示例,实际实现需根据您的数据库连接和ORM库进行调整):
<?php
// 假设这是一个简化的数据库连接和操作类
class Insert_Taxonomy {
    private $pdo; // PDO数据库连接对象
    public function __construct() {
        // 实际应用中,这里会建立数据库连接
        // 例如:$this->pdo = new PDO("mysql:host=localhost;dbname=your_db", "user", "pass");
        // 为简化示例,我们假设连接已存在或通过其他方式获取
        // 这里只是一个占位符,实际需要您自己的数据库连接代码
        $this->pdo = null; // 实际应为有效的PDO对象
        error_log("Insert_Taxonomy class initialized. (Database connection assumed)");
    }
    /**
     * 检查分类名称是否存在
     * @param string $name 分类名称
     * @return bool 如果存在则返回true,否则返回false
     */
    public function Exists_Taxonomy_Name($name) {
        if ($name === 0) return true; // 根节点始终被认为“存在”
        // 实际查询数据库
        // $stmt = $this->pdo->prepare("SELECT COUNT(*) FROM taxonomy_table WHERE taxonomy_name = ?");
        // $stmt->execute([$name]);
        // return $stmt->fetchColumn() > 0;
        error_log("Checking if taxonomy name '{$name}' exists.");
        // 模拟返回
        return false; // 假设默认不存在,以便创建
    }
    /**
     * 根据分类名称获取ID
     * @param string|int $name 分类名称或0(表示根)
     * @return int 分类ID,如果不存在或为0则返回0
     */
    public function Get_Taxonomy_Id($name) {
        if ($name === 0) {
            return 0; // 根分类的ID为0
        }
        // 实际查询数据库
        // $stmt = $this->pdo->prepare("SELECT taxonomy_id FROM taxonomy_table WHERE taxonomy_name = ?");
        // $stmt->execute([$name]);
        // $result = $stmt->fetch(PDO::FETCH_ASSOC);
        // return $result ? $result['taxonomy_id'] : 0;
        error_log("Getting taxonomy ID for name '{$name}'.");
        // 模拟返回一个ID,实际应从数据库获取
        // 在实际运行中,如果名称不存在,这里应返回0或抛出错误
        return crc32($name); // 简单模拟一个ID
    }
    /**
     * 创建新的分类记录
     * @param array $args 包含 [name, parent_id, type]
     * @return bool 插入成功返回true,否则返回false
     */
    public function create_taxonomy($args) {
        list($name, $parent_id, $type) = $args;
        // 实际插入数据库
        // $stmt = $this->pdo->prepare("INSERT INTO taxonomy_table (taxonomy_name, taxonomy_parent, taxonomy_type) VALUES (?, ?, ?)");
        // return $stmt->execute([$name, $parent_id, $type]);
        error_log("Creating taxonomy: Name='{$name}', ParentID='{$parent_id}', Type='{$type}'.");
        return true; // 模拟成功
    }
}
?>通过本教程,您应该已经掌握了如何将扁平的、分隔符连接的JSON数据转换为MySQL数据库中的分层结构。关键在于正确解析数据,并巧妙地管理父级ID的跟踪逻辑,确保每个子类别都能准确地关联到其直接父级。这种方法不仅适用于产品类别,也适用于任何需要构建树状或层级关系的数据导入场景。
以上就是将扁平JSON数据转换为MySQL分层结构教程的详细内容,更多请关注php中文网其它相关文章!
 
                        
                        每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
 
                Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号