Java实现视频网站的视频上传、视频转码、视频关键帧抽图, 及视频播放功能(转)...

news/2024/5/20 2:06:32/文章来源:https://blog.csdn.net/weixin_30594001/article/details/97703680
转自:http://www.cnblogs.com/dennisit/archive/2013/02/16/2913287.html

视频网站中提供的在线视频播放功能,播放的都是FLV格式的文件,它是Flash动画文件,可通过Flash制作的播放器来播放该文件.项目中用制作的player.swf播放器.

多媒体视频处理工具FFmpeg有非常强大的功能包括视频采集功能、视频格式转换、视频抓图、给视频加水印等。  

ffmpeg视频采集功能非常强大,不仅可以采集视频采集卡或USB摄像头的图像,还可以进行屏幕录制,同时还支持以RTP方式将视频流传送给支持RTSP的流媒体服务器,支持直播应用。

1.能支持的格式

ffmpeg能解析的格式:(asx,asf,mpg,wmv,3gp,mp4,mov,avi,flv等)

2.不能支持的格式

对ffmpeg无法解析的文件格式(wmv9,rm,rmvb等),可以先用别的工具(mencoder)转换为avi(ffmpeg能解析的)格式.

实例是将上传视频转码为flv格式,该格式ffmpeg支持,所以我们实例中需要ffmpeg视频处理工具.

 

数据库MySQL5.5

实例所需要的数据库脚本

复制代码
drop database if exists db_mediaplayer;
create database db_mediaplayer;
use db_mediaplayer;create table tb_media(id int not null primary key auto_increment comment '主键' , title varchar(50) not null comment '视频名称' ,src varchar(200) not null comment '视频存放地址' ,picture varchar(200) not null comment '视频截图' ,descript varchar(400) comment '视频描述' ,uptime varchar(40) comment '上传时间'
);desc tb_media;
复制代码

项目结构图:

上传视频界面设计

在上传文件时,Form表单中 enctype属性值必须为"multipart/form-data".模块界面设计如下图:

enctype属性值说明

application/x-www-form-urlencoded

表单数据被编码为名称/值对,这是标准的编码格式

multipart/form-data

表单数据被编码为一条消息,页面上每个控件对应消息中的一部分

text/plain

表单数据以纯文本形式进行编码,其中不含任何控件格式的字符

 

业务接口定义

面向接口编程,接口中定义系统功能模块.这样方便理清业务,同时接口的对象必须由实现了该接口的对象来创建.这样就避免编码中的某些业务遗漏等,同时扩展性也增强了.

 

复制代码
package com.webapp.dao;
import java.util.List;
import com.webapp.entity.Media;/****  MediaDao.java    **  @version : 1.1*  *  @author  : 苏若年    <a href="mailto:DennisIT@163.com">发送邮件</a>*    *  @since     : 1.0        创建时间:    2013-2-07        上午10:19:54*     *  TODO     :    interface MediaDao.java is used for ...**/public interface MediaDao {/*** 视频转码* @param ffmpegPath    转码工具的存放路径* @param upFilePath    用于指定要转换格式的文件,要截图的视频源文件* @param codcFilePath    格式转换后的的文件保存路径* @param mediaPicPath    截图保存路径* @return* @throws Exception*/public boolean executeCodecs(String ffmpegPath,String upFilePath, String codcFilePath, String mediaPicPath)throws Exception;/*** 保存文件* @param media* @return* @throws Exception*/public boolean saveMedia(Media media)throws Exception;/*** 查询本地库中所有记录的数目* @return* @throws Exception*/public int getAllMediaCount()throws Exception;/*** 带分页的查询* @param firstResult* @param maxResult* @return*/public List<Media> queryALlMedia(int firstResult, int maxResult)throws Exception;/*** 根据Id查询视频* @param id* @return* @throws Exception*/public Media queryMediaById(int id)throws Exception;
}
复制代码

 

接口的实现,这里列出ffmpeg视频转码与截图模块

 

