-
FOUC in Safari (번역중)카테고리 없음 2011. 7. 14. 12:13
The FOUC Problem
Posted by Dave Hyatt on Friday, September 1st, 2006 at 10:59 pm
(translate by gyuseong)FOUC stands for Flash of Unstyled Content. This situation occurs whenever a Web browser ends up showing your Web page’s content without having any style information yet. It’s an interesting technical problem, because when/how a browser ends up committing the crime of FOUCing depends heavily on how the browser’s engine is architected and on interesting assumptions made by Web site authors when designing their sites.
When a browser loads a Web page, it first fetches an HTML file from the Web site. As that HTML file is downloaded from the Web site, the browser engine begins to feed the file’s contents to its HTML parser. The browser engine builds up a tree that represents the structure of the Web page (the DOM tree) as it encounters elements and nodes in the file. At some point while parsing, the engine is going to hit a stylesheet loading directive, either inside a <style> block or in a <link> element.
It is at this point that things get interesting. The browser engine now has a choice to make. It can either stall the parsing process while it waits for the remote stylesheet to load, or it can plunge bravely ahead. Let’s examine each option.Stall (멈춤)
If the browser engine stalls parsing then the processing of the entire page is now held up. Although the network layer (typically on another thread) can still provide data to the engine, this data basically queues up unprocessed. In effect the entire engine is doing nothing while it waits for this stylesheet to finish. Gecko has this behavior.
만약 브라우저 엔진이 파싱을 멈추면 전체 페이지의 처리과정은 멈추게 된다. 비록 네트워크 레이어 (보통 다른 쓰레드)는 아직 데이터를 엔진으로 공급할 수 있지만, 이 데이터들은 기본적으로 처리되지 않은 상태로 큐에 쌓이게 된다. 사실 모든 엔진은 이 스타일시트(making style sturct)가 끝나기를 기다리는 동안 아무것도 하지 않는다. Gecko엔진은 이렇게 동작한다.
One major disadvantage of this approach is that stylesheets and scripts cannot load in parallel. If a Web page has a link element followed by a script element, the engine won’t even begin loading the script element until it has completely loaded the stylesheet. The situation gets even worse if the stylesheet itself wants to import other stylesheets, since the engine will continue to stall until the entire stylesheet tree represented by the directive in the HTML file has loaded.
이런 접근방식의 중요한 불이익은 스타일시트와 스크립트를 병렬로딩할 수 없다는 것이다. 만약 웹페이지가 스크립트 요소 뒤에 링크(스타일시트링크) 요소를 가지고 있다면, 엔진은 스타일시트를 로딩할 때까지 스크립트 로딩을 시작하지 않을 것이다. 스타일시트 안에서 다른 스타일 시트를 불러오게 된다면 상황은 더 악화 될 수 있는데, 엔진은 로딩된 HTML파일 내 지시어에 의해 전체 스타일시트 트리가 불려지기 전까지 계속 멈추고 있을 것이다.This choice effectively makes stylesheets behave the same as scripts and effectively guarantees that no load parallelization will occur on anything in the head of the Web page. This sounds bad, so why do it? Well, the fundamental reason is that Web authors have essentially already written code that makes the assumption that style information accessed via scripts will be current.
이런 선택은 효과적으로 스타일시트가 스크립트와 같게 동작하도록 만들고 ~~~ (번역안됨).
For example, consider a script following a link element. If that script asks for the offsetHeight of an element, and the height was specified in the CSS file, Web page authors expect to get the right height. Even though this behavior is not part of any standard, Web designers have implicitly built the serial loading assumption into their scripts.How did this happen? Well, primarily because existing engines behaved this way, but also because of the synchronous nature of script elements. Because scripts already stall parsing, authors just made the natural assumption that stylesheets would stall parsing too.
Verdict: Thumbs down.
Don’t Stall
WebKit has the opposite behavior and will continue parsing a page even after a stylesheet directive has been encountered so that stylesheet and script loads can be parallelized. That way everything can be ready for display much earlier.
The problem with this behavior is what to do when a script attempts to access a property that involves having accurate layout/style information to answer. Shipping Safari’s current behavior when this happens is as follows: it will go ahead and lay out what it’s got even though it doesn’t have the stylesheet yet. It will also display it. This means you see FOUC whenever a script tries to access properties like scrollHeight or offsetWidth before the stylesheet has loaded.
We’ve improved things in the WebKit nightlies a bit so that at least the rendering of unstyled content will be suppressed. However we will still give back information that the page might consider to be incorrect, since the stylesheet that supplies the up-to-date information hasn’t loaded yet.
This approach ends up being superior in speed to the stalling approach in the case where no style/layout properties are accessed. However it ends up possibly being inferior in speed to the stalling approach if such a property is accessed, depending on how severe the contortions the engine has to perform between the unstyled rendering and the final styled rendering are.
Verdict: Thumbs down.
On-Demand Stall
The ideal technical solution to this problem involves a mixture of the two approaches, and that is on-demand stalling. In this approach the engine still parallelizes loads and doesn’t stall parsing. However if a layout or style property is accessed, the JavaScript engine suspends itself and awaits the load of any pending stylesheets. Once all the stylesheets have loaded, the script picks up where it left off.
The hurdle with this approach is that you have to be able to halt scripts mid-execution and resume them later. A technically inferior solution might be to just stall the execution of *all* scripts, but given the preponderance of script elements on most Web sites, that approach is really not much better than stalling all the time.
Verdict: Thumbs up.
Why does this matter?
The FOUC problem would normally be a minor occurrence. However with the advent of Google AdSense, FOUC has become a Safari epidemic. Because these Google ads not only execute inline script but access layout information that they often don’t even end up using in the page, the problem of FOUC is much more severe than it should be.
It is poor coding to access style/layout information repeatedly during the loading cycle of a page, because in order for the engine to supply you with an answer, it has to ensure that its layout and style are up to date. Think of every use of a property like offsetWidth as a bottleneck that stalls Web page display while the engine computes an accurate snapshot of the entire page just to give the script a correct answer. If a script accesses such properties repeatedly while making numerous other layout/style changes between accesses, it can cripple the load speed of a Web site in every modern browser.
http://www.webkit.org/blog/66/the-fouc-problem/