React 深度学习:ReactFiber

packages/react-reconciler/src/ReactFiber.js

Fiber

// A Fiber is work on a Component that needs to be done or was done. There can
// be more than one per component.
// Fiber 是 Component 上需要完成或已经完成工作。每个组件可以有多个 fiber。
export type Fiber = {|// These first fields are conceptually members of an Instance. This used to// be split into a separate type and intersected with the other Fiber fields,// but until Flow fixes its intersection bugs, we've merged them into a// single type.// 这些字段在概念上是实例的成员。// 这曾经被分割成一个单独的类型,并与其他 fiber 字段相交,// 但是直到 Flow 修复了它的交集错误,我们才将它们合并成一个单一的类型。// An Instance is shared between all versions of a component. We can easily// break this out into a separate object to avoid copying so much to the// alternate versions of the tree. We put this on a single object for now to// minimize the number of objects created during the initial render.// 在一个组件的所有版本之间共享的实例。// 我们可以很容易地将其拆分为一个单独的对象,// 以避免过多地复制到树的其他版本。// 现在我们将它放在一个对象上,以最小化在初始渲染期间创建的对象的数量。// 标记 fiber 类型的标签。tag: WorkTag,// 此子元素的唯一标识符。key: null | string,// The value of element.type which is used to preserve the identity during// reconciliation of this child.// element.type 的值,用于在此子元素的协调(reconciliation)过程中保存标识。elementType: any,// The resolved function/class/ associated with this fiber.//与此 fiber 相关联的已解析的 function/class/。type: any,// The local state associated with this fiber.// 与此 fiber 相关联的本地状态。stateNode: any,// Conceptual aliases// parent : Instance -> return The parent happens to be the same as the// return fiber since we've merged the fiber and instance.// 概念的别名// parent:Instance -> return// parent 恰好与 return 的 fiber 相同,因为我们已经合并了 fiber 和实例。// Remaining fields belong to Fiber// 剩余的字段属于 Fiber// The Fiber to return to after finishing processing this one.// This is effectively the parent, but there can be multiple parents (two)// so this is only the parent of the thing we're currently processing.// It is conceptually the same as the return address of a stack frame.// 这条 Fiber 处理完后完后要返回。// 这实际上是 parent,但是可以有多个 parent (两个),所以这只是我们当前正在处理的东西的父类 fiber。// 在概念上与堆栈帧的返回地址相同。return: Fiber | null,// Singly Linked List Tree Structure.// 单链表树结构。child: Fiber | null,sibling: Fiber | null,index: number,// The ref last used to attach this node.// 最后用于附加此节点的 ref。// 我将避免为 prod 和 model 添加一个 owner 字段作为函数。// I'll avoid adding an owner field for prod and model that as functions.ref: null | (((handle: mixed) => void) & {_stringRef: ?string}) | RefObject,// Input is the data coming into process this fiber. Arguments. Props.// 输入是处理该 fiber 的数据。Arguments. Props.pendingProps: any, // This type will be more specific once we overload the tag.一旦我们重载了标签,这种类型将会更加具体。memoizedProps: any, // The props used to create the output.用于创建输出的 props。// A queue of state updates and callbacks.// 状态更新和回调的队列。updateQueue: UpdateQueue<any> | null,// The state used to create the output// 用于创建输出的状态memoizedState: any,// A linked-list of contexts that this fiber depends on// 此 fiber 所依赖的上下文的链表contextDependencies: ContextDependencyList | null,// Bitfield that describes properties about the fiber and its subtree. E.g.// the ConcurrentMode flag indicates whether the subtree should be async-by-// default. When a fiber is created, it inherits the mode of its// parent. Additional flags can be set at creation time, but after that the// value should remain unchanged throughout the fiber's lifetime, particularly// before its child fibers are created.// 位字段,描述 fiber 及其子树的属性。// 例如,ConcurrentMode 标志指示子树是否默认为 async。// 当创建一个 fiber 时,它继承其父的 mode 。// 可以在创建时设置附加标志,但在此之后,值应该在整个 fiber 的生命周期内保持不变,特别是在创建子 fiber 之前。mode: TypeOfMode,// EffecteffectTag: SideEffectTag,// Singly linked list fast path to the next fiber with side-effects.// 单链表快速路径到下一个具有副作用的 fiber。nextEffect: Fiber | null,// The first and last fiber with side-effect within this subtree. This allows// us to reuse a slice of the linked list when we reuse the work done within// this fiber.// 这个子树中的第一个和最后一个有副作用的 fiber。// 这允许我们在重用在这个 fiber 中完成的工作时重用链表的一部分。firstEffect: Fiber | null,lastEffect: Fiber | null,// Represents a time in the future by which this work should be completed.// Does not include work found in its subtree.// 表示将来完成这项工作的时间。// 不包括子树中的工作。expirationTime: ExpirationTime,// This is used to quickly determine if a subtree has no pending changes.// 这用于快速确定子树是否没有挂起的更改。childExpirationTime: ExpirationTime,// This is a pooled version of a Fiber. Every fiber that gets updated will// eventually have a pair. There are cases when we can clean up pairs to save// memory if we need to.// 这是 fiber 的混合版本。// 每一个被更新的 fiber 最终都会有一对。// 在某些情况下,如果需要,我们可以清理它以节省内存。alternate: Fiber | null,// Time spent rendering this Fiber and its descendants for the current update.// This tells us how well the tree makes use of sCU for memoization.// It is reset to 0 each time we render and only updated when we don't bailout.// This field is only set when the enableProfilerTimer flag is enabled.// 为当前更新进行渲染此 fiber 及其后代所花费的时间。// 这告诉我们这棵树如何很好地利用 sCU 进行记忆。// 每次渲染时它都被重置为 0,只有在不进行紧急救援(bailout)时才会更新。// 只有在启用 enableProfilerTimer 标志时才设置此字段。actualDuration?: number,// If the Fiber is currently active in the "render" phase,// This marks the time at which the work began.// This field is only set when the enableProfilerTimer flag is enabled.// 如果 fiber 目前处于“渲染”阶段,// 这标志着这项工作(work)开始的时间。// 只有在启用 enableProfilerTimer 标志时才设置此字段。actualStartTime?: number,// Duration of the most recent render time for this Fiber.// This value is not updated when we bailout for memoization purposes.// This field is only set when the enableProfilerTimer flag is enabled.// 此 fiber 最近渲染的持续时间。// 当我们出于记忆目的进行紧急救助(bailout)时,此值不会更新。// 只有在启用 enableProfilerTimer 标志时才设置此字段。selfBaseDuration?: number,// Sum of base times for all descedents of this Fiber.// This value bubbles up during the "complete" phase.// This field is only set when the enableProfilerTimer flag is enabled.// 这个 fiber 的所有 descedents 的基时间之和。// 这个值在“完成”阶段出现。// 只有在启用 enableProfilerTimer 标志时才设置此字段。treeBaseDuration?: number,// Conceptual 的别名// workInProgress : Fiber ->  alternate// 用于重用的替代方法恰好与正在进行的 work 相同。// 对于复用的替代方法恰好与正在进行的 work 相同。// The alternate used for reuse happens// to be the same as work in progress.// __DEV__ only_debugID?: number,_debugSource?: Source | null,_debugOwner?: Fiber | null,_debugIsCurrentlyTiming?: boolean,// Used to verify that the order of hooks does not change between renders.// 用于验证钩子的顺序在两次渲染之间没有发生变化。_debugHookTypes?: Array<HookType> | null,
|};
复制代码

