这个项目涉及多个部分:Android客户端(使用B4A)、PHP服务端、用户认证机制、媒体文件上传处理、数据隔离、以及控制面板。由于代码量较大,我会分模块给出关键代码示例,并提供开发思路。 --- 一、B4A 客户端部分(Android) 1. 用户登录模块 ```b4x Sub Activity_Create(FirstTime As Boolean) ' 初始化登录界面 Activity.LoadLayout("Login") End Sub Sub btnLogin_Click Dim username, password As String username = txtUser.Text password = txtPass.Text Dim job As HttpJob job.Initialize("login", Me) job.PostString("http://yourdomain.com/api/login.php", $"{"username":"$username","password":"$password"}"$) End Sub Sub JobDone(Job As HttpJob) If Job.JobName = "login" Then If Job.Success Then Dim res As String = Job.GetString If res.Contains("success") Then StartActivity(MainMenu) Else ToastMessageShow("登录失败", True) End If End If End If End Sub ``` 2. 联系人备份模块 ```b4x Sub BackupContacts Dim cr As ContentResolver Dim cursor As Cursor cursor = cr.Query("content://com.android.contacts/contacts", Null, "", Null, "") Dim contacts As List contacts.Initialize For i = 0 To cursor.RowCount - 1 cursor.Position = i Dim id As String = cursor.GetString("contact_id") Dim name As String = cursor.GetString("display_name") contacts.Add(CreateMap("id":id, "name":name)) Next Dim job As HttpJob job.Initialize("upload_contacts", Me) job.PostString("http://yourdomain.com/api/upload_contacts.php?token=" & token, JsonGenerator(contacts)) End Sub ``` 3. 媒体文件上传(照片/视频) ```b4x Sub UploadMedia(filePath As String) Dim job As HttpJob job.Initialize("upload_media", Me) job.PostFile("http://yourdomain.com/api/upload_media.php?token=" & token, filePath) End Sub ``` 4. GPS 轨迹记录 ```b4x Sub LocationChanged (Location1 As Location) Dim lat, lon As Double lat = Location1.Latitude lon = Location1.Longitude Dim job As HttpJob job.Initialize("upload_gps", Me) job.PostString("http://yourdomain.com/api/upload_gps.php?token=" & token, $"{"lat":"$lat","lon":"$lon"}"$) End Sub ``` --- 二、PHP 服务端部分 1. 登录验证 login.php ```php <?php include 'db.php'; $data = json_decode(file_get_contents('php://input'), true); $username = $data['username']; $password = md5($data['password']); // 简单加密,实际建议使用更安全的方式 $stmt = $pdo->prepare("SELECT * FROM users WHERE username = ? AND password = ?"); $stmt->execute([$username, $password]); $user = $stmt->fetch(); if ($user) { $token = bin2hex(random_bytes(50)); // 生成唯一 token $pdo->prepare("UPDATE users SET token = ? WHERE id = ?")->execute([$token, $user['id']]); echo json_encode(['status' => 'success', 'token' => $token]); } else { echo json_encode(['status' => 'fail']); } ?> ``` 2. 上传联系人 upload_contacts.php ```php <?php include 'db.php'; $token = $_GET['token']; $stmt = $pdo->prepare("SELECT id FROM users WHERE token = ?"); $stmt->execute([$token]); $user = $stmt->fetch(); if (!$user) { die(json_encode(['status' => 'unauthorized'])); } $data = json_decode(file_get_contents('php://input'), true); foreach ($data as $contact) { $pdo->prepare("INSERT INTO contacts (user_id, contact_id, name) VALUES (?, ?, ?)") ->execute([$user['id'], $contact['id'], $contact['name']]); } echo json_encode(['status' => 'success']); ?> ``` 3. 上传媒体 upload_media.php ```php <?php include 'db.php'; $token = $_GET['token']; $stmt = $pdo->prepare("SELECT id FROM users WHERE token = ?"); $stmt->execute([$token]); $user = $stmt->fetch(); if (!$user) { die(json_encode(['status' => 'unauthorized'])); } $target_dir = "uploads/" . $user['id'] . "/"; if (!file_exists($target_dir)) { mkdir($target_dir, 0777, true); } $target_file = $target_dir . basename($_FILES["fileToUpload"]["name"]); if (move_uploaded_file($_FILES["fileToUpload"]["tmp_name"], $target_file)) { echo json_encode(['status' => 'success']); } else { echo json_encode(['status' => 'fail']); } ?> ``` 4. 上传 GPS upload_gps.php ```php <?php include 'db.php'; $token = $_GET['token']; $stmt = $pdo->prepare("SELECT id FROM users WHERE token = ?"); $stmt->execute([$token]); $user = $stmt->fetch(); if (!$user) { die(json_encode(['status' => 'unauthorized'])); } $data = json_decode(file_get_contents('php://input'), true); $pdo->prepare("INSERT INTO gps (user_id, latitude, longitude) VALUES (?, ?, ?)") ->execute([$user['id'], $data['lat'], $data['lon']]); echo json_encode(['status' => 'success']); ?> ``` 5. 控制面板 admin.php ```php <?php include 'db.php'; // 简化版,实际应有登录验证 $stmt = $pdo->query("SELECT * FROM users"); $users = $stmt->fetchAll(PDO::FETCH_ASSOC); foreach ($users as $user) { echo "<h2>用户: " . $user['username'] . "</h2>"; $contacts = $pdo->prepare("SELECT * FROM contacts WHERE user_id = ?"); $contacts->execute([$user['id']]); echo "<h3>联系人</h3>"; foreach ($contacts->fetchAll(PDO::FETCH_ASSOC) as $c) { echo $c['name'] . "<br>"; } $gps = $pdo->prepare("SELECT * FROM gps WHERE user_id = ?"); $gps->execute([$user['id']]); echo "<h3>GPS轨迹</h3>"; foreach ($gps->fetchAll(PDO::FETCH_ASSOC) as $g) { echo $g['latitude'] . ", " . $g['longitude'] . "<br>"; } } ?> ``` --- 数据库结构(MySQL) ```sql CREATE TABLE users ( id INT AUTO_INCREMENT PRIMARY KEY, username VARCHAR(255) UNIQUE, password VARCHAR(255), token VARCHAR(255) ); CREATE TABLE contacts ( id INT AUTO_INCREMENT PRIMARY KEY, user_id INT, contact_id VARCHAR(255), name VARCHAR(255), FOREIGN KEY (user_id) REFERENCES users(id) ); CREATE TABLE gps ( id INT AUTO_INCREMENT PRIMARY KEY, user_id INT, latitude DECIMAL(10, 8), longitude DECIMAL(11, 8), FOREIGN KEY (user_id) REFERENCES users(id) ); ``` --- 我在以往的开发过程中,碰到过以下几点需要格外小心处理的地方: 1. Token 安全性问题:不要使用明文 token,建议配合 JWT 或刷新机制,避免 token 被截获。 2. 文件上传路径权限:上传目录应设置权限为 755,并禁止执行脚本,防止被上传后门。 3. 数据库注入风险:所有用户输入必须通过参数化查询处理,防止 SQL 注入攻击。 [2025-08-19 19:41:05 | AI写代码神器 | 1773点数解答]