{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "b6b62950-26a9-4d22-91b8-40901fb9418a",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2025-05-15 03:12:59.013114: I tensorflow/core/util/port.cc:113] oneDNN custom operations are on. You may see slightly different numerical results due to floating-point round-off errors from different computation orders. To turn them off, set the environment variable `TF_ENABLE_ONEDNN_OPTS=0`.\n",
      "2025-05-15 03:12:59.074124: E external/local_xla/xla/stream_executor/cuda/cuda_dnn.cc:9261] Unable to register cuDNN factory: Attempting to register factory for plugin cuDNN when one has already been registered\n",
      "2025-05-15 03:12:59.074151: E external/local_xla/xla/stream_executor/cuda/cuda_fft.cc:607] Unable to register cuFFT factory: Attempting to register factory for plugin cuFFT when one has already been registered\n",
      "2025-05-15 03:12:59.076150: E external/local_xla/xla/stream_executor/cuda/cuda_blas.cc:1515] Unable to register cuBLAS factory: Attempting to register factory for plugin cuBLAS when one has already been registered\n",
      "2025-05-15 03:12:59.086427: I tensorflow/core/platform/cpu_feature_guard.cc:182] This TensorFlow binary is optimized to use available CPU instructions in performance-critical operations.\n",
      "To enable the following instructions: AVX2 AVX512F AVX512_VNNI FMA, in other operations, rebuild TensorFlow with the appropriate compiler flags.\n",
      "2025-05-15 03:12:59.977410: W tensorflow/compiler/tf2tensorrt/utils/py_utils.cc:38] TF-TRT Warning: Could not find TensorRT\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "WARNING:tensorflow:From /tmp/ipykernel_37256/566002682.py:26: is_gpu_available (from tensorflow.python.framework.test_util) is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "Use `tf.config.list_physical_devices('GPU')` instead.\n",
      "True\n",
      "[PhysicalDevice(name='/physical_device:GPU:0', device_type='GPU')]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2025-05-15 03:13:01.530539: I tensorflow/core/common_runtime/gpu/gpu_device.cc:1929] Created device /device:GPU:0 with 22288 MB memory:  -> device: 0, name: NVIDIA GeForce RTX 4090, pci bus id: 0000:99:00.0, compute capability: 8.9\n",
      "2025-05-15 03:13:01.539325: I tensorflow/core/common_runtime/gpu/gpu_device.cc:1929] Created device /job:localhost/replica:0/task:0/device:GPU:0 with 22288 MB memory:  -> device: 0, name: NVIDIA GeForce RTX 4090, pci bus id: 0000:99:00.0, compute capability: 8.9\n"
     ]
    }
   ],
   "source": [
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "import os\n",
    "os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"0\"\n",
    "\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "import tensorflow as tf \n",
    "import random\n",
    "from tensorflow.keras import backend as K\n",
    "from tensorflow.keras.utils import plot_model\n",
    "\n",
    "\n",
    "import os\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import tensorflow as tf\n",
    "from tensorflow import keras\n",
    "from tensorflow.keras import layers, Model\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "\n",
    "print(tf.test.is_gpu_available())\n",
    "print(tf.config.list_physical_devices('GPU'))\n",
    "config = tf.compat.v1.ConfigProto()  \n",
    "config.gpu_options.allow_growth=True\n",
    "sess = tf.compat.v1.Session(config=config)\n",
    "\n",
    "def seed_tensorflow(seed):\n",
    "    random.seed(seed)\n",
    "    os.environ['PYTHONHASHSEED'] = str(seed)\n",
    "    np.random.seed(seed)\n",
    "    tf.random.set_seed(seed)\n",
    "    session_conf = tf.compat.v1.ConfigProto(intra_op_parallelism_threads=1, inter_op_parallelism_threads=1)\n",
    "    sess = tf.compat.v1.Session(config=session_conf)\n",
    "    tf.compat.v1.keras.backend.set_session(sess)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "3cdd3e69-206c-4d7a-8646-949873cd4f2c",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2025-05-15 03:13:01.561433: I tensorflow/core/common_runtime/gpu/gpu_device.cc:1929] Created device /job:localhost/replica:0/task:0/device:GPU:0 with 22288 MB memory:  -> device: 0, name: NVIDIA GeForce RTX 4090, pci bus id: 0000:99:00.0, compute capability: 8.9\n"
     ]
    }
   ],
   "source": [
    "seed_tensorflow(1024)\n",
    "dataname = \"sports\"\n",
    "save_path = dataname+\".h5\"\n",
    "data = pd.read_csv(\"./\"+dataname+\"/\"+dataname+\".inter\", sep=\"\\t\", encoding=\"utf-8\")\n",
    "item_mm = np.load(\"./\"+dataname+\"/image_feat.npy\")\n",
    "user_mm = np.load(\"./\"+dataname+\"/user_mm.npy\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "6e6106c6-51bb-47a3-8b03-7e768b4c5682",
   "metadata": {},
   "outputs": [],
   "source": [
    "data['y'] = (data['rating']>= 5).astype(\"int32\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "01da215a-23d0-4c44-9558-ac9a709dbbdf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0         0\n",
       "1         1\n",
       "2         0\n",
       "3         1\n",
       "4         1\n",
       "         ..\n",
       "296332    1\n",
       "296333    1\n",
       "296334    1\n",
       "296335    1\n",
       "296336    1\n",
       "Name: y, Length: 296337, dtype: int32"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "data['y']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "96c75d54-a6d6-45c4-bf3f-ce589613fe73",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2025-05-15 03:13:02.860388: I tensorflow/core/common_runtime/gpu/gpu_device.cc:1929] Created device /job:localhost/replica:0/task:0/device:GPU:0 with 22288 MB memory:  -> device: 0, name: NVIDIA GeForce RTX 4090, pci bus id: 0000:99:00.0, compute capability: 8.9\n",
      "2025-05-15 03:13:02.893224: I tensorflow/core/common_runtime/gpu/gpu_device.cc:1929] Created device /job:localhost/replica:0/task:0/device:GPU:0 with 22288 MB memory:  -> device: 0, name: NVIDIA GeForce RTX 4090, pci bus id: 0000:99:00.0, compute capability: 8.9\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "uid: 35598, vid: 18357\n",
      "训练集样本数: 218409\n",
      "验证集样本数: 37899\n",
      "测试集样本数: 40029\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import tensorflow as tf\n",
    "from tensorflow import keras\n",
    "from tensorflow.keras import layers, Model\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "# ------------------------------\n",
    "# 0. 设置随机种子\n",
    "\n",
    "    \n",
    "seed_tensorflow(1024)\n",
    "\n",
    "# ------------------------------\n",
    "# 1. 数据加载与预处理\n",
    "# ------------------------------\n",
    "# 假设 data 是你的 DataFrame，且包含 'userID', 'itemID', 'y' 和 'x_label'\n",
    "# 例如: data = pd.read_csv(\"your_data.csv\")\n",
    "num_uid = data['userID'].nunique()\n",
    "num_vid = data['itemID'].nunique()\n",
    "print(f\"uid: {num_uid}, vid: {num_vid}\")\n",
    "\n",
    "# 根据 x_label 划分数据集\n",
    "train_df = data[data.x_label == 0]\n",
    "val_df   = data[data.x_label == 1]\n",
    "test_df  = data[data.x_label == 2]\n",
    "\n",
    "print(\"训练集样本数:\", len(train_df))\n",
    "print(\"验证集样本数:\", len(val_df))\n",
    "print(\"测试集样本数:\", len(test_df))\n",
    "\n",
    "# 将 DataFrame 转换为 tf.data.Dataset（返回 features 字典和标签张量）\n",
    "def df_to_dataset(df, feature_cols, label_col, shuffle=True, batch_size=128):\n",
    "    features = { key: df[key].values for key in feature_cols }\n",
    "    labels   = df[label_col].values\n",
    "    ds = tf.data.Dataset.from_tensor_slices((features, labels))\n",
    "    if shuffle:\n",
    "         ds = ds.shuffle(buffer_size=len(df))\n",
    "    ds = ds.batch(batch_size)\n",
    "    ds = ds.prefetch(tf.data.AUTOTUNE)\n",
    "    return ds\n",
    "\n",
    "feature_cols = ['userID', 'itemID']\n",
    "label_col    = 'y'\n",
    "BATCH_SIZE   = 32\n",
    "\n",
    "train_dataset = df_to_dataset(train_df, feature_cols, label_col, shuffle=True, batch_size=BATCH_SIZE)\n",
    "val_dataset   = df_to_dataset(val_df, feature_cols, label_col, shuffle=False, batch_size=BATCH_SIZE)\n",
    "test_dataset  = df_to_dataset(test_df, feature_cols, label_col, shuffle=False, batch_size=BATCH_SIZE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "a0d88134-cc15-4443-bfdd-8d0bc8b08c57",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0         11981\n",
       "1         15852\n",
       "2         17787\n",
       "3             0\n",
       "4          3369\n",
       "          ...  \n",
       "296328    18245\n",
       "296329    18336\n",
       "296332    18340\n",
       "296333    18335\n",
       "296334    18329\n",
       "Name: itemID, Length: 218409, dtype: int64"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# 0.6875\n",
    "train_df['itemID']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "98af7a58-2ab6-4d83-b5c6-2ca582f23055",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0             0\n",
       "1             0\n",
       "2             0\n",
       "3             0\n",
       "4             0\n",
       "          ...  \n",
       "296328    35596\n",
       "296329    35596\n",
       "296332    35597\n",
       "296333    35597\n",
       "296334    35597\n",
       "Name: userID, Length: 218409, dtype: int64"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train_df['userID']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "c9c17975-7602-4b72-a240-40202d1b3dbb",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<_PrefetchDataset element_spec=({'userID': TensorSpec(shape=(None,), dtype=tf.int64, name=None), 'itemID': TensorSpec(shape=(None,), dtype=tf.int64, name=None)}, TensorSpec(shape=(None,), dtype=tf.int32, name=None))>"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train_dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7dc149c2-d15d-4d51-9e64-7b26d8c47df6",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2025-05-15 03:13:04.892678: I tensorflow/core/common_runtime/gpu/gpu_device.cc:1929] Created device /job:localhost/replica:0/task:0/device:GPU:0 with 22288 MB memory:  -> device: 0, name: NVIDIA GeForce RTX 4090, pci bus id: 0000:99:00.0, compute capability: 8.9\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "uid: 35598, vid: 18357\n",
      "训练集样本数: 218409\n",
      "验证集样本数: 37899\n",
      "测试集样本数: 40029\n",
      "Model: \"model\"\n",
      "__________________________________________________________________________________________________\n",
      " Layer (type)                Output Shape                 Param #   Connected to                  \n",
      "==================================================================================================\n",
      " userID (InputLayer)         [(None, 1)]                  0         []                            \n",
      "                                                                                                  \n",
      " itemID (InputLayer)         [(None, 1)]                  0         []                            \n",
      "                                                                                                  \n",
      " id_uid_emb (Embedding)      (None, 1, 768)               2733926   ['userID[0][0]']              \n",
      "                                                          4                                       \n",
      "                                                                                                  \n",
      " id_vid_emb (Embedding)      (None, 1, 768)               1409817   ['itemID[0][0]']              \n",
      "                                                          6                                       \n",
      "                                                                                                  \n",
      " mm_uid_emb (Embedding)      (None, 1, 768)               2733926   ['userID[0][0]']              \n",
      "                                                          4                                       \n",
      "                                                                                                  \n",
      " mm_vid_emb (Embedding)      (None, 1, 768)               1409817   ['itemID[0][0]']              \n",
      "                                                          6                                       \n",
      "                                                                                                  \n",
      " id_uid_flatten (Flatten)    (None, 768)                  0         ['id_uid_emb[0][0]']          \n",
      "                                                                                                  \n",
      " id_vid_flatten (Flatten)    (None, 768)                  0         ['id_vid_emb[0][0]']          \n",
      "                                                                                                  \n",
      " mm_uid_flatten (Flatten)    (None, 768)                  0         ['mm_uid_emb[0][0]']          \n",
      "                                                                                                  \n",
      " mm_vid_flatten (Flatten)    (None, 768)                  0         ['mm_vid_emb[0][0]']          \n",
      "                                                                                                  \n",
      " id_concat (Concatenate)     (None, 1536)                 0         ['id_uid_flatten[0][0]',      \n",
      "                                                                     'id_vid_flatten[0][0]']      \n",
      "                                                                                                  \n",
      " mm_concat (Concatenate)     (None, 1536)                 0         ['mm_uid_flatten[0][0]',      \n",
      "                                                                     'mm_vid_flatten[0][0]']      \n",
      "                                                                                                  \n",
      " id_dense1 (Dense)           (None, 128)                  196736    ['id_concat[0][0]']           \n",
      "                                                                                                  \n",
      " mm_dense1 (Dense)           (None, 128)                  196736    ['mm_concat[0][0]']           \n",
      "                                                                                                  \n",
      " id_dense2 (Dense)           (None, 64)                   8256      ['id_dense1[0][0]']           \n",
      "                                                                                                  \n",
      " mm_dense2 (Dense)           (None, 64)                   8256      ['mm_dense1[0][0]']           \n",
      "                                                                                                  \n",
      " final_classifier (FinalCla  ((None, 1),                  129       ['id_dense2[0][0]',           \n",
      " ssifier)                     (None, 1),                             'mm_dense2[0][0]']           \n",
      "                              (None, 1))                                                          \n",
      "                                                                                                  \n",
      "==================================================================================================\n",
      "Total params: 83284993 (317.71 MB)\n",
      "Trainable params: 41847553 (159.64 MB)\n",
      "Non-trainable params: 41437440 (158.07 MB)\n",
      "__________________________________________________________________________________________________\n",
      "\n",
      "Start of epoch 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2025-05-15 03:13:09.386962: I external/local_xla/xla/service/service.cc:168] XLA service 0x1953f540 initialized for platform CUDA (this does not guarantee that XLA will be used). Devices:\n",
      "2025-05-15 03:13:09.386988: I external/local_xla/xla/service/service.cc:176]   StreamExecutor device (0): NVIDIA GeForce RTX 4090, Compute Capability 8.9\n",
      "2025-05-15 03:13:09.395187: I tensorflow/compiler/mlir/tensorflow/utils/dump_mlir_util.cc:269] disabling MLIR crash reproducer, set env var `MLIR_CRASH_REPRODUCER_DIRECTORY` to enable.\n",
      "2025-05-15 03:13:09.423720: I external/local_xla/xla/stream_executor/cuda/cuda_dnn.cc:454] Loaded cuDNN version 8904\n",
      "WARNING: All log messages before absl::InitializeLog() is called are written to STDERR\n",
      "I0000 00:00:1747249989.588624   37256 device_compiler.h:186] Compiled cluster using XLA!  This line is logged at most once for the lifetime of the process.\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "WARNING:tensorflow:5 out of the last 5 calls to <function _BaseOptimizer._update_step_xla at 0x7fd5705092d0> triggered tf.function retracing. Tracing is expensive and the excessive number of tracings could be due to (1) creating @tf.function repeatedly in a loop, (2) passing tensors with different shapes, (3) passing Python objects instead of tensors. For (1), please define your @tf.function outside of the loop. For (2), @tf.function has reduce_retracing=True option that can avoid unnecessary retracing. For (3), please refer to https://www.tensorflow.org/guide/function#controlling_retracing and https://www.tensorflow.org/api_docs/python/tf/function for  more details.\n",
      "WARNING:tensorflow:6 out of the last 6 calls to <function _BaseOptimizer._update_step_xla at 0x7fd5705092d0> triggered tf.function retracing. Tracing is expensive and the excessive number of tracings could be due to (1) creating @tf.function repeatedly in a loop, (2) passing tensors with different shapes, (3) passing Python objects instead of tensors. For (1), please define your @tf.function outside of the loop. For (2), @tf.function has reduce_retracing=True option that can avoid unnecessary retracing. For (3), please refer to https://www.tensorflow.org/guide/function#controlling_retracing and https://www.tensorflow.org/api_docs/python/tf/function for  more details.\n",
      "Step 0: total_loss = 0.9801, AUC = 0.5269, gamma_id = 0.5813, omega_id = 1.0000, omega_mm = 0.0021\n",
      "Step 200: total_loss = 0.6687, AUC = 0.5696, gamma_id = 0.5218, omega_id = 1.0000, omega_mm = 0.0009\n",
      "Step 400: total_loss = 0.6557, AUC = 0.5816, gamma_id = 0.5189, omega_id = 1.0000, omega_mm = 0.0009\n",
      "Step 600: total_loss = 0.6507, AUC = 0.5886, gamma_id = 0.5217, omega_id = 1.0000, omega_mm = 0.0009\n",
      "Step 800: total_loss = 0.6468, AUC = 0.5957, gamma_id = 0.4985, omega_id = 1.0000, omega_mm = 0.0007\n",
      "Epoch 1: Train Loss = 0.6461, Train AUC = 0.5975, Val Loss = 0.6324, Val AUC = 0.6295\n",
      "Best model updated at epoch 1 with val loss inf\n",
      "\n",
      "Start of epoch 2\n",
      "Step 0: total_loss = 0.6024, AUC = 0.7181, gamma_id = 0.5099, omega_id = 1.0000, omega_mm = 0.0008\n",
      "Step 200: total_loss = 0.5925, AUC = 0.7047, gamma_id = 0.4959, omega_id = 1.0000, omega_mm = 0.0006\n",
      "Step 400: total_loss = 0.5845, AUC = 0.7164, gamma_id = 0.4902, omega_id = 1.0000, omega_mm = 0.0006\n",
      "Step 600: total_loss = 0.5793, AUC = 0.7243, gamma_id = 0.4866, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Step 800: total_loss = 0.5761, AUC = 0.7292, gamma_id = 0.4803, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Epoch 2: Train Loss = 0.5754, Train AUC = 0.7304, Val Loss = 0.5914, Val AUC = 0.7106\n",
      "Best model updated at epoch 2 with val loss inf\n",
      "\n",
      "Start of epoch 3\n",
      "Step 0: total_loss = 0.4798, AUC = 0.8374, gamma_id = 0.4813, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Step 200: total_loss = 0.4671, AUC = 0.8426, gamma_id = 0.4694, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 400: total_loss = 0.4780, AUC = 0.8339, gamma_id = 0.4780, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Step 600: total_loss = 0.4846, AUC = 0.8280, gamma_id = 0.4787, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Step 800: total_loss = 0.4882, AUC = 0.8243, gamma_id = 0.4879, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Epoch 3: Train Loss = 0.4892, Train AUC = 0.8233, Val Loss = 0.6118, Val AUC = 0.7148\n",
      "Best model updated at epoch 3 with val loss inf\n",
      "\n",
      "Start of epoch 4\n",
      "Step 0: total_loss = 0.4589, AUC = 0.8516, gamma_id = 0.4833, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Step 200: total_loss = 0.4217, AUC = 0.8746, gamma_id = 0.4756, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 400: total_loss = 0.4305, AUC = 0.8681, gamma_id = 0.4721, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 600: total_loss = 0.4378, AUC = 0.8626, gamma_id = 0.4621, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Step 800: total_loss = 0.4454, AUC = 0.8569, gamma_id = 0.4556, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Epoch 4: Train Loss = 0.4464, Train AUC = 0.8561, Val Loss = 0.6388, Val AUC = 0.7133\n",
      "\n",
      "Start of epoch 5\n",
      "Step 0: total_loss = 0.4015, AUC = 0.8959, gamma_id = 0.4696, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 200: total_loss = 0.3856, AUC = 0.8960, gamma_id = 0.4168, omega_id = 1.0000, omega_mm = 0.0001\n",
      "Step 400: total_loss = 0.3956, AUC = 0.8897, gamma_id = 0.4497, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Step 600: total_loss = 0.4042, AUC = 0.8841, gamma_id = 0.4813, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Step 800: total_loss = 0.4111, AUC = 0.8793, gamma_id = 0.4747, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Epoch 5: Train Loss = 0.4125, Train AUC = 0.8784, Val Loss = 0.7005, Val AUC = 0.7076\n",
      "\n",
      "Start of epoch 6\n",
      "Step 0: total_loss = 0.3269, AUC = 0.9266, gamma_id = 0.4488, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Step 200: total_loss = 0.3454, AUC = 0.9177, gamma_id = 0.4457, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Step 400: total_loss = 0.3543, AUC = 0.9124, gamma_id = 0.4624, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 600: total_loss = 0.3614, AUC = 0.9083, gamma_id = 0.4798, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Step 800: total_loss = 0.3684, AUC = 0.9044, gamma_id = 0.4583, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Epoch 6: Train Loss = 0.3692, Train AUC = 0.9038, Val Loss = 0.8176, Val AUC = 0.7036\n",
      "\n",
      "Start of epoch 7\n",
      "Step 0: total_loss = 0.2750, AUC = 0.9540, gamma_id = 0.4627, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 200: total_loss = 0.2882, AUC = 0.9436, gamma_id = 0.4483, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Step 400: total_loss = 0.2979, AUC = 0.9393, gamma_id = 0.4262, omega_id = 1.0000, omega_mm = 0.0002\n",
      "Step 600: total_loss = 0.3064, AUC = 0.9356, gamma_id = 0.4643, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 800: total_loss = 0.3126, AUC = 0.9327, gamma_id = 0.4913, omega_id = 1.0000, omega_mm = 0.0006\n",
      "Epoch 7: Train Loss = 0.3144, Train AUC = 0.9319, Val Loss = 0.9559, Val AUC = 0.6949\n",
      "\n",
      "Start of epoch 8\n",
      "Step 0: total_loss = 0.2247, AUC = 0.9770, gamma_id = 0.4428, omega_id = 1.0000, omega_mm = 0.0002\n",
      "Step 200: total_loss = 0.2267, AUC = 0.9665, gamma_id = 0.4531, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Step 400: total_loss = 0.2363, AUC = 0.9629, gamma_id = 0.4541, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Step 600: total_loss = 0.2424, AUC = 0.9606, gamma_id = 0.4596, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Step 800: total_loss = 0.2490, AUC = 0.9583, gamma_id = 0.4837, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Epoch 8: Train Loss = 0.2507, Train AUC = 0.9577, Val Loss = 1.2034, Val AUC = 0.6890\n",
      "\n",
      "Start of epoch 9\n",
      "Step 0: total_loss = 0.1983, AUC = 0.9811, gamma_id = 0.4782, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Step 200: total_loss = 0.1684, AUC = 0.9821, gamma_id = 0.4652, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 400: total_loss = 0.1751, AUC = 0.9801, gamma_id = 0.4666, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 600: total_loss = 0.1822, AUC = 0.9782, gamma_id = 0.4683, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 800: total_loss = 0.1889, AUC = 0.9764, gamma_id = 0.4395, omega_id = 1.0000, omega_mm = 0.0002\n",
      "Epoch 9: Train Loss = 0.1901, Train AUC = 0.9761, Val Loss = 1.6263, Val AUC = 0.6824\n",
      "\n",
      "Start of epoch 10\n",
      "Step 0: total_loss = 0.1419, AUC = 0.9886, gamma_id = 0.4315, omega_id = 1.0000, omega_mm = 0.0002\n",
      "Step 200: total_loss = 0.1214, AUC = 0.9908, gamma_id = 0.4677, omega_id = 1.0000, omega_mm = 0.0004\n",
      "Step 400: total_loss = 0.1253, AUC = 0.9899, gamma_id = 0.4840, omega_id = 1.0000, omega_mm = 0.0005\n",
      "Step 600: total_loss = 0.1300, AUC = 0.9890, gamma_id = 0.4408, omega_id = 1.0000, omega_mm = 0.0002\n",
      "Step 800: total_loss = 0.1357, AUC = 0.9879, gamma_id = 0.4477, omega_id = 1.0000, omega_mm = 0.0003\n",
      "Epoch 10: Train Loss = 0.1367, Train AUC = 0.9877, Val Loss = 1.8250, Val AUC = 0.6750\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABKUAAAHqCAYAAADVi/1VAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/P9b71AAAACXBIWXMAAA9hAAAPYQGoP6dpAADJ8klEQVR4nOzdd3RU1d7G8e+kh4QklBRCSSA06UiJdJBAKCJFJGChSLOLEb0gSrPgq4LxIoqFIgoCKmJBqUoTpBdBikAgtAQCpELqnPePkdHcBGmBk/J81pp15+zZs+eZyZXs/GaffSyGYRiIiIiIiIiIiIjcRg5mBxARERERERERkeJHRSkREREREREREbntVJQSEREREREREZHbTkUpERERERERERG57VSUEhERERERERGR205FKRERERERERERue1UlBIRERERERERkdtORSkREREREREREbntVJQSEREREREREZHbTkUpEZFi5ujRo1gsFt5++22zo4iIiIgUCgMHDsTT09PsGCJFjopSIsLs2bOxWCxs3brV7ChFwuWiz5Vub7zxhtkRRUREJJ+9//77WCwWQkND83z8al8Kvf3221gsFo4ePZrrsW+++YbOnTtTtmxZXFxcCAwMpE+fPvz888/5+RZMNXDgwCvOndzc3MyOJyK3iJPZAUREiqp+/frRpUuXXO0NGzY0IY2IiIjcSnPnziU4OJjNmzdz6NAhqlatetNjGobBI488wuzZs2nYsCGRkZEEBARw+vRpvvnmG9q3b8+vv/5K8+bN8+EdmM/V1ZVPPvkkV7ujo6MJaUTkdlBRSkTkBqSmpuLh4fGvfe68804eeuih25RIREREzBIdHc2GDRtYtGgRw4cPZ+7cuYwbN+6mx508eTKzZ89mxIgRTJkyBYvFYn9szJgxfPbZZzg5FY4/6QzDIC0tDXd39yv2cXJy0txJpJjR6Xsics127NhB586d8fLywtPTk/bt2/Pbb7/l6JOZmcmECROoVq0abm5ulClThpYtW7JixQp7n9jYWAYNGkSFChVwdXWlXLlydO/ePc/l6v/r559/plWrVnh4eODj40P37t3Zt2+f/fGvvvoKi8XCmjVrcj33ww8/xGKxsGfPHnvb/v376d27N6VLl8bNzY3GjRvz3Xff5Xje5dMb16xZw+OPP46fnx8VKlS41o/tXwUHB3PPPfewfPlyGjRogJubG7Vq1WLRokW5+h45coT777+f0qVLU6JECe666y6WLFmSq19aWhrjx4+nevXquLm5Ua5cOXr16sXhw4dz9f3oo48ICQnB1dWVJk2asGXLlhyP38zPSkREpLiYO3cupUqVomvXrvTu3Zu5c+fe9JiXLl1i0qRJ1KxZ035q3/96+OGHadq06b+Ok5qaynPPPUfFihVxdXWlRo0avP322xiGYe9Tp04d2rVrl+u5VquV8uXL07t37xxtUVFR1K5dGzc3N/z9/Rk+fDgXLlzI8dzLc5xly5bRuHFj3N3d+fDDD6/3Y8jl8rxs7dq1DB8+nDJlyuDl5UX//v1zZQDbaZW1a9fG1dWVwMBAnnjiCRISEnL127RpE126dKFUqVJ4eHhQr1493n333Vz9Tp48SY8ePfD09MTX15eRI0eSnZ2do8/8+fNp1KgRJUuWxMvLi7p16+Y5lohopZSIXKO9e/fSqlUrvLy8eOGFF3B2dubDDz+kbdu2rFmzxr5/wvjx45k0aRJDhgyhadOmJCUlsXXrVrZv306HDh0AuO+++9i7dy9PPfUUwcHBnDlzhhUrVhATE0NwcPAVM6xcuZLOnTtTpUoVxo8fz6VLl5g6dSotWrRg+/btBAcH07VrVzw9PVm4cCFt2rTJ8fwFCxZQu3Zt6tSpY39PLVq0oHz58owaNQoPDw8WLlxIjx49+Prrr+nZs2eO5z/++OP4+voyduxYUlNTr/qZXbx4kfj4+FztPj4+Ob7V/PPPP4mIiODRRx9lwIABzJo1i/vvv5+lS5faP7O4uDiaN2/OxYsXefrppylTpgyffvop9957L1999ZU9a3Z2Nvfccw+rVq2ib9++PPPMMyQnJ7NixQr27NlDSEiI/XXnzZtHcnIyw4cPx2Kx8Oabb9KrVy+OHDmCs7PzTf2sREREipO5c+fSq1cvXFxc6NevHx988AFbtmyhSZMmNzzm+vXrOX/+PCNGjLjh09cMw+Dee+/ll19+YfDgwTRo0IBly5bx/PPPc/LkSd555x0AIiIiGD9+PLGxsQQEBOTIcOrUKfr27WtvGz58OLNnz2bQoEE8/fTTREdH895777Fjxw5+/fVX+xwC4MCBA/Tr14/hw4czdOhQatSocdXMec2dXFxc8PLyytH25JNP4uPjw/jx4zlw4AAffPABx44dY/Xq1fYC3vjx45kwYQJhYWE89thj9n5btmzJkXXFihXcc889lCtXjmeeeYaAgAD27dvHDz/8wDPPPGN/zezsbMLDwwkNDeXtt99m5cqVTJ48mZCQEB577DH7WP369aN9+/b83//9HwD79u3j119/zTGWiPzFEJFib9asWQZgbNmy5Yp9evToYbi4uBiHDx+2t506dcooWbKk0bp1a3tb/fr1ja5du15xnAsXLhiA8dZbb113zgYNGhh+fn7GuXPn7G27du0yHBwcjP79+9vb+vXrZ/j5+RlZWVn2ttOnTxsODg7GxIkT7W3t27c36tata6SlpdnbrFar0bx5c6NatWr2tsufT8uWLXOMeSXR0dEGcMXbxo0b7X2DgoIMwPj666/tbYmJiUa5cuWMhg0b2ttGjBhhAMa6devsbcnJyUblypWN4OBgIzs72zAMw5g5c6YBGFOmTMmVy2q15shXpkwZ4/z58/bHv/32WwMwvv/+e8Mwbu5nJSIiUlxs3brVAIwVK1YYhmH7fVuhQgXjmWeeydHv8u/fK/1efeuttwzAiI6ONgzDMN59910DML755psbzrZ48WIDMF599dUc7b179zYsFotx6NAhwzAM48CBAwZgTJ06NUe/xx9/3PD09DQuXrxoGIZhrFu3zgCMuXPn5ui3dOnSXO2X5zhLly69pqwDBgy44twpPDzc3u/yvKxRo0ZGRkaGvf3NN980AOPbb781DMMwzpw5Y7i4uBgdO3a0z5MMwzDee+89AzBmzpxpGIZhZGVlGZUrVzaCgoKMCxcu5Mh0ee70z3z/nEsahmE0bNjQaNSokf34mWeeMby8vK5pzigihqHT90TkqrKzs1m+fDk9evSgSpUq9vZy5crxwAMPsH79epKSkgDbKqC9e/fy559/5jmWu7s7Li4urF69Os8l1ldy+vRpdu7cycCBAyldurS9vV69enTo0IEff/zR3hYREcGZM2dYvXq1ve2rr77CarUSEREBwPnz5/n555/p06cPycnJxMfHEx8fz7lz5wgPD+fPP//k5MmTOTIMHTr0ur6pHDZsGCtWrMh1q1WrVo5+gYGBOVZlXV6CvmPHDmJjYwH48ccfadq0KS1btrT38/T0ZNiwYRw9epQ//vgDgK+//pqyZcvy1FNP5crzv8v+IyIiKFWqlP24VatWgO00Qbjxn5WIiEhxMnfuXPz9/e2nv1ks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kTPIYhiG4esg/I3L5cLhcFBfX3/WytBFRERkcNC4oP/0sxIRERHo/5hAPaVERERERERERGTAKSklIiIiIiIiIiIDTkkpEREREREREREZcEpKiYiIiIiIiIjIgFNSSkREREREREREBpySUiIiIiIiIiIiMuCUlBIRERERERERkQGnpJSIiIiIiIiIiAw4JaVERERERERERGTAKSklIiIiIiIiIiIDTkkpEREREREREREZcEpKiYiIiIiIiIjIgFNSSkREREREREREBpySUiIiIiIiIiIiMuACfB2APzIMAwCXy+XjSERERMTXjo4Hjo4P5Pg0hhIRERHo//hJSak+NDQ0AJCWlubjSERERMRfNDQ04HA4fB2GX9MYSkRERLo62fjJYuhjv148Hg+HDx8mIiICi8Xi63D8nsvlIi0tjYMHDxIZGenrcKQHvT/+T++Rf9P74//O9ntkGAYNDQ2kpKRgtarzwYloDNV/+t3i//Qe+Te9P/5P75F/85fxkyql+mC1Whk2bJivwxh0IiMj9cvGj+n98X96j/yb3h//dzbfI1VI9Y/GUKdOv1v8n94j/6b3x//pPfJvvh4/6eM+EREREREREREZcEpKiYiIiIiIiIjIgFNSSr40u93O0qVLsdvtvg5F+qD3x//pPfJven/8n94jGYz0363/03vk3/T++D+9R/7NX94fNToXEREREREREZEBp0opEREREREREREZcEpKiYiIiIiIiIjIgFNSSkREREREREREBpySUnJali9fzowZM4iIiCAhIYH58+dTWFjo67DkBH75y19isVi46667fB2KdCotLeXGG28kNjaWkJAQzjvvPD755BNfhyWdOjo6eOCBB8jKyiIkJIQRI0bwi1/8ArVi9J0PPviAq666ipSUFCwWC6tXr+523DAMHnzwQZKTkwkJCWHOnDns2bPHN8GKHIfGUIOLxk/+SWMo/6YxlH/x9/GTklJyWt5//30WLVrEhg0beOutt2hvb+fyyy/H7Xb7OjTpw+bNm/nd737HpEmTfB2KdKqtrWX27NkEBgby5ptvsnPnTn7zm98QHR3t69Ck069+9SueffZZnn76aXbt2sWvfvUrHnvsMZ566ilfhzZkud1uJk+ezDPPPNPn8ccee4wnn3ySlStXsnHjRsLCwpg7dy4tLS0DHKnI8WkMNXho/OSfNIbyfxpD+Rd/Hz/p6XtyRlRWVpKQkMD777/PRRdd5OtwpIvGxkamTp3Kb3/7Wx5++GGys7NZsWKFr8Ma8u6//37Wr1/Phx9+6OtQ5Di+/vWvk5iYyPPPP+/dd8011xASEsKf//xnH0YmABaLhVWrVjF//nzA/JQvJSWFe+65h3vvvReA+vp6EhMTefHFF7nuuut8GK3I8WkM5Z80fvJfGkP5P42h/Jc/jp9UKSVnRH19PQAxMTE+jkR6WrRoEV/72teYM2eOr0ORLtasWcP06dP5zne+Q0JCAlOmTOH3v/+9r8OSLs4//3zy8vLYvXs3ANu3b+ejjz7iiiuu8HFk0pf9+/fjdDq7/a5zOBzk5OSQn5/vw8hETkxjKP+k8ZP/0hjK/2kMNXj4w/gpYEC+i5zTPB4Pd911F7Nnz2bixIm+Dke6+Otf/8qnn37K5s2bfR2K9LBv3z6effZZFi9ezE9/+lM2b97Mj370I4KCgli4cKGvwxPMT2JdLhdjx47FZrPR0dHBI488wg033ODr0KQPTqcTgMTExG77ExMTvcdE/I3GUP5J4yf/pjGU/9MYavDwh/GTklLypS1atIjPP/+cjz76yNehSBcHDx7kzjvv5K233iI4ONjX4UgPHo+H6dOn8+ijjwIwZcoUPv/8c1auXKkBlZ/429/+xl/+8hdefvllJkyYwLZt27jrrrtISUnReyQiZ4TGUP5H4yf/pzGU/9MYSk6Fpu/Jl3L77bfzz3/+k3fffZdhw4b5OhzpYsuWLVRUVDB16lQCAgIICAjg/fff58knnyQgIICOjg5fhzikJScnM378+G77xo0bR0lJiY8ikp5+/OMfc//993Pddddx3nnncdNNN3H33XezfPlyX4cmfUhKSgKgvLy82/7y8nLvMRF/ojGUf9L4yf9pDOX/NIYaPPxh/KSklJwWwzC4/fbbWbVqFe+88w5ZWVm+Dkl6uPTSS9mxYwfbtm3zLtOnT+eGG25g27Zt2Gw2X4c4pM2ePbvXI8B3795NRkaGjyKSnpqamrBau/8zabPZ8Hg8PopITiQrK4ukpCTy8vK8+1wuFxs3biQ3N9eHkYl0pzGUf9P4yf9pDOX/NIYaPPxh/KTpe3JaFi1axMsvv8zrr79ORESEd76pw+EgJCTEx9EJQERERK/+FGFhYcTGxqpvhR+4++67Of/883n00Uf57ne/y6ZNm3juued47rnnfB2adLrqqqt45JFHSE9PZ8KECWzdupUnnniCf/u3f/N1aENWY2MjRUVF3u39+/ezbds2YmJiSE9P56677uLhhx9m1KhRZGVl8cADD5CSkuJ9woyIP9AYyr9p/OT/NIbyfxpD+Re/Hz8ZIqcB6HP5wx/+4OvQ5AS+8pWvGHfeeaevw5BO//jHP4yJEycadrvdGDt2rPHcc8/5OiTpwuVyGXfeeaeRnp5uBAcHG8OHDzd+9rOfGa2trb4Obch69913+/y3Z+HChYZhGIbH4zEeeOABIzEx0bDb7call15qFBYW+jZokR40hhp8NH7yPxpD+TeNofyLv4+fLIZhGAOT/hIRERERERERETGpp5SIiIiIiIiIiAw4JaVERERERERERGTAKSklIiIiIiIiIiIDTkkpEREREREREREZcEpKiYiIiIiIiIjIgFNSSkREREREREREBpySUiIiIiIiIiIiMuCUlBIRERERERERkQGnpJSIyBlmsVhYvXq1r8MQERERGTQ0fhIZmpSUEpFzyve//30sFkuvZd68eb4OTURERMQvafwkIr4S4OsARETOtHnz5vGHP/yh2z673e6jaERERET8n8ZPIuILqpQSkXOO3W4nKSmp2xIdHQ2YpeHPPvssV1xxBSEhIQwfPpzXXnut2+t37NjBJZdcQkhICLGxsdxyyy00NjZ2O+eFF15gwoQJ2O12kpOTuf3227sdr6qq4uqrryY0NJRRo0axZs2as3vTIiIiIl+Cxk8i4gtKSonIkPPAAw9wzTXXsH37dm644Qauu+46du3aBYDb7Wbu3LlER0ezefNmXn31Vd5+++1ug6Znn32WRYsWccstt7Bjxw7WrFnDyJEju32PZcuW8d3vfpfPPvuMK6+8khtuuIGampoBvU8RERGRM0XjJxE5KwwRkXPIwoULDZvNZoSFhXVbHnnkEcMwDAMwfvjDH3Z7TU5OjnHrrbcahmEYzz33nBEdHW00NjZ6j7/xxhuG1Wo1nE6nYRiGkZKSYvzsZz87bgyA8V//9V/e7cbGRgMw3nzzzTN2nyIiIiJnisZPIuIr6iklIueciy++mGeffbbbvpiYGO96bm5ut2O5ubls27YNgF27djF58mTCwsK8x2fPno3H46GwsBCLxcLhw4e59NJLTxjDpEmTvOthYWFERkZSUVFxurckIiIiclZp/CQivqCklIicc8LCwnqVg58pISEh/TovMDCw27bFYsHj8ZyNkERERES+NI2fRMQX1FNKRIacDRs29NoeN24cAOPGjWP79u243W7v8fXr12O1WhkzZgwRERFkZmaSl5c3oDGLiIiI+JLGTyJyNqhSSkTOOa2trTidzm77AgICiIuLA+DVV19l+vTpXHDBBfzlL39h06ZNPP/88wDccMMNLF26lIULF/LQQw9RWVnJHXfcwU033URiYiIADz30ED/84Q9JSEjgiiuuoKGhgfXr13PHHXcM7I2KiIiInCEaP4mILygpJSLnnLVr15KcnNxt35gxYygoKADMJ7v89a9/5bbbbiM5OZn//d//Zfz48QCEhoaybt067rzzTmbMmEFoaCjXXHMNTzzxhPdaCxcupKWlhf/+7//m3nvvJS4ujm9/+9sDd4MiIiIiZ5jGTyLiCxbDMAxfByEiMlAsFgurVq1i/vz5vg5FREREZFDQ+ElEzhb1lBIRERERERERkQGnpJSIiIiIiIiIiAw4Td8TEREREREREZEBp0opEREREREREREZcEpKiYiIiIiIiIjIgFNSSkREREREREREBpySUiIiIiIiIiIiMuCUlBIRERERERERkQGnpJSIiIiIiIiIiAw4JaVERERERERERGTAKSklIiIiIiIiIiIDTkkpEREREREREREZcP8/M9x/ArsYbQoAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1200x500 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test Loss: 0.6118, Test AUC: 0.7148\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import tensorflow as tf\n",
    "from tensorflow import keras\n",
    "from tensorflow.keras import layers, Model\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "# ------------------------------\n",
    "# 0. 设置随机种子\n",
    "\n",
    "    \n",
    "seed_tensorflow(1024)\n",
    "\n",
    "# ------------------------------\n",
    "# 1. 数据加载与预处理\n",
    "# ------------------------------\n",
    "# 假设 data 是你的 DataFrame，且包含 'userID', 'itemID', 'y' 和 'x_label'\n",
    "# 例如: data = pd.read_csv(\"your_data.csv\")\n",
    "num_uid = data['userID'].nunique()\n",
    "num_vid = data['itemID'].nunique()\n",
    "print(f\"uid: {num_uid}, vid: {num_vid}\")\n",
    "\n",
    "# 根据 x_label 划分数据集\n",
    "train_df = data[data.x_label == 0]\n",
    "val_df   = data[data.x_label == 1]\n",
    "test_df  = data[data.x_label == 2]\n",
    "\n",
    "print(\"训练集样本数:\", len(train_df))\n",
    "print(\"验证集样本数:\", len(val_df))\n",
    "print(\"测试集样本数:\", len(test_df))\n",
    "\n",
    "# 将 DataFrame 转换为 tf.data.Dataset（返回 features 字典和标签张量）\n",
    "def df_to_dataset(df, feature_cols, label_col, shuffle=True, batch_size=128):\n",
    "    features = { key: df[key].values for key in feature_cols }\n",
    "    labels   = df[label_col].values\n",
    "    ds = tf.data.Dataset.from_tensor_slices((features, labels))\n",
    "    if shuffle:\n",
    "         ds = ds.shuffle(buffer_size=len(df))\n",
    "    ds = ds.batch(batch_size)\n",
    "    ds = ds.prefetch(tf.data.AUTOTUNE)\n",
    "    return ds\n",
    "\n",
    "feature_cols = ['userID', 'itemID']\n",
    "label_col    = 'y'\n",
    "BATCH_SIZE   = 256\n",
    "\n",
    "train_dataset = df_to_dataset(train_df, feature_cols, label_col, shuffle=True, batch_size=BATCH_SIZE)\n",
    "val_dataset   = df_to_dataset(val_df, feature_cols, label_col, shuffle=False, batch_size=BATCH_SIZE)\n",
    "test_dataset  = df_to_dataset(test_df, feature_cols, label_col, shuffle=False, batch_size=BATCH_SIZE)\n",
    "\n",
    "# ------------------------------\n",
    "# 2. 定义 Embedding 层及 MLP 模块（用 MLP 替代 DCN）\n",
    "# ------------------------------\n",
    "uid_emb_dim = 768\n",
    "vid_emb_dim = 768\n",
    "\n",
    "# 这里假设 user_mm 与 item_mm 已经预加载好相应的权重（类型为 numpy.array）\n",
    "# 如果名称不同，请替换为你实际的变量名\n",
    "# user_mm = ...  \n",
    "# item_mm = ...\n",
    "\n",
    "def build_mlp_module(x, name_prefix):\n",
    "    mlp = layers.Dense(128, activation='relu', name=f\"{name_prefix}_dense1\")(x)\n",
    "    mlp = layers.Dense(64, activation='relu', name=f\"{name_prefix}_dense2\")(mlp)\n",
    "    return mlp\n",
    "\n",
    "# ------------------------------\n",
    "# 3. 构建模型（删除独立分量分类，仅使用最终分类器，并输出中间表示用于对齐损失）\n",
    "# ------------------------------\n",
    "# 输入层\n",
    "uid_in = keras.Input(shape=(1,), name=\"userID\")\n",
    "vid_in = keras.Input(shape=(1,), name=\"itemID\")\n",
    "\n",
    "# --- ID 分支（使用可训练 embedding，对 userID 与 itemID 均使用） ---\n",
    "id_uid_emb = layers.Embedding(input_dim=num_uid, output_dim=uid_emb_dim,\n",
    "                              weights=[user_mm],\n",
    "                              trainable=True,\n",
    "                              name=\"id_uid_emb\")(uid_in)    # shape: (batch, 1, uid_emb_dim)\n",
    "id_vid_emb = layers.Embedding(input_dim=num_vid, output_dim=vid_emb_dim,\n",
    "                              weights=[item_mm],\n",
    "                              trainable=True,\n",
    "                              name=\"id_vid_emb\")(vid_in)     # shape: (batch, 1, vid_emb_dim)\n",
    "id_uid_feat = layers.Flatten(name=\"id_uid_flatten\")(id_uid_emb)   # (batch, uid_emb_dim)\n",
    "id_vid_feat = layers.Flatten(name=\"id_vid_flatten\")(id_vid_emb)   # (batch, vid_emb_dim)\n",
    "id_concat   = layers.Concatenate(name=\"id_concat\")([id_uid_feat, id_vid_feat])  # (batch, uid_emb_dim+vid_emb_dim)\n",
    "id_mlp      = build_mlp_module(id_concat, name_prefix=\"id\")\n",
    "id_bn       = id_mlp#layers.BatchNormalization(name=\"id_bn\")(id_mlp)  # 中间表示，用于对齐损失\n",
    "\n",
    "# --- MM 分支（使用冻结 embedding，对 userID 与 itemID 均使用） ---\n",
    "mm_uid_emb = layers.Embedding(input_dim=num_uid, output_dim=uid_emb_dim,\n",
    "                              weights=[user_mm],\n",
    "                              trainable=False,\n",
    "                              name=\"mm_uid_emb\")(uid_in)   # (batch, 1, uid_emb_dim)\n",
    "mm_vid_emb = layers.Embedding(input_dim=num_vid, output_dim=vid_emb_dim,\n",
    "                              weights=[item_mm],\n",
    "                              trainable=False,\n",
    "                              name=\"mm_vid_emb\")(vid_in)    # (batch, 1, vid_emb_dim)\n",
    "mm_uid_feat = layers.Flatten(name=\"mm_uid_flatten\")(mm_uid_emb)   # (batch, uid_emb_dim)\n",
    "mm_vid_feat = layers.Flatten(name=\"mm_vid_flatten\")(mm_vid_emb)   # (batch, vid_emb_dim)\n",
    "mm_concat   = layers.Concatenate(name=\"mm_concat\")([mm_uid_feat, mm_vid_feat])  # (batch, uid_emb_dim+vid_emb_dim)\n",
    "mm_mlp      = build_mlp_module(mm_concat, name_prefix=\"mm\")\n",
    "mm_bn       = mm_mlp#layers.BatchNormalization(name=\"mm_bn\")(mm_mlp)  # 中间表示\n",
    "\n",
    "# --- 最终分类器（自定义层，将两个分支的归一化输出转换为 logit 并相加）\n",
    "class FinalClassifier(layers.Layer):\n",
    "    def __init__(self, **kwargs):\n",
    "        super(FinalClassifier, self).__init__(**kwargs)\n",
    "    def build(self, input_shape):\n",
    "        # input_shape: list [id_shape, mm_shape]\n",
    "        id_shape, mm_shape = input_shape\n",
    "        d_id = int(id_shape[-1])\n",
    "        d_mm = int(mm_shape[-1])\n",
    "        self.W_id = self.add_weight(shape=(d_id, 1),\n",
    "                                    initializer='glorot_uniform',\n",
    "                                    trainable=True,\n",
    "                                    name='W_id')\n",
    "        self.W_mm = self.add_weight(shape=(d_mm, 1),\n",
    "                                    initializer='glorot_uniform',\n",
    "                                    trainable=True,\n",
    "                                    name='W_mm')\n",
    "        self.bias = self.add_weight(shape=(1,),\n",
    "                                    initializer='zeros',\n",
    "                                    trainable=True,\n",
    "                                    name='bias')\n",
    "        super(FinalClassifier, self).build(input_shape)\n",
    "    def call(self, inputs):\n",
    "        x_id, x_mm = inputs\n",
    "        logit_id = tf.matmul(x_id, self.W_id)\n",
    "        logit_mm = tf.matmul(x_mm, self.W_mm)\n",
    "        logits = logit_id + logit_mm + self.bias/2.0\n",
    "        output = tf.sigmoid(logits)\n",
    "        # 返回最终预测和各分支的 logit\n",
    "        return output, logit_id, logit_mm\n",
    "\n",
    "final_output, logit_id, logit_mm = FinalClassifier(name=\"final_classifier\")([id_bn, mm_bn])\n",
    "# 构造模型：输出包含最终预测、logit 和两个分支的归一化输出（用于计算对齐损失）\n",
    "model = Model(inputs=[uid_in, vid_in],\n",
    "              outputs=[final_output, logit_id, logit_mm, id_bn, mm_bn])\n",
    "model.summary()\n",
    "\n",
    "# ------------------------------\n",
    "# ------------------------------\n",
    "# 5. 自定义训练循环，加入梯度调制和对齐损失\n",
    "# ------------------------------\n",
    "# 超参数\n",
    "eta = 2.0           # 梯度调制的调制强度\n",
    "alpha_value = 0.1   # 对齐损失权重\n",
    "learning_rate = 1e-3\n",
    "\n",
    "optimizer = tf.keras.optimizers.Adam(learning_rate=learning_rate)\n",
    "loss_fn = tf.keras.losses.BinaryCrossentropy()\n",
    "\n",
    "# 指标\n",
    "train_loss_metric = tf.keras.metrics.Mean()\n",
    "train_auc_metric  = tf.keras.metrics.AUC()\n",
    "val_loss_metric   = tf.keras.metrics.Mean()\n",
    "val_auc_metric    = tf.keras.metrics.AUC()\n",
    "\n",
    "# 为了记录每个 epoch 的指标，便于后续可视化\n",
    "epoch_train_loss_list = []\n",
    "epoch_train_auc_list  = []\n",
    "epoch_val_loss_list   = []\n",
    "epoch_val_auc_list    = []\n",
    "best_val_loss = float('inf')\n",
    "best_val_auc = 0\n",
    "best_epoch = -1\n",
    "\n",
    "# 用于根据变量名称判断属于哪个分支，对梯度进行调制\n",
    "def modulate_gradients(grads, vars, omega_id, omega_mm):\n",
    "    modulated_grads = []\n",
    "    for g, v in zip(grads, vars):\n",
    "        if g is None:\n",
    "            modulated_grads.append(None)\n",
    "        elif isinstance(g, tf.IndexedSlices):\n",
    "            if ('id_' in v.name) or ('W_id' in v.name):\n",
    "                scaled_values = g.values * omega_id\n",
    "            elif ('mm_' in v.name) or ('W_mm' in v.name):\n",
    "                scaled_values = g.values * omega_mm\n",
    "            else:\n",
    "                scaled_values = g.values\n",
    "            modulated_grads.append(tf.IndexedSlices(scaled_values, g.indices, g.dense_shape))\n",
    "        else:\n",
    "            if ('id_' in v.name) or ('W_id' in v.name):\n",
    "                modulated_grads.append(g * omega_id)\n",
    "            elif ('mm_' in v.name) or ('W_mm' in v.name):\n",
    "                modulated_grads.append(g * omega_mm)\n",
    "            else:\n",
    "                modulated_grads.append(g)\n",
    "    return modulated_grads\n",
    "def build_pi_and_size(fisher_list):\n",
    "    flt_list = [tf.reshape(t, [-1]) for t in fisher_list if t is not None]\n",
    "    if len(flt_list) == 0:\n",
    "        return None, 0          # 该分支没有可更新参数\n",
    "    flat = tf.concat(flt_list, axis=0)\n",
    "    epsilon = 1e-8                                  # 防止出现零概率\n",
    "    flat = flat + epsilon\n",
    "    pi = flat / tf.reduce_sum(flat)                 # 重新归一化\n",
    "    return pi, flat.shape[0]\n",
    "# -------------------------------------------------\n",
    "# 为每个可训练变量准备一个 Fisher 累积器 (与变量同形状)\n",
    "# -------------------------------------------------\n",
    "def sample_indices(pi_tensor, total_size, budget):\n",
    "    if total_size == 0 or budget == 0:\n",
    "        return np.array([], dtype=np.int64)\n",
    "\n",
    "    # ① 预算不能超过参数个数\n",
    "    budget = min(budget, total_size)\n",
    "\n",
    "    # ② 概率重新归一化，确保 sum == 1 且全部 > 0\n",
    "    pi_np = pi_tensor.numpy()\n",
    "    pi_np = pi_np / pi_np.sum()\n",
    "\n",
    "    return np.random.choice(total_size,\n",
    "                            size=budget,\n",
    "                            replace=False,\n",
    "                            p=pi_np)\n",
    "\n",
    "val_AUC = []\n",
    "# 自定义训练循环\n",
    "EPOCHS = 10  # 根据需要调整\n",
    "c_eps = 1\n",
    "for epoch in range(EPOCHS):\n",
    "    print(f\"\\nStart of epoch {epoch+1}\")\n",
    "    train_loss_metric.reset_states()\n",
    "    train_auc_metric.reset_states()\n",
    "    \n",
    "    for step, (features, labels) in enumerate(train_dataset):\n",
    "        with tf.GradientTape() as tape:\n",
    "            final_pred, logit_id_batch, logit_mm_batch, id_bn_batch, mm_bn_batch = model(\n",
    "                [features['userID'], features['itemID']], training=True)\n",
    "            ce_loss = loss_fn(labels, final_pred)\n",
    "            total_loss = ce_loss\n",
    "        \n",
    "        grads = tape.gradient(total_loss, model.trainable_variables)\n",
    "        \n",
    "        labels = tf.cast(labels, tf.float32)\n",
    "        p_id = tf.sigmoid(logit_id_batch)\n",
    "        p_mm = tf.sigmoid(logit_mm_batch)\n",
    "        s_id = labels * p_id + (1 - labels) * (1 - p_id)\n",
    "        s_mm = labels * p_mm + (1 - labels) * (1 - p_mm)\n",
    "        sum_s_id = tf.reduce_sum(s_id)\n",
    "        sum_s_mm = tf.reduce_sum(s_mm) + 1e-8\n",
    "        #gamma_id = sum_s_id / sum_s_mm\n",
    "        \n",
    "        rho_id   = sum_s_id / sum_s_mm   \n",
    "        rho_mm   = 1.0 / rho_id   \n",
    "        \n",
    "        # 采样比例\n",
    "        gamma_id = tf.exp(rho_mm) / (tf.exp(rho_id) + tf.exp(rho_mm))       ### NEW\n",
    "        gamma_mm = 1.0 - gamma_id     \n",
    "        \n",
    "        # TODO\n",
    "        # ---------- Fisher 信息 & π ----------\n",
    "        fisher_id_tensors = []\n",
    "        fisher_mm_tensors = []\n",
    "\n",
    "        for g, v in zip(grads, model.trainable_variables):\n",
    "            v_name = v.name\n",
    "            # ① 跳过 None 梯度、Embedding 以及共享/非分支参数\n",
    "            if (g is None) or ('emb' in v_name):\n",
    "                fisher_id_tensors.append(None)\n",
    "                fisher_mm_tensors.append(None)\n",
    "                continue\n",
    "\n",
    "            # ② 把 IndexedSlices 转成 dense 便于逐参数操作\n",
    "            if isinstance(g, tf.IndexedSlices):\n",
    "                g_dense = tf.convert_to_tensor(\n",
    "                    tf.scatter_nd(tf.expand_dims(g.indices, 1), g.values, v.shape))\n",
    "            else:\n",
    "                g_dense = g\n",
    "\n",
    "            fisher_tensor = tf.square(g_dense)      # (∂log p / ∂θ_j)^2\n",
    "\n",
    "            # ③ 根据变量归属(ID / MM)分别存\n",
    "            if ('id_' in v_name) or ('W_id' in v_name):\n",
    "                fisher_id_tensors.append(fisher_tensor)\n",
    "                fisher_mm_tensors.append(None)\n",
    "            elif ('mm_' in v_name) or ('W_mm' in v_name):\n",
    "                fisher_mm_tensors.append(fisher_tensor)\n",
    "                fisher_id_tensors.append(None)\n",
    "            else:                                   # 其它权重保持不动\n",
    "                fisher_id_tensors.append(None)\n",
    "                fisher_mm_tensors.append(None)\n",
    "                \n",
    "        \n",
    "        \n",
    "        # ---------- Fisher → π ----------\n",
    "        pi_id, size_id = build_pi_and_size(fisher_id_tensors)\n",
    "        pi_mm, size_mm = build_pi_and_size(fisher_mm_tensors)\n",
    "\n",
    "        # Step-3 依据 γ 分配预算并随机采样生成 mask\n",
    "        # ================================================================\n",
    "        # ① 预算（要更新多少个参数）\n",
    "        #gamma_id = 0.9\n",
    "        #gamma_mm = 0.4\n",
    "        budget_id = int(tf.round(gamma_id * tf.cast(size_id, tf.float32)))\n",
    "        budget_mm = int(tf.round(gamma_mm * tf.cast(size_mm, tf.float32)))\n",
    "        \n",
    "        \n",
    "        idx_id_keep = sample_indices(pi_id, size_id, budget_id)\n",
    "        idx_mm_keep = sample_indices(pi_mm, size_mm, budget_mm)\n",
    "\n",
    "        # ③ 生成展平的缩放系数  w_j = 1/(π_j + c)   —— 未采样参数为 0\n",
    "        weight_id_flat = np.zeros(size_id, dtype=np.float32)\n",
    "        weight_mm_flat = np.zeros(size_mm, dtype=np.float32)\n",
    "\n",
    "        if size_id:\n",
    "            pi_id_np = pi_id.numpy()\n",
    "            weight_id_flat[idx_id_keep] = 1.0 / (pi_id_np[idx_id_keep] + c_eps)\n",
    "\n",
    "        if size_mm:\n",
    "            pi_mm_np = pi_mm.numpy()\n",
    "            weight_mm_flat[idx_mm_keep] = 1.0 / (pi_mm_np[idx_mm_keep] + c_eps)\n",
    "\n",
    "        # ④ 把扁平权重按原张量形状恢复并施加到梯度\n",
    "        masked_grads = []\n",
    "        cursor_id = cursor_mm = 0\n",
    "        for g, v in zip(grads, model.trainable_variables):\n",
    "            v_name = v.name\n",
    "            if g is None or ('emb' in v_name):\n",
    "                masked_grads.append(g)\n",
    "                continue\n",
    "\n",
    "            # dense 化\n",
    "            g_dense = (tf.convert_to_tensor(\n",
    "                          tf.scatter_nd(tf.expand_dims(g.indices, 1), g.values, v.shape))\n",
    "                       if isinstance(g, tf.IndexedSlices) else g)\n",
    "            num_param = tf.size(g_dense).numpy()\n",
    "\n",
    "            if ('id_' in v_name) or ('W_id' in v.name):\n",
    "                w_slice = weight_id_flat[cursor_id: cursor_id + num_param]\n",
    "                cursor_id += num_param\n",
    "            elif ('mm_' in v_name) or ('W_mm' in v.name):\n",
    "                w_slice = weight_mm_flat[cursor_mm: cursor_mm + num_param]\n",
    "                cursor_mm += num_param\n",
    "            else:                          # 其它共享权重全部保留（权重=1）\n",
    "                w_slice = np.ones(num_param, dtype=np.float32)\n",
    "\n",
    "            w_tensor = tf.reshape(tf.convert_to_tensor(w_slice), v.shape)\n",
    "            g_scaled = g_dense * w_tensor      # 关键：乘以 1/(π_j+c) 或 0\n",
    "\n",
    "            # 若原梯度是 IndexedSlices，需要再转回稀疏格式\n",
    "            if isinstance(g, tf.IndexedSlices):\n",
    "                nz_idx = tf.where(tf.not_equal(w_tensor, 0.0))\n",
    "                g_scaled = tf.IndexedSlices(\n",
    "                    values=tf.gather_nd(g_scaled, nz_idx),\n",
    "                    indices=nz_idx[:, 0],\n",
    "                    dense_shape=g.dense_shape)\n",
    "\n",
    "            masked_grads.append(g_scaled)\n",
    "            \n",
    "        optimizer.apply_gradients(zip(masked_grads, model.trainable_variables))\n",
    "        \n",
    "        omega_id = tf.cond(gamma_id > 1.0,\n",
    "                           lambda: 1.0 - tf.tanh(eta * gamma_id),\n",
    "                           lambda: 1.0)\n",
    "        omega_mm = tf.cond(1/gamma_id > 1.0,\n",
    "                           lambda: 1.0 - tf.tanh(eta / gamma_id),\n",
    "                           lambda: 1.0)\n",
    "        \n",
    "        #optimizer.apply_gradients(zip(masked_grads, model.trainable_variables))\n",
    "        #modulated_grads = modulate_gradients(grads, model.trainable_variables, omega_id, omega_mm)\n",
    "        #optimizer.apply_gradients(zip(grads, model.trainable_variables))\n",
    "        \n",
    "        train_loss_metric.update_state(total_loss)\n",
    "        train_auc_metric.update_state(labels, final_pred)\n",
    "        \n",
    "        if step % 200 == 0:\n",
    "            print(f\"Step {step}: total_loss = {train_loss_metric.result().numpy():.4f}, \"\n",
    "                  f\"AUC = {train_auc_metric.result().numpy():.4f}, \"\n",
    "                  f\"gamma_id = {gamma_id:.4f}, omega_id = {omega_id:.4f}, omega_mm = {omega_mm:.4f}\")\n",
    "    \n",
    "    # 验证阶段\n",
    "    for features_val, labels_val in test_dataset:\n",
    "        final_pred_val, _, _, _, _ = model([features_val['userID'], features_val['itemID']], training=False)\n",
    "        val_loss_metric.update_state(loss_fn(labels_val, final_pred_val))\n",
    "        val_auc_metric.update_state(labels_val, final_pred_val)\n",
    "    \n",
    "    current_train_loss = train_loss_metric.result().numpy()\n",
    "    current_train_auc  = train_auc_metric.result().numpy()\n",
    "    current_val_loss   = val_loss_metric.result().numpy()\n",
    "    current_val_auc    = val_auc_metric.result().numpy()\n",
    "    \n",
    "    epoch_train_loss_list.append(current_train_loss)\n",
    "    epoch_train_auc_list.append(current_train_auc)\n",
    "    epoch_val_loss_list.append(current_val_loss)\n",
    "    epoch_val_auc_list.append(current_val_auc)\n",
    "    \n",
    "    print(f\"Epoch {epoch+1}: Train Loss = {current_train_loss:.4f}, \"\n",
    "          f\"Train AUC = {current_train_auc:.4f}, \"\n",
    "          f\"Val Loss = {current_val_loss:.4f}, \"\n",
    "          f\"Val AUC = {current_val_auc:.4f}\")\n",
    "    val_AUC.append(current_val_auc)\n",
    "    \n",
    "    # 保存最优模型权重（以验证损失为准）\n",
    "    if current_val_auc > best_val_auc:\n",
    "        best_val_auc = current_val_auc\n",
    "        best_epoch = epoch\n",
    "        model.save_weights(save_path)\n",
    "        print(f\"Best model updated at epoch {epoch+1} with val loss {best_val_loss:.4f}\")\n",
    "    \n",
    "    val_loss_metric.reset_states()\n",
    "    val_auc_metric.reset_states()\n",
    "\n",
    "# ------------------------------\n",
    "# 6. 可视化训练和验证指标\n",
    "# ------------------------------\n",
    "epochs_range = range(1, EPOCHS+1)\n",
    "plt.figure(figsize=(12, 5))\n",
    "\n",
    "plt.subplot(1, 2, 1)\n",
    "plt.plot(epochs_range, epoch_train_loss_list, label=\"Train Loss\")\n",
    "plt.plot(epochs_range, epoch_val_loss_list, label=\"Val Loss\")\n",
    "plt.xlabel(\"Epoch\")\n",
    "plt.ylabel(\"Loss\")\n",
    "plt.title(\"Loss over Epochs\")\n",
    "plt.legend()\n",
    "\n",
    "plt.subplot(1, 2, 2)\n",
    "plt.plot(epochs_range, epoch_train_auc_list, label=\"Train AUC\")\n",
    "plt.plot(epochs_range, epoch_val_auc_list, label=\"Val AUC\")\n",
    "plt.xlabel(\"Epoch\")\n",
    "plt.ylabel(\"AUC\")\n",
    "plt.title(\"AUC over Epochs\")\n",
    "plt.legend()\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.show()\n",
    "\n",
    "# ------------------------------\n",
    "# 7. 加载最优模型权重，并在测试集上评估\n",
    "# ------------------------------\n",
    "model.load_weights(save_path)\n",
    "test_loss_metric = tf.keras.metrics.Mean()\n",
    "test_auc_metric  = tf.keras.metrics.AUC()\n",
    "\n",
    "for features_test, labels_test in test_dataset:\n",
    "    final_pred_test, _, _, _, _ = model([features_test['userID'], features_test['itemID']], training=False)\n",
    "    test_loss_metric.update_state(loss_fn(labels_test, final_pred_test))\n",
    "    test_auc_metric.update_state(labels_test, final_pred_test)\n",
    "\n",
    "print(f\"Test Loss: {test_loss_metric.result().numpy():.4f}, Test AUC: {test_auc_metric.result().numpy():.4f}\")\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
