pxt-calliope/docs/js/variables.html

277 lines
70 KiB
HTML
Raw Normal View History

2017-08-19 17:16:35 +02:00
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml" xmlns:og="http://ogp.me/ns#" xmlns:fb="http://www.facebook.com/2008/fbml">
<head>
<meta charset="UTF-8">
<title>calliope mini - Blocks / Javascript editor - calliope</title>
<meta name="Description" content="A Blocks / JavaScript code editor for the calliope mini." />
<!-- include meta.html -->
<!-- This file is typically overriden by the target, with own Twitter
account etc. Most data is taken from the theme file though.
-->
<meta name="twitter:card" content="summary" />
<meta name="twitter:site" content="@mspxtio" />
<meta name="twitter:title" content="calliope mini - Blocks / Javascript editor - calliope" />
<meta name="twitter:description" content="A Blocks / JavaScript code editor for the calliope mini." />
<meta name="twitter:image" content="
<meta property="og:title" content="calliope mini - Blocks / Javascript editor - calliope" />
<meta property="og:site_name" content="PXT" />
<meta property="og:type" content="website" />
<meta property="og:description" content="A Blocks / JavaScript code editor for the calliope mini." />
<meta property="fb:app_id" content="" />
<meta property="og:image" content="
<link rel="apple-touch-icon" href="
<link rel="icon" type="image/png" href="
<link rel="shortcut icon" href="
<meta name="theme-color" content="#249899">
<!-- end include meta.html -->
<!-- include head.html -->
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1" />
<meta name="viewport" content="width=device-width,height=device-height,user-scalable=no,initial-scale=1.0,maximum-scale=1.0,minimum-scale=1.0">
<link rel="stylesheet" href="/pxt-calliope/semantic.css" />
<link rel="stylesheet" href="/pxt-calliope/docfiles/style.css" />
<link rel="stylesheet" href="/pxt-calliope/docfiles/vs.css" />
<link rel="stylesheet" href="/pxt-calliope/docfiles/target.css" />
<link rel="stylesheet" href="/pxt-calliope/docfiles/fork.css" />
<script src="/pxt-calliope/jquery.js"></script>
<script src="/pxt-calliope/semantic.js"></script>
<script src="/pxt-calliope/lzma/lzma_worker-min.js"></script>
<script src="/pxt-calliope/embed.js" type="text/javascript"></script>
<script src="/pxt-calliope/docfiles/docs.js" type="text/javascript"></script>
<script src="/pxt-calliope/docfiles/target.js" type="text/javascript"></script>
<script src="/pxt-calliope/docfiles/fork.js" type="text/javascript"></script>
<style>
.ui.accent { color: #249899; }
.ui.inverted.accent { background: #249899; }
</style>
<script>
$(document).ready(function() {
// patch youtube in semantic
if ($ && $.fn && $.fn.embed && $.fn.embed.settings && $.fn.embed.settings.sources && $.fn.embed.settings.sources.youtube) {
$.fn.embed.settings.sources.youtube.url = '//www.youtube.com/embed/{id}?rel=0'
}
$('.ui.embed').embed();
})
</script>
<!-- end include head.html -->
</head>
<body id='root' class='root'>
<!-- include header.html -->
<div class="ui fixed accent inverted menu">
<div class="ui container">
<a href="/pxt-calliope/docs/.html" class="header item">
<img class="ui mini image" src="
</a>
<!-- Menu -->
</div>
</div>
<!-- end include header.html -->
<div id="docs" class="ui main container mainbody">
<div class="ui breadcrumb">
<a class=" section"
href="/pxt-calliope/docs/js.html">js</a><i class="right chevron icon divider"></i><a class="active section"
href="/pxt-calliope/docs/js/variables.md.html">variables.md</a>
</div>
<div class="ui text">
<h1 id="variable-declarations">Variable Declarations</h1>
<p>Declaring a variable in JavaScript has always traditionally been done with the <code>var</code> keyword.</p>
<pre><code class="lang-typescript"><span class="hljs-keyword">var</span> a = <span class="hljs-number">10</span>;</code></pre>
<p>The <code>var</code> construct has some <a href="http://www.typescriptlang.org/docs/handbook/variable-declarations.html">problems</a>,
which is why <code>let</code> statements were introduced. Apart from the keyword used, <code>let</code> statements are written
the same way <code>var</code> statements are.</p>
<pre><code class="lang-typescript"><span class="hljs-keyword">let</span> a = <span class="hljs-number">10</span>;</code></pre>
<p>The key difference is not in the syntax, but in the semantics, which well now dive into.</p>
<h2 id="block-scoping">Block-scoping</h2>
<p>When a variable is declared using <code>let</code>, it uses what some call <em>lexical-scoping</em> or <em>block-scoping</em>.
Unlike variables declared with <code>var</code> whose scopes leak out to their containing function,
block-scoped variables are not visible outside of their nearest containing block or <code>for</code>-loop.</p>
<pre><code class="lang-typescript"><span class="hljs-function"><span class="hljs-keyword">function</span> <span class="hljs-title">f</span>(<span class="hljs-params">input: <span class="hljs-built_in">boolean</span></span>) </span>{
<span class="hljs-keyword">let</span> a = <span class="hljs-number">100</span>;
<span class="hljs-keyword">if</span> (input) {
<span class="hljs-comment">// Still okay to reference 'a'</span>
<span class="hljs-keyword">let</span> b = a + <span class="hljs-number">1</span>;
<span class="hljs-keyword">return</span> b;
}
<span class="hljs-comment">// Error: 'b' doesn't exist here</span>
<span class="hljs-keyword">return</span> b;
}</code></pre>
<p>Here, we have two local variables <code>a</code> and <code>b</code>.
<code>a</code>s scope is limited to the body of <code>f</code> while <code>b</code>s scope is limited to the containing <code>if</code> statements block.</p>
<p>Another property of block-scoped variables is that they cant be read or written to before theyre actually declared.
While these variables are “present” throughout their scope, all points up until their declaration are part of their <em>temporal dead zone</em>.
This is just a sophisticated way of saying you cant access them before the <code>let</code> statement, and luckily TypeScript will let you know that.</p>
<pre><code class="lang-typescript-ignore">a++; <span class="hljs-comment">// illegal to use 'a' before it's declared;</span>
<span class="hljs-keyword">let</span> a;</code></pre>
<h2 id="re-declarations">Re-declarations</h2>
<p>With <code>var</code> declarations, it doesnt matter how many times you declare your variables, you just get one:</p>
<pre><code class="lang-typescript"><span class="hljs-keyword">var</span> x = <span class="hljs-number">10</span>;
<span class="hljs-keyword">var</span> x = <span class="hljs-number">20</span>;</code></pre>
<p>In the above example, all declarations of <code>x</code> actually refer to the <em>same</em> <code>x</code>, and this is perfectly valid.
This often ends up being a source of bugs. Thankfully, <code>let</code> declarations are not as forgiving.</p>
<pre><code class="lang-typescript"><span class="hljs-keyword">let</span> x = <span class="hljs-number">10</span>;
<span class="hljs-keyword">let</span> x = <span class="hljs-number">20</span>; <span class="hljs-comment">// error: can't re-declare 'x' in the same scope</span></code></pre>
<h2 id="shadowing">Shadowing</h2>
<p>The act of introducing a new name in a more deeply nested scope is called <em>shadowing</em>.
It is a bit of a double-edged sword in that it can introduce certain bugs on its own in the
event of accidental shadowing, while also preventing certain bugs.
For instance, imagine a <code>sumMatrix</code> function using <code>let</code> variables.</p>
<pre><code class="lang-typescript"><span class="hljs-function"><span class="hljs-keyword">function</span> <span class="hljs-title">sumMatrix</span>(<span class="hljs-params">matrix: <span class="hljs-built_in">number</span>[][]</span>) </span>{
<span class="hljs-keyword">let</span> sum = <span class="hljs-number">0</span>;
<span class="hljs-keyword">for</span> (<span class="hljs-keyword">let</span> i = <span class="hljs-number">0</span>; i &lt; matrix.length; i++) {
<span class="hljs-keyword">var</span> currentRow = matrix[i];
<span class="hljs-keyword">for</span> (<span class="hljs-keyword">let</span> i = <span class="hljs-number">0</span>; i &lt; currentRow.length; i++) {
sum += currentRow[i];
}
}
<span class="hljs-keyword">return</span> sum;
}</code></pre>
<p>This version of the loop will actually perform the summation correctly because the inner loops <code>i</code> shadows <code>i</code> from the outer loop.
Shadowing should <em>usually</em> be avoided in the interest of write clearer code, such as</p>
<pre><code class="lang-typescript"><span class="hljs-function"><span class="hljs-keyword">function</span> <span class="hljs-title">sumMatrix</span>(<span class="hljs-params">matrix: <span class="hljs-built_in">number</span>[][]</span>) </span>{
<span class="hljs-keyword">let</span> sum = <span class="hljs-number">0</span>;
<span class="hljs-keyword">for</span> (<span class="hljs-keyword">let</span> i = <span class="hljs-number">0</span>; i &lt; matrix.length; i++) {
<span class="hljs-keyword">var</span> currentRow = matrix[i];
<span class="hljs-keyword">for</span> (<span class="hljs-keyword">let</span> j = <span class="hljs-number">0</span>; j &lt; currentRow.length; j++) {
sum += currentRow[j];
}
}
<span class="hljs-keyword">return</span> sum;
}</code></pre>
<p>While there are some scenarios where it may be fitting to take advantage of it, you should use your best judgement.</p>
<h1 id="-const-declarations"><code>const</code> declarations</h1>
<p><code>const</code> declarations are another way of declaring variables.</p>
<pre><code class="lang-typescript"><span class="hljs-keyword">const</span> numLivesForCat = <span class="hljs-number">9</span>;</code></pre>
<p>They are like <code>let</code> declarations but, as their name implies, their value cannot be changed once they are bound.
In other words, they have the same scoping rules as <code>let</code>, but you cant re-assign to them.</p>
<a class="ui primary button" href="/pxt-calliope/docs/js/operators.html">
<p>NEXT: Operators</p>
</a>
</div>
<p style="margin-top:1em"><a href="https://github.com/Microsoft/pxt-calliope/blob/master/docs/docs/js/variables.md"><i class="write icon"></i>Edit this page on GitHub</a></p>
</div>
<!-- include footer.html -->
<div class="ui inverted accent vertical footer segment">
<div class="ui center aligned container">
<div class="ui horizontal inverted small divided link list">
<a class="item" href="https://www.pxt.io/" title="Programming Experience Toolkit">Powered by PXT</a>
<a class="item" href="https://www.pxt.io/contact">Contact Us</a>
<a class="item" href="https://www.pxt.io/privacy">Privacy &amp; Cookies</a>
<a class="item" href="https://www.pxt.io/legal">Terms Of Use</a>
<a class="item" href="https://www.pxt.io/trademarks">Trademarks</a>
<div class="item">© 2016 Microsoft</div>
<!-- we need to force the browser to load this font -->
<div style='font-family: Icons; color: #010101;' aria-hidden="true">.</div>
</div>
<a class="item" href="https://www.microsoft.com/"><img class="ui centered image" src="https://az851932.vo.msecnd.net/pub/pmapoirq" /></a>
</div>
</div>
<!-- end include footer.html -->
<!-- include macros.html -->
<!-- macro button -->
<!-- macro vimeo -->
<!-- macro youtube -->
<!-- macro section -->
<!-- macro hide -->
<!-- macro avatar -->
<!-- macro hint -->
<!-- wrapped around ordinary content -->
<!-- macro main-container -->
<!-- used for 'column' box - they are collected and wrapped in 'column-container' -->
<!-- macro column -->
<!-- macro column-container -->
<!-- Menu on the top of the page -->
<!-- macro item -->
<!-- macro divider -->
<!-- macro top-dropdown -->
<!-- macro inner-dropdown -->
<!-- end include macros.html -->
<!-- include tracking.html -->
<script type="text/javascript">
var appInsights=window.appInsights||function(config){
function i(config){t[config]=function(){var i=arguments;t.queue.push(function(){t[config].apply(t,i)})}}var t={config:config},u=document,e=window,o="script",s="AuthenticatedUserContext",h="start",c="stop",l="Track",a=l+"Event",v=l+"Page",y=u.createElement(o),r,f;y.src=config.url||"https://az416426.vo.msecnd.net/scripts/a/ai.0.js";u.getElementsByTagName(o)[0].parentNode.appendChild(y);try{t.cookie=u.cookie}catch(p){}for(t.queue=[],t.version="1.0",r=["Event","Exception","Metric","PageView","Trace","Dependency"];r.length;)i("track"+r.pop());return i("set"+s),i("clear"+s),i(h+a),i(c+a),i(h+v),i(c+v),i("flush"),config.disableExceptionTracking||(r="onerror",i("_"+r),f=e[r],e[r]=function(config,i,u,e,o){var s=f&&f(config,i,u,e,o);return s!==!0&&t["_"+r](config,i,u,e,o),s}),t
}({
instrumentationKey:"9801ed01-c40f-46ec-aa40-2a1742a9e71c",
disableAjaxTracking: true,
overridePageViewDuration: false,
disableExceptionTracking: true,
isCookieUseDisabled: true,
isStorageUseDisabled: true
});
window.appInsights=appInsights;
appInsights.queue.push(function () {
appInsights.context.addTelemetryInitializer(function (envelope) {
if (typeof pxtConfig === "undefined") return;
var telemetryItem = envelope.data.baseData;
telemetryItem.properties = telemetryItem.properties || {};
telemetryItem.properties["target"] = pxtConfig.targetId;
telemetryItem.properties["version"] = pxtConfig.targetVersion;
telemetryItem.properties["stage"] = (pxtConfig.relprefix || "/--").replace(/[^a-z]/ig, '')
if (typeof window !== "undefined" && window.location && /[?&]electron=1/i.test(window.location.href))
telemetryItem.properties["electron"] = 1;
});
});
appInsights.trackPageView();
</script>
<!-- end include tracking.html -->
</body>
</html>