gpt4 book ai didi

php - 递归函数查找MySQL表的所有依赖项

转载 作者:行者123 更新时间:2023-11-30 22:12:54 26 4
gpt4 key购买 nike

我有一个数据库,其中有几个相关的表。例如,用户表包含系统中的所有用户。然后我有一个名为 User_friend 的索引表,其中包含用户与其好友之间的关系。

我有一个函数 loadObject($class, $id) 其调用如下:

 loadObject('User', 1);

并将 id = 1 的用户作为具有以下格式的数组返回:

array(
'id' => 1,
'username' => 'My user',
// the following array contains all the entries in User_invited
'friends' => [2, 3, 4, 5],
// same for comments
'comments' => [6, 7]
'type' => 'User'
);

我试图想出一个递归函数来检查 id = 1 的用户,找到所有 friend (在“ friend ”数组内),然后循环遍历每个值,找到那些用户及其 friend ,直到它到达链的末端而不复制任何条目。

这看起来很简单。问题在于,除了 friend 之外,我们还可以与评论、事件和许多其他表建立其他关系。

棘手的部分是这个函数不仅适用于“User”类,而且适用于我们定义的任何类。

我正在做的是使用某种索引数组来定义哪些索引表引用哪些主表。

例如:

$dependencies = [
'friends' => 'User'
];

这意味着,当我们找到“ friend ”键时,我们应该查询“用户”表。

这是我的代码:

<?php
$class = $_GET['class'];
// if we receive a collection of ids, find each individual object
$ids = explode(",", $_GET['ids']);

// load all main objects first
foreach($ids as $id) {
$error = isNumeric($id);
$results[] = loadObject($class,$id);
}

$preload = $results;
$output = [];

$output = checkPreload($preload);
print json_encode($output);

function checkPreload($preload)
{
$dependencies = [
'comment' => 'Comment',
'friend' => 'User',
'enemy' => 'User',
'google' => 'GoogleCalendarService',
'ical' => 'ICalCalendarService',
'owner' => 'User',
'invited' => 'User'
];

foreach($preload as $key => $object)
{
foreach($object as $property => $values)
{
// if the property is an array (has dependencies)
// i.e. event: [1, 2, 3]
if(is_array($values) && count($values) > 0)
{
// and if the dependency exists in our $dependencies array, find
// the next Object we have to retrieve
// i.e. event => CatchAppCalendarEvent
if(array_key_exists($property, $dependencies))
{
$dependentTable = $dependencies[$property];
// find all the ids inside that array of dependencies
// i.e. event: [1, 2, 3]
// and for each ID load th the object:
// i.e. CatchAppCalendarEvent.id = 1, CatchAppCalendarEvent.id = 2, CatchAppCalendarEvent.id = 3
foreach($values as $id)
{
$dependantObject = loadObject($dependencies[$property], $id);
// if the object doesn't exist in our $preload array, add it and call the
// function again
if(!objectDoesntExist($preload, $dependantObject)) {
$preload[] = $dependantObject;
reset($preload);
checkPreload($preload);
}
}
}
}
}
}
return $preload;
}

// 'id' and 'type' together are unique for each entry in the database
function objectDoesntExist($preload, $object)
{
foreach($preload as $element)
{
if($element['type'] == $object['type'] && $element['id'] == $object['id']) {
return true;
}
}
return false;
}

我很确定我已经接近解决方案,但我无法理解为什么不起作用。即使我使用函数检查对象是否已插入 $preload 数组中,似乎也会陷入无限循环。此外,有时不检查下一组元素。会不会是因为我将数据附加到 $preload 变量?

我们非常欢迎任何帮助。我一直在尝试寻找解决依赖关系的算法,但没有应用于 MySQL 数据库。

谢谢

最佳答案

在一些失败的测试之后,我决定不使用递归方法,而是使用迭代方法。

我所做的是从一个元素开始并将其放入“队列”(一个数组)中,找到该元素的依赖项,将它们附加到“队列”,然后退后一步并重新检查相同的元素元素以查看是否还有更多依赖项。

检查依赖关系的函数现在有点不同:

/**
* This is the code function of our DRA. This function contains an array of dependencies where the keys are the
* keys of the object i.e. User.id, User.type, etc. and the values are the dependent classes (tables). The idea
* is to iterate through this array in our queue of objects. If we find a property in one object that that matches
* the key, we go to the appropriate class/table (value) to find more dependencies (loadObject2 injects the dependency
* with it's subsequent dependencies)
*
*/
function findAllDependenciesFor($element)
{
$fields = [
'property' => 'tableName',
...
];

$output = [];

foreach($element as $key => $val) {
if(array_key_exists($key, $fields)) {
if(is_array($val)) {
foreach($val as $id) {
$newElement = loadObject($fields[$key], $id);
$output[] = $newElement;
}
}
else {
// there's been a field conversion at some point in the app and some 'location'
// columns contain text and numbers (i.e. 'unknown'). Let's force all values to be
// and integer and avoid loading 0 values.
$val = (int) $val;
if($val != 0) {
$newElement = loadObject($fields[$key], $val);
$output[] = $newElement;
}
}

}
}

return $output;
}

我还使用与以前相同的函数来检查“队列”是否已包含该元素(我已将该函数重命名为“objectExists”而不是“objectDoesntExist”。如您所见,我检查了类型( table) 和 id,因为这两个属性的组合对于整个系统/数据库是唯一的。

function objectExists($object, $queue)
{
foreach($queue as $element) {
if($object['type'] == $element['type'] && $object['id'] == $element['id']) {
return true;
}
}
return false;
}

最后是main函数:

// load all main objects first
foreach($ids as $id) {
$error = isNumeric($id);
$results[] = loadObject($class,$id);
}

$queue = $results;
for($i = 0; $i < count($queue); $i++)
{
// find all dependencies of element
$newElements = findAllDependenciesFor($queue[$i]);
foreach($newElements as $object) {
if(!objectExists($object, $queue)) {
$queue[] = $object;

// instead of skipping to the next object in queue, we have to re-check
// the same object again because is possible that it included new dependencies
// so let's step back on to re-check the object
$i--;
}
}
$i++;
}

如您所见,我使用的是常规“for”而不是“foreach”。这是因为我需要能够在我的“队列”中前进/后退。

关于php - 递归函数查找MySQL表的所有依赖项,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39475209/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com