请求go web后端接口 java安卓端播放视频

前端代码

添加gradle依赖

复制代码
implementation 'com.squareup.retrofit2:retrofit:2.9.0'
implementation 'com.squareup.retrofit2:converter-gson:2.9.0'

添加访问网络权限

复制代码
<uses-permission android:name="android.permission.INTERNET" />

允许http 请求请求

复制代码
android:usesCleartextTraffic="true"

video类

java 复制代码
package com.example.myapplication;
public class Video {
    private String url;

    public String getUrl() {
        return url;
    }
}

VideoService 接口类

java 复制代码
package com.example.myapplication;

import java.util.List;

import retrofit2.Call;
import retrofit2.http.GET;

public interface VideoService {
    @GET("/videos")
    Call<List<Video>> getVideos();
}

MainActivity 类

java 复制代码
package com.example.myapplication;

import android.net.Uri;
import android.os.Bundle;
import android.util.Log;
import android.widget.VideoView;
import android.widget.Toast;
import android.media.MediaPlayer;
import androidx.appcompat.app.AppCompatActivity;

import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;

import java.util.List;

public class MainActivity extends AppCompatActivity {

    private VideoView videoView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        videoView = findViewById(R.id.videoView);

        // 初始化 Retrofit
        Retrofit retrofit = new Retrofit.Builder()
                .baseUrl("http://192.168.110.148:8080") // 修改为你的 Go 后端地址
                .addConverterFactory(GsonConverterFactory.create())
                .build();

        VideoService service = retrofit.create(VideoService.class);
        Call<List<Video>> call = service.getVideos();

        showToast("Starting Retrofit call to get videos");

        call.enqueue(new Callback<List<Video>>() {
            @Override
            public void onResponse(Call<List<Video>> call, Response<List<Video>> response) {
                if (response.isSuccessful() && response.body() != null) {
                    List<Video> videos = response.body();
                    if (!videos.isEmpty()) {
                        String videoUrl = videos.get(0).getUrl();
                        showToast("Received video URL: " + videoUrl); // 打印 URL 进行调试
                        playVideo(videos.get(0).getUrl()); // 播放第一个视频
                    } else {
                        showToast("No videos found in the response");
                    }
                } else {
                    showToast("Failed to fetch videos, Response code: " + response.code());
                }
            }

            @Override
            public void onFailure(Call<List<Video>> call, Throwable t) {
                showToast("Error fetching videos: " + t.getMessage());
                t.printStackTrace();
            }
        });
    }

    private void playVideo(String videoUrl) {
        showToast("Attempting to play video with URL: " + videoUrl);

        String fullUrl = "http://192.168.110.148:8080" + videoUrl; // 完整视频 URL
        showToast("Full video URL: " + fullUrl);

        Uri videoUri = Uri.parse(fullUrl);
        showToast("Parsed URI: " + videoUri.toString());

        videoView.setVideoURI(videoUri); // 使用 setVideoURI 方法

        videoView.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
            @Override
            public void onPrepared(MediaPlayer mp) {
                showToast("Video is prepared, starting playback.");
                videoView.start(); // 开始播放
            }
        });

        videoView.setOnErrorListener(new MediaPlayer.OnErrorListener() {
            @Override
            public boolean onError(MediaPlayer mp, int what, int extra) {
                showToast("Error occurred while playing video. What: " + what + ", Extra: " + extra);
                return true; // 返回 true 表示我们已经处理了错误
            }
        });
    }

    // 用于显示 Toast 信息的辅助方法
    private void showToast(String message) {
        Toast.makeText(MainActivity.this, message, Toast.LENGTH_LONG).show();
    }
}

go后端代码

Go 复制代码
package main

import (
	"encoding/json"
	"net/http"
)

// Video 代表一个视频结构
type Video struct {
	URL string `json:"url"`
}

// VideoList 返回的视频列表
var videoList = []Video{
	{URL: "/videos/video1.mp4"}, // 本地视频地址
	{URL: "/videos/video2.mp4"}, // 本地视频地址 http://192.168.0.104:8080/videos/video2.mp4
}

func main() {
	// 在根目录下提供静态文件服务
	http.Handle("/videos/", http.StripPrefix("/videos/", http.FileServer(http.Dir("./videos"))))
	http.HandleFunc("/videos", getVideos)
	http.ListenAndServe(":8080", nil)
}

// getVideos 处理视频请求
func getVideos(w http.ResponseWriter, r *http.Request) {
	w.Header().Set("Content-Type", "application/json")
	json.NewEncoder(w).Encode(videoList)
}
相关推荐
qq_422152579 分钟前
PDF 加水印工具怎么选?2026 年文档版权保护方案对比
前端·pdf·github
kyriewen21 分钟前
手写 Promise.all、race、any:不到 30 行代码,解决并发异步的所有姿势
前端·javascript·面试
2601_961765291 小时前
【分享】PlayerPro媒体音乐播放器 完整专业版
android·媒体
摇滚侠1 小时前
SpringMVC 入门到实战 文件上传 75-77
java·后端·spring·maven·intellij-idea
brucelee1861 小时前
OpenClaw 浏览器控制(Chrome MCP)完整教程
前端·chrome
GIS数据转换器1 小时前
城市排水生命线安全运行监测平台深度解析
java·运维·人工智能·python·安全·数据挖掘·无人机
ct9781 小时前
React 状态管理方案深度对比
开发语言·前端·react
胡志辉的博客2 小时前
深入浅出理解浏览器事件循环:从一道输出题讲到 Chrome 源码
前端·javascript·chrome·chromium·event loop
代码不加糖2 小时前
js中不会冒泡的事件有哪些?
前端·javascript·vue.js
华如锦2 小时前
面了很多 Java转AI Agent方向,一些面试题总结
java·开发语言·人工智能·python·ai