FiberNode

function FiberNode(tag: WorkTag,pendingProps: mixed,key: null | string,mode: TypeOfMode,
) {// Instancethis.tag = tag;this.key = key;this.elementType = null;this.type = null;this.stateNode = null;// Fiberthis.return = null;this.child = null;this.sibling = null;this.index = 0;this.ref = null;this.pendingProps = pendingProps;this.memoizedProps = null;this.updateQueue = null;this.memoizedState = null;this.contextDependencies = null;this.mode = mode;// Effectsthis.effectTag = NoEffect;this.nextEffect = null;this.firstEffect = null;this.lastEffect = null;this.expirationTime = NoWork;this.childExpirationTime = NoWork;this.alternate = null;if (enableProfilerTimer) {// Note: The following is done to avoid a v8 performance cliff.//// Initializing the fields below to smis and later updating them with// double values will cause Fibers to end up having separate shapes.// This behavior/bug has something to do with Object.preventExtension().// Fortunately this only impacts DEV builds.// Unfortunately it makes React unusably slow for some applications.// To work around this, initialize the fields below with doubles.//// Learn more about this here:// https://github.com/facebook/react/issues/14365// https://bugs.chromium.org/p/v8/issues/detail?id=8538this.actualDuration = Number.NaN;this.actualStartTime = Number.NaN;this.selfBaseDuration = Number.NaN;this.treeBaseDuration = Number.NaN;// It's okay to replace the initial doubles with smis after initialization.// This won't trigger the performance cliff mentioned above,// and it simplifies other profiler code (including DevTools).this.actualDuration = 0;this.actualStartTime = -1;this.selfBaseDuration = 0;this.treeBaseDuration = 0;}if (__DEV__) {// do something}
}
复制代码

