科技未来网站建设pinfinity wordpress

news/2025/9/23 15:48:39/文章来源:
科技未来网站建设,pinfinity wordpress,响应式公司网站,网页设计动画网站说明#xff1a;在实际的业务中#xff0c;难免会跟第三方系统进行数据的交互与传递#xff0c;那么如何保证数据在传输过程中的安全呢#xff08;防窃取#xff09;#xff1f;除了https的协议之外#xff0c;能不能加上通用的一套算法以及规范来保证传输的安全性呢在实际的业务中难免会跟第三方系统进行数据的交互与传递那么如何保证数据在传输过程中的安全呢防窃取除了https的协议之外能不能加上通用的一套算法以及规范来保证传输的安全性呢 下面我们就来讨论下常用的一些API设计的安全方法可能不一定是最好的有更牛逼的实现方式但是这篇是我自己的经验分享. 一、token 简介 Token访问令牌access token, 用于接口中, 用于标识接口调用者的身份、凭证减少用户名和密码的传输次数。一般情况下客户端(接口调用方)需要先向服务器端申请一个接口调用的账号服务器会给出一个appId和一个key, key用于参数签名使用注意key保存到客户端需要做一些安全处理防止泄露。 Token的值一般是UUID服务端生成Token后需要将token做为key将一些和token关联的信息作为value保存到缓存服务器中(redis)当一个请求过来后服务器就去缓存服务器中查询这个Token是否存在存在则调用接口不存在返回接口错误一般通过拦截器或者过滤器来实现Token分为两种 API Token(接口令牌): 用于访问不需要用户登录的接口如登录、注册、一些基本数据的获取等。获取接口令牌需要拿appId、timestamp和sign来换sign加密(timestampkey) USER Token(用户令牌): 用于访问需要用户登录之后的接口如获取我的基本信息、保存、修改、删除等操作。获取用户令牌需要拿用户名和密码来换 关于Token的时效性token可以是一次性的、也可以在一段时间范围内是有效的具体使用哪种看业务需要。 一般情况下接口最好使用https协议如果使用http协议Token机制只是一种减少被黑的可能性其实只能防君子不能防小人。 一般token、timestamp和sign 三个参数会在接口中会同时作为参数传递每个参数都有各自的用途。 二、timestamp 简介 timestamp: 时间戳是客户端调用接口时对应的当前时间戳时间戳用于防止DoS攻击。当黑客劫持了请求的url去DoS攻击每次调用接口时接口都会判断服务器当前系统时间和接口中传的的timestamp的差值如果这个差值超过某个设置的时间(假如5分钟)那么这个请求将被拦截掉如果在设置的超时时间范围内是不能阻止DoS攻击的。 timestamp机制只能减轻DoS攻击的时间缩短攻击时间。如果黑客修改了时间戳的值可通过sign签名机制来处理。 DoS DoS是Denial of Service的简称即拒绝服务造成DoS的攻击行为被称为DoS攻击其目的是使计算机或网络无法提供正常的服务。最常见的DoS攻击有计算机网络带宽攻击和连通性攻击。 DoS攻击是指故意的攻击网络协议实现的缺陷或直接通过野蛮手段残忍地耗尽被攻击对象的资源目的是让目标计算机或网络无法提供正常的服务或资源访问使目标系统服务系统停止响应甚至崩溃而在此攻击中并不包括侵入目标服务器或目标网络设备。这些服务资源包括网络带宽文件系统空间容量开放的进程或者允许的连接。这种攻击会导致资源的匮乏无论计算机的处理速度多快、内存容量多大、网络带宽的速度多快都无法避免这种攻击带来的后果。 Pingflood: 该攻击在短时间内向目的主机发送大量ping包造成网络堵塞或主机资源耗尽。 Synflood: 该攻击以多个随机的源主机地址向目的主机发送SYN包而在收到目的主机的SYN ACK后并不回应这样目的主机就为这些源主机建立了大量的连接队列而且由于没有收到ACK一直维护着这 些队列造成了资源的大量消耗而不能向正常请求提供服务。 Smurf该攻击向一个子网的广播地址发一个带有特定请求如ICMP回应请求的包并且将源地址伪装成想要攻击的主机地址。子网上所有主机都回应广播包请求而向被攻击主机发包使该主机受到攻击。 Land-based攻击者将一个包的源地址和目的地址都设置为目标主机的地址然后将该包通过IP欺骗的方式发送给被攻击主机这种包可以造成被攻击主机因试图与自己建立连接而陷入死循环从而很大程度地降低了系统性能。 Ping of Death根据TCP/IP的规范一个包的长度最大为65536字节。尽管一个包的长度不能超过65536字节但是一个包分成的多个片段的叠加却能做到。当一个主机收到了长度大于65536字节的包时就是受到了Ping of Death攻击该攻击会造成主机的宕机。 TeardropIP数据包在网络传递时数据包可以分成更小的片段。攻击者可以通过发送两段或者更多数据包来实现TearDrop攻击。第一个包的偏移量为0长度为N第二个包的偏移量小于N。为了合并这些数据段TCP/IP堆栈会分配超乎寻常的巨大资源从而造成系统资源的缺乏甚至机器的重新启动。 PingSweep使用ICMP Echo轮询多个主机。 三、sign 简介 nonce随机值是客户端随机生成的值作为参数传递过来随机值的目的是增加sign签名的多变性。随机值一般是数字和字母的组合6位长度随机值的组成和长度没有固定规则。 sign: 一般用于参数签名防止参数被非法篡改最常见的是修改金额等重要敏感参数 sign的值一般是将所有非空参数按照升续排序然后tokenkeytimestampnonce(随机数)拼接在一起然后使用某种加密算法进行加密作为接口中的一个参数sign来传递也可以将sign放到请求头中。接口在网络传输过程中如果被黑客挟持并修改其中的参数值然后再继续调用接口虽然参数的值被修改了但是因为黑客不知道sign是如何计算出来的不知道sign都有哪些值构成不知道以怎样的顺序拼接在一起的最重要的是不知道签名字符串中的key是什么所以黑客可以篡改参数的值但没法修改sign的值当服务器调用接口前会按照sign的规则重新计算出sign的值然后和接口传递的sign参数的值做比较如果相等表示参数值没有被篡改如果不等表示参数被非法篡改了就不执行接口了。 四、防止重复提交 对于一些重要的操作需要防止客户端重复提交的(如非幂等性重要操作)具体办法是当请求第一次提交时将sign作为key保存到redis并设置超时时间超时时间和Timestamp中设置的差值相同。当同一个请求第二次访问时会先检测redis是否存在该sign如果存在则证明重复提交了接口就不再继续调用了。如果sign在缓存服务器中因过期时间到了而被删除了此时当这个url再次请求服务器时因token的过期时间和sign的过期时间一直sign过期也意味着token过期那样同样的url再访问服务器会因token错误会被拦截掉这就是为什么sign和token的过期时间要保持一致的原因。拒绝重复调用机制确保URL被别人截获了也无法使用如抓取数据。 对于哪些接口需要防止重复提交可以自定义个注解来标记。注意所有的安全措施都用上的话有时候难免太过复杂在实际项目中需要根据自身情况作出裁剪比如可以只使用签名机制就可以保证信息不会被篡改或者定向提供服务的时候只用Token机制就可以了。如何裁剪全看项目实际情况和对接口安全性的求。 注意所有的安全措施都用上的话有时候难免太过复杂在实际项目中需要根据自身情况作出裁剪比如可以只使用签名机制就可以保证信息不会被篡改或者定向提供服务的时候只用Token机制就可以了。如何裁剪全看项目实际情况和对接口安全性的要求。 五、使用流程 接口调用方(客户端)向接口提供方(服务器)申请接口调用账号申请成功后接口提供方会给接口调用方一个appId和一个key参数 客户端携带参数appId、timestamp、sign去调用服务器端的API token其中sign加密(appId timestamp key) 客户端拿着api_token 去访问不需要登录就能访问的接口 当访问用户需要登录的接口时客户端跳转到登录页面通过用户名和密码调用登录接口登录接口会返回一个usertoken, 客户端拿着usertoken 去访问需要登录才能访问的接口 sign的作用是防止参数被篡改客户端调用服务端时需要传递sign参数服务器响应客户端时也可以返回一个sign用于客户度校验返回的值是否被非法篡改了。客户端传的sign和服务器端响应的sign算法可能会不同。 示例代码 dependency dependencygroupIdorg.springframework.boot/groupIdartifactIdspring-boot-starter-data-redis/artifactId /dependency dependencygroupIdredis.clients/groupIdartifactIdjedis/artifactIdversion2.9.0/version /dependencydependencygroupIdorg.springframework.boot/groupIdartifactIdspring-boot-starter-web/artifactId /dependency RedisConfiguration Configuration public class RedisConfiguration {Beanpublic JedisConnectionFactory jedisConnectionFactory(){return new JedisConnectionFactory();}/*** 支持存储对象* return*/Beanpublic RedisTemplateString, String redisTemplate(){RedisTemplateString, String redisTemplate new StringRedisTemplate();redisTemplate.setConnectionFactory(jedisConnectionFactory());Jackson2JsonRedisSerializer jackson2JsonRedisSerializer new Jackson2JsonRedisSerializer(Object.class);ObjectMapper objectMapper new ObjectMapper();objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);jackson2JsonRedisSerializer.setObjectMapper(objectMapper);redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);redisTemplate.afterPropertiesSet();return redisTemplate;} }TokenController Slf4j RestController RequestMapping(/api/token) public class TokenController {Autowiredprivate RedisTemplate redisTemplate;/*** API Token** param sign* return*/PostMapping(/api_token)public ApiResponseAccessToken apiToken(String appId, RequestHeader(timestamp) String timestamp, RequestHeader(sign) String sign) {Assert.isTrue(!StringUtils.isEmpty(appId) !StringUtils.isEmpty(timestamp) !StringUtils.isEmpty(sign), 参数错误);long reqeustInterval System.currentTimeMillis() - Long.valueOf(timestamp);Assert.isTrue(reqeustInterval 5 * 60 * 1000, 请求过期请重新请求);// 1\. 根据appId查询数据库获取appSecretAppInfo appInfo new AppInfo(1, 12345678954556);// 2\. 校验签名String signString timestamp appId appInfo.getKey();String signature MD5Util.encode(signString);log.info(signature);Assert.isTrue(signature.equals(sign), 签名错误);// 3\. 如果正确生成一个token保存到redis中如果错误返回错误信息AccessToken accessToken this.saveToken(0, appInfo, null);return ApiResponse.success(accessToken);}NotRepeatSubmit(5000)PostMapping(user_token)public ApiResponseUserInfo userToken(String username, String password) {// 根据用户名查询密码, 并比较密码(密码可以RSA加密一下)UserInfo userInfo new UserInfo(username, 81255cb0dca1a5f304328a70ac85dcbd, 111111);String pwd password userInfo.getSalt();String passwordMD5 MD5Util.encode(pwd);Assert.isTrue(passwordMD5.equals(userInfo.getPassword()), 密码错误);// 2\. 保存TokenAppInfo appInfo new AppInfo(1, 12345678954556);AccessToken accessToken this.saveToken(1, appInfo, userInfo);userInfo.setAccessToken(accessToken);return ApiResponse.success(userInfo);}private AccessToken saveToken(int tokenType, AppInfo appInfo, UserInfo userInfo) {String token UUID.randomUUID().toString();// token有效期为2小时Calendar calendar Calendar.getInstance();calendar.setTime(new Date());calendar.add(Calendar.SECOND, 7200);Date expireTime calendar.getTime();// 4\. 保存tokenValueOperationsString, TokenInfo operations redisTemplate.opsForValue();TokenInfo tokenInfo new TokenInfo();tokenInfo.setTokenType(tokenType);tokenInfo.setAppInfo(appInfo);if (tokenType 1) {tokenInfo.setUserInfo(userInfo);}operations.set(token, tokenInfo, 7200, TimeUnit.SECONDS);AccessToken accessToken new AccessToken(token, expireTime);return accessToken;}public static void main(String[] args) {long timestamp System.currentTimeMillis();System.out.println(timestamp);String signString timestamp 1 12345678954556;String sign MD5Util.encode(signString);System.out.println(sign);System.out.println(-------------------);signString password123456username112345678954556 ff03e64b-427b-45a7-b78b-47d9e8597d3b1529815393153sdfsdfsfs timestamp A1scr6;sign MD5Util.encode(signString);System.out.println(sign);} } WebMvcConfiguration Configuration public class WebMvcConfiguration extends WebMvcConfigurationSupport {private static final String[] excludePathPatterns {/api/token/api_token};Autowiredprivate TokenInterceptor tokenInterceptor;Overridepublic void addInterceptors(InterceptorRegistry registry) {super.addInterceptors(registry);registry.addInterceptor(tokenInterceptor).addPathPatterns(/api/**).excludePathPatterns(excludePathPatterns);} } 5. TokenInterceptor Component public class TokenInterceptor extends HandlerInterceptorAdapter {Autowiredprivate RedisTemplate redisTemplate;/**** param request* param response* param handler 访问的目标方法* return* throws Exception*/Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {String token request.getHeader(token);String timestamp request.getHeader(timestamp);// 随机字符串String nonce request.getHeader(nonce);String sign request.getHeader(sign);Assert.isTrue(!StringUtils.isEmpty(token) !StringUtils.isEmpty(timestamp) !StringUtils.isEmpty(sign), 参数错误);// 获取超时时间NotRepeatSubmit notRepeatSubmit ApiUtil.getNotRepeatSubmit(handler);long expireTime notRepeatSubmit null ? 5 * 60 * 1000 : notRepeatSubmit.value();// 2\. 请求时间间隔long reqeustInterval System.currentTimeMillis() - Long.valueOf(timestamp);Assert.isTrue(reqeustInterval expireTime, 请求超时请重新请求);// 3\. 校验Token是否存在ValueOperationsString, TokenInfo tokenRedis redisTemplate.opsForValue();TokenInfo tokenInfo tokenRedis.get(token);Assert.notNull(tokenInfo, token错误);// 4\. 校验签名(将所有的参数加进来防止别人篡改参数) 所有参数看参数名升续排序拼接成url// 请求参数 token timestamp nonceString signString ApiUtil.concatSignString(request) tokenInfo.getAppInfo().getKey() token timestamp nonce;String signature MD5Util.encode(signString);boolean flag signature.equals(sign);Assert.isTrue(flag, 签名错误);// 5\. 拒绝重复调用(第一次访问时存储过期时间和请求超时时间保持一致), 只有标注不允许重复提交注解的才会校验if (notRepeatSubmit ! null) {ValueOperationsString, Integer signRedis redisTemplate.opsForValue();boolean exists redisTemplate.hasKey(sign);Assert.isTrue(!exists, 请勿重复提交);signRedis.set(sign, 0, expireTime, TimeUnit.MILLISECONDS);}return super.preHandle(request, response, handler);} } public class MD5Util {private static final String hexDigits[] { 0, 1, 2, 3, 4, 5,6, 7, 8, 9, a, b, c, d, e, f };private static String byteArrayToHexString(byte b[]) {StringBuffer resultSb new StringBuffer();for (int i 0; i b.length; i)resultSb.append(byteToHexString(b[i]));return resultSb.toString();}private static String byteToHexString(byte b) {int n b;if (n 0)n 256;int d1 n / 16;int d2 n % 16;return hexDigits[d1] hexDigits[d2];}public static String encode(String origin) {return encode(origin, UTF-8);}public static String encode(String origin, String charsetname) {String resultString null;try {resultString new String(origin);MessageDigest md MessageDigest.getInstance(MD5);if (charsetname null || .equals(charsetname))resultString byteArrayToHexString(md.digest(resultString.getBytes()));elseresultString byteArrayToHexString(md.digest(resultString.getBytes(charsetname)));} catch (Exception exception) {}return resultString;} } NotRepeatSubmit -----自定义注解防止重复提交。 /*** 禁止重复提交*/ Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) public interface NotRepeatSubmit {/** 过期时间单位毫秒 **/long value() default 5000; } AccessToken Data AllArgsConstructor public class AccessToken {/** token */private String token;/** 失效时间 */private Date expireTime; } AppInfo Data NoArgsConstructor AllArgsConstructor public class AppInfo {/** App id */private String appId;/** API 秘钥 */private String key; } TokenInfo Data public class TokenInfo {/** token类型: api:0 、user:1 */private Integer tokenType;/** App 信息 */private AppInfo appInfo;/** 用户其他数据 */private UserInfo userInfo; } UserInfo Data public class UserInfo {/** 用户名 */private String username;/** 手机号 */private String mobile;/** 邮箱 */private String email;/** 密码 */private String password;/** 盐 */private String salt;private AccessToken accessToken;public UserInfo(String username, String password, String salt) {this.username username;this.password password;this.salt salt;} } ApiCodeEnum /*** 错误码code可以使用纯数字,使用不同区间标识一类错误也可以使用纯字符也可以使用前缀编号** 错误码ERR 编号** 可以使用日志级别的前缀作为错误类型区分 Info(I) Error(E) Warning(W)** 或者以业务模块 错误号** TODO 错误码设计** Alipay 用了两个code两个msg(https://docs.open.alipay.com/api_1/alipay.trade.pay)*/ public enum ApiCodeEnum {SUCCESS(10000, success),UNKNOW_ERROR(ERR0001,未知错误),PARAMETER_ERROR(ERR0002,参数错误),TOKEN_EXPIRE(ERR0003,认证过期),REQUEST_TIMEOUT(ERR0004,请求超时),SIGN_ERROR(ERR0005,签名错误),REPEAT_SUBMIT(ERR0006,请不要频繁操作),;/** 代码 */private String code;/** 结果 */private String msg;ApiCodeEnum(String code, String msg) {this.code code;this.msg msg;}public String getCode() {return code;}public String getMsg() {return msg;} } ApiResult Data NoArgsConstructor AllArgsConstructor public class ApiResult {/** 代码 */private String code;/** 结果 */private String msg; } ApiUtil -------这个参考支付宝加密的算法写的.我直接Copy过来了。 public class ApiUtil {/*** 按参数名升续拼接参数* param request* return*/public static String concatSignString(HttpServletRequest request) {MapString, String paramterMap new HashMap();request.getParameterMap().forEach((key, value) - paramterMap.put(key, value[0]));// 按照key升续排序然后拼接参数SetString keySet paramterMap.keySet();String[] keyArray keySet.toArray(new String[keySet.size()]);Arrays.sort(keyArray);StringBuilder sb new StringBuilder();for (String k : keyArray) {// 或略掉的字段if (k.equals(sign)) {continue;}if (paramterMap.get(k).trim().length() 0) {// 参数值为空则不参与签名sb.append(k).append().append(paramterMap.get(k).trim()).append();}}return sb.toString();}public static String concatSignString(MapString, String map) {MapString, String paramterMap new HashMap();map.forEach((key, value) - paramterMap.put(key, value));// 按照key升续排序然后拼接参数SetString keySet paramterMap.keySet();String[] keyArray keySet.toArray(new String[keySet.size()]);Arrays.sort(keyArray);StringBuilder sb new StringBuilder();for (String k : keyArray) {if (paramterMap.get(k).trim().length() 0) {// 参数值为空则不参与签名sb.append(k).append().append(paramterMap.get(k).trim()).append();}}return sb.toString();}/*** 获取方法上的NotRepeatSubmit注解* param handler* return*/public static NotRepeatSubmit getNotRepeatSubmit(Object handler) {if (handler instanceof HandlerMethod) {HandlerMethod handlerMethod (HandlerMethod) handler;Method method handlerMethod.getMethod();NotRepeatSubmit annotation method.getAnnotation(NotRepeatSubmit.class);return annotation;}return null;} } ApiResponse Data Slf4j public class ApiResponseT {/** 结果 */private ApiResult result;/** 数据 */private T data;/** 签名 */private String sign;public static T ApiResponse success(T data) {return response(ApiCodeEnum.SUCCESS.getCode(), ApiCodeEnum.SUCCESS.getMsg(), data);}public static ApiResponse error(String code, String msg) {return response(code, msg, null);}public static T ApiResponse response(String code, String msg, T data) {ApiResult result new ApiResult(code, msg);ApiResponse response new ApiResponse();response.setResult(result);response.setData(data);String sign signData(data);response.setSign(sign);return response;}private static T String signData(T data) {// TODO 查询keyString key 12345678954556;MapString, String responseMap null;try {responseMap getFields(data);} catch (IllegalAccessException e) {return null;}String urlComponent ApiUtil.concatSignString(responseMap);String signature urlComponent key key;String sign MD5Util.encode(signature);return sign;}/*** param data 反射的对象,获取对象的字段名和值* throws IllegalArgumentException* throws IllegalAccessException*/public static MapString, String getFields(Object data) throws IllegalAccessException, IllegalArgumentException {if (data null) return null;MapString, String map new HashMap();Field[] fields data.getClass().getDeclaredFields();for (int i 0; i fields.length; i) {Field field fields[i];field.setAccessible(true);String name field.getName();Object value field.get(data);if (field.get(data) ! null) {map.put(name, value.toString());}}return map;} } ThreadLocal ThreadLocal是线程内的全局上下文。就是在单个线程中方法之间共享的内存每个方法都可以从该上下文中获取值和修改值。 最后感谢每一个认真阅读我文章的人礼尚往来总是要有的虽然不是什么很值钱的东西如果你用得到的话可以直接拿走 软件测试面试小程序 被百万人刷爆的软件测试题库谁用谁知道全网最全面试刷题小程序手机就可以刷题地铁上公交上卷起来 涵盖以下这些面试题板块 1、软件测试基础理论 2、web,app接口功能测试 3、网络 4、数据库 5、linux 6、web,app接口自动化 7、性能测试 8、编程基础9、hr面试题 10、开放性测试题11、安全测试12、计算机基础 这些资料对于【软件测试】的朋友来说应该是最全面最完整的备战仓库这个仓库也陪伴上万个测试工程师们走过最艰难的路程希望也能帮助到你

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

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

