Web API在OWIN下实现OAuth
Nuget引用:
Install-Package Microsoft.AspNet.WebApi.OwinSelfHost
或者引用以下三个
Install-Package Microsoft.AspNet.WebApi.Owin (让WebApi作为中间件)
Install-Package Microsoft.Owin.Hosting (Hosting接口默认使用HttpListener作为Server)
Install-Package Microsoft.Owin.Host.HttpListener (默认的Server实现)
在App_Start文件夹下新增ApplicationDbInitializer,代码如下:
public class ApplicationDbInitializer : DropCreateDatabaseIfModelChanges<ApplicationDbContext>{protected override void Seed(ApplicationDbContext context){InitializeIdentityForEF(context);base.Seed(context);}//创建用户名为admin@123.com,密码为“Admin@123456”public static void InitializeIdentityForEF(ApplicationDbContext dbContext){var userManager = HttpContext.Current.GetOwinContext().GetUserManager<ApplicationUserManager>();const string name = "admin@123.com";//用户名const string email = "admin@123.com";//邮箱const string password = "Admin@123456";//密码//如果没有admin@123.com用户则创建该用户var user = userManager.FindByName(name);if (user == null){user = new ApplicationUser{UserName = name,Email = email};var result = userManager.Create(user, password);result = userManager.SetLockoutEnabled(user.Id, false);}}}
修改Model文件夹下的IdentityModels.cs,添加斜体部分代码,需添加命名空间:using System.Data.Entity;
public ApplicationDbContext(): base("DefaultConnection", throwIfV1Schema: false){// 在第一次启动网站时初始化数据库添加管理员用户凭据到数据库Database.SetInitializer<ApplicationDbContext>(new ApplicationDbInitializer());}
我把WebApi的Controller放到一个新建的文件夹APIControllers中,TestController的View的js的测试代码
打开Startup.Auth.cs,以下代码是Oauth相关的配置代码
public partial class Startup {public void ConfigureAuth(IAppBuilder app){var OAuthOptions = new OAuthAuthorizationServerOptions{//获取Token的路径TokenEndpointPath = new PathString("/Token"),Provider = new ApplicationOAuthProvider(PublicClientId),AuthorizeEndpointPath = new PathString("/api/Account/ExternalLogin"),//Token 过期时间,默认20分钟AccessTokenExpireTimeSpan = TimeSpan.FromDays(1), //在生产模式下设 AllowInsecureHttp = falseAllowInsecureHttp = true};app.UseOAuthBearerTokens(OAuthOptions);} }
使用Client Credentials Grant的授权方式( grant_type= client_credentials)获取 Access Token,并以这个 Token 调用与用户相关的 Web API。
public class ApplicationOAuthProvider : OAuthAuthorizationServerProvider{public override Task ValidateClientAuthentication(OAuthValidateClientAuthenticationContext context){string clientId, clientSecret;context.TryGetBasicCredentials(out clientId, out clientSecret);if (clientId == "Mobile" && clientSecret == "Xiaomi"){context.Validated();}return Task.FromResult<object>(null);}public override Task GrantClientCredentials(OAuthGrantClientCredentialsContext context){var oAuthIdentity = new ClaimsIdentity(context.Options.AuthenticationType);oAuthIdentity.AddClaim(new Claim(ClaimTypes.Name, "Xiaomi"));var ticket = new AuthenticationTicket(oAuthIdentity, new AuthenticationProperties());context.Validated(ticket);return base.GrantClientCredentials(context);}}
在 ValidateClientAuthentication() 方法中获取客户端的 client_id 与 client_secret 进行验证。在 GrantClientCredentials() 方法中对客户端进行授权,授了权就能发 access token 。这样,OAuth的ClientCredentials授权服务端代码就完成了。在ASP.NET Web API中启用OAuth的Access Token验证非常简单,只需在相应的Controller或Action加上[Authorize]标记,VS已生成部分代码,详细查看APIController文件夹下的ValuesController
下面我们在客户端调用一下,添加TestController,生成Index的View,然后在View中添加如下
$(function () {$("#clientCredentials").on("click", function () {GetClientCredentialsAccessToken();});});function GetClientCredentialsAccessToken() {$("#clientResult").html("Requesting");var clientId = "Mobile";var clientSecret = "Xiaomi";$.ajax({url: "/Token",type: "post",data: { "grant_type": "client_credentials" },dataType: "json",headers: {"Authorization": "Basic " + Base64_Encode(clientId + ":" + clientSecret)},success: function (data) {var accessToken = data.access_token;GetValues(accessToken);}});}function GetValues(accessToken) {var html = "Token:" + accessToken + "<br/><br/>";$.ajax({url: "/api/Values",type: "get",dataType: "json",headers: {"Authorization": "Bearer " + accessToken},success: function (values) {for (var i = 0; i < values.length; i++) {html += "values[" + i + "] :" + values[i] + "<br/>";}$("#clientResult").html(html);}});}function Base64_Encode(str) {var c1, c2, c3;var base64EncodeChars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";var i = 0, len = str.length, string = '';while (i < len) {c1 = str.charCodeAt(i++) & 0xff;if (i === len) {string += base64EncodeChars.charAt(c1 >> 2);string += base64EncodeChars.charAt((c1 & 0x3) << 4);string += "==";break;}c2 = str.charCodeAt(i++);if (i === len) {string += base64EncodeChars.charAt(c1 >> 2);string += base64EncodeChars.charAt(((c1 & 0x3) << 4) | ((c2 & 0xF0) >> 4));string += base64EncodeChars.charAt((c2 & 0xF) << 2);string += "=";break;}c3 = str.charCodeAt(i++);string += base64EncodeChars.charAt(c1 >> 2);string += base64EncodeChars.charAt(((c1 & 0x3) << 4) | ((c2 & 0xF0) >> 4));string += base64EncodeChars.charAt(((c2 & 0xF) << 2) | ((c3 & 0xC0) >> 6));string += base64EncodeChars.charAt(c3 & 0x3F);}return string;}
Ps:
传递clientId与clientSecret有两种方式,上例使用BasicAuthentication,服务端使用TryGetBasicCredentials();另外一种方式是普通From的,把参数放到Ajax的data中,如:
{“clientId”: id ,” clientSecret”:”secret”, "grant_type":"client_credentials"}
对应服务端使用TryGetFormCredentials()获取clientId和clientSecret;
推荐使用Basic Authentication方式;使用Resource Owner Password Credentials Grant 的授权方式( grant_type=password )获取 Access Token,并以这个 Token 调用与用户相关的 Web API。
Resource Owner Password Credentials Grant 授权方式(需要验证登录用户)
因为我们刚开始时已经初始化EF,添加了一个用户信息。ApplicationOAuthProvider.cs 的GrantResourceOwnerCredentials()方法(VS帮我们自动生成了),已经实现了先关的代码
public override async Task GrantResourceOwnerCredentials(OAuthGrantResourceOwnerCredentialsContext context){//调用后台的登录服务验证用户名与密码var userManager = context.OwinContext.GetUserManager<ApplicationUserManager>();ApplicationUser user = await userManager.FindAsync(context.UserName, context.Password);if (user == null){context.SetError("invalid_grant", "用户名或密码不正确。");return;}ClaimsIdentity oAuthIdentity = await user.GenerateUserIdentityAsync(userManager, OAuthDefaults.AuthenticationType);ClaimsIdentity cookiesIdentity = await user.GenerateUserIdentityAsync(userManager, CookieAuthenticationDefaults.AuthenticationType);AuthenticationProperties properties = CreateProperties(user.UserName);AuthenticationTicket ticket = new AuthenticationTicket(oAuthIdentity, properties);context.Validated(ticket);context.Request.Context.Authentication.SignIn(cookiesIdentity);}
在Test的index.cshtml 中新增测试的代码,如下
function GetResourceOwnerCredentialsAccessToken() {$("#resourceOwnerresult").html("Requesting");var clientId = "Mobile";var clientSecret = "Xiaomi";$.ajax({url: "/Token",type: "post",data: {"grant_type": "password","username": "admin@123.com","password": "Admin@123456"},dataType: "json",headers: {"Authorization": "Basic " + Base64_Encode(clientId + ":" + clientSecret)},success: function (data) {var accessToken = data.access_token;GetCurrentUserName(accessToken);}});}function GetCurrentUserName(accessToken) {var html = "Token:" + accessToken + "<br/><br/>";$.ajax({url: "/api/User",type: "get",dataType: "text",headers: {"Authorization": "Bearer " + accessToken},success: function (userName) {html += "CurrentUserName:" + userName + "<br/>";$("#resourceOwnerresult").html(html);}});}
至此,使用WebApi 的两种授权方式发放Token和两种授权方式区别,以及使用Token调用受保护的api已经介绍完了,Oauth其实还有一个refresh token,refresh token 是专用于刷新 access token 的 token。一是因为 access token 是有过期时间的,到了过期时间这个 access token 就失效,需要刷新;二是因为一个 access token 会关联一定的用户权限,如果用户授权更改了,这个 access token 需要被刷新以关联新的权限。
这个就不单独介绍了,有兴趣的可以自己研究下。转:https://www.cnblogs.com/YamatAmain/p/5029466.html
api测试:
/// <summary>///客户端模式【Client Credentials Grant】///http://www.asp.net/web-api/overview/security/individual-accounts-in-web-api/// </summary>[RoutePrefix("api/v1/oauth2")]public class OAuth2Controller : ApiController{/// <summary>/// 获得资讯/// </summary>/// <returns></returns> [Authorize][Route("news")]public async Task<IHttpActionResult> GetNewsAsync(){var authentication = HttpContext.Current.GetOwinContext().Authentication;var ticket = authentication.AuthenticateAsync("Bearer").Result;var claimsIdentity = User.Identity as ClaimsIdentity;var data = claimsIdentity.Claims.Where(c => c.Type == "urn:oauth:scope").ToList();var claims = ((ClaimsIdentity)Thread.CurrentPrincipal.Identity).Claims;return Ok(new { IsError = true, Msg = string.Empty, Data = Thread.CurrentPrincipal.Identity.Name + " It's about news !!! token expires: " + ticket.Properties.Dictionary.ToJson() });}}
/// <summary>///客户端模式【Client Credentials Grant】///http://www.asp.net/web-api/overview/security/individual-accounts-in-web-api/// </summary>[RoutePrefix("api/v1/oauth2")]public class OAuth2Controller : ApiController{/// <summary>/// 获取token/// </summary>/// <returns></returns>[Route("token")]public async Task<IHttpActionResult> GetTokenAsync(){//获得tokenvar dict = new SortedDictionary<string, string>();dict.Add("client_id", "irving");dict.Add("client_secret", "123456");dict.Add("grant_type", "client_credentials");var data = await (@"http://" + Request.RequestUri.Authority + @"/token").PostUrlEncodedAsync(dict).ReceiveJson<Token>();//根据token获得咨询信息 [Authorization: Bearer {THE TOKEN}]//var news = await (@"http://" + Request.RequestUri.Authority + @"/api/v1/oauth2/news").WithHeader("Authorization", "Bearer " + data.access_token).GetAsync().ReceiveString();var news = await (@"http://" + Request.RequestUri.Authority + @"/api/v1/oauth2/news").WithOAuthBearerToken(data.access_token).GetAsync().ReceiveString();return Ok(new { IsError = true, Msg = data, Data = news });}}public class Token{public string access_token { get; set; }public string token_type { get; set; }public string expires_in { get; set; }}
全:https://www.cnblogs.com/xsj1989/p/5557251.html
https://www.cnblogs.com/Irving/p/4607104.html
/************ 自己测试的 ****************/
using Microsoft.Owin.Security.OAuth; using System; using System.Collections.Generic; using System.Linq; using System.Web.Http;namespace WebApplication1 {public static class WebApiConfig{public static void Register(HttpConfiguration config){// Web API 配置和服务 //添加的配置 //匿名身份验证//config.SuppressDefaultHostAuthentication(); //config.Filters.Add(new HostAuthenticationFilter(OAuthDefaults.AuthenticationType)); // Web API 路由 config.MapHttpAttributeRoutes();config.Routes.MapHttpRoute(name: "DefaultApi",routeTemplate: "api/{controller}/{id}",defaults: new { id = RouteParameter.Optional });}} }
var
jsonFormatter = config.Formatters.OfType<JsonMediaTypeFormatter>().First();
jsonFormatter.SerializerSettings.ContractResolver =
new
CamelCasePropertyNamesContractResolver();
using System; using System.Threading.Tasks; using Microsoft.Owin; using Owin; using System.Web.Http;[assembly: OwinStartup(typeof(WebApplication1.Startup))]namespace WebApplication1 {public partial class Startup{public void Configuration(IAppBuilder app){// 有关如何配置应用程序的详细信息,请访问 http://go.microsoft.com/fwlink/?LinkID=316888var config = new HttpConfiguration(); //去掉?WebApiConfig.Register(config); // 去掉?? ConfigureAuth(app);//开启OAuth服务 app.UseWebApi(config);//这一行代码必须放在ConfiureOAuth(app)之后 }} }
using Microsoft.Owin; using Microsoft.Owin.Security.OAuth; using Owin; using System; using System.Collections.Generic; using System.Linq; using System.Web; using WebApplication1.Providers;namespace WebApplication1 {public partial class Startup{public static OAuthBearerAuthenticationOptions OAuthBearerOptions { get; private set; }public void ConfigureAuth(IAppBuilder app){//OAuthBearerOptions = new OAuthBearerAuthenticationOptions(); //匿名身份验证//Token 生成配置var oAuthOptions = new OAuthAuthorizationServerOptions{AllowInsecureHttp = true, //允许客户端使用Http协议请求AuthenticationMode = Microsoft.Owin.Security.AuthenticationMode.Active,TokenEndpointPath = new PathString("/token"),AccessTokenExpireTimeSpan = TimeSpan.FromMinutes(5),//提供认证策略Provider = new OpenAuthorizationServerProvider()//RefreshTokenProvider = new RefreshAuthenticationTokenProvider() };//app.UseOAuthAuthorizationServer(oAuthOptions);//app.UseOAuthBearerAuthentication(OAuthBearerOptions); //匿名身份验证 app.UseOAuthBearerTokens(oAuthOptions);}} }
using Microsoft.Owin.Security.OAuth; using System; using System.Collections.Generic; using System.Linq; using System.Threading.Tasks; using System.Web;namespace WebApplication1.Providers {public class OpenAuthorizationServerProvider : OAuthAuthorizationServerProvider{public OpenAuthorizationServerProvider(){}/// <summary>/// 客户端授权[生成access token]/// </summary>/// <param name="context"></param>/// <returns></returns>public override Task GrantClientCredentials(OAuthGrantClientCredentialsContext context){var oAuthIdentity = new System.Security.Claims.ClaimsIdentity(context.Options.AuthenticationType);oAuthIdentity.AddClaim(new System.Security.Claims.Claim(System.Security.Claims.ClaimTypes.Name, context.OwinContext.Get<string>("as:client_id")));var ticket = new Microsoft.Owin.Security.AuthenticationTicket(oAuthIdentity, new Microsoft.Owin.Security.AuthenticationProperties { AllowRefresh = true });context.Validated(ticket);return base.GrantClientCredentials(context);}/// <summary>/// 验证客户端/// </summary>/// <param name="context"></param>public override Task ValidateClientAuthentication(OAuthValidateClientAuthenticationContext context){string clientId;string clientSecret;context.TryGetFormCredentials(out clientId, out clientSecret);//clienti认证//context.TryGetBasicCredentials(out clientId, out clientSecret); //Basic认证//TODO:读库,验证if (clientId != "malfy" && clientSecret != "111111"){context.SetError("invalid_client", "client is not valid");return Task.FromResult<object>(null);}context.OwinContext.Set("as:client_id", clientId);//ILifetimeScope scope = context.OwinContext.GetAutofacLifetimeScope();//var authService = scope.Resolve<IAuthService>();//var client = authService.GetClient(clientId);//context.OwinContext.Set<string>("as:clientRefreshTokenLifeTime", client.RefreshTokenLifeTime.ToString()); context.Validated(clientId);return Task.FromResult<object>(null);}} }
Controller : [Authorize]
//********** end ***********/
Provider :提供具体的认证策略;
///添加新的RefreshTokenProvider
public class SimpleRefreshTokenProvider : IAuthenticationTokenProvider {public async Task CreateAsync(AuthenticationTokenCreateContext context){var refreshTokenId = Guid.NewGuid().ToString("n");using (AuthRepository _repo = new AuthRepository()){var token = new RefreshToken(){Id = refreshTokenId.GetHash(),Subject = context.Ticket.Identity.Name,IssuedUtc = DateTime.UtcNow,ExpiresUtc = DateTime.UtcNow.AddMinutes(30)};context.Ticket.Properties.IssuedUtc = token.IssuedUtc;context.Ticket.Properties.ExpiresUtc = token.ExpiresUtc;token.ProtectedTicket = context.SerializeTicket();var result = await _repo.AddRefreshToken(token);if (result){context.SetToken(refreshTokenId);}}}public async Task ReceiveAsync(AuthenticationTokenReceiveContext context){string hashedTokenId = context.Token.GetHash();using (AuthRepository _repo = new AuthRepository()){var refreshToken = await _repo.FindRefreshToken(hashedTokenId);if (refreshToken != null){//Get protectedTicket from refreshToken class context.DeserializeTicket(refreshToken.ProtectedTicket);var result = await _repo.RemoveRefreshToken(hashedTokenId);}}}public void Create(AuthenticationTokenCreateContext context){throw new NotImplementedException();}public void Receive(AuthenticationTokenReceiveContext context){throw new NotImplementedException();}}
我们实现了其中两个异步方法,对两个同步方法不做实现。其中CreateAsync用来生成RefreshToken值,生成后需要持久化在数据库中,客户端需要拿RefreshToken来请求刷新token,此时ReceiveAsync方法将拿客户的RefreshToken和数据库中RefreshToken做对比,验证成功后删除此refreshToken。
2、重新请求token
可以看到这次请求不但得到了token,还得到了refresh_token
3、当token过期后,凭借上次得到的refresh_token重新获取token
此次请求又得到了新的refresh_token,每次refresh_token只能用一次,因为在方法ReceiveAsync中我们一旦拿到refresh_token就删除了记录。
七、总结
此文重点介绍了OAuth2.0中resource owner password credentials模式的使用,此模式可以实现资源服务为自己的客户端授权。另外文章中也提到模式4-client credentials也可以实现这种场景,但用来给有服务端的客户端使用-区别于纯html+js客户端。原因在于模式4-client credentials使用appKey+appSecrect来验证客户端,如果没有服务端的话appSecrect将暴露在js中。
同样的道理:模式1-授权码模式(authorization code)和模式2-简化模式(implicit)的区别也在于模式2-简化模式(implicit)用在无服务端的场景下,请求头中不用带appSecrect。
在webApi中使用owin来实现OAuth2.0是最简单的解决方案,另外一个方案是使用DotNetOpenOauth,这个方案的实现稍显复杂,可用的文档也较少,源码中带有几个例子我也没有直接跑起来,最后无奈之下几乎读完了整个源码才理解。
八、客户端的实现
我们将采用jquery和angular两种js框架来调用本文实现的服务端。下一篇将实现此功能,另外还要给我们的服务端加上CORS(同源策略)支持。
代码都同步更新在 https://git.oschina.net/richieyangs/OAuthPractice.git
转:https://www.cnblogs.com/Leo_wl/p/4919783.html
数据并行:http://www.cnblogs.com/Leo_wl/p/4919814.html
转载于:https://www.cnblogs.com/love201314/p/7902720.html
如若内容造成侵权/违法违规/事实不符,请联系编程学习网邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
相关文章
- 使用Oauth构建webapp应用
一、什么是OAuth OAuth是一个关于授权(Authorization)的开放网络标准,目前的版本是2.0版。注意是Authorization(授权),而不是Authentication(认证)。用来做Authentication(认证)的标准叫做openid connect,我们将在以后…...
2024/4/24 4:33:58 - 要用Identity Server 4 -- OAuth 2.0 超级简介
OAuth 2.0 简介OAuth有一些定义:OAuth 2.0是一个委托协议, 它可以让那些控制资源的人允许某个应用以代表他们来访问他们控制的资源, 注意是代表这些人, 而不是假冒或模仿这些人. 这个应用从资源的所有者那里获得到授权(Authorization)和access token, 随后就可以使用这个access…...
2024/4/21 14:43:38 - [译] Spring REST API + OAuth2 + AngularJS
www.baeldung.com/rest-api-sp…作者:Eugen Paraschiv译者:oopsguy.com 1、概述 在本教程中,我们将使用 OAuth 来保护 REST API,并以一个简单的 AngularJS 客户端进行示范。 我们要建立的应用程序将包含了四个独立模块:…...
2024/4/27 3:31:12 - Spring Cloud :: Security :: OAuth2 - ⓪ OAuth2模型详述
本系列文章 Spring Cloud :: Security :: Oauth2 - ⓪ OAuth2模型详述Spring Cloud :: Security :: Oauth2 - ① 待定... 写在前面 文中很多专有名词在初次使用时我会标注一下中文,但是在后面使用过程中我还是会使用英文来表示,因为所有的标准的出处都是…...
2024/4/21 14:43:38 - 双眼皮长什么样
...
2024/4/21 14:43:35 - 双眼皮长度能改吗
...
2024/4/26 1:27:16 - 双眼皮拆线肿会消除吗
...
2024/4/23 22:11:54 - 双眼皮拆线有多疼
...
2024/4/20 15:17:14 - 双眼皮拆线完注意事项
...
2024/4/20 15:17:13 - 在ArcGIS Desktop中进行三参数或七参数精确投影转换
转自 在ArcGIS Desktop中进行三参数或七参数精确投影转换 ArcGIS中定义的投影转换方法,在对数据的空间信息要求较高的工程中往往不能适用,有比较明显的偏差。在项目的前期数据准备工作中,需要进行更加精确的三参数或七参数投影转换。下面介绍…...
2024/4/21 14:43:34 - 品优购电商系统 3 - 规格及模板管理
本文部分内容来自黑马49期项目实战-品优购电商系统开发 文章目录课程目标前端分层开发需求分析代码分离前端基础层前端服务层前端控制层修改页面控制器继承需求分析前端代码建立父控制器修改品牌控制器层测试导入生成的代码规格管理需求及表结构分析需求表结构规格列表引入JS加…...
2024/4/21 14:43:34 - AngularJS 1.x 实现进销存系统感悟与反思
背景 作为单页大型应用的mv*框架AngularJS爱好者的我,不实现个大型web应用练练手怎么赶说自己熟练ng,于是尝试实现了大部分erp的进销存功能。本文主要对实现业务需求中遇到的问题以及解决方案进行阐述。不确保对其他项目有可移植性。 问题与解决方案 Q:目…...
2024/4/21 22:49:41 - 基于SSM框架大型分布式电商系统开发(9-10)
目录前言第九章 网站前台-搜索页面-搜索解决方案-Solr【1】1.Solr安装与配置1.1什么是Solr1.2 Solr安装1.3中文分析器IK Analyzer1.3.1 IK Analyzer简介1.3.2 IK Analyzer配置1.4配置域1.4.1域1.4.2复制域1.4.3动态域2.品优购-批量数据导入2.1 需求分析2.2 查询商品数据列表2.2…...
2024/4/21 14:43:33 - Django 优秀资源大全项目资源非 Python 包工具贡献
Awesome Django 介绍 Awesome-Django 是由 Roberto Rosario 发起和维护的 Django 资源列表。该列表收集了大量 Django 相关的优秀应用、项目等资源,方便了 Django 用户参考查阅。 Django 优秀资源大全 则是依据 Awesome-Django 翻译而来。也欢迎你帮助推荐和提供建议…...
2024/4/21 14:43:31 - JetBrains Rider 2018.3.4
etBrAIns Rider是一个基于IntelliJ平台和ReSharper的跨平台.NET IDE。支持许多.NET项目类型,使您可以开发各种应用程序,包括.NET桌面应用程序,服务和库,Unity游戏,Xamarin应用程序,ASP.NET和ASP.NET Core W…...
2024/4/21 14:43:30 - 品优购-Day04
学习目标 目标1:了解代码生成迅速完成基础代码 目标2:完成模板管理的CRUD操作 目标3:完成商品分类的CRUD操作 一、模版列表得显示,调整好界面,即后端及前端文件都已拷贝到项目中 1.type_template.html导入步骤一:js…...
2024/4/21 14:43:29 - 品优购-Day03
学习目标: 目标1:理解和运用angularJS的service 目标2:理解和运用控制器继承 目标3:掌握代码生成器的使用 目标4:实现规格管理 目标5:实现模板管理 一、自定义服务:就是将之前的所有与后台请求相关代码抽…...
2024/4/27 8:45:24 - 双眼皮拆线全过程视频
...
2024/4/20 15:17:28 - 双眼皮拆线前忌口
...
2024/4/20 15:17:27 - 一统江湖的大前端(7)React.js-从开发者到工程师
目录 一. 前端打怪升级指南1.1 我应该从哪个框架开始学?1.2 一次转职1.3 二次转职1.4 转职-其他二. 为什么你应该学习React2.1 技术栈的延伸2.2 组件化开发2.3 思想的提升三. 没有实战项目,我应该如何学习React四. 资料推荐首发链接:https://bbs.huaweic…...
2024/4/27 16:39:38
最新文章
- spring Ai框架调用openai大模型简易demo
前提: 需要科学上网,key没有官方的,就找中转的key1 pom依赖,注意添加的依赖和仓库配置 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xm…...
2024/5/3 6:49:14 - 梯度消失和梯度爆炸的一些处理方法
在这里是记录一下梯度消失或梯度爆炸的一些处理技巧。全当学习总结了如有错误还请留言,在此感激不尽。 权重和梯度的更新公式如下: w w − η ⋅ ∇ w w w - \eta \cdot \nabla w ww−η⋅∇w 个人通俗的理解梯度消失就是网络模型在反向求导的时候出…...
2024/3/20 10:50:27 - 3d representation的一些基本概念
顶点(Vertex):三维空间中的一个点,可以有多个属性,如位置坐标、颜色、纹理坐标和法线向量。它是构建三维几何形状的基本单元。 边(Edge):连接两个顶点形成的直线段,它定…...
2024/5/2 17:20:39 - 外包干了4年,技术退步明显。。。。
说一下自己的情况,本科生,19年通过校招进入上海某软件公司,干了接近4年的功能测试,今年年初,感觉自己不能够在这样下去了,长时间呆在一个舒适的环境会让一个人堕落!而我已经在一个企业干了四年的功能测试&a…...
2024/5/2 4:18:31 - 【外汇早评】美通胀数据走低,美元调整
原标题:【外汇早评】美通胀数据走低,美元调整昨日美国方面公布了新一期的核心PCE物价指数数据,同比增长1.6%,低于前值和预期值的1.7%,距离美联储的通胀目标2%继续走低,通胀压力较低,且此前美国一季度GDP初值中的消费部分下滑明显,因此市场对美联储后续更可能降息的政策…...
2024/5/1 17:30:59 - 【原油贵金属周评】原油多头拥挤,价格调整
原标题:【原油贵金属周评】原油多头拥挤,价格调整本周国际劳动节,我们喜迎四天假期,但是整个金融市场确实流动性充沛,大事频发,各个商品波动剧烈。美国方面,在本周四凌晨公布5月份的利率决议和新闻发布会,维持联邦基金利率在2.25%-2.50%不变,符合市场预期。同时美联储…...
2024/5/2 16:16:39 - 【外汇周评】靓丽非农不及疲软通胀影响
原标题:【外汇周评】靓丽非农不及疲软通胀影响在刚结束的周五,美国方面公布了新一期的非农就业数据,大幅好于前值和预期,新增就业重新回到20万以上。具体数据: 美国4月非农就业人口变动 26.3万人,预期 19万人,前值 19.6万人。 美国4月失业率 3.6%,预期 3.8%,前值 3…...
2024/4/29 2:29:43 - 【原油贵金属早评】库存继续增加,油价收跌
原标题:【原油贵金属早评】库存继续增加,油价收跌周三清晨公布美国当周API原油库存数据,上周原油库存增加281万桶至4.692亿桶,增幅超过预期的74.4万桶。且有消息人士称,沙特阿美据悉将于6月向亚洲炼油厂额外出售更多原油,印度炼油商预计将每日获得至多20万桶的额外原油供…...
2024/5/2 9:28:15 - 【外汇早评】日本央行会议纪要不改日元强势
原标题:【外汇早评】日本央行会议纪要不改日元强势近两日日元大幅走强与近期市场风险情绪上升,避险资金回流日元有关,也与前一段时间的美日贸易谈判给日本缓冲期,日本方面对汇率问题也避免继续贬值有关。虽然今日早间日本央行公布的利率会议纪要仍然是支持宽松政策,但这符…...
2024/4/27 17:58:04 - 【原油贵金属早评】欧佩克稳定市场,填补伊朗问题的影响
原标题:【原油贵金属早评】欧佩克稳定市场,填补伊朗问题的影响近日伊朗局势升温,导致市场担忧影响原油供给,油价试图反弹。此时OPEC表态稳定市场。据消息人士透露,沙特6月石油出口料将低于700万桶/日,沙特已经收到石油消费国提出的6月份扩大出口的“适度要求”,沙特将满…...
2024/4/27 14:22:49 - 【外汇早评】美欲与伊朗重谈协议
原标题:【外汇早评】美欲与伊朗重谈协议美国对伊朗的制裁遭到伊朗的抗议,昨日伊朗方面提出将部分退出伊核协议。而此行为又遭到欧洲方面对伊朗的谴责和警告,伊朗外长昨日回应称,欧洲国家履行它们的义务,伊核协议就能保证存续。据传闻伊朗的导弹已经对准了以色列和美国的航…...
2024/4/28 1:28:33 - 【原油贵金属早评】波动率飙升,市场情绪动荡
原标题:【原油贵金属早评】波动率飙升,市场情绪动荡因中美贸易谈判不安情绪影响,金融市场各资产品种出现明显的波动。随着美国与中方开启第十一轮谈判之际,美国按照既定计划向中国2000亿商品征收25%的关税,市场情绪有所平复,已经开始接受这一事实。虽然波动率-恐慌指数VI…...
2024/4/30 9:43:09 - 【原油贵金属周评】伊朗局势升温,黄金多头跃跃欲试
原标题:【原油贵金属周评】伊朗局势升温,黄金多头跃跃欲试美国和伊朗的局势继续升温,市场风险情绪上升,避险黄金有向上突破阻力的迹象。原油方面稍显平稳,近期美国和OPEC加大供给及市场需求回落的影响,伊朗局势并未推升油价走强。近期中美贸易谈判摩擦再度升级,美国对中…...
2024/4/27 17:59:30 - 【原油贵金属早评】市场情绪继续恶化,黄金上破
原标题:【原油贵金属早评】市场情绪继续恶化,黄金上破周初中国针对于美国加征关税的进行的反制措施引发市场情绪的大幅波动,人民币汇率出现大幅的贬值动能,金融市场受到非常明显的冲击。尤其是波动率起来之后,对于股市的表现尤其不安。隔夜美国股市出现明显的下行走势,这…...
2024/5/2 15:04:34 - 【外汇早评】美伊僵持,风险情绪继续升温
原标题:【外汇早评】美伊僵持,风险情绪继续升温昨日沙特两艘油轮再次发生爆炸事件,导致波斯湾局势进一步恶化,市场担忧美伊可能会出现摩擦生火,避险品种获得支撑,黄金和日元大幅走强。美指受中美贸易问题影响而在低位震荡。继5月12日,四艘商船在阿联酋领海附近的阿曼湾、…...
2024/4/28 1:34:08 - 【原油贵金属早评】贸易冲突导致需求低迷,油价弱势
原标题:【原油贵金属早评】贸易冲突导致需求低迷,油价弱势近日虽然伊朗局势升温,中东地区几起油船被袭击事件影响,但油价并未走高,而是出于调整结构中。由于市场预期局势失控的可能性较低,而中美贸易问题导致的全球经济衰退风险更大,需求会持续低迷,因此油价调整压力较…...
2024/4/26 19:03:37 - 氧生福地 玩美北湖(上)——为时光守候两千年
原标题:氧生福地 玩美北湖(上)——为时光守候两千年一次说走就走的旅行,只有一张高铁票的距离~ 所以,湖南郴州,我来了~ 从广州南站出发,一个半小时就到达郴州西站了。在动车上,同时改票的南风兄和我居然被分到了一个车厢,所以一路非常愉快地聊了过来。 挺好,最起…...
2024/4/29 20:46:55 - 氧生福地 玩美北湖(中)——永春梯田里的美与鲜
原标题:氧生福地 玩美北湖(中)——永春梯田里的美与鲜一觉醒来,因为大家太爱“美”照,在柳毅山庄去寻找龙女而错过了早餐时间。近十点,向导坏坏还是带着饥肠辘辘的我们去吃郴州最富有盛名的“鱼头粉”。说这是“十二分推荐”,到郴州必吃的美食之一。 哇塞!那个味美香甜…...
2024/4/30 22:21:04 - 氧生福地 玩美北湖(下)——奔跑吧骚年!
原标题:氧生福地 玩美北湖(下)——奔跑吧骚年!让我们红尘做伴 活得潇潇洒洒 策马奔腾共享人世繁华 对酒当歌唱出心中喜悦 轰轰烈烈把握青春年华 让我们红尘做伴 活得潇潇洒洒 策马奔腾共享人世繁华 对酒当歌唱出心中喜悦 轰轰烈烈把握青春年华 啊……啊……啊 两…...
2024/5/1 4:32:01 - 扒开伪装医用面膜,翻六倍价格宰客,小姐姐注意了!
原标题:扒开伪装医用面膜,翻六倍价格宰客,小姐姐注意了!扒开伪装医用面膜,翻六倍价格宰客!当行业里的某一品项火爆了,就会有很多商家蹭热度,装逼忽悠,最近火爆朋友圈的医用面膜,被沾上了污点,到底怎么回事呢? “比普通面膜安全、效果好!痘痘、痘印、敏感肌都能用…...
2024/4/27 23:24:42 - 「发现」铁皮石斛仙草之神奇功效用于医用面膜
原标题:「发现」铁皮石斛仙草之神奇功效用于医用面膜丽彦妆铁皮石斛医用面膜|石斛多糖无菌修护补水贴19大优势: 1、铁皮石斛:自唐宋以来,一直被列为皇室贡品,铁皮石斛生于海拔1600米的悬崖峭壁之上,繁殖力差,产量极低,所以古代仅供皇室、贵族享用 2、铁皮石斛自古民间…...
2024/4/28 5:48:52 - 丽彦妆\医用面膜\冷敷贴轻奢医学护肤引导者
原标题:丽彦妆\医用面膜\冷敷贴轻奢医学护肤引导者【公司简介】 广州华彬企业隶属香港华彬集团有限公司,专注美业21年,其旗下品牌: 「圣茵美」私密荷尔蒙抗衰,产后修复 「圣仪轩」私密荷尔蒙抗衰,产后修复 「花茵莳」私密荷尔蒙抗衰,产后修复 「丽彦妆」专注医学护…...
2024/4/30 9:42:22 - 广州械字号面膜生产厂家OEM/ODM4项须知!
原标题:广州械字号面膜生产厂家OEM/ODM4项须知!广州械字号面膜生产厂家OEM/ODM流程及注意事项解读: 械字号医用面膜,其实在我国并没有严格的定义,通常我们说的医美面膜指的应该是一种「医用敷料」,也就是说,医用面膜其实算作「医疗器械」的一种,又称「医用冷敷贴」。 …...
2024/5/2 9:07:46 - 械字号医用眼膜缓解用眼过度到底有无作用?
原标题:械字号医用眼膜缓解用眼过度到底有无作用?医用眼膜/械字号眼膜/医用冷敷眼贴 凝胶层为亲水高分子材料,含70%以上的水分。体表皮肤温度传导到本产品的凝胶层,热量被凝胶内水分子吸收,通过水分的蒸发带走大量的热量,可迅速地降低体表皮肤局部温度,减轻局部皮肤的灼…...
2024/4/30 9:42:49 - 配置失败还原请勿关闭计算机,电脑开机屏幕上面显示,配置失败还原更改 请勿关闭计算机 开不了机 这个问题怎么办...
解析如下:1、长按电脑电源键直至关机,然后再按一次电源健重启电脑,按F8健进入安全模式2、安全模式下进入Windows系统桌面后,按住“winR”打开运行窗口,输入“services.msc”打开服务设置3、在服务界面,选中…...
2022/11/19 21:17:18 - 错误使用 reshape要执行 RESHAPE,请勿更改元素数目。
%读入6幅图像(每一幅图像的大小是564*564) f1 imread(WashingtonDC_Band1_564.tif); subplot(3,2,1),imshow(f1); f2 imread(WashingtonDC_Band2_564.tif); subplot(3,2,2),imshow(f2); f3 imread(WashingtonDC_Band3_564.tif); subplot(3,2,3),imsho…...
2022/11/19 21:17:16 - 配置 已完成 请勿关闭计算机,win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机...
win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机”问题的解决方法在win7系统关机时如果有升级系统的或者其他需要会直接进入一个 等待界面,在等待界面中我们需要等待操作结束才能关机,虽然这比较麻烦,但是对系统进行配置和升级…...
2022/11/19 21:17:15 - 台式电脑显示配置100%请勿关闭计算机,“准备配置windows 请勿关闭计算机”的解决方法...
有不少用户在重装Win7系统或更新系统后会遇到“准备配置windows,请勿关闭计算机”的提示,要过很久才能进入系统,有的用户甚至几个小时也无法进入,下面就教大家这个问题的解决方法。第一种方法:我们首先在左下角的“开始…...
2022/11/19 21:17:14 - win7 正在配置 请勿关闭计算机,怎么办Win7开机显示正在配置Windows Update请勿关机...
置信有很多用户都跟小编一样遇到过这样的问题,电脑时发现开机屏幕显现“正在配置Windows Update,请勿关机”(如下图所示),而且还需求等大约5分钟才干进入系统。这是怎样回事呢?一切都是正常操作的,为什么开时机呈现“正…...
2022/11/19 21:17:13 - 准备配置windows 请勿关闭计算机 蓝屏,Win7开机总是出现提示“配置Windows请勿关机”...
Win7系统开机启动时总是出现“配置Windows请勿关机”的提示,没过几秒后电脑自动重启,每次开机都这样无法进入系统,此时碰到这种现象的用户就可以使用以下5种方法解决问题。方法一:开机按下F8,在出现的Windows高级启动选…...
2022/11/19 21:17:12 - 准备windows请勿关闭计算机要多久,windows10系统提示正在准备windows请勿关闭计算机怎么办...
有不少windows10系统用户反映说碰到这样一个情况,就是电脑提示正在准备windows请勿关闭计算机,碰到这样的问题该怎么解决呢,现在小编就给大家分享一下windows10系统提示正在准备windows请勿关闭计算机的具体第一种方法:1、2、依次…...
2022/11/19 21:17:11 - 配置 已完成 请勿关闭计算机,win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机”的解决方法...
今天和大家分享一下win7系统重装了Win7旗舰版系统后,每次关机的时候桌面上都会显示一个“配置Windows Update的界面,提示请勿关闭计算机”,每次停留好几分钟才能正常关机,导致什么情况引起的呢?出现配置Windows Update…...
2022/11/19 21:17:10 - 电脑桌面一直是清理请关闭计算机,windows7一直卡在清理 请勿关闭计算机-win7清理请勿关机,win7配置更新35%不动...
只能是等着,别无他法。说是卡着如果你看硬盘灯应该在读写。如果从 Win 10 无法正常回滚,只能是考虑备份数据后重装系统了。解决来方案一:管理员运行cmd:net stop WuAuServcd %windir%ren SoftwareDistribution SDoldnet start WuA…...
2022/11/19 21:17:09 - 计算机配置更新不起,电脑提示“配置Windows Update请勿关闭计算机”怎么办?
原标题:电脑提示“配置Windows Update请勿关闭计算机”怎么办?win7系统中在开机与关闭的时候总是显示“配置windows update请勿关闭计算机”相信有不少朋友都曾遇到过一次两次还能忍但经常遇到就叫人感到心烦了遇到这种问题怎么办呢?一般的方…...
2022/11/19 21:17:08 - 计算机正在配置无法关机,关机提示 windows7 正在配置windows 请勿关闭计算机 ,然后等了一晚上也没有关掉。现在电脑无法正常关机...
关机提示 windows7 正在配置windows 请勿关闭计算机 ,然后等了一晚上也没有关掉。现在电脑无法正常关机以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容,让我们赶快一起来看一下吧!关机提示 windows7 正在配…...
2022/11/19 21:17:05 - 钉钉提示请勿通过开发者调试模式_钉钉请勿通过开发者调试模式是真的吗好不好用...
钉钉请勿通过开发者调试模式是真的吗好不好用 更新时间:2020-04-20 22:24:19 浏览次数:729次 区域: 南阳 > 卧龙 列举网提醒您:为保障您的权益,请不要提前支付任何费用! 虚拟位置外设器!!轨迹模拟&虚拟位置外设神器 专业用于:钉钉,外勤365,红圈通,企业微信和…...
2022/11/19 21:17:05 - 配置失败还原请勿关闭计算机怎么办,win7系统出现“配置windows update失败 还原更改 请勿关闭计算机”,长时间没反应,无法进入系统的解决方案...
前几天班里有位学生电脑(windows 7系统)出问题了,具体表现是开机时一直停留在“配置windows update失败 还原更改 请勿关闭计算机”这个界面,长时间没反应,无法进入系统。这个问题原来帮其他同学也解决过,网上搜了不少资料&#x…...
2022/11/19 21:17:04 - 一个电脑无法关闭计算机你应该怎么办,电脑显示“清理请勿关闭计算机”怎么办?...
本文为你提供了3个有效解决电脑显示“清理请勿关闭计算机”问题的方法,并在最后教给你1种保护系统安全的好方法,一起来看看!电脑出现“清理请勿关闭计算机”在Windows 7(SP1)和Windows Server 2008 R2 SP1中,添加了1个新功能在“磁…...
2022/11/19 21:17:03 - 请勿关闭计算机还原更改要多久,电脑显示:配置windows更新失败,正在还原更改,请勿关闭计算机怎么办...
许多用户在长期不使用电脑的时候,开启电脑发现电脑显示:配置windows更新失败,正在还原更改,请勿关闭计算机。。.这要怎么办呢?下面小编就带着大家一起看看吧!如果能够正常进入系统,建议您暂时移…...
2022/11/19 21:17:02 - 还原更改请勿关闭计算机 要多久,配置windows update失败 还原更改 请勿关闭计算机,电脑开机后一直显示以...
配置windows update失败 还原更改 请勿关闭计算机,电脑开机后一直显示以以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容,让我们赶快一起来看一下吧!配置windows update失败 还原更改 请勿关闭计算机&#x…...
2022/11/19 21:17:01 - 电脑配置中请勿关闭计算机怎么办,准备配置windows请勿关闭计算机一直显示怎么办【图解】...
不知道大家有没有遇到过这样的一个问题,就是我们的win7系统在关机的时候,总是喜欢显示“准备配置windows,请勿关机”这样的一个页面,没有什么大碍,但是如果一直等着的话就要两个小时甚至更久都关不了机,非常…...
2022/11/19 21:17:00 - 正在准备配置请勿关闭计算机,正在准备配置windows请勿关闭计算机时间长了解决教程...
当电脑出现正在准备配置windows请勿关闭计算机时,一般是您正对windows进行升级,但是这个要是长时间没有反应,我们不能再傻等下去了。可能是电脑出了别的问题了,来看看教程的说法。正在准备配置windows请勿关闭计算机时间长了方法一…...
2022/11/19 21:16:59 - 配置失败还原请勿关闭计算机,配置Windows Update失败,还原更改请勿关闭计算机...
我们使用电脑的过程中有时会遇到这种情况,当我们打开电脑之后,发现一直停留在一个界面:“配置Windows Update失败,还原更改请勿关闭计算机”,等了许久还是无法进入系统。如果我们遇到此类问题应该如何解决呢࿰…...
2022/11/19 21:16:58 - 如何在iPhone上关闭“请勿打扰”
Apple’s “Do Not Disturb While Driving” is a potentially lifesaving iPhone feature, but it doesn’t always turn on automatically at the appropriate time. For example, you might be a passenger in a moving car, but your iPhone may think you’re the one dri…...
2022/11/19 21:16:57