function [X_hat, err_hist, loss_hist] = GD_TC_ES(X_star, Y, Omega, p, eta, k, T)
% GD_TC_ES  Tensor completion via (unscaled) GD with validation early-stop


    rate = 0.95;                % 训练/验证划分比例
    rel_tol_inc = 7e-3;         % 判定"上升"的相对容差（避免浮点抖动）
    stop_on_increase = true;    % 一旦验证损失上升就提前停止

    [n1,n2,n3] = size(Y);

    % 低秩分解初始化
    L = rand(n1,k,n3)/k*1e-5;
    R = rand(n2,k,n3)/k*1e-5;
    X = tprod(L, tran(R));

    % 训练/验证划分
    [Omega_train, Omega_val, Y_train, Y_val] = split_logical_tensor(Omega, Y, rate);
    p = p * rate;   % 采样率按训练集比例修正（与你原逻辑一致）

    % 历史记录
    err_hist  = nan(T,1);
    time_hist = nan(T,1);
    loss_hist = nan(T,1);       %#ok<NASGU> % 若你后续想把验证曲线也画出来可返回
    tacc = 0;

    X_hat   = X;    % 当前最优解
    best_val = Inf; % 最优验证损失

    den = norm(X_star(:)); % 误差归一化常数

    for t = 1:T
        ticIter = tic;

        % 当前残差（训练/验证）
        Z_train = Omega_train .* X - Y_train;
        Z_val   = Omega_val   .* X - Y_val;

        % 训练目标（仅用于监控；更新使用 Z_train）
        % obj = (norm(Z_train,'fro')^2) / (2*p);

        % 梯度更新（未缩放 GD）
        L = L - (eta/p) * tprod(Z_train,      R);
        R = R - (eta/p) * tprod(tran(Z_train), L);

        % 重建
        X = tprod(L, tran(R));

        % 记录误差 & 验证损失
        re   = norm(X(:) - X_star(:)) / den;
        vlos = norm(Z_val(:));
        err_hist(t)  = re;
        loss_hist(t) = vlos;

        % best-so-far（验证更优就更新）
        if vlos < best_val 
            best_val = vlos;
            X_hat    = X;
        end

        % 早停：验证损失上升（相对容差判断）
        if stop_on_increase
           
            if vlos > best_val * (1 + rel_tol_inc)
                % 提前停止：保留已记录的历史与 best-so-far 的 X_hat
                tacc = tacc + toc(ticIter);
                time_hist(t) = tacc;
                t
                break;
            end
        end

        % 累计时间
        tacc = tacc + toc(ticIter);
        time_hist(t) = tacc;

        % 数值保护的停止条件（可选）
        if ~isfinite(re) || re < 1e-14 || re > 1e8
            re
            break;
        end
    end


    % 裁剪有效长度
    last = find(isfinite(err_hist), 1, 'last');
    if isempty(last), last = 1; end
    err_hist  = err_hist(1:last);
    time_hist = time_hist(1:last);
end