相关文章

营销型网站建设作用网站建设课程设计报告图文

一、引言 我昨天写了《安卓应用开发学习:获取经纬度及地理位置描述信息》日志,今天再接再厉,记录一下跟着《Android App 开发进阶与项目实战》一书,实现获取导航卫星信息,并在手机上显示的功能的情况。先上实现后的在…

环保类网站建设电商网站建设成本

对于本题 我感觉还是链表做起来舒服 数组也可以做 但是数组需要去控制循环 不太好控制 我之前搞了 最后看别人的实现 但是链表搞了一次就搞好了 香的嘞~ 下面是代码 用单链表实现循环 再去删除要删除的人 5个人 数到2 你们在纸上画图 我就不画了 对于数组实现你们可以去…

网站建设 广州佛山市北区小型网页设计培训

前言 最近工作比较忙,没怎么记录东西了。Android的Handler重要性不必赘述,之前也写过几篇关于hanlder的文章了: Handler有多深?连环二十七问Android多线程:深入分析 Handler机制源码(二) And…

洛谷P10288 [GESP样题 八级] 区间

原题 题目描述 小杨有一个长度为 \(n\) 的正整数序列 \(A\)。 小杨有 \(q\) 次询问。第 \(i\) 次(\(1\le i\le q\))询问时,小杨会给出 \(l_i,r_i,x_i\),请你求出 \(x_i\) 在 \(A_{l_i}, A_{l_i+1}, \dots A_{r_i}\…

百度如何搜索网址网站推广优化趋势

1. nuScenes 数据集 1.1 概述 nuScenes 数据集 (pronounced /nu:ːsiː:nz/) 是由 Motional (以前称为 nuTonomy) 团队开发的自动驾驶公共大型数据集。nuScenes 数据集的灵感来自于开创性的 KITTI 数据集。 nuScenes 是第一个提供自动驾驶车辆整个传感器套件 (6 个摄像头、1 …

AI 时代下,开发流程的重塑:从“代码先行”到“文档驱动”

本文探讨了AI编程工具在提升效率的同时,因缺乏顶层设计而导致项目混乱的困境。并提出一种“文档驱动”的AI原生开发新范式,强调高质量、结构化的文档是驾驭AI编码,实现高质量交付的关键。文章标题 引言:AI 编程工具…

P13617 [ICPC 2025 APC] Bit Counting Sequenc

题意:给定长度 $n \leq 5 \times 10^5$ 的 popcount 序列 $\{a_i\}$,求其对应的原序列。思路:观察发现 popcount 序列具有倍增构造性质,将原序列 $[0,2^k - 1]$ 的第 $k$ 位改成 $1$ 可得到 $[2^k,2^{k + 1} - 1]$…

perl -MCPAN -e install GD;

001、 Package gdlib was not found in the pkg-config search path.Perhaps you should add the directory containing `gdlib.pcto the PKG_CONFIG_PATH environment variableNo package gdlib found

Day 02 HTML的基础 - 教程

pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", …

如何在网站添加代码大气精美网站设计工作室织梦模板

【芯片DFX】万字长文带你搞懂JTAG的门门道道【芯片DFX】ARM:CoreSight、ETM、PTM、ITM、HTM、ETB等常用术语解析

网络建站的费用微网站开发一般费用多少钱

Navicat Premium(16.3.3 Windows 版或以上)正式支持 GaussDB 分布式数据库。GaussDB 分布式模式更适合对系统可用性和数据处理能力要求较高的场景。Navicat 工具不仅提供可视化数据查看和编辑功能,还提供强大的高阶功能(如模型、结…

什么网站专做店铺wordpress 一栏主题

首先已经创建好了 Vue 框架,安装好了 node.js。 没有完成的可按照此博客搭建:搭建Vue项目 之后打开终端,使用命令。 1、命令安装 axios 和 vue-axios npm install axios --save npm install vue-axios --save2、package.json 查看版本 在 p…

做静态网站的步骤怎么下载网页视频到本地

一、类型转换 C语言中的类型转换比较松散,C新增4个类型转换运算符,更加严格的显示类型转换,使转换的效率更加规范 1、static_cast static_cast,用于仅在编译时检查的强制转换。 如果编译器检测到你尝试在完全不兼容的类型之间强制…

微信公众号手机网站做网站有哪些需求

在工作中,偶尔看到有些机器的网口名字是以ethX命令,有些则以enpXsX这种名字命名。网上的资料说的都不太明白,资料也无据可查,很难让人信服。于是决定自己查了下官方的资料和源码,把这些搞清楚。 官方文档:Predictable…

P3959 [NOIP 2017 提高组] 宝藏 题解

link 题目要求任选图中一点为根,通过拓展道路最终形成一棵树,使得代价总和最小,代价受深度和边权两个因素影响。 容易想到一种爆搜,任选一点为根,每次扫描已选点来不断尝试拓展道路,但这样做太蛋疼了,我们尝试优…

(二)若依前后端分离版本二次开发 代码生成、目录添加、数据字典维护

(二)若依前后端分离版本二次开发 代码生成、目录添加、数据字典维护一 摘要 主要介绍添加目录、菜单、代码自动生成等功能。 二 目标 2.1 掌握添加目录,菜单 2.2 掌握添加数据字典 2.3 掌握代码生成 三 实施 已标签…

C#与Access数据库操作简易指南:增删改查及类封装

简介:C#是面向对象的编程语言,常用于Windows应用开发,而Microsoft Access是一种数据库管理系统。本教程指导如何结合C#操作Access数据库进行基本的增删改查(CRUD)操作,使用ADO.NET通过OLEDB类连接和操作数据库。…

对之前部署hbase总结

部署路程为看黑马大数据视频 部署直到yarn集群部署之后看尚硅谷的视频完成部署即可 之后参考有可能会报错直接复制报错到csdn即可 启动代码和停止指令: 注意每次启动检查每一个主机启动是否完成 慢一些 否则会报错

网站排版图片网站禁止右键

一、前言 在大多数传统的web系统中,使用Redis一般都是作为缓存使用,在大数据查询时作为缓解性能的一种解决方案。博主的的系统中使用Redis也主要使用到缓存的作用,还有做了注册中心,分布式事务。其他的强大的功能,没有…