一、解析快手无水印视频链接原理
共分三个步骤:
1.通过视频分享获得视频地址短链接;如:
https://www.kuaishou.com/f/X7tIV0jIivYUyTk
2.通过TNetHTTPClient重定向获得视频地址长链接;如:
3.通过地址长链接得到视频ID;如:
本视频的ID为:3xvx3jg7ffi3xrs
4.通过TNetHTTPClient发送一个包含视频ID的post请求。请求成功服务器返回(response)一个包含无水印视频地址,封面,标题等信息的JSON数据;
5.解析response的JSON数据,得到无水印视频地址,就可以下载了。
二、解析无水印视频代码
介绍了原理,废话不多说,直接上代码:
unit uKuaishou;
interface
uses
windows,classes,System.Net.URLClient, System.Net.HttpClient, System.Net.HttpClientComponent,
System.SysUtils,strutils,System.RegularExpressions,system.JSON;
const
wm_user=$0400;
wm_downfile=wm_user+100+1;
//定义UserAgent为计算机操作系统:
USER_AGENT='Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36';
//定义Cookies为你本人的Cookies,可以从浏览器的取得;不懂的+V:byc6352
KUAISHOU_COOKIES='你本人的Cookies';
//这个API_URL为发送post请求的服务器地址:
API_URL='https://video.kuaishou.com/graphql';
//KUAISHOU_ARGS为发送post请求的数据体,其中 photoId为视频唯一ID;
KUAISHOU_ARGS='{"operationName":"visionVideoDetail","variables":{"photoId":"%s","page":"detail","webPageArea":"profilexxnull"},'+
'"query":"query visionVideoDetail($photoId: String, $type: String, $page: String, $webPageArea: String) {visionVideoDetail(photoId: $photoId, type: $type, page: $page, webPageArea: $webPageArea) {photo{id duration timestamp caption coverUrl photoUrl}}}"}';
type
TKuaishou=class(TThread)
private
FId:cardinal; //线程编号;
Furl:string; //视频分享地址(短地址);
FRedirectedUrl:string; //重定向后的地址(长地址);
Fvideourl:string; //解析成功后的视频地址;
FvideoId:string; //视频ID;
Ffilename:string; //保存到本机的视频文件名;
FvideoTitle:string; //视频标题;
FcoverUrl:string; //视频封面地址;
Fmsg:string; //解析成功与否的消息;
class var Fform: HWND; //发送数据到窗体的HWND;
procedure SetId(id:cardinal); //设置线程编号;
class procedure SetForm(const hForm: HWND); static; //设置窗体句柄;
protected
procedure Execute; override;
public
//id:为线程编号;url为视频分享短链接地址;
constructor Create(id:cardinal;url:string);
destructor Destroy;
property id:cardinal read FId write SetId;
property url:string read Furl;
property msg:string read Fmsg;
property filename:string read Ffilename;
property videourl:string read Fvideourl;
property videoTitle:string read FvideoTitle;
property RedirectedUrl:string read FRedirectedUrl;
property videoId:string read FvideoId;
property coverUrl:string read FcoverUrl;
//向服务器发送post请求,返回视频信息的JSON数据;
function getPostResult(data:string):string;
//向服务器发送GET请求,获取重定向长链接地址;
function getRedirectedUrl(url:string):string;
//从长链接解析出视频ID;
function getVideoId(txt:string):string;
//获取无水印视频地址,封面链接,视频标题;
function getVideoUrl():string;
//解析服务器返回的JSON数据
function parseJson(jo:string):string;
class property form: HWND read Fform write SetForm;
end;
implementation
constructor TKuaishou.Create(id:cardinal;url:string);
begin
inherited Create(True);
FId:=id;
Furl:=url;
end;
destructor TKuaishou.Destroy;
begin
inherited Destroy;
end;
//放在线程中执行,不影响主窗体响应;
procedure TKuaishou.Execute;
begin
getVideoUrl();
end;
//获取无水印视频地址,封面链接,视频标题;
function TKuaishou.getVideoUrl():string;
var
args,jo:string;
begin
result:='';
//向服务器发送GET请求,获取重定向长链接地址;
FRedirectedUrl:=getRedirectedUrl(Furl);
if(FRedirectedUrl)='' then exit;
//从长链接解析出视频ID;
FvideoId:=getVideoId(FRedirectedUrl);
if(FvideoId)='' then exit;
//构造post请求的data数据体;
args:=format(KUAISHOU_ARGS,[FvideoId]);
//向服务器发送post请求,获得视频信息的json;
jo:=getPostResult(args);
if(jo)='' then exit;
Fmsg:=jo;
//解析json获取无水印地址;
result:=parseJson(jo);
end;
//解析服务器返回的JSON数据
function TKuaishou.parseJson(jo:string):string;
var
json,j1: TJSONObject;
begin
result:='';
if(pos('errors',jo)>0)then exit;
try
json := TJSONObject.ParseJSONValue(jo) as TJSONObject;
if json = nil then exit;
j1:=json.GetValue('data') as TJSONObject;
j1:=j1.GetValue('visionVideoDetail') as TJSONObject;
j1:=j1.GetValue('photo') as TJSONObject;
FvideoTitle:=j1.GetValue('caption').Value;
FcoverUrl:=j1.GetValue('coverUrl').Value;
FvideoUrl:=j1.GetValue('photoUrl').Value;
result:='#100#'+FvideoUrl+'#'+FcoverUrl+'#'+FvideoTitle;
finally
if json <> nil then json.Free;
end;
end;
//向服务器发送post请求,返回视频信息的JSON数据;
function TKuaishou.getPostResult(data:string):string;
var
client: TNetHTTPClient;
ss,args: TStringStream;
begin
result:='';
try
client := TNetHTTPClient.Create(nil);
SS := TStringStream.Create('', TEncoding.UTF8);
ss.Clear;
args := TStringStream.Create(data, TEncoding.UTF8);
with client do
begin
ConnectionTimeout := 10000; // 10秒
ResponseTimeout := 10000; // 10秒
AcceptCharSet := 'utf-8';
UserAgent := USER_AGENT; //1
client.AllowCookies:=true;
client.HandleRedirects:=true;
Accept:='*/*';
client.ContentType:='application/json'; //2
//这儿写入你电脑的cookies,技术支持:+V:byc6352
client.CustomHeaders['Cookie'] := KUAISHOU_COOKIES;
client.CustomHeaders['Referer'] := Furl;
//发送post请求:
client.Post(API_URL,args,ss);
result:=ss.DataString;
end;
finally
ss.Free;
args.Free;
client.Free;
end;
end;
//向服务器发送GET请求,获取重定向长链接地址;
function TKuaishou.getRedirectedUrl(url:string):string;
var
client: TNetHTTPClient;
ss: TStringStream;
s,id:string;
AResponse:IHTTPResponse;
i:integer;
begin
try
client := TNetHTTPClient.Create(nil);
SS := TStringStream.Create('', TEncoding.UTF8);
ss.Clear;
with client do
begin
ConnectionTimeout := 2000; // 2秒
ResponseTimeout := 2000; // 2秒
AcceptCharSet := 'utf-8';
UserAgent := USER_AGENT;
client.AllowCookies:=true;
client.HandleRedirects:=false; //这儿一定要设置不允许重定向;
Accept:='*/*';
try
AResponse:=Get(url, ss);
s:=AResponse.HeaderValue['Location']; //重定向地址在Location头中;
if(s='')then exit;
result:=s;
except
on E: Exception do
Log(e.Message);
end;
end;
finally
ss.Free;
client.Free;
end;
end;
//从长地址获取视频ID;
function TKuaishou.getVideoId(txt:string):string;
var
m:TMatch;
i:integer;
begin
result:='';
if(length(txt)<15)then exit;
result:=midstr(txt,14,15);
end;
//------------------------------------------属性方法-------------------------------------
procedure TKuaishou.SetId(Id:cardinal);
begin
FId:=Id;
end;
class procedure TKuaishou.SetForm(const hForm: HWND);
begin
Fform:=hForm;
end;
end.
以上类TKuaishou可以直接使用;
三、解析类使用方法:
procedure TfMain.btnParseClick(Sender: TObject);
var
kuaishou:TKuaishou;
url,result:string;
begin
url:=trim(edtUrl.Text);
kuaishou:=TKuaishou.Create(1,url);
//为了演示简单,这儿没有使用线程解析。
result:=kuaishou.getVideoUrl();
if(result='')then
begin
memo1.Text:=kuaishou.msg;
edit3.Text:='';
showmessage('解析失败!');
exit;
end;
//视频连接
edit1.Text:=kuaishou.videourl;
//封面链接
edit2.Text:=kuaishou.coverurl;
edit3.Text:=result;
memo1.Text:=kuaishou.msg;
end;
四、解析程序
1、解析后无水印视频链接地址
2、程序界面:

3、程序下载:
4、另外推荐一款(tubeget高清视频下载器),支持全球1000多个视频站点:
我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
我正在尝试设置一个puppet节点,但rubygems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由rubygems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby
我想了解Ruby方法methods()是如何工作的。我尝试使用“ruby方法”在Google上搜索,但这不是我需要的。我也看过ruby-doc.org,但我没有找到这种方法。你能详细解释一下它是如何工作的或者给我一个链接吗?更新我用methods()方法做了实验,得到了这样的结果:'labrat'代码classFirstdeffirst_instance_mymethodenddefself.first_class_mymethodendendclassSecond使用类#returnsavailablemethodslistforclassandancestorsputsSeco
尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub
我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer
我有一个包含模块的模型。我想在模块中覆盖模型的访问器方法。例如:classBlah这显然行不通。有什么想法可以实现吗? 最佳答案 您的代码看起来是正确的。我们正在毫无困难地使用这个确切的模式。如果我没记错的话,Rails使用#method_missing作为属性setter,因此您的模块将优先,阻止ActiveRecord的setter。如果您正在使用ActiveSupport::Concern(参见thisblogpost),那么您的实例方法需要进入一个特殊的模块:classBlah
设置:狂欢ruby1.9.2高线(1.6.13)描述:我已经相当习惯在其他一些项目中使用highline,但已经有几个月没有使用它了。现在,在Ruby1.9.2上全新安装时,它似乎不允许在同一行回答提示。所以以前我会看到类似的东西:require"highline/import"ask"Whatisyourfavoritecolor?"并得到:Whatisyourfavoritecolor?|现在我看到类似的东西:Whatisyourfavoritecolor?|竖线(|)符号是我的终端光标。知道为什么会发生这种变化吗? 最佳答案
我正在使用puppet为ruby程序提供一组常量。我需要提供一组主机名,我的程序将对其进行迭代。在我之前使用的bash脚本中,我只是将它作为一个puppet变量hosts=>"host1,host2"我将其提供给bash脚本作为HOSTS=显然这对ruby不太适用——我需要它的格式hosts=["host1","host2"]自从phosts和putsmy_array.inspect提供输出["host1","host2"]我希望使用其中之一。不幸的是,我终其一生都无法弄清楚如何让它发挥作用。我尝试了以下各项:我发现某处他们指出我需要在函数调用前放置“function_”……这