createFiber

// This is a constructor function, rather than a POJO constructor, still
// please ensure we do the following:
// 1) Nobody should add any instance methods on this. Instance methods can be
//    more difficult to predict when they get optimized and they are almost
//    never inlined properly in static compilers.
// 2) Nobody should rely on `instanceof Fiber` for type testing. We should
//    always know when it is a fiber.
// 3) We might want to experiment with using numeric keys since they are easier
//    to optimize in a non-JIT environment.
// 4) We can easily go from a constructor to a createFiber object literal if that
//    is faster.
// 5) It should be easy to port this to a C struct and keep a C implementation
//    compatible.
const createFiber = function(tag: WorkTag,pendingProps: mixed,key: null | string,mode: TypeOfMode,
): Fiber {// $FlowFixMe: the shapes are exact here but Flow doesn't like constructorsreturn new FiberNode(tag, pendingProps, key, mode);
};
复制代码

createWorkInProgress

// This is used to create an alternate fiber to do work on.
// 用于创建进行工作的备用 fiber。
export function createWorkInProgress(current: Fiber,pendingProps: any,expirationTime: ExpirationTime,
): Fiber {let workInProgress = current.alternate;if (workInProgress === null) {// We use a double buffering pooling technique because we know that we'll// only ever need at most two versions of a tree. We pool the "other" unused// node that we're free to reuse. This is lazily created to avoid allocating// extra objects for things that are never updated. It also allow us to// reclaim the extra memory if needed.workInProgress = createFiber(current.tag,pendingProps,current.key,current.mode,);workInProgress.elementType = current.elementType;workInProgress.type = current.type;workInProgress.stateNode = current.stateNode;if (__DEV__) {// DEV-only fieldsworkInProgress._debugID = current._debugID;workInProgress._debugSource = current._debugSource;workInProgress._debugOwner = current._debugOwner;workInProgress._debugHookTypes = current._debugHookTypes;}workInProgress.alternate = current;current.alternate = workInProgress;} else {workInProgress.pendingProps = pendingProps;// 我们已经有一个 alternate.// 重置效果标签。workInProgress.effectTag = NoEffect;// The effect list is no longer valid.// 效果列表不再有效。workInProgress.nextEffect = null;workInProgress.firstEffect = null;workInProgress.lastEffect = null;if (enableProfilerTimer) {// We intentionally reset, rather than copy, actualDuration & actualStartTime.// This prevents time from endlessly accumulating in new commits.// This has the downside of resetting values for different priority renders,// But works for yielding (the common case) and should support resuming.workInProgress.actualDuration = 0;workInProgress.actualStartTime = -1;}}workInProgress.childExpirationTime = current.childExpirationTime;workInProgress.expirationTime = current.expirationTime;workInProgress.child = current.child;workInProgress.memoizedProps = current.memoizedProps;workInProgress.memoizedState = current.memoizedState;workInProgress.updateQueue = current.updateQueue;workInProgress.contextDependencies = current.contextDependencies;// These will be overridden during the parent's reconciliationworkInProgress.sibling = current.sibling;workInProgress.index = current.index;workInProgress.ref = current.ref;if (enableProfilerTimer) {workInProgress.selfBaseDuration = current.selfBaseDuration;workInProgress.treeBaseDuration = current.treeBaseDuration;}return workInProgress;
}
复制代码

createHostRootFiber

export function createHostRootFiber(isConcurrent: boolean): Fiber {let mode = isConcurrent ? ConcurrentMode | StrictMode : NoContext;if (enableProfilerTimer && isDevToolsPresent) {// Always collect profile timings when DevTools are present.// This enables DevTools to start capturing timing at any point–// Without some nodes in the tree having empty base times.mode |= ProfileMode;}return createFiber(HostRoot, null, null, mode);
}
复制代码

createFiberFromTypeAndProps

/*** 根据 type 和 props 创建 fiber* @param type* @param key* @param pendingProps* @param owner* @param mode* @param expirationTime* @returns {Fiber|*}*/
export function createFiberFromTypeAndProps(type: any, // React$ElementTypekey: null | string,pendingProps: any,owner: null | Fiber,mode: TypeOfMode,expirationTime: ExpirationTime,
): Fiber {let fiber;let fiberTag = IndeterminateComponent;// The resolved type is set if we know what the final type will be. I.e. it's not lazy.let resolvedType = type;if (typeof type === 'function') {if (shouldConstruct(type)) {fiberTag = ClassComponent;}} else if (typeof type === 'string') {fiberTag = HostComponent;} else {getTag: switch (type) {case REACT_FRAGMENT_TYPE:return createFiberFromFragment(pendingProps.children,mode,expirationTime,key,);case REACT_CONCURRENT_MODE_TYPE:return createFiberFromMode(pendingProps,mode | ConcurrentMode | StrictMode,expirationTime,key,);case REACT_STRICT_MODE_TYPE:return createFiberFromMode(pendingProps,mode | StrictMode,expirationTime,key,);case REACT_PROFILER_TYPE:return createFiberFromProfiler(pendingProps, mode, expirationTime, key);case REACT_SUSPENSE_TYPE:return createFiberFromSuspense(pendingProps, mode, expirationTime, key);default: {if (typeof type === 'object' && type !== null) {switch (type.$$typeof) {case REACT_PROVIDER_TYPE:fiberTag = ContextProvider;break getTag;case REACT_CONTEXT_TYPE:// This is a consumerfiberTag = ContextConsumer;break getTag;case REACT_FORWARD_REF_TYPE:fiberTag = ForwardRef;break getTag;case REACT_MEMO_TYPE:fiberTag = MemoComponent;break getTag;case REACT_LAZY_TYPE:fiberTag = LazyComponent;resolvedType = null;break getTag;}}let info = '';if (__DEV__) {if (type === undefined ||(typeof type === 'object' &&type !== null &&Object.keys(type).length === 0)) {info +=' You likely forgot to export your component from the file ' +"it's defined in, or you might have mixed up default and " +'named imports.';}const ownerName = owner ? getComponentName(owner.type) : null;if (ownerName) {info += '\n\nCheck the render method of `' + ownerName + '`.';}}invariant(false,'Element type is invalid: expected a string (for built-in ' +'components) or a class/function (for composite components) ' +'but got: %s.%s',type == null ? type : typeof type,info,);}}}fiber = createFiber(fiberTag, pendingProps, key, mode);fiber.elementType = type;fiber.type = resolvedType;fiber.expirationTime = expirationTime;return fiber;
}
复制代码

createFiberFromElement

/*** 根据 element 创建 fiber* @param element* @param mode* @param expirationTime* @returns {Fiber}*/
export function createFiberFromElement(element: ReactElement,mode: TypeOfMode,expirationTime: ExpirationTime,
): Fiber {let owner = null;if (__DEV__) {owner = element._owner;}const type = element.type;const key = element.key;const pendingProps = element.props;const fiber = createFiberFromTypeAndProps(type,key,pendingProps,owner,mode,expirationTime,);if (__DEV__) {fiber._debugSource = element._source;fiber._debugOwner = element._owner;}return fiber;
}
复制代码

createFiberFromFragment


/*** 根据 Fragment 创建 Fiber* @param elements* @param mode* @param expirationTime* @param key* @returns {Fiber}*/
export function createFiberFromFragment(elements: ReactFragment,mode: TypeOfMode,expirationTime: ExpirationTime,key: null | string,
): Fiber {const fiber = createFiber(Fragment, elements, key, mode);fiber.expirationTime = expirationTime;return fiber;
}
复制代码

createFiberFromProfiler

/*** 根据 Profiler 创建 Fiber* @param pendingProps* @param mode* @param expirationTime* @param key* @returns {Fiber}*/
function createFiberFromProfiler(pendingProps: any,mode: TypeOfMode,expirationTime: ExpirationTime,key: null | string,
): Fiber {if (__DEV__) {if (typeof pendingProps.id !== 'string' ||typeof pendingProps.onRender !== 'function') {warningWithoutStack(false,'Profiler must specify an "id" string and "onRender" function as props',);}}const fiber = createFiber(Profiler, pendingProps, key, mode | ProfileMode);// TODO: The Profiler fiber shouldn't have a type. It has a tag.fiber.elementType = REACT_PROFILER_TYPE;fiber.type = REACT_PROFILER_TYPE;fiber.expirationTime = expirationTime;return fiber;
}
复制代码

createFiberFromMode

/*** 根据 mode 创建 fiber* @param pendingProps* @param mode* @param expirationTime* @param key* @returns {Fiber}*/
function createFiberFromMode(pendingProps: any,mode: TypeOfMode,expirationTime: ExpirationTime,key: null | string,
): Fiber {const fiber = createFiber(Mode, pendingProps, key, mode);// TODO: The Mode fiber shouldn't have a type. It has a tag.const type =(mode & ConcurrentMode) === NoContext? REACT_STRICT_MODE_TYPE: REACT_CONCURRENT_MODE_TYPE;fiber.elementType = type;fiber.type = type;fiber.expirationTime = expirationTime;return fiber;
}
复制代码

createFiberFromSuspense

/*** 根据 Suspense 创建 fiber* @param pendingProps* @param mode* @param expirationTime* @param key* @returns {Fiber}*/
export function createFiberFromSuspense(pendingProps: any,mode: TypeOfMode,expirationTime: ExpirationTime,key: null | string,
) {const fiber = createFiber(SuspenseComponent, pendingProps, key, mode);// TODO: The SuspenseComponent fiber shouldn't have a type. It has a tag.const type = REACT_SUSPENSE_TYPE;fiber.elementType = type;fiber.type = type;fiber.expirationTime = expirationTime;return fiber;
}
复制代码

createFiberFromText

/*** 基于文本创建 fiber* @param content* @param mode* @param expirationTime* @returns {Fiber}*/
export function createFiberFromText(content: string,mode: TypeOfMode,expirationTime: ExpirationTime,
): Fiber {const fiber = createFiber(HostText, content, null, mode);fiber.expirationTime = expirationTime;return fiber;
}
复制代码

createFiberFromHostInstanceForDeletion

export function createFiberFromHostInstanceForDeletion(): Fiber {const fiber = createFiber(HostComponent, null, null, NoContext);// TODO: These should not need a type.fiber.elementType = 'DELETED';fiber.type = 'DELETED';return fiber;
}
复制代码

createFiberFromPortal


/*** 创建来自 portal 的fiber* @param portal* @param mode* @param expirationTime* @returns {Fiber}*/
export function createFiberFromPortal(portal: ReactPortal,mode: TypeOfMode,expirationTime: ExpirationTime,
): Fiber {const pendingProps = portal.children !== null ? portal.children : [];const fiber = createFiber(HostPortal, pendingProps, portal.key, mode);fiber.expirationTime = expirationTime;fiber.stateNode = {containerInfo: portal.containerInfo,pendingChildren: null, // Used by persistent updatesimplementation: portal.implementation,};return fiber;
}
复制代码

转载于:https://juejin.im/post/5d1a17d86fb9a07ea803e28f

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

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

相关文章

JMM模型到并发编程

电脑&#xff1a;内存 L1 L2 L3 缓存 CPU ctrlatldel就可以看到 Java 有线程内存&#xff0c;在执行线程的时候&#xff0c;会从主内存把变量加载到工作内存&#xff08;缓存&#xff09;&#xff0c;所以&#xff0c;在多线程同时改变一个静态变量时候&#xff0c;实际是分开…

焦虑的移动互联网开发者如何破局?专题解析

尴尬的35岁 不知道是哪个人提出的职场35岁就要面临被淘汰的定律&#xff0c;因为35岁定律本来就是个伪命题&#xff0c;尤其是在IT行业! 现在年八九百万的大学生毕业&#xff0c;他们虽然年轻活力&#xff0c;但是很多企业也将之“拒之门外”。 35岁的不要&#xff0c;二十几…

17.前端路由router-07keep-alive

keep-alive是Vue提供的一个抽象组件&#xff0c;用来对组件进行缓存&#xff0c;从而节省性能&#xff0c; 由于是一个抽象组件&#xff0c;所以在v页面渲染完毕后不会被渲染成一个DOM元素 当组件在keep-alive内被切换时组件的activated、deactivated这两个生命 周期钩子函数会…

jvm指令码

建议直接复制保存至本地&#xff0c;研究jvm运行&#xff0c;执行指令非常有帮助 一、未归类系列A 此系列暂未归类。 指令码 助记符 说明 0x00 nop 什么都不做 0x01 aconst_null 将null推送至栈顶 二、const系列 该系列命令主要负责把简单的数值类型送到栈顶。该系列命令不带参…

来一份全面的面试宝典练练手,面试真题解析

前言 下面的题目都是大家在面试字节跳动或者其它大厂面试时经常遇到的&#xff0c;如果大家有好的题目或者好的见解欢迎分享。 参考解析&#xff1a;郭霖、鸿洋 内容特点&#xff1a;条理清晰&#xff0c;含图像化表示更加易懂。 内容概要&#xff1a;包括 Handler、Activi…

使用IDEA创建Maven项目和Maven使用入门(配图详解)

本文详解的讲解了使用IDEA创建Maven项目&#xff0c;及Maven的基础入门。 1、打开IDEA&#xff0c;右上角选择File->New->Project 2、如图中所示选择Maven&#xff08;可按自己所需添加&#xff0c;否则加载时速度很慢&#xff09; 3、添加项目所需Groupld,ArtifactId,Ve…

jvisualvm安装Visual GC插件

给jdk自带的jvisualvm安装Visual GC插件&#xff0c;遇到Were sorry the java.net site has closed&#xff08;我们很抱歉java.net网站已经关闭&#xff09; 1、找到新的更新地址 visualvm新访问地址&#xff1a;https://visualvm.github.io/index.html 进入“Plugins”&…

来自阿里巴巴佛系安卓程序员的指南,专题解析

开头 中国互联网发展的这些年&#xff0c;如今90后程序员是中国程序员的主力军&#xff0c;互联网的热潮也让一批批00后蠢蠢欲动&#xff0c;尝试涌入互联网圈。 当程序员容易&#xff0c;当一个优秀的程序员需要不断学习&#xff0c;从初级程序员到高级程序员&#xff0c;从…

C#在WinForm中打开控制台显示

引用&#xff1a; namespace 测试使用 {public partial class Form1 : Form{[System.Runtime.InteropServices.DllImport("kernel32.dll", SetLastError true)][return: System.Runtime.InteropServices.MarshalAs(System.Runtime.InteropServices.UnmanagedType.Bo…

Redis+AQS

前言 对于java的单进程应用来说&#xff0c;存在资源竞争的场景可以使用synchronized关键字和Lock来对资源进行加锁&#xff0c;使整个操作具有原子性。但是对于多进程或者分布式的应用来说&#xff0c;上面提到的锁不共享&#xff0c;做不到互相通讯&#xff0c;所以就需要分…

Laravel 除了首页能正常访问,其它页面均404

在宝塔系统上通过一键源码配置laravel框架后&#xff0c;发现除了首页能够正常访问&#xff0c;其它的页面均返回的404&#xff0c;后经过比对已经网上查资料,发现是nginx的配置文件出了问题 1.找到配置文件 宝塔系统地址&#xff1a;/www/server/panel/vhost/nginx 2.在新建网…

disruptor 介绍

一、背景 1.来源 Disruptor是英国外汇交易公司LMAX开发的一个高性能队列&#xff0c;研发的初衷是解决内部的内存队列的延迟问题&#xff0c;而不是分布式队列。基于Disruptor开发的系统单线程能支撑每秒600万订单&#xff0c;2010年在QCon演讲后&#xff0c;获得了业界关注。…

算法题+JVM+自定义View,详细的Android学习指南

前言 想要成为一名优秀的Android开发&#xff0c;你需要一份完备的知识体系&#xff0c;在这里&#xff0c;让我们一起成长为自己所想的那样~。 学算法真的很痛苦&#xff0c;虽然大数据现在很火&#xff0c;但找到适合自己定位的职业也未尝不是一种合理选择。 投百度的经历非…

用过的前端插件合集

用过的前端插件合集 FontAwesome字体 Font Awesome详细用法参见上述站点的Examples。 SweetAlert系列 SweetAlertSweetAlert2SweetAlert 到 SweetAlert2 升级指南示例&#xff1a; 基本使用&#xff1a; swal("标题","内容","success);使用SweetAlert…

CAS和AQS

CAS 全称&#xff08;Compare And Swap&#xff09;,比较交换 Unsafe类是CAS的核心类&#xff0c;提供硬件级别的原子操作。 // 对象、对象的地址、预期值、修改值 public final native boolean compareAndSwapInt(Object var1, long var2, int var4, int var5);缺点&#xf…

系统盘点Android开发者必须掌握的知识点,全网疯传

最近在知乎上&#xff0c;有许多人在邀请我去回答“Android前景怎么样、是不是要凉了、是不是应该考虑要转行&#xff1f;”等一系列的问题。 想着可能有很多人都有这样的担心&#xff0c;于是就赶紧写篇文章&#xff0c;来跟你们谈下Android开发的前景到底怎么样&#xff1f;…

CountDownLatch的理解和使用 多线程同步器

CountDownLatch的理解和使用 在笔者想要了解Thrift时候&#xff0c;找到一个博主写的系统间通信技术的架构设计&#xff0c;在了解和学习的过程中遇到很多小问题和基础知识&#xff0c;自己还是不够清楚&#xff0c;就查询和总结下。 因为笔者也都是从网上找的一些资料&#…

数据库操作DDL

show database; 查看所有数据库 drop database db_name; 删除数据库 create database db_name;创建数据库 一个数据库对应一个文件夹 create database if not exists db_name; show warnings; 查看所有警告 show create databae db_name;查看创建的数据库 create database if n…

细数Android开发者的艰辛历程,已拿offer附真题解析

笼统来说&#xff0c;中年程序员容易被淘汰的原因其实不外乎三点。 1、输出能力已到顶点。这个人奋斗十来年了&#xff0c;依旧碌碌无为&#xff0c;很明显这人的天花板就这样了&#xff0c;说白了&#xff0c;天赋就这样。 2、适应能力越来越差。年纪大&#xff0c;有家庭&…

原子操作类AtomicInteger详解

为什么需要AtomicInteger原子操作类&#xff1f; 对于Java中的运算操作&#xff0c;例如自增或自减&#xff0c;若没有进行额外的同步操作&#xff0c;在多线程环境下就是线程不安全的。num解析为numnum1&#xff0c;明显&#xff0c;这个操作不具备原子性&#xff0c;多线程并…