复制代码
    /*** 视频转码* @param ffmpegPath    转码工具的存放路径* @param upFilePath    用于指定要转换格式的文件,要截图的视频源文件* @param codcFilePath    格式转换后的的文件保存路径* @param mediaPicPath    截图保存路径* @return* @throws Exception*/public boolean executeCodecs(String ffmpegPath, String upFilePath, String codcFilePath,String mediaPicPath) throws Exception {// 创建一个List集合来保存转换视频文件为flv格式的命令List<String> convert = new ArrayList<String>();convert.add(ffmpegPath); // 添加转换工具路径convert.add("-i"); // 添加参数"-i",该参数指定要转换的文件convert.add(upFilePath); // 添加要转换格式的视频文件的路径convert.add("-qscale");     //指定转换的质量convert.add("6");convert.add("-ab");        //设置音频码率convert.add("64");convert.add("-ac");        //设置声道数convert.add("2");convert.add("-ar");        //设置声音的采样频率convert.add("22050");convert.add("-r");        //设置帧频convert.add("24");convert.add("-y"); // 添加参数"-y",该参数指定将覆盖已存在的文件        convert.add(codcFilePath);// 创建一个List集合来保存从视频中截取图片的命令List<String> cutpic = new ArrayList<String>();cutpic.add(ffmpegPath);cutpic.add("-i");cutpic.add(upFilePath); // 同上(指定的文件即可以是转换为flv格式之前的文件,也可以是转换的flv文件)cutpic.add("-y");cutpic.add("-f");cutpic.add("image2");cutpic.add("-ss"); // 添加参数"-ss",该参数指定截取的起始时间cutpic.add("17"); // 添加起始时间为第17秒cutpic.add("-t"); // 添加参数"-t",该参数指定持续时间cutpic.add("0.001"); // 添加持续时间为1毫秒cutpic.add("-s"); // 添加参数"-s",该参数指定截取的图片大小cutpic.add("800*280"); // 添加截取的图片大小为350*240cutpic.add(mediaPicPath); // 添加截取的图片的保存路径boolean mark = true;ProcessBuilder builder = new ProcessBuilder();try {builder.command(convert);builder.redirectErrorStream(true);builder.start();builder.command(cutpic);builder.redirectErrorStream(true);// 如果此属性为 true,则任何由通过此对象的 start() 方法启动的后续子进程生成的错误输出都将与标准输出合并,//因此两者均可使用 Process.getInputStream() 方法读取。这使得关联错误消息和相应的输出变得更容易            builder.start();} catch (Exception e) {mark = false;System.out.println(e);e.printStackTrace();}return mark;}
复制代码

 

系统中可能存在多个模块,这些模块的业务DAO可以通过工厂来管理,需要的时候直接提供即可.

因为如果对象new太多,会不必要的浪费资源.所以工厂,采用单例模式,私有构造,提供对外可访问的方法即可.

 

复制代码
package com.webapp.dao;
import com.webapp.dao.impl.MediaDaoImpl;/****  DaoFactory.java    **  @version : 1.1*  *  @author  : 苏若年    <a href="mailto:DennisIT@163.com">发送邮件</a>*    *  @since     : 1.0        创建时间:    2013-2-07        下午02:18:51*     *  TODO     :    class DaoFactory.java is used for ...**/public class DaoFactory { //工厂模式,生产Dao对象,面向接口编程,返回实现业务接口定义的对象private static DaoFactory daoFactory = new DaoFactory();//单例设计模式, 私有构造,对外提供获取创建的对象的唯一接口,private DaoFactory(){}public static DaoFactory getInstance(){return daoFactory;}public static MediaDao getMediaDao(){return new MediaDaoImpl();}}
复制代码

 

视图提交请求,给控制器,控制器分析请求参数,进行相应的业务调用处理.servlet控制器相关代码如下

 

复制代码
package com.webapp.service;import java.io.File;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.List;import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;import com.webapp.dao.DaoFactory;
import com.webapp.dao.MediaDao;
import com.webapp.entity.Media;
import com.webapp.util.DateTimeUtil;/****  MediaService.java    **  @version : 1.1*  *  @author  : 苏若年    <a href="mailto:DennisIT@163.com">发送邮件</a>*    *  @since     : 1.0        创建时间:    2013-2-08        下午02:24:47*     *  TODO     :    class MediaService.java is used for ...**/public class MediaService extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {doPost(request, response);}public void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {PrintWriter out = response.getWriter();MediaDao mediaDao = DaoFactory.getMediaDao();String message = "";String uri = request.getRequestURI();String path = uri.substring(uri.lastIndexOf("/"), uri.lastIndexOf("."));if("/uploadFile".equals(path)){//提供解析时的一些缺省配置DiskFileItemFactory factory = new DiskFileItemFactory();//创建一个解析器,分析InputStream,该解析器会将分析的结果封装成一个FileItem对象的集合//一个FileItem对象对应一个表单域ServletFileUpload sfu = new ServletFileUpload(factory);try {Media media = new Media();List<FileItem> items = sfu.parseRequest(request);boolean flag = false;    //转码成功与否的标记for(int i=0; i<items.size(); i++){FileItem item = items.get(i);//要区分是上传文件还是普通的表单域if(item.isFormField()){//isFormField()为true,表示这不是文件上传表单域//普通表单域String paramName = item.getFieldName();/*String paramValue = item.getString();System.out.println("参数名称为:" + paramName + ", 对应的参数值为: " + paramValue);*/if(paramName.equals("title")){media.setTitle(new String(item.getString().getBytes("ISO8859-1"),"UTF-8"));}if(paramName.equals("descript")){media.setDescript(new String(item.getString().getBytes("ISO8859-1"),"UTF-8"));}}else{//上传文件//System.out.println("上传文件" + item.getName());ServletContext sctx = this.getServletContext();//获得保存文件的路径String basePath = sctx.getRealPath("videos");//获得文件名String fileUrl= item.getName();//在某些操作系统上,item.getName()方法会返回文件的完整名称,即包括路径String fileType = fileUrl.substring(fileUrl.lastIndexOf(".")); //截取文件格式//自定义方式产生文件名String serialName = String.valueOf(System.currentTimeMillis());//待转码的文件File uploadFile = new File(basePath+"/temp/"+serialName + fileType);item.write(uploadFile);if(item.getSize()>500*1024*1024){message = "<li>上传失败!您上传的文件太大,系统允许最大文件500M</li>";}String codcFilePath = basePath + "/" + serialName + ".flv";                //设置转换为flv格式后文件的保存路径String mediaPicPath = basePath + "/images" +File.separator+ serialName + ".jpg";    //设置上传视频截图的保存路径// 获取配置的转换工具(ffmpeg.exe)的存放路径String ffmpegPath = getServletContext().getRealPath("/tools/ffmpeg.exe");media.setSrc("videos/" + serialName + ".flv");media.setPicture("videos/images/" +serialName + ".jpg");media.setUptime(DateTimeUtil.getYMDHMSFormat());//转码                        flag = mediaDao.executeCodecs(ffmpegPath, uploadFile.getAbsolutePath(), codcFilePath, mediaPicPath);}}if(flag){//转码成功,向数据表中添加该视频信息                    mediaDao.saveMedia(media);message = "<li>上传成功!</li>";}request.setAttribute("message", message);request.getRequestDispatcher("media_upload.jsp").forward(request,response);} catch (Exception e) {e.printStackTrace();throw new ServletException(e);}}if("/queryAll".equals(path)){List<Media> mediaList;try {mediaList = mediaDao.queryALlMedia(0,5);request.setAttribute("mediaList", mediaList);request.getRequestDispatcher("media_list.jsp").forward(request, response);} catch (Exception e) {e.printStackTrace();}}if("/play".equals(path)){String idstr = request.getParameter("id");int mediaId = -1;Media media = null;if(null!=idstr){mediaId = Integer.parseInt(idstr);}try {media = mediaDao.queryMediaById(mediaId);} catch (Exception e) {e.printStackTrace();}request.setAttribute("media", media);request.getRequestDispatcher("media_player.jsp").forward(request, response);}}}
复制代码

 

可以通过分页查找,显示最新top5,展示到首页.相应特效可以使用JS实现.

 

 

 

 

相关代码如下:

 

复制代码
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ page import="com.webapp.entity.*"%>
<%@ page import="java.util.*"%>
<%String path = request.getContextPath();String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<title>视频列表</title>
<link rel="stylesheet" type="text/css" href="skin/css/style.css" ></link><script type="text/javascript" src="skin/js/jquery1.3.2.js"></script>
<script type="text/javascript">
$(function() {var sWidth = $("#focus").width(); //获取焦点图的宽度(显示面积)var len = $("#focus ul li").length; //获取焦点图个数var index = 0;var picTimer;//以下代码添加数字按钮和按钮后的半透明条,还有上一页、下一页两个按钮var btn = "<div class='btnBg'></div><div class='btn'>";for(var i=0; i < len; i++) {btn += "<span></span>";}btn += "</div><div class='preNext pre'></div><div class='preNext next'></div>";$("#focus").append(btn);$("#focus .btnBg").css("opacity",0.5);//为小按钮添加鼠标滑入事件,以显示相应的内容$("#focus .btn span").css("opacity",0.4).mouseenter(function() {index = $("#focus .btn span").index(this);showPics(index);}).eq(0).trigger("mouseenter");//上一页、下一页按钮透明度处理$("#focus .preNext").css("opacity",0.2).hover(function() {$(this).stop(true,false).animate({"opacity":"0.5"},300);},function() {$(this).stop(true,false).animate({"opacity":"0.2"},300);});//上一页按钮$("#focus .pre").click(function() {index -= 1;if(index == -1) {index = len - 1;}showPics(index);});//下一页按钮$("#focus .next").click(function() {index += 1;if(index == len) {index = 0;}showPics(index);});//本例为左右滚动,即所有li元素都是在同一排向左浮动,所以这里需要计算出外围ul元素的宽度$("#focus ul").css("width",sWidth * (len));//鼠标滑上焦点图时停止自动播放,滑出时开始自动播放$("#focus").hover(function() {clearInterval(picTimer);},function() {picTimer = setInterval(function() {showPics(index);index++;if(index == len) {index = 0;}},4000); //此4000代表自动播放的间隔,单位:毫秒}).trigger("mouseleave");//显示图片函数,根据接收的index值显示相应的内容function showPics(index) { //普通切换var nowLeft = -index*sWidth; //根据index值计算ul元素的left值$("#focus ul").stop(true,false).animate({"left":nowLeft},300); //通过animate()调整ul元素滚动到计算出的position//$("#focus .btn span").removeClass("on").eq(index).addClass("on"); //为当前的按钮切换到选中的效果$("#focus .btn span").stop(true,false).animate({"opacity":"0.4"},300).eq(index).stop(true,false).animate({"opacity":"1"},300); //为当前的按钮切换到选中的效果    }
});</script>
</head><body>
<div class="wrapper"><h1>最新视频</h1><div id="focus"><ul><%List<Media> mediaList = (List<Media>)request.getAttribute("mediaList");if(mediaList.size()>0&&mediaList!=null){for(int i=0; i<mediaList.size(); i++){Media media = mediaList.get(i);%><li><a href="play.action?id=<%=media.getId() %>"><img src="<%=basePath%><%=media.getPicture() %>" alt="" /></a></li><% }}else{%><li><h3 style="color:white;margin-left: 352px;margin-top: 130px;">没有记录</h3></li><% }%></ul></div></div>
</body>
</html>
复制代码

 

首页展示的图片都是带ID的链接请求.图片为视频转码过程中拉取到的图片.点击图片即可发送播放视频请求,

视频播放页面效果如下图所示.

 

视频播放页面需要在页面中嵌入Flash播放器

代码如下:

复制代码
<!-- 嵌入Flash播放器 -->
<td align="center" width="455"><object width="452" height="339" classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"><param name="movie"value="<%=basePath%>tools/player.swf?fileName=<%=basePath%><%=media.getSrc()%>" /><embedsrc="<%=basePath%>tools/player.swf?fileName=<%=basePath%><%=media.getSrc()%>"width="98%" height="90%"></embed> </object>
</td>
复制代码

相关说明:

<object>元素,加载ActiveX控件,classid属性则指定了浏览器使用的ActiveX空间.因为使用Flash制作的播放器来播放视频文件,所以classid的值必须为”clsid:D27CDB6E-AE6D-11cf-96B8-444553540000”

 

<param>元素,value属性指定被加载的视频文件.实例中用的是flash制作的视频播放器.在value属性值中向player.swf播放器传递了一个file参数.该参数指定了要播放的视频的路径.

<embed>元素,src属性也是用来加载影片,与<param>标记的value属性值具体相同的功能.



来自为知笔记(Wiz)


转载于:https://www.cnblogs.com/ismallboy/p/6785326.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_746238.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

网站整站开发小例子

之前学习的前端知识都只是局限于htmljs编写的静态网页&#xff0c;所以前几天试着加上模拟数据把前端和后台串联起来。在这个demo中&#xff0c;我运用了node.js、koa及vue.js。首先我们需要了解node.js&#xff0c;简单的说 Node.js 就是运行在服务端的 JavaScript。node.js安…

让网站Http重定向Https并进行百度Https认证

先为大家带来一点福利。阿里云最近开始发放代金券了&#xff0c;新老用户均可免费获取&#xff0c;1880元代金券&#xff0c;建议大家都领取一份&#xff0c;反正是免费领的&#xff0c;说不定以后需要呢&#xff1f;阿里云代金卷链接https://promotion.aliyun.com/ntms/yunpar…

Jekyll 动态地建立静态博客网站 (Get Started)

提前声明&#xff1a;Jekyll并不简单&#xff0c;必须要正确的看待它。把它和PHP&#xff0c;JSP和Django等放在一起讨论会减少很多失落感。它的学习曲线几乎相当于Wordpress&#xff0c;工作流程和结构也几乎一样。Jekyll与Wordpress最大不同的就是&#xff0c;没有数据库。但…

一步步带你,如何网站架构

何为大型网站 大型网站特性 既然说的是大型网站架构&#xff0c;那么架构的背后自然是解决人因面对大型网站特性而带来的问题。这样可以先给大家说下大型网站的特性&#xff0c;这些特性带来的问题就是人要解决的问题&#xff1a; 高并发、大流量&#xff1a;PV 量巨大&#xf…

什么是域名?什么网站名?什么是URL?

什么是域名&#xff1f;现实中&#xff0c;域名使用的实在太多了&#xff0c;但域名的详细含义你理解么&#xff1f;你知道域名是什么吗&#xff1f;下面&#xff0c;小编将带你详解域名的含义&#xff01; 方法/步骤 域名&#xff0c;相信大家都不默认&#xff0c;也使用过无数…

《大型网站技术架构》读书笔记一:大型网站架构演化

一、大型网站系统特点 &#xff08;1&#xff09;高并发、大流量&#xff1a;PV量巨大 &#xff08;2&#xff09;高可用&#xff1a;7*24小时不间断服务 &#xff08;3&#xff09;海量数据&#xff1a;文件数目分分钟xxTB &#xff08;4&#xff09;用户分布广泛&#xff0c;…

java 大型网站架构_大型网站架构系列:电商网站架构案例(1)

大型网站架构系列&#xff1a;电商网站架构案例(1)大型网站架构是一个系列文档&#xff0c;欢迎大家关注。本次分享主题&#xff1a;电商网站架构案例。从电商网站的需求&#xff0c;到单机架构&#xff0c;逐步演变为常用的&#xff0c;可供参考的分布式架构的原型。除具备功能…

推荐的五款市面上常用的免费CMS建站系统

我做设计也有不少年头了&#xff0c;很多客户或者朋友找我做网站的时候&#xff0c;一般问我的是用什么软件系统给他们做。大部分客户希望用的软件是免费的。所以今天给大家介绍五款我自己用过还不错的&#xff0c;重点是还免费的建站系统。 MetInfo MetInfo这个系统是一个客户…

iis网站服务器端口设置,iis服务器端口设置方法

iis服务器端口设置方法 内容精选换一换获取当前系统健康状态。设置成“GET”。该接口仅支持POST和GET方法&#xff0c;不支持PUT和DELTE等方法。https://ip:port/ccbms/health.jsp其中&#xff0c;ip为CTI平台的CCBMS服务器地址&#xff0c;port为CCBMS服务器的HTTPS端口号。无…

16个时髦的扁平化设计+视觉滚差的网站模板

Persona – HTML5 Parallax Portfolio Theme 充满吸引力的一款网站模板&#xff0c;扁平化设计风格&#xff0c;动感的视差滚动效果。 在线演示 模板下载 Superb – Responsive One-Page Portfolio Superb 是一个基于 HTML5 & CSS3 的响应式单页作品集网站模板。 在线演…

两分钟学会最新的用github+便携版hexo做网站服务器教程+绑定域名

前段时间在阿里云买了一个域名&#xff0c;便想着如何利用这个域名来做自己的个人网站。网上教程特别多&#xff0c;但是大多数都不能用了&#xff0c;因为github在不断的进行更新。所以碰到了很多很多的坑。本人自己也是git小白&#xff0c;git的有些命令也是比较复杂。但是&a…

Jenkins与网站代码上线解决方案【转】

转自 Jenkins与网站代码上线解决方案 - 惨绿少年 https://www.nmtui.com/clsn/lx524.html 1.1 前言 Jenkins是一个用Java编写的开源的持续集成工具。在与Oracle发生争执后&#xff0c;项目从Hudson项目独立。 Jenkins提供了软件开发的持续集成服务。它运行在Servlet容器中&…

[源码和文档分享]基于JAVA EE和豆瓣API实现的电影评论网站

本网站通过对网上最新的热点影片的分门别类&#xff0c;采取简洁、直观的方式让观看者体验到前所未有的舒适感&#xff0c;通过对网站界面布局&#xff0c;信息采集&#xff0c;会员管理&#xff0c;观看者的互动等功能的规划来实现崭新的视听环境。 本网站提供的电影资源和内容…

5 ,emr 入门 : 网址 ,定价 ,学习网站

一 &#xff0c;相关网址 &#xff1a; 1 &#xff0c;emr 文档入口 &#xff1a; https://docs.aws.amazon.com/zh_cn/emr/index.html 2 &#xff0c;emr 开发指南 &#xff1a; https://docs.amazonaws.cn/emr/latest/ManagementGuide/emr-what-is-emr.html 3 &#xff…

1 ,spark 电影点评项目 : 简介,项目历史 ,元数据下载 ,电影数据集网站 ( 非常好的网站 )

1 &#xff0c;可以采用的技术 &#xff1a;选哪种方式都行&#xff0c;我们都用用 RDDDataFrameDataFrame RDDDataSet 2 &#xff0c;需求 &#xff1a; 某电影 &#xff1a; 男性不同年龄观看者人数某电影 &#xff1a; 女性不同年龄观看者人数所有电影 &#xff1a; 评分…

在百度里指定网站搜索:site:developer.mozilla.org 数组(后面关键字“数组”可以换成任意要搜索的关键字)...

打开百度&#xff1a; 在百度搜索框 输入&#xff1a;site:developer.mozilla.org 数组 第一个就是&#xff0c;实现效果如下&#xff1a; 转载于:https://www.cnblogs.com/lszw/p/11067159.html

借助 Lucene.Net 构建站内搜索引擎(上)

前言&#xff1a;最近翻开了之前老杨&#xff08;杨中科&#xff09;的Lucene.Net站内搜索项目的教学视频&#xff0c;于是作为老杨脑残粉的我又跟着复习了一遍&#xff0c;学习途中做了一些笔记也就成了接下来您看到的这篇博文&#xff0c;仅仅是我的个人笔记&#xff0c;大神…

github可以刷星吗_教你一招即使没有服务器,也可以免费拥有自己的专属网站

现在固定公网IP资源稀缺&#xff0c;虽然说IPv6已经开始使用&#xff0c;但是用的人并不多&#xff0c;大部分用的都是IPv4&#xff0c;想要搞一个自己网站&#xff0c;首先就是需要一个固定的公网IP&#xff0c;但是每年却需要缴纳不少的money&#xff0c;而且服务器的成本&am…

oss批量上传工具_使用阿里云OSS与github actions自动部署个人网站

OSS 即 object storage service&#xff0c;对象存储服务。我们可以通过阿里云的 OSS 来托管自己的前端应用&#xff0c;个人网站或者博客在 [使用 netlify 托管你的前端应用]( "使用 netlify 托管你的前端应用") 中我也介绍到另一种专业的网站托管服务平台 netlify。…

Java旅游网站源码+页面

旅游网站[源码前后端页面] 在线预览&#xff1a;旅游网站TourismPro 账号&#xff1a;朱利尔&#xff0c;密码&#xff1a;123 管理员账号&#xff1a;admin&#xff0c;密码&#xff1a;123456 Github开源地址&#xff1a;https://github.com/jwwam/tourismPro下载 or 部署问…