<html>

<head>
<meta http-equiv="Content-Language" content="en-us">
<meta http-equiv="Content-Type" content="text/html; charset=windows-1252">
<title>Perl Style</title>
<link rel="stylesheet" type="text/css" href="../../CSS/main.css">
</head>

<body>

<basefont face="Times" size="3">
<table border="0" width="100%" id="table1" cellspacing="1" cellpadding="4"  >
<tr>
<td rowspan="2" width="40%">

<p align="center"><img border="0" src="/Images/splogo.gif" width="45"  ><font face="Bernard MT Condensed" color="#FF0000" size="7">
<font color="#FF0000"><a href="/index.shtml">Softpanorama</a></font></font></p>

<center>

<font size="2" face="Arial"><b>May the source be with you, but remember the <a href="/SE/kiss.shtml">KISS principle</a> ;-) </b></font></center>
</td>
<td align="center">

<b><a href="/index.shtml"><font size="2">Home</font></a></b></td>
<td align="center">

<b><a href="/switchboard.shtml"><font size="2">Switchboard</font></a></b></td>
<td align="center">

<b><a href="/Admin/index.shtml"><font size="2">Unix Administration</font></a></b></td>
<td align="center">

<b><a href="/Commercial_linuxes/RHEL/index.shtml"><font size="2">Red Hat</font></a></b></td>
<td align="center">

<b><a href="/Net/index.shtml"><font size="2">TCP/IP Networks</font></a></b></td>
<td align="center">

<b><a href="/Skeptics/Political_skeptic/Neoliberalism/index.shtml"><font size="2">Neoliberalism</font></a></b></td>
<td align="center">

<b><a href="/Social/Toxic_managers/index.shtml"><font size="2">Toxic Managers</font></a></b></td>
</tr>
<tr>
<td colspan="7" align="center">

<b><font size="2">(slightly skeptical) Educational society promoting &quot;<a href="/Back_to_basics/index.shtml">Back to basics</a>&quot; movement against IT overcomplexity 
and&nbsp; <a href="Commercial_linuxes/RHEL/index.shtml">bastardization of classic Unix</a></font></b></td>
</tr>
</table>
<hr width="100%" noshade size="5" color="#FF0000">
<center>
<script async src="https://pagead2.googlesyndication.com/pagead/js/adsbygoogle.js"></script>
<!-- Top_banner2021 -->
<ins class="adsbygoogle"
     style="display:block"
     data-ad-client="ca-pub-4031247137266443"
     data-ad-slot="3389635737"
     data-ad-format="auto"
     data-full-width-responsive="true"></ins>
<script>
     (adsbygoogle = window.adsbygoogle || []).push({});
</script></center>


<h1>Perl Style: defensive programming in Perl</h1>

<table border="1" width="100%" bgcolor="#FFFFCC">
   <tr>
      <td width="14%" align="center"><a href="#News">News</a></td>
      <td width="14%" align="center"><b><a target="_blank" href="../../SE/programming_style.shtml">Programming Style</a></b></td>
      <td width="14%" align="center"><a href="#Recommended_Links">Recommended Links</a></td>
      <td width="14%" align="center"><b><a href="perl_debugging.shtml">Perl Debugging</a></b></td>
      <td width="14%" align="center"><b><a href="Debugging/perl_error_checklist.shtml">Perl Error Checklist</a></b></td>
      <td width="15%" align="center"><b><a href="perl_in_command_line.shtml">Perl as a command line utility tool</a></b></td>
      <td width="15%" align="center"><b><a href="perl_programming_environment.shtml">Perl Programming Environment</a></b></td>
   </tr>
   <tr>
      <td width="14%" align="center" height="25"><b><a href="pod_documentation.shtml">Perl POD as literate programming tool</a></b></td>
      <td width="14%" bgcolor="#FFFFCC" align="center" height="25"><b><a href="../../SE/literate_programming.shtml">Literate Programming</a></b></td>
      <td width="14%" bgcolor="#FFFFCC" align="center" height="25"><b><a href="../../SE/defensive_programming.shtml">Defensive programming</a></b></td>
      <td width="14%" bgcolor="#FFFFCC" align="center" height="25"><b><a href="../../SE/conway_law.shtml">Conway Law</a></b></td>
      <td width="14%" bgcolor="#FFFFCC" align="center" height="25"><b><a href="../../SE/brooks_law.shtml">Brooks law</a></b></td>
      <td width="15%" bgcolor="#FFFFCC" align="center" height="25"><b><a href="../../SE/conceptual_integrity.shtml">Conceptual Integrity</a></b></td>
      <td width="15%" bgcolor="#FFFFCC" align="center" height="25"><b><a href="../../SE/programming_style.shtml">Programming style</a></b></td>
   </tr>
   <tr>
      <td width="14%" align="center"><b><a href="../../Utilities/beautifiers.shtml">Pretty-printers</a></b></td>
      <td width="14%" bgcolor="#FFFFCC" align="center"><b><a href="../../Admin/admin_horror_stories.shtml">Sysadmin Horror Stories</a></b></td>
      <td width="14%" bgcolor="#FFFFCC" align="center"><b><a target="_blank" href="history_and_philosophy.shtml">History and philosophy</a></b></td>
      <td width="14%" bgcolor="#FFFFCC" align="center"><b><a target="_blank" href="perl_tips.shtml">Tips</a></b></td>
      <td width="14%" bgcolor="#FFFFCC" align="center"><b><a href="../../Lang/quotes.shtml">Language Design and Programming Quotes</a></b></td>
      <td width="15%" bgcolor="#FFFFCC" align="center"><a target="_blank" href="../../People/Wall/perl_related_humor.shtml">Humor</a></td>
      <td width="15%" bgcolor="#FFFFCC" align="center"><a href="#Etc">Etc</a></td>
   </tr>
</table>


<table border="0" width="178" height="620" align="right" cellspacing="4" bgcolor="#FFFFFF">
   <tr>
      <td>

      <table border="1" width="174" height="616" align="center" bgcolor="#FF0000">
         <tr>
            <td>
<script async src="//pagead2.googlesyndication.com/pagead/js/adsbygoogle.js"></script>
<!-- Upper skyscaper -->
<ins class="adsbygoogle"
     style="display:inline-block;width:160px;height:600px"
     data-ad-client="ca-pub-4031247137266443"
     data-ad-slot="0371843916"></ins>
<script>
(adsbygoogle = window.adsbygoogle || []).push({});
</script>
            </td>
         </tr>
      </table></td>
   </tr>
</table>
<ul>

   <li><b><a href="#Introduction">Introduction</a></b></li>

   <li><b><a href="#Defensive_programming">Defensive programming</a></b></li>

   <li><b><a href="#General_recommendation_from_The_Elements_of_Programming_Style">General recommendation from The Elements of Programming 
   Style</a></b></li>

   <li><b><a href="#Lexical_style">Lexical style</a></b></li>

   <li><b><a href="#Programming_style">Programming style</a></b><ul>

      <li><a href="#Avoid_dogmatism_and_religious__zealotry">Avoid dogmatism and religious&nbsp; zealotry</a></li>
   
      <li><a href="#Do_not_imitate_complexity_junkies">Do not imitate complexity junkies</a></li>
   </ul>
   </li>

</ul>
<hr noshade color="#FF0000" width="33%" size="7">

<h3><a name="Introduction">Introduction</a></h3>

<p>Perl gives you enough rope to hang yourself, so it is prudent to practice defensive programming. </p>

<p>Programming style is not a dogma and textbooks with the word &#39;style&quot; on the cover are better to be avoided ;-). And they key feature 
of style is consistency.</p>

<p>Wikipedia&nbsp; defines&nbsp; <a target="_blank" href="https://en.wikipedia.org/wiki/Programming_style">Programming style</a> rather 
narrowly, emphasizing&nbsp; lexical style but in reality the concept is broader then that</p>
<blockquote>

   <p><b>Programming style</b> is a set of rules or guidelines used when writing the
   <a title="Source code" href="https://en.wikipedia.org/wiki/Source_code">source code</a> for a
   <a title="Computer program" href="https://en.wikipedia.org/wiki/Computer_program">computer program</a>. It is often claimed that following 
   a particular programming style will help <a title="Programmer" href="https://en.wikipedia.org/wiki/Programmer">programmers</a> to 
   read and understand source code conforming to the style, and help to avoid introducing errors.</p>

   <p>A classic work on the subject was <i>
   <a title="The Elements of Programming Style (book)" href="https://en.wikipedia.org/wiki/The_Elements_of_Programming_Style_(book)">
   The Elements of Programming Style</a></i>, written in the 1970s, and illustrated with examples from the
   <a title="Fortran" href="https://en.wikipedia.org/wiki/Fortran">Fortran</a> and
   <a title="PL/I" href="https://en.wikipedia.org/wiki/PL/I">PL/I</a> languages prevalent at the time.</p>

   <p>The programming style used in a particular program may be derived from the <b>coding standards</b> or <b>
   <a title="Code conventions" href="https://en.wikipedia.org/wiki/Code_conventions">code conventions</a></b> of a company or other computing 
   organization, as well as the preferences of the author of the code. Programming styles are often designed for a specific
   <a title="Programming language" href="https://en.wikipedia.org/wiki/Programming_language">programming language</a> (or language family): 
   style considered good in <a title="C (programming language)" href="https://en.wikipedia.org/wiki/C_(programming_language)">C</a> source 
   code may not be appropriate for
   <a title="BASIC programming language" href="https://en.wikipedia.org/wiki/BASIC_programming_language">BASIC</a> source code, and so 
   on. However, some rules are commonly applied to many languages.</p>
</blockquote>
We will distinguish three elements of style:<ul>

   <li><a href="#Defensive_programming">Defensive programming related elements of style</a>. </li>

   <li><a href="#Lexical_style">Lexical style.</a> lexical formatting now is often usually enforced by pretty printer, but within those 
   constrains you can </li>

   <li><a href="#Programming_style">Programming style.</a> This is pretty individual thing, but excessive &quot;cleverness&quot; can byte you 
   later. </li>
</ul>

<h3><a name="Defensive_programming">Defensive programming</a></h3>

<p>One important trend is programming style is so called <em>defensive programming.</em> </p>

<p>For example the first thing you need to do when you use somebody else code, the first thing to do is read it. That will get you general 
idea if it is OK to use it or it is an overkill. Often&nbsp; it is an overkill. In this case it might be better to create a simpler 
derivative. </p>

<p>Also with SPAN modules, each time release new version you need to diff it. As this advice is difficult to follow, overuse of CPAN 
modules can lower your quality of code. Unmaintained SPAN modules have another danger -- the fall behind changes in OS, and you become 
their maintainer. </p>

<p>Number of bugs in your scripts depends on the style you consciously or unconsciously adopted. Also each language has typical errors 
which need to be consistently checked for as they crop in again and again. See <b><a href="Debugging/perl_error_checklist.shtml">Perl 
Error Checklist</a></b></p>

<p>As an example of early attempt to formulate some principles of defensive programming style we can list&nbsp; Tom Christiansen recommendations 
(Jan 1, 1998) for Perl language Perl does not have strict typing of variables and, by default, does not&nbsp; require any declaration 
of variables, creating potential for misspelled variables slipping into production version of the program. Unless you use <tt>strict</tt> 
pragma -- the use the latter became standard in modern Perl). While they are more then 20 years old they are still relevant:&nbsp;&nbsp;
</p>
<ul>

   <li><tt>use strict </tt></li>

   <li><tt>#!/usr/bin/perl -w </tt></li>

   <li><em>Check all syscall return values, printing</em> $! </li>

   <li><em>Watch for external program failures in</em> $? </li>

   <li>Check $@ after <code>eval&quot;&quot;</code> or <code>s///ee</code>. </li>

   <li>[Use] Parameter asserts </li>

   <li><tt>#!/usr/bin/perl -T</tt>&nbsp; (taint mode in which Perl distrust any data from outside world, see below)</li>

   <li>Always have an <code>else</code> after a chain of <code>elsif</code>s </li>

   <li>Put commas at the end of lists to so your program won&#39;t break if someone inserts another item at the end of the list.</li>
</ul>

<h3><a name="General_recommendation_from_The_Elements_of_Programming_Style">General recommendation from The Elements of Programming 
Style</a></h3>

<p style="margin: 0.5em 0px; color: rgb(32, 33, 34); font-family: sans-serif; font-size: 14px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">
Its lessons are summarized at the end of each section in
<a title="Aphorism" style="text-decoration: none; color: rgb(11, 0, 128); background: none" href="https://en.wikipedia.org/wiki/Aphorism">
pithy maxims</a>, such as &quot;Let the machine do the dirty work&quot; (<a href="https://en.wikipedia.org/wiki/The_Elements_of_Programming_Style">The 
Elements of Programming Style - Wikipedia</a></p>
<ol style="list-style-image: none; color: rgb(32, 33, 34); font-family: sans-serif; font-size: 14px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; text-decoration-style: initial; text-decoration-color: initial; margin-left: 3.2em; margin-right: 0px; margin-top: 0.3em; margin-bottom: 0px; padding: 0px; background-color: rgb(255, 255, 255)">

   <li style="margin-bottom: 0.1em;">Write clearly — don&#39;t be too clever.</li>

   <li style="margin-bottom: 0.1em;">Say what you mean, simply and directly.</li>

   <li style="margin-bottom: 0.1em;">Use library functions whenever feasible.</li>

   <li style="margin-bottom: 0.1em;">Avoid too many temporary variables.</li>

   <li style="margin-bottom: 0.1em;">Write clearly — don&#39;t sacrifice clarity for efficiency.</li>

   <li style="margin-bottom: 0.1em;">Let the machine do the dirty work.</li>

   <li style="margin-bottom: 0.1em;">Replace repetitive expressions by calls to common functions.</li>

   <li style="margin-bottom: 0.1em;">Parenthesize to avoid ambiguity.</li>

   <li style="margin-bottom: 0.1em;">Choose variable names that won&#39;t be confused.</li>

   <li style="margin-bottom: 0.1em;">Avoid unnecessary branches.</li>

   <li style="margin-bottom: 0.1em;">If a logical expression is hard to understand, try transforming it.</li>

   <li style="margin-bottom: 0.1em;">Choose a data representation that makes the program simple.</li>

   <li style="margin-bottom: 0.1em;">Write first in easy-to-understand pseudo language; then translate into whatever language you have 
   to use.</li>

   <li style="margin-bottom: 0.1em;">Modularize. Use procedures and functions.</li>

   <li style="margin-bottom: 0.1em;">Avoid gotos completely if you can keep the program readable.</li>

   <li style="margin-bottom: 0.1em;">Don&#39;t patch bad code — rewrite it.</li>

   <li style="margin-bottom: 0.1em;">Write and test a big program in small pieces.</li>

   <li style="margin-bottom: 0.1em;">Use recursive procedures for recursively-defined data structures.</li>

   <li style="margin-bottom: 0.1em;">Test input for plausibility and validity.</li>

   <li style="margin-bottom: 0.1em;">Make sure input doesn&#39;t violate the limits of the program.</li>

   <li style="margin-bottom: 0.1em;">Terminate input by end-of-file marker, not by count.</li>

   <li style="margin-bottom: 0.1em;">Identify bad input; recover if possible.</li>

   <li style="margin-bottom: 0.1em;">Make input easy to prepare and output self-explanatory.</li>

   <li style="margin-bottom: 0.1em;">Use uniform input formats.</li>

   <li style="margin-bottom: 0.1em;">Make input easy to proofread.</li>

   <li style="margin-bottom: 0.1em;">Use self-identifying input. Allow defaults. Echo both on output.</li>

   <li style="margin-bottom: 0.1em;">Make sure all variables are initialized before use.</li>

   <li style="margin-bottom: 0.1em;">Don&#39;t stop at one bug.</li>

   <li style="margin-bottom: 0.1em;">Use debugging compilers.</li>

   <li style="margin-bottom: 0.1em;">Watch out for off-by-one errors.</li>

   <li style="margin-bottom: 0.1em;">Take care to branch the right way on equality.</li>

   <li style="margin-bottom: 0.1em;">Be careful if a loop exits to the same place from the middle and the bottom.</li>

   <li style="margin-bottom: 0.1em;">Make sure your code does &quot;nothing&quot; gracefully.</li>

   <li style="margin-bottom: 0.1em;">Test programs at their boundary values.</li>

   <li style="margin-bottom: 0.1em;">Check some answers by hand.</li>

   <li style="margin-bottom: 0.1em;">10.0 times 0.1 is hardly ever 1.0.</li>

   <li style="margin-bottom: 0.1em;">7/8 is zero while 7.0/8.0 is not zero.</li>

   <li style="margin-bottom: 0.1em;">Don&#39;t compare floating point numbers solely for equality.</li>

   <li style="margin-bottom: 0.1em;">Make it right before you make it faster.</li>

   <li style="margin-bottom: 0.1em;">Make it fail-safe before you make it faster.</li>

   <li style="margin-bottom: 0.1em;">Make it clear before you make it faster.</li>

   <li style="margin-bottom: 0.1em;">Don&#39;t sacrifice clarity for small gains in efficiency.</li>

   <li style="margin-bottom: 0.1em;">Let your compiler do the simple optimizations.</li>

   <li style="margin-bottom: 0.1em;">Don&#39;t strain to re-use code; reorganize instead.</li>

   <li style="margin-bottom: 0.1em;">Make sure special cases are truly special.</li>

   <li style="margin-bottom: 0.1em;">Keep it simple to make it faster.</li>

   <li style="margin-bottom: 0.1em;">Don&#39;t diddle code to make it faster — find a better algorithm.</li>

   <li style="margin-bottom: 0.1em;">Instrument your programs. Measure before making efficiency changes.</li>

   <li style="margin-bottom: 0.1em;">Make sure comments and code agree.</li>

   <li style="margin-bottom: 0.1em;">Don&#39;t just echo the code with comments — make every comment count.</li>

   <li style="margin-bottom: 0.1em;">Don&#39;t comment bad code — rewrite it.</li>

   <li style="margin-bottom: 0.1em;">Use variable names that mean something.</li>

   <li style="margin-bottom: 0.1em;">Use statement labels that mean something.</li>

   <li style="margin-bottom: 0.1em;">Format a program to help the reader understand it.</li>

   <li style="margin-bottom: 0.1em;">Document your data layouts.</li>

   <li style="margin-bottom: 0.1em;">Don&#39;t over-comment</li>
</ol>

<h3><a name="Lexical_style">Lexical style</a></h3>

<p>The key idea of proper lexical style is make spotting typical lexical errors more quickly. There are several types of errors rampant 
in C-style languages that are diagnosed easier if the program source is <a href="../../Utilities/beautifiers.shtml">pretty-printed</a>. 
Among them:</p>
<ul>

   <li>Missing closing curve bracket</li>

   <li>Unclosed single quotes of double quoted literal</li>

   <li>Missing regular round bracket in expression</li>
</ul>

<p>Perl inherited C-language design decisions to use curvy brackets for blocks and that creates great problems with determining which 
block a particular curvy bracket closes. Indenting blocks via pretty-printer makes this more apparent. Which means that a good pretty 
printer is must for Perl.&nbsp; Similar wart exist due to &quot;overuse&quot; of &quot;regular&quot; round brackets :</p>

<pre>if ($tar=substr($text,1,3) {
# <font color="#FF0000">In the line above ) - bracket is missing  </font>
... 
}</pre>
Here the fact that one closing bracket&nbsp; is missing is not immediately apparent and Perl diagnostic is typically simply horrible 
(aka misleading) as for this type of errors. In a way in this area designers of Perl interpreters not only failed to make the progress 
with the level of compilers that existed in early 70th, but managed to make a step back ;-). Making spaces for surrounding bracket makes 
this situation better, but not by much:

<pre>if ( $tar=substr($text,1,3) ) {
... 
}</pre>
The key requirement to lexical style is than it should be consistent. It is better achieved using automatic pretty-printing programs. 
So they have great value for easy spotting the types of errors mentioned above and generally improve readability of programs.&nbsp;&nbsp;

<p>That does not mean that each type of lexical errors are addressed equally well. Missing curve bracket problem is addressed very well 
indeed. Bu the problem with missing round brackets need some additional efforts to detect and solve. Here there is a difference between 
a good pretty-printer and an excellent one. As for literals this problem is now solved on editor level as syntax highlighting makes 
this error apparent. Also better editors change the font and color of matching bracket when you an opening or closing bracket.&nbsp;
</p>

<p>Still the problem of&nbsp; &quot;unclosed&quot; single of double quote is badly diagnosed on the interpreter level. This handling can be improved. 
For example many years ago PL/1 pioneered a special parameter that represented a max length of the literals in the program. Other languages 
provide pragma that makes literals limited to a single line (actually concatenation of literals can be done at compile time so this 
solution makes more sense that commonly accepted. None of this solutions are available for Perl as of Perl 5.10. Here a good pretty 
printing program can definitely help as it can find and mark lines with unbalanced quotes quite easily. Perl lint is another good option.
</p>

<p>One interesting aspect of lexical style is that a better style makes programs less verbose, in a sense that the program occupies 
less lines, while not compromising readability. In this sense putting an opening curly bracket on the same line as the keyword and using 
&quot;cuddled else style&quot; (&nbsp;&nbsp; <tt>} else {</tt> ) are more compact and equally readable in comparison with alternatives. For example:</p>

<p><b>5 lines:</b></p>

<pre>if ($a&gt;$b) {
   $max=$a;
} else {
   $max=$b;
}</pre>

<p><b>8 lines: </b>&nbsp;</p>

<pre>if ($a&gt;$b)
{
   $max=$a;
}
else
{
   $max=$b;
}</pre>
This example demonstrates certain, desirable from my point of view, elements of Perl lexical style with the following characteristics:
<ul>

   <li>Three space indents.&nbsp; </li>

   <li>Blanks in stead of tabs for nesting.</li>

   <li>Cuddled else: <tt>} else {</tt> </li>

   <li>Opening curly on the same line as the keyword it belongs to </li>

   <li>Closing bracket should be put vertically aligned with that keyword \</li>

   <li>No comments after opening curvy bracket</li>
</ul>

<p>The only exception that I would consider worthwhile is to use separate line for opening curvy bracket in procedures to mark procedures 
as different entities then regular blocks</p>

<pre>sub test 
{ 

}</pre>

<p>Another important element of lexical style is that <font color="#FF0000"><i><b>you should indent with spaces, not tabs.</b></i></font>&nbsp; 
The problem is that different editors treat tabs differently and often have different default tab settings. That &quot;mangles&quot; your layout 
if you open the program in a new editor.&nbsp; </p>

<h3><a name="Programming_style">Programming style</a></h3>

<p>Programming style is set of rules as for using language constructions. Pioneering book in this respect the book
<a target="_blank" href="https://www.amazon.com/Elements-Programming-Style-Brian-Kernighan/dp/0070342075/ref=sr_1_1?ie=UTF8&qid=1342218846&sr=8-1&keywords=programming+style">
The Elements of Programming Style</a> by
<a target="_blank" href="https://www.amazon.com/Elements-Programming-Style-Brian-Kernighan/dp/0070342075/ref=sr_1_1?ie=UTF8&qid=1342218846&sr=8-1&keywords=programming+style">
Brian W. Kernighan and P. J. Plauger</a> published in 1978. </p>

<p>In reality style recommendations are far from being absolute. Much depends on tools and language used.&nbsp; Some extremely like 
structured programming or verification movement while containing some useful bits more distract from writing a good program then help. 
Like Knuth noted the weakness of structure programming is that it emphasize absence of goto, not the presence of structure.&nbsp;
</p>

<p>Style can easily become a religious issue. And that&#39;s a big big danger, as fanatics poison everything they touch. Like Rob Pike aptly 
noted:</p>
<blockquote>

   <p>&nbsp;&quot;Under no circumstances should you program the way I say to because I say to; program the way you think expresses best what 
   you&#39;re trying to accomplish in the program. And do so consistently and ruthlessly.&quot;</p>
</blockquote>

<p>As Dennis Ritchie noted &quot;</p>
<blockquote>

   <p><font color="#FF0000"><i><b>It is very hard to get things both right (coherent and correct) and usable (consistent enough, attractive 
   enough)&quot;. </b></i></font></p>
</blockquote>

<p>That&#39;s the matter of elegance. Most time making code shorter improves maintainability. But with Perl it is important overusing/abusing 
idioms, an addiction some Perl books author suffer from.&nbsp; </p>

<p>One important aspect of programming style is related to what is called <i>defensive programming</i>. Here are some recommendations 
from Tom Christiansen: </p>
<ul>

   <li>Use option -w to find misspelled variables and get more meaningful diagnostic. For example:<pre>#!/usr/bin/perl -wc your_script</pre>
   </li>

   <li>Check <b>all</b> syscall return values, printing $! </li>

   <li>Watch for external program failures in $? </li>

   <li>Check $@ after <tt>eval&quot;&quot;</tt>&nbsp; or <tt>s///ee</tt>. </li>

   <li>Use parameters asserts </li>

   <li>#!/usr/bin/perl -T </li>

   <li>Always have an <tt>else</tt>&nbsp; after a chain of <tt>elsif</tt>s </li>

   <li>Put commas at the end of lists to so your program won&#39;t break if someone inserts another item at the end of the list.</li>
</ul>

<p>&nbsp;advice on the naming of variables and functions is highly subjective. I use the following rules: </p>
<ul>

   <li><b>While short identifiers like <tt>$gotit</tt>&nbsp; are probably ok for my variables they are bad idea for glabals.</b></li>

   <li><b>You can use undersores for local variables and </b>mixed case for globals <tt>$VarNamesLikeThis. constnts can be put in all 
   Capspa </tt>&nbsp;especially for non-native speakers of English. It&#39;s also a simple rule that works consistently with VAR_NAMES_LIKE_THIS.
   </li>

   <li>You may find it helpful to use letter case to indicate the scope or nature of a variable. For example:<pre>   $ALL_CAPS_HERE   constants only (beware clashes with perl vars!)
   $Some_Caps_Here  package-wide global/static
   $no_caps_here    function scope my() or local() variables</pre>
   </li>

   <li><b>Functions and methods names seem to work best as all lowercase. </b>E.g., <tt>$obj</tt>-&gt;as_string().</li>
</ul>
<ul>

   <li><b>`Procedure names should reflect what they do; function names should reflect what they return.&#39;</b> --Rob Pike. Name objects 
   so that they read well in English. For example, predicate functions should usually be named with `is&#39;, `does&#39;, `can&#39;, or `has&#39;. Thus,
   <tt>&amp;is_ready</tt>&nbsp; is better than <tt>&amp;ready</tt>&nbsp; for the same function.&nbsp; Therefore, <tt>&amp;canonize</tt>&nbsp; as 
   a void function (procedure), <tt>&amp;canonical_version</tt>&nbsp; as a value-returning function, and <tt>&amp;is_canonical</tt>&nbsp; for 
   a boolean check. </li>

   <li>The <tt>&amp;abc2xyz</tt>&nbsp; and <tt>&amp;abc_to_xyz</tt>&nbsp; forms are also well established for conversion functions or hash mappings.
   </li>

   <li>Hashes usually express some <i>property</i> of the keys, and are used with the English word `of&#39; or the possessive form. Name 
   hashes for their values, not their keys.<pre>   GOOD:
&nbsp;&nbsp;&nbsp;&nbsp;    %color = (&#39;apple&#39; =&gt; &#39;red&#39;, &#39;banana&#39; =&gt; &#39;yellow&#39;);
&nbsp;&nbsp;&nbsp;&nbsp;    print $color{&#39;apple&#39;};&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;    # Prints `red&#39;</pre>

   <pre>   BAD:
&nbsp;&nbsp;&nbsp;&nbsp;    %fruit = (&#39;apple&#39; =&gt; &#39;red&#39;, &#39;banana&#39; =&gt; &#39;yellow&#39;);
&nbsp;&nbsp;&nbsp;&nbsp;    print $fruit{&#39;apple&#39;};&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;    # Prints `red&#39;</pre>
   </li>
</ul>

<h3><a name="Avoid_dogmatism_and_religious__zealotry">Avoid dogmatism and religious&nbsp; zealotry</a></h3>

<p>Those thing helped to kill Perl popularity, promoting languages which is many respect is one step forward two steps back. </p>

<p>Here is a couple of other useful point which help to avoid dogmatism, which is a real danger&nbsp; for programmer. Religious zealot 
is usually a very bad, dangerous programmer ready to mold the world in his often comply faulty vision.&nbsp; Proliferation of OO zealots 
is a sign of degeneration of programming as a profession. may be it has something to do with neoliberalism with its fake metrics for 
everything, I do not know ;-) They are usually only half-clever. </p>

<p>Below is a great post which expands on those points (and actually shows that Permonks community contians, or used to contain,&nbsp; 
real wizards): </p>
<ol>

   <li><a target="_blank" href="http://www.perlmonks.org/?node_id=777376">Best practices, revisited</a> on Jul 05, 2009 at 14:20 by 
   ELISHEVA
   <blockquote>

      <p><font color="#FF0000"><b><i>... ... ...</i></b></font></p>

      <p><font color="#FF0000"><i><b>In the last 10 years the term &quot;best practice&quot; has lost much of its association with the process 
      of learning. Instead best practice has become a buzz word that is increasingly associated with a laundry list of rules and procedures.
      </b></i></font>Perhaps it is our innate need to measure ourselves against a standard. Or perhaps it is the word &quot;best&quot;. There 
      can only be one best, even if it takes a process to find it. Why reinvent the wheel once the best has been found?</p>

      <p>Nowhere is this more clear than in the way many organizations and some monks seem to use Damian Conway&#39;s book on Perl best 
      practices. The best practice in Damian Conway&#39;s book refers (or should refer) to the process that Damian Conway went through while 
      developing his coding practice. He wrote this book in part because, over the years, his own coding style had come to resemble 
      an archeological dig though his own coding history.
      <a target="_blank" href="http://www.theperlreview.com/Interviews/damian-bpp-20050622.html?pm">Interview with Damian Conway, Brian 
      d foy</a>. However, few people talk about his process, whereas many preach (or complain about) his rule list.</p>

      <p><em>It may be human nature to turn best practices into best rules, but it isn&#39;t good management:</em></p>

      <p><b>1. Best practice by the rulebook oversimplifies the knowledge transfer process.</b> Knowledge consists of several components: 
      facts, recipes, thinking processes, information gathering skills, and methods of evaluation. Rules are only effective in transferring 
      the first two of these. However, all the rest are essential. Without them rules get out of date or will be applied in counter 
      productive ways.</p>

      <p>Facts, recipes, and coding standards are like wheels and brakes. But they do not drive the car. If the driver doesn&#39;t know 
      the difference between the brake and the accelerator, the car will crash no matter how wonderful the wheels.<font color="#FF0000"><i><b> 
      Hard to communicate skills like information gathering and methods of evaluation are what drive the coding car, not the rules capturing 
      layout and syntax.</b></i></font></p>

      <p>If we focus only on rules, it is natural to assume that knowledge will be transferred simply by giving people enough motivation 
      to follow rules. But this doesn&#39;t turn out to be the case.</p>

      <p>In 1996 (Strategic Management Journal), Gabriel Szulanski (The Wharton School) published a study analyzing the impediments 
      to knowledge transfer. (see <a target="_blank" href="http://management.uta.edu/Casper/MultiStat/Articles%20Fall%202007/Sz.pdf">
      Exploring internal stickiness: Impediments to the transfer of best practice in the firm</a>). He considered many factors that 
      might get in the way. The study concluded that motivation was overshadowed by three other issues: &quot;lack of absorptive capacity&quot;, 
      &quot;causal ambiguity&quot;, and &quot;arduousness of the relationship&quot;.</p>

      <p>If rules alone were enough none of these would matter. &quot;Lack of absorptive capacity&quot; means that the necessary background knowledge 
      to understand and value the rule is missing. Causal ambiguity means insufficient knowledge of how the rules relate to outcomes. 
      Put in plain English: we aren&#39;t very good at applying rules without reasons or context.</p>

      <p>However, explaining rules also means transferring judgment - something that cannot be captured purely in the rules themselves. 
      And this brings us to the last barrier to knowledge transfer: &quot;arduousness of the relationship&quot;. This awkward term refers to how 
      well the knowledge provider and receiver work together. Do they have a mentoring relationship that can answer questions and provide 
      background information? Or are they simply conduits for authority, insisting on the value of the rules without helping show how 
      the knowledge can be adapted to exceptional situations?</p>

      <p><b>2. An overemphasis on rules is a short-term investment in a long-term illusion.</b> Software is full of symbols and a great 
      deal of code is boiler plate. It is easy to imagine that rules play a large role in software and the right set of rules will have 
      a large payback.</p>

      <p>This might be true if writing software were merely a transformation process. But if it were, we&#39;d have developed software to 
      automatically translate business processes, math books, and motion studies into software long ago. To be sure <i>some</i> of the 
      coding today could probably be done by software, but not all of it. In every human endeavor there is a certain amount of boiler 
      plate activity that passes for intellectual labour. This can be automated. But there is also a certain amount of genuine creativity 
      and analysis. It takes a human being to know which is which. It takes a human being to do the later.</p>

      <p>If we want superior development teams, we need to spend our energy nurturing what only we humans can do. This is where our 
      investment needs to sit. As for the things we can do with rules: if we focus our skills on the creative portions we <i>will</i> 
      figure out a way to write software that makes the boiler plate things go away. It is only a matter of time.</p>

      <p><b>3. Rules that free us from thinking do not provide for change.</b> Rules that free us from thinking are, by their very nature, 
      static. In 1994 a management book &quot;Built to Last&quot; took the management world by storm and became a knock out best seller for several 
      years thereafter. 10 years later, the magazine &quot;Fast Company&quot; wrote an article reviewing the impact of the book and the companies 
      featured in that book. <a target="_blank" href="http://www.fastcompany.com/magazine/88/built-to-last.html?page=0%2C0">Was Build 
      to Last Built to Last</a> - in 2004 about half the companies described no longer would qualify as built to last. When interviewed 
      for the article, one of the authors of the book, James C. Collins, argued that these companies had lost sight of what had made 
      them great. He emphasized &quot;Theeee most important part of the book is chapter four! ... Preserve the core! And! Stimulate progress! 
      To be built to last, you have to be built for change!&quot;</p>

      <p><b>4. If it isn&#39;t abnormal it can&#39;t produce abnormal returns.</b> The things that can be reduced to judgment-free rules offer 
      no competitive advantage because they can be easily reproduced. No matter how hard we try we <i>cannot</i> build the best coding 
      shop by following everybody else&#39;s rules. To excel, our practices need to be closely matched to our team&#39;s strengths and weaknesses.</p>

      <p>Some of the more recent management literature has begun stressing the concept of &quot;signature practices&quot;. Signature practices 
      are practices that are unique to an organization. They capture its special ethos and talents and serve as a focal point around 
      which the company (or coding team) can develop its competitive edge. (See, for example &quot;Beyond Best Practice&quot;, by Linda Gratton 
      and Sumatra Ghoshal, Sloan Management Review, April 15, 2005).</p>

      <p>I don&#39;t mean to be knocking rules. They have their place. But if we want to have an outstanding development team, our definition 
      of best practice needs to expand beyond rules. We need to think about what makes our teams thrive. What helps them be at their 
      most creative? What gets them into flow? When are they best at sharing knowledge with each other? At understanding each others 
      code? Incorporating new team members? At meeting customers&#39; needs? And then we have to be prepared to be <i>ruthless</i> in getting 
      rid of <i>anything</i> that gets in the way of that. Even if it is the rules themselves.</p>

      <p>Best, beth</p>

      <p><b>Update</b>: Clarification of point #4, in response to <a href="?node=mzedeler">mzedeler</a> below.</p>
   </blockquote>
   </li>
</ol>

<h3><a name="Do_not_imitate_complexity_junkies">Do not imitate complexity junkies</a></h3>

<p>Complexity junkies are determined to show us their inventiveness, often to detriment of clarity. By saving&nbsp; a couple of line&nbsp; 
here and there, they create difficulties in maintainance&nbsp; and their style should be avoided. Here is one example from standard 
module Std.pm</p>

<pre>sub getopts ($;$) {
    my ($argumentative, $hash) = @_;
    my (@args,$first,$rest,$exit);
    my $errs = 0;
    local $_;
    local @EXPORT;

    @args = split( / */, $argumentative );
    while(@ARGV && ($_ = $ARGV[0]) =~ /^-(.)(.*)/s) {
	($first,$rest) = ($1,$2);
	if (/^--$/) {	# early exit if --
	    shift @ARGV;
	    last;
	}
	my $pos = index($argumentative,$first);
	if ($pos >= 0) {
	    if (defined($args[$pos+1]) and ($args[$pos+1] eq ':')) {
		shift(@ARGV);
		if ($rest eq '') {
		    ++$errs unless @ARGV;
		    $rest = shift(@ARGV);
		}
		if (ref $hash) {
		    $$hash{$first} = $rest;
		}
		else {
		    ${"opt_$first"} = $rest;
		    push( @EXPORT, "\$opt_$first" );
		}
	    }
	    else {
		if (ref $hash) {
		    $$hash{$first} = 1;
		}
		else {
		    ${"opt_$first"} = 1;
		    push( @EXPORT, "\$opt_$first" );
		}
		if ($rest eq '') {
		    shift(@ARGV);
		}
		else {
		    $ARGV[0] = "-$rest";
		}
	    }
	}
	else {
	    if ($first eq '-' and $rest eq 'help') {
		version_mess($argumentative, 'main');
		help_mess($argumentative, 'main');
		try_exit();
		shift(@ARGV);
		next;
	    } elsif ($first eq '-' and $rest eq 'version') {
		version_mess($argumentative, 'main');
		try_exit();
		shift(@ARGV);
		next;
	    }
	    warn "Unknown option: $first\n";
	    ++$errs;
	    if ($rest ne '') {
		$ARGV[0] = "-$rest";
	    }
	    else {
		shift(@ARGV);
	    }
	}
    }
    unless (ref $hash) { 
	local $Exporter::ExportLevel = 1;
	import Getopt::Std;
    }
    $errs == 0;
}</pre>

Can be simplified to 
<pre>
sub getopts
{
my ($options_def,$options_hash)=@_;
my ($first,$rest,$pos,$cur_opt);
   while(@ARGV){
      $cur_opt=$ARGV[0];
      last if( substr($cur_opt,0,1) ne '-' );
      if ($cur_opt eq '--'){
          shift @ARGV;
          last;
      }
      $first=substr($cur_opt,1,1);
      $pos = index($options_def,$first);
      if( $pos==-1) {
         warn("Undefined option -$first skipped without processing\n");
         shift(@ARGV);
         next;
      }
      $rest=substr($cur_opt,2);
      if( $pos&lt;length($options_def)-1 && substr($options_def,$pos+1,1) eq ':' ){
         # option with parameters
         if( $rest eq ''){
           shift(@ARGV); # get the value of option
           unless( @ARGV ){
              warn("End of line reached for option -$first which requires argument\n");
              $$options_hash{$first}='';
              last;
           }
           if ( $ARGV[0] =~/^-/ ) {
               warn("Option -$first requires argument\n");
               $$options_hash{$first} = '';
           }else{
               $$options_hash{$first}=$ARGV[0];
               shift(@ARGV); # get next chunk
           }
         } else {
            #value is concatenated with option like -ddd
            if( ($first x length($rest)) eq $rest ){
               $$options_hash{$first} = length($rest)+1;
            }else{
               $$options_hash{$first}=$rest;
            }
            shift(@ARGV);
         }
      }else {
         $$options_hash{$first} = 1; # set the option
         if ($rest eq '') {
            shift(@ARGV);
         } else {
            $ARGV[0] = "-$rest"; # there can be other options without arguments after the first
         }
      }
   }
}
</pre>

<h3><b>Larry Wall guidance&nbsp; (from Perl Style documentation) </b></h3>

<p>Regarding aesthetics of code lay out, about the only thing Larry cares strongly about is that the closing curly brace of a multi-line
<font size="-1">BLOCK</font> should line up with the keyword that started the construct. </p>

<p>Beyond that, he has other preferences that aren&#39;t so strong: </p>
<ul>

   <li>4-column indent. </li>

   <li><em>Opening curly on same line as keyword, if possible, otherwise line up. </em></li>

   <li>Space before the opening curly of a multi-line <font size="-1">BLOCK.</font> </li>

   <li><em>One-line <font size="-1">BLOCK</font> may be put on one line, including curlies. </em></li>

   <li>No space before the semicolon. </li>

   <li>Semicolon omitted in &quot;short&#39;&#39; one-line <font size="-1">BLOCK.</font> </li>

   <li>Space around most operators. </li>

   <li>Space around a &quot;complex&#39;&#39; subscript (inside brackets). </li>

   <li><em>Blank lines between chunks that do different things</em><i><font color="#FF0000">. </font></i></li>

   <li>Uncuddled elses. </li>

   <li>No space between function name and its opening parenthesis. </li>

   <li><font color="#FF0000"><i><b>Space after each comma.</b></i></font> </li>

   <li>Long lines broken after an operator (except &quot;and&#39;&#39; and &quot;or&#39;&#39;). </li>

   <li>Space after last parenthesis matching on current line. </li>

   <li>Line up corresponding items vertically. </li>

   <li>Omit redundant punctuation as long as clarity doesn&#39;t suffer</li>
</ul>
<hr>
<table border="1" width="100%" height="350">
   <tr>
      <td width="100%" align="center" colspan="5" rowspan="4">
      <center>
      <b>Top Visited</b></center>
      <iframe src="/topupdates.shtml" width="100%" height="320">

      <p>Your browser does not support iframes.</p>
      </iframe>
      </td>
      <td bgcolor="#FFFF00" align="center"><b><a href="switchboard.shtml">Switchboard</a></b></td>
   </tr>
   <tr>
      <td bgcolor="#FFFF00" align="center"><b><a href="/index.shtml#Latest">Latest</a></b></td>
   </tr>
   <tr>
      <td bgcolor="#FFFF00" align="center"><b>
      <a href="https://www.google.com/search?q=site:www.softpanorama.org&hl=en&lr=&tbo=1&prmd=ivns&source=lnt&tbs=qdr:y&sa=X&ei=aML0TaHgO4fUgAedxtjzCw&ved=0CAsQpwUoBQ&biw=1256&bih=801&cad=cbv#q=site:www.softpanorama.org&hl=en&lr=&tbo=1&prmd=ivns&source=lnt&tbs=qdr:w&sa=X&ei=csL0TYHsI4LagAeKu7DPCw&ved=0CAwQpwUoAw&fp=fb58e14853731932&biw=1256&bih=801">
      Past week</a></b></td>
   </tr>
   <tr>
      <td bgcolor="#FFFF00" align="center"><b>
      <a href="https://www.google.com/search?q=site:www.softpanorama.org&hl=en&lr=&tbo=1&prmd=ivns&source=lnt&tbs=qdr:m&sa=X&ei=IMD0TdjLHMXUgQe9t6zfCw&ved=0CA0QpwUoBA">
      Past month</a></b></td>
   </tr>
</table
<hr noshade color="#FF0000" size="5">




<hr>

<h2><a name="NEWS_TOC">NEWS CONTENTS</a></h2>
<ul>
<li>20200930 : <a href="#n20200930X_postfix_if_conditions_are_useful_mainly_to_specify_exist_condition_in_the_loop_rarely">Postfix if conditions are useful mainly to specify exist condition in the loop, rarely elsewhere</a> <i></i>  ( Sep 30, 2020 , <a target="_blank" href= "https://perlmonks.org/?node_id=11122274">perlmonks.org</a> ) </li>
<li>20200929 : <a href="#n20200929X_at_what_point_excessive_syntactic_flexibility_stimulates_perverted_programming_style">At what point excessive syntactic flexibility stimulates perverted programming style which is reflected in the derogative term "complexity junkies"?</a> <i></i>  ( Sep 29, 2020 , <a target="_blank" href= "https://perlmonks.org/?node_id=11122345">perlmonks.org</a> ) </li>
<li>20191121 : <a href="#n20191121X_tux_nl_style_and_layout">Tux.nl - Style and   Layout</a> <i></i>  ( Nov 21, 2019 , <a target="_blank" href="http://tux.nl/style.html">tux.nl</a> ) </li>
<li>20171129 : <a href="#n20171129X_how_can_i_have_variable_assertions_in_perl">How can I have variable assertions in Perl </a> <i></i>  ( Nov 29, 2017 , <a target="_blank" href= "https://stackoverflow.com/questions/1023813/how-can-i-have-variable-assertions-in-perl">stackoverflow.com</a> ) </li>
<li>20171117 : <a href="#n20171117X_bruce_gray_your_perl_5_brain_on_perl_6">Bruce   Gray - Your Perl 5 Brain, on Perl 6 </a> <i> by Bruce Gray</i>  ( Nov 17, 2017 , <a target="_blank" href= "https://www.youtube.com/watch?v=EUXclZ_d0DQ">www.youtube.com</a> ) </li>
<li>20171113 : <a href="#n20171113X_perls_worst_best_practices">Perls Worst Best Practices</a> <i> by Daina Pettit </i>  ( <a target="_blank" href="https://www.youtube.com/watch?v=Y1I1KcKvz9Q">&#8206;Perl&#39;s Worst Best Practices&#8206;</a>, Nov 13, 2017 ) </li>
<li>20120126 : <a href="#n20120126X_appendix_b_perl_best_practices_oreilly_media">Appendix B Perl Best  Practices - OReilly Media</a> <i></i>  ( <a target="_blank" href="http://oreilly.com/perl/excerpts/perl-best-practices/appendix-b.html">Appendix B Perl Best  Practices - O&#39;Reilly Media</a>, Jan 26, 2012 ) </li>
<li>20111027 : <a href="#n20111027X_how_can_i_convert_perl_code_into_html_with_syntax_highlighting">How can I convert Perl code into HTML with syntax  highlighting?</a> <i></i>  ( Jun 16, 2002 , <a target="_blank" href="http://www.perlmonks.org">www.perlmonks.org</a> ) </li>
<li>20111027 : <a href="#n20111027X_togotutor">togotutor</a> <i></i>  ( softpanorama.org,  ) </li>
<li>20111027 : <a href="#n20111027X_perl_style_guide_juerds_site">Perl style guide - Juerds site </a> <i></i>  ( <a target="_blank" href="http://juerd.nl/site.plp/perlstyle">Perl style guide - Juerd&#39;s site </a>,  ) </li>
<li>20050714 : <a href="#n20050714X_perl_com_ten_essential_development_practices">Perl.com- Ten Essential Development  Practices</a> <i> by Damian Conway</i>  ( <a target="_blank" href="http://www.perl.com/lpt/a/2005/07/14/bestpractices.html">Perl.com- Ten Essential Development  Practices</a>, July 14, 2005 ) </li>
<li>20031017 : <a href="#n20031017X_part_13_perl_style">Part 13 Perl Style</a> <i> by Dan Richter </i>  ( <a target="_blank" href="http://mailman.linuxchix.org/pipermail/courses/2003-October/001349.html">Part 13 Perl Style</a>, Oct 17,2003 ) </li>
<li>20031017 : <a href="#n20031017X_the_road_to_better_programming_introduction_and_chapter_1">The road to better programming  Introduction and chapter 1</a> <i> by Teodor Zlatanov</i>  ( November 1, 2001 ) </li>
<li>20031017 : <a href="#n20031017X_the_road_to_better_programming">The road to better programming</a> <i></i>  ( <a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road1.html">The road to better programming</a>,  ) </li>
</ul>
<h2><a name="News">Old News ;-)</a></h2>

<center><table border="0" width="100"><tr>
<td align="center"><a href="#NEWS_TOC"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20200930X_postfix_if_conditions_are_useful_mainly_to_specify_exist_condition_in_the_loop_rarely" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20200929X_at_what_point_excessive_syntactic_flexibility_stimulates_perverted_programming_style"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[Sep 30, 2020] <a target="_blank" href="https://perlmonks.org/?node_id=11122274">Postfix if conditions are useful mainly to specify exist condition in the loop, rarely elsewhere</a></h4>

<blockquote>
<h6>Sep 30, 2020 |  <a target="_blank" href= "https://perlmonks.org/?node_id=11122274">perlmonks.org</a></h6>
 

    <p><a target="_blank" href="https://perlmonks.org/?node_id=11122274">Re^6: What esteemed monks
    think about changes necessary/desirable in Perl 7 outside of OO staff</a><br>
    by <a target="_blank" href="https://perlmonks.org/?node_id=1203274">likbez</a> on Sep 27, 2020
    at 22:37 UTC</p>

    <p>We need not "assume that somebody uses this formatting". I do it frequently, and I have
    often seen it in other people's code. That fact that you use it and saw it in other people code
    means nothing. People often adopt and use bad programming style. Even talented programmers do.
    Look at classic <b><i>The Elements of Programming Style</i></b> , by Brian W. Kernighan and P.
    J. Plauger. They include such recommendations as ( cited from
    https://en.wikipedia.org/wiki/The_Elements_of_Programming_Style ) :</p>

    <ul>
      <li>Write clearly -- don't be too clever.</li>

      <li>Say what you mean, simply and directly.</li>

      <li>... ... ...</li>

      <li>Write clearly -- don't sacrifice clarity for efficiency.</li>

      <li>... ... ...</li>

      <li>Parenthesize to avoid ambiguity.</li>

      <li>... ... ...</li>

      <li>Make sure special cases are truly special.</li>

      <li>... ... ...</li>
    </ul>

    <p>The real question is <i>whether the use you advocate represents a good Perl programming
    style or not.</i></p>

    <p>I would understand the use of post-fix if construct in a loop to specify exit condition.
    Something like:</p>
    <pre>
return if ($exit_condition);
</pre>They make code more readable in comparison with the regular if statement as as such have
certain value and IMHO represent a good programming style.

    <p>In many other cases the desire to save two curly braces looks to me a very questionable
    practice and a bad programming style. Your mileage may vary.</p>
 <!--TAGS: .  ~/Likbez -->
</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20200930X_postfix_if_conditions_are_useful_mainly_to_specify_exist_condition_in_the_loop_rarely"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20200929X_at_what_point_excessive_syntactic_flexibility_stimulates_perverted_programming_style" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20191121X_tux_nl_style_and_layout"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[Sep 29, 2020] <a target="_blank" href="https://perlmonks.org/?node_id=11122345">At what point excessive syntactic flexibility stimulates perverted programming style which is reflected in the derogative term "complexity junkies"?</a></h4>

<h5>Notable quotes:</h5>
<h5>&quot;...  In your private role you are free to do whatever you wish. After all programming open         source is about fun, not so much about discipline. ...&quot;</h5>
<h5>&quot;...  The situation radically changes in commercial projects. If you are a manager of a large         project you need to ensure a uniform and preferably simple style via guidelines that         explicitly prohibit such "excesses" and to step on the throat of such "excessively         creative" people to make them "behave". ...&quot;</h5>
<h5>&quot;...  That's why languages that allow too much syntactic freedom are generally not welcomed in         large commercial projects, even if they are able to manage large namespaces more or less         OK. ...&quot;</h5>
<blockquote>
<h6>Sep 29, 2020 |  <a target="_blank" href= "https://perlmonks.org/?node_id=11122345">perlmonks.org</a></h6>
 

    <p><b><a target="_blank" href="https://perlmonks.org/?node_id=1203274">likbez</a> on Sep 29,
    2020 at 18:01 UTC </b></p>

    <blockquote>
      <a target="_blank" href="https://perlmonks.org/?node_id=11122345">Re^11: What esteemed monks
      think about changes necessary/desirable in Perl 7 outside of OO staff</a>

      <p>by <a target="_blank" href="https://perlmonks.org/?node_id=1203274">likbez</a> on Sep 29, 2020 at 18:01 UTC </p>


        <blockquote>
          If I have to maintain (as only maintainer) a piece of perl code, I will *rewrite* *all*
          statements as you state from action if expression; to expression and action; as that (to
          me) is waaaaaaaaaay easier to read/understand/maintain. Nothing to do with "idiomatic
          perl". Nothing at all!
        </blockquote>People are extremely flexible. The same is true for programmers. Many of
        talented programmers I encountered have a somewhat idiosyncratic style...

        <p><em>In your private role you are free to do whatever you wish. After all programming open
        source is about fun, not so much about discipline.</em></p>

        <p><em>The situation radically changes in commercial projects. If you are a manager of a large
        project you need to ensure a uniform and preferably simple style via guidelines that
        explicitly prohibit such "excesses" and to step on the throat of such "excessively
        creative" people to make them "behave".</em></p>

        <p><em>That's why languages that allow too much syntactic freedom are generally not welcomed in
        large commercial projects, even if they are able to manage large namespaces more or less
        OK.</em></p>

        <p>Let's be frank: Perl lost Web applications development more or less completely. The
        reasons are not clear and can be argued, but the fact is indisputable. But the problem that
        I see is that Perl can lose attraction among sysadmins because of excessive push of OO
        programming style by OO-fanatics and the second rate book authors, as well as due to
        inability of distribute remaining scarce development resources toward modest and not fancy
        (unlike closures, accessors, frameworks and other fancy staff) improvements in the
        procedural programming arena (the area which this post is all about).</p>

        <p>An interesting question is: at what point excessive syntactic flexibility stimulates
        perverted programming style which is reflected in the derogative term "complexity junkies"?
        When in the program simple things look complex, and complex unmanageable. "Object oriented
        spaghetti" ('Lasagna code' with too many layers) is another term that addresses the same
        problem. See, for example, discussion at
        <a target="_blank" href="https://medium.com/better-programming/is-object-oriented-programming-garbage-66c4f41adcaa">https://medium.com/better-programming/is-object-oriented-programming-garbage-66c4f41adcaa</a> </p>

        <p>Also
        <a target="_blank" href="https://www.youtube.com/watch?time_continue=9&v=V6VP-2aIcSc&feature=emb_logo">https://www.youtube.com/watch?time_continue=9&amp;v=V6VP-2aIcSc&amp;feature=emb_logo</a> </p>
    </blockquote>
 <!--TAGS: .  ~/Likbez -->
</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20200929X_at_what_point_excessive_syntactic_flexibility_stimulates_perverted_programming_style"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20191121X_tux_nl_style_and_layout" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20171129X_how_can_i_have_variable_assertions_in_perl"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[Nov 21, 2019] <a target="_blank" href="http://tux.nl/style.html">Tux.nl - Style and   Layout</a></h4>

<blockquote>
<h6>Nov 21, 2019 |  <a target="_blank" href="http://tux.nl/style.html">tux.nl</a></h6>
 

    <p><b>Why my style is best better</b></p>

    <p>I will try to explain the logic behind the style decisions taken over that last 35+ years of
    programming in different languages.</p>

    <p>About programming style and layout there are as many opinions as there are people. Most
    important in my opinion is to <i>think</i> about the reasoning behind what you, your team or
    your company chooses to follow as guides.</p>

    <p>I seriously think that way too many (young) programmers leave school, brainwashed with
    GNU-style coding without realizing that the amount of indentation and the placing of braces,
    brackets and parentheses were well thought about.</p>

    <p>Several well known styles (including mine) are discussed at <a target="_blank" href=
    "https://secure.wikimedia.org/wikipedia/en/wiki/Indent_style">wikimedia</a> . It is worth
    reading through them to see the pros and cons of each.</p>

    <p>For me personally, the GNU coding style is one of the reasons I do <i>NOT</i> contribute a
    lot to these projects. The style does not fit my logic, and if I send patches that are rejected
    simply because I wrote them in a style/layout that I think is way better because I then
    understand the underlying logic, I give up.</p>

    <p>Here I will take a tour through what I think is the only correct way of (perl) code layout,
    and why. Most of this can be achieved with <a target="_blank" href=
    "http://search.cpan.org/dist/Perl-Tidy">Perl::Tidy</a> and a correct <a target="_blank" href=
    "http://tux.nl/Files/.perltidyrc">.perltidyrc</a> . I'll use their configuration definitions as
    a guide.</p>

    <dl>
      <dt>Indentation in code blocks</dt>

      <dt>Opening Block Brace Right or Left</dt>

      <dt>Braces Left</dt>

      <dd>
        Because braces are just syntactic sugar to keep a block together, it should visually also
        bind to the block, and not to the conditional. As the closing brace - or <code>END</code>
        in languages like PASCAL - is visually showing me the end of the block, it should obviously
        have the same indent as the block itself. An advantage is that the alignment of the closing
        brace with the block emphasizes the fact that the entire block is conceptually (as well as
        programmatically) a single compound statement.<br>
        In other words: I see the braces being part of the block, and as all statements inside a
        block share the same indentation, in my opinion the brace - being part of the block -
        should have the same indentation too.

        <ul>
          <li>Indent width is 4, tabs are allowed (when set to 8). I prefer having it being spaces
          only, but as I cannot <i>see</i> the difference with good editors, I do not really
          care.</li>

          <li>Opening brace should be on the same line as the conditional</li>

          <li>Block should be indented</li>

          <li>Closing brace should have the same indent as the block</li>
        </ul>
        <pre>
  if ($flag eq "a") {
      $anchor = $header;
      }
</pre>This style is also referred to as <a target="_blank" href=
"https://en.wikipedia.org/wiki/Indent_style#Ratliff_style">Ratliff style</a> on wikipedia or
<a target="_blank" href="https://secure.wikimedia.org/wikipedia/en/wiki/Indent_style#Banner_style">
        Banner style</a> on wikimedia.
      </dd>

      <dt>Continuation Indentation</dt>

      <dd>
        <pre>
  if ($flag eq "a") {
      $anchor = substr ($header, 0, 6) .
                substr ($char_list, $place_1, 1) .
                substr ($char_list, $place_2, 1);
      }
</pre>
      </dd>

      <dd>
        Or, also acceptable:
        <pre>
  if ($flag eq "a") {
      $anchor =
          substr ($header, 0, 6) .
          substr ($char_list, $place_1, 1) .
          substr ($char_list, $place_2, 1);
      }
</pre>
      </dd>

      <dt>Braces Right</dt>

      <dd>
        <pre>
  if ($bigwasteofspace1 &amp;&amp; $bigwasteofspace2 ||
      $bigwasteofspace3 &amp;&amp; $bigwasteofspace4) {
      big_waste_of_time ();
      }
</pre>
      </dd>

      <dd>
        also acceptable:
        <pre>
  if (   $bigwasteofspace1 &amp;&amp; $bigwasteofspace2
      || $bigwasteofspace3 &amp;&amp; $bigwasteofspace4) {
      big_waste_of_time ();
      }
</pre>
      </dd>

      <dd>
        also acceptable:
        <pre>
  if (  $bigwasteofspace1 &amp;&amp; $bigwasteofspace2 ||
        $bigwasteofspace3 &amp;&amp; $bigwasteofspace4) {
      big_waste_of_time ();
      }
</pre>
      </dd>

      <dt>(No) Cuddled Else</dt>

      <dd>
        Of course cuddled else is not the way to go, as it makes removing either branch more
        difficult and makes the indent of the closing brace go wrong. The only right way to use
        if/else indent is uncuddled:
        <pre>
  if ($flag eq "h") {
      $headers = 0;
      }
  elsif ($flag eq "f") {
      $sectiontype = 3;
      }
  else {
      print "invalid option: " . substr ($arg, $i, 1) . "\n";
      dohelp ();
      }
</pre>
      </dd>

      <dt>Vertical tightness</dt>

      <dd>
        <pre>
  sub _directives
  {
      {   ENDIF =&gt; \&amp;_endif,
          IF    =&gt; \&amp;_if,
          };
      } # _directives
</pre>the opening brace of a sub may optionally be put on a new line. If so, it should be in column
one, for all those that use 'vi' or one of it's clones, so }, {, ]], and [[ work as expected.<br>

        if the opening brace is on the same line, which I prefer, it requires a single leading
        space
        <pre>
  sub _directives {
      {   ENDIF =&gt; \&amp;_endif,
          IF    =&gt; \&amp;_if,
          };
      } # _directives
</pre>
      </dd>

      <dt>Indentation Style for Other Containers</dt>

      <dt>Opening Vertical Tightness</dt>

      <dd>
        <pre>
  $dbh = DBI-&gt;connect (undef, undef, undef, {
      PrintError =&gt; 0,
      RaiseError =&gt; 1,
      });
</pre>
      </dd>

      <dd>
        <pre>
  if (!defined (start_slip ($DEVICE, $PHONE,  $ACCOUNT, $PASSWORD,
                            $LOCAL,  $REMOTE, $NETMASK, $MTU)) &amp;&amp;
       $continuation_flag) {
      do_something_about_it ();
      }
</pre>
      </dd>

      <dt>Closing Token Placement</dt>

      <dd>
        <pre>
  my @month_of_year = ( "Jan", "Feb", "Mar", "Apr", "May", "Jun",
                        "Jul", "Aug", "Sep", "Oct", "Nov", "Dec",
                        );
</pre>
      </dd>

      <dd>
        also acceptable:
        <pre>
  my @month_of_year = (qw(
      Jan Feb Mar Apr May Jun
      Jul Aug Sep Oct Nov Dec
      ));
</pre>As with the closing brace of a block, the closing parenthesis belongs to the data in the
container it closes, and thus should have the same indentation.
      </dd>

      <dt>Define Horizontal Tightness</dt>

      <dd>
        Of course <code>function &lt;space&gt; &lt;paren&gt; &lt;no-space&gt; &lt;first-arg&gt;
        &lt;comma&gt; &lt;space&gt;</code>
        <pre>
  if ((my $duration = travel ($target, $means)) &gt; 1200) {
</pre>One of my pet-peeves. Having white-space between the function name and its opening
parenthesis is the best match to how we think. As an example, if I would ask someone to describe
his/her day, he/she might answer
        <pre>
  I woke up
  I freshened myself
  I had breakfast
  I got to work
  I worked
  I had lunch
  I worked again
  I went home
  I had diner
  I watched TV
  I brushed my teeth
  I went to bed
</pre>In computer-speak
        <pre>
  wake_up ();
  wash ($self);
  eat ("breakfast");
  goto ("work")
  work ();
  eat ("lunch");
  work ();
  goto ("home");
  eat ("diner");
  watch_tv ();
  wash ($teeth);
  sleep ();
</pre>In which the seasoned programmer might see
        <pre>
  for $day in (qw( Mon Tue Wed Thu Fri )) {
     wake_up ();
     wash ($self);
     eat ("breakfast");
     :
     :
</pre>Or, more extreme to show the sequence of <b>actions</b>
        <pre>
  for $day in (qw( Mon Tue Wed Thu Fri )) {
     wake_up ();
     wash    ($self);
     eat     ("breakfast");
     :
     :
</pre>Where it, IMHO, clearly shows that the actions are far more important than what it takes to
perform the action. When I read through the process, I don't care about what transport the person
uses to get to work and if eggs are part of the breakfast. These are the parameters to the actions
        <pre>
  for $day in (qw( Mon Tue Wed Thu Fri )) {
     wake_up ();
     wash    ($day eq "Fri" ? "bath" : "shower", water_temp =&gt; "47");
     eat     (type   =&gt; "breakfast", eggs  =&gt; 2, toast =&gt; 4, Tea =&gt; "yes");
     travel  (target =&gt; $work,       means =&gt; "train");
     :
     :
</pre>I will only have a look at the function's argument if I need to. In reading that I <code>
        eat</code> , I see what <b>action</b> is taken. That's enough for understanding the program
        flow. The arguments to the function have to be grouped together using parenthesis for the
        function to know that all the arguments are for the function: the parenthesis are there to
        group the arguments, not to make the function a function so the parenthesis belong to the
        arguments and not to the function and therefor are to be close to the arguments ant not to
        the function.<br>
        Arguments are separated by a comma <b>and</b> a space, just to separate the arguments more
        for better readability
        <pre>
  my $width = $col[$j + $k] - $col[$j];

  my %bf = map { $_ =&gt; -M $_ } grep { m/\.deb$/ } dirents ".";
</pre>
      </dd>

      <dt>Statement modifiers</dt>

      <dd>
        <pre>
  $work_done and go_home ();
</pre>A rule of thumb is to NEVER use statement modifiers like
        <pre>
  go_home () unless $work_done; # WRONG!
</pre>As it will draw the attention to going home (unconditionally) instead of to the condition,
which is more important. This is especially annoying when using exit, die, croak or return. Any of
these will visually end the current scope, so you do not have to read on. Unless there is a
statement modifier and you need to re-read the entire section.
      </dd>

      <dt>No else after return/exit/die/croak/throw</dt>

      <dd>
        <pre>
  if (expression) {
      return;
      }
  else {
      return 42;
      }
</pre>As any of return, exit, die, croak, or throw will immediately exit the current scope, the
mind will read the code as to stop processing it right there, which is exactly what those keywords
are for.<br>
        In an if/else construct, the code <b>after</b> the construct is supposed to be executed
        when either if the if/else branches where followed. If the if-branch exits the current
        scope, there is no need to run the code after the construct, so the else is useless.<br>
        This is the main reason why these keywords should <b>never</b> have a statement modifier
        (and no, you cannot come up with a valid exception to this rule).
      </dd>

      <dt>Statement Termination Semicolon Spaces</dt>

      <dd>
        <pre>
  my $i = 1;
</pre>
      </dd>

      <dt>For Loop Semicolon Spaces</dt>

      <dd>
        <pre>
  for (@a = @$ap, $u = shift @a; @a; $u = $v) {
</pre>
      </dd>

      <dt>Block Comment Indentation</dt>

      <dd>
        <ul>
          <li>If comment is aligned to the left margin, leave it there</li>

          <li>If the original comment was indented, match the indent to the surrounding code.</li>

          <li>Never reformat comments itself. Do not wrap</li>
        </ul>
      </dd>

      <dt>Outdenting Long Quotes</dt>

      <dd>
        <pre>
  if ($source_stream) {
      if (@ARGV &gt; 0) {
          die "You may not specify any filenames when a source array is given\n";
          }
      }
</pre>
      </dd>

      <dd>
        <pre>
  if ($source_stream) {
      if (@ARGV &gt; 0) {
          die "You may not specify any filenames ".
              "when a source array is given\n";
          }
      }
</pre>
      </dd>

      <dd>
        <pre>
  for (@methods) {
      push @results, {
          name =&gt; $_-&gt;name,
          help =&gt; $_-&gt;help,
          };
      }
</pre>
      </dd>
    </dl>
 <!--TAGS: .  -->
</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20191121X_tux_nl_style_and_layout"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20171129X_how_can_i_have_variable_assertions_in_perl" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20171117X_bruce_gray_your_perl_5_brain_on_perl_6"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[Nov 29, 2017] 
  <a target="_blank" href=
  "https://stackoverflow.com/questions/1023813/how-can-i-have-variable-assertions-in-perl">How can I have variable assertions in Perl </a></h4>

  <blockquote>
    <h6>Nov 29, 2017 | <a target="_blank" href=
    "https://stackoverflow.com/questions/1023813/how-can-i-have-variable-assertions-in-perl">stackoverflow.com</a></h6>

    <p><b><a target="_blank" href="https://stackoverflow.com/users/85185/alex">Alex</a></b> ,Jun
    21, 2009 at 12:45</p>

    <blockquote>
      How can I check that a variable has a specific value in Perl? Is there a command to stop a
      script's execution to look up some of it's variables?

      <p>I wonder if I can use the Pythonic practice of inserting:</p>
      <pre>
    assert 0, (foo, bar)</pre>

      <p>to debug scripts in a debuger-less way?</p>
    </blockquote>

    <p><b><a target="_blank" href=
    "https://stackoverflow.com/users/26702/telemachus">Telemachus</a></b> ,Jun 21, 2009 at
    12:58</p>

    <blockquote>
      A quick CPAN search suggests <a target="_blank" href=
      "http://search.cpan.org/perldoc?Carp::Assert">Carp::Assert</a> .
    </blockquote>

    <p><b><a target="_blank" href=
    "https://stackoverflow.com/users/100754/sinan-%C3%9Cn%C3%BCr">Sinan &Uuml;n&uuml;r</a></b> ,Jun
    21, 2009 at 13:30</p>

    <blockquote>
      +1 for typing in more characters faster than I did. I am going to edit the URL to be version
      agnostic though. &ndash; <a target="_blank" href=
      "https://stackoverflow.com/users/100754/sinan-%C3%9Cn%C3%BCr">Sinan &Uuml;n&uuml;r</a>
      <a target="_blank" href=
      "https://stackoverflow.com/questions/1023813/how-can-i-have-variable-assertions-in-perl#comment833727_1023839">
      Jun 21 '09 at 13:30</a>
    </blockquote>

    <p><b><a target="_blank" href=
    "https://stackoverflow.com/users/26702/telemachus">Telemachus</a></b> ,Jun 21, 2009 at
    13:50</p>

    <blockquote>
      I was going to comment on your answer about the photo finish. As for the URL, I constantly
      forget that, so thanks. &ndash; <a target="_blank" href=
      "https://stackoverflow.com/users/26702/telemachus">Telemachus</a> <a target="_blank" href=
      "https://stackoverflow.com/questions/1023813/how-can-i-have-variable-assertions-in-perl#comment833762_1023839">
      Jun 21 '09 at 13:50</a>
    </blockquote>

    <p><b><a target="_blank" href=
    "https://stackoverflow.com/users/100754/sinan-%C3%9Cn%C3%BCr">Sinan &Uuml;n&uuml;r</a></b> ,Jun
    21, 2009 at 12:58</p>

    <blockquote>
      See <a target="_blank" href="http://search.cpan.org/perldoc?Carp::Assert">Carp::Assert</a> .
    </blockquote>

    <p><b><a target="_blank" href="https://stackoverflow.com/users/17279/zoul">zoul</a></b> ,Jun
    21, 2009 at 13:44</p>

    <blockquote>
      <a target="_blank" href="http://search.cpan.org/perldoc?Smart::Comments">Smart::Comments</a>
      are nice.
    </blockquote>

    <p><b><a target="_blank" href="https://stackoverflow.com/users/14750/ret">RET</a></b> ,Jun 22,
    2009 at 3:28</p>

    <blockquote>
      <em>Smart::Comments++ When used with the -ENV switch, it's a fantastic tool for this sort of
      thing. Much better than having to strip all the tests out before going to production, as
      someone else suggested. </em>

      <p>&ndash; <a target="_blank" href=
      "https://stackoverflow.com/users/14750/ret">RET</a> <a target="_blank" href=
      "https://stackoverflow.com/questions/1023813/how-can-i-have-variable-assertions-in-perl#comment835316_1023925">
      Jun 22 '09 at 3:28</a>
    </p>
    </blockquote>

    <p><b><a target="_blank" href="https://stackoverflow.com/users/70482/nik">nik</a></b> ,Jun 21,
    2009 at 12:54</p>

    <blockquote>
      There is a script at <a target="_blank" href=
      "http://www.perlmonks.org/?node_id=391780">PerlMonks</a> that introduces a fast assert
      method.

      <p>Speed is important since Perl is interpreted and any inline checks will impact performance
      (unlike simple C macros for example)</p>
      <hr>

      <p>I am not sure if these things are going to be directly usable.</p>

      <ul>
        <li>there is <a target="_blank" href=
        "http://search.cpan.org/perldoc?Test::Harness">Test::Harness</a> in default installs. Here
        <a target="_blank" href="http://tomacorp.com/perl/harness/index.html">is a starter
        tutorial</a> . The more recent module is <a target="_blank" href=
        "http://search.cpan.org/perldoc?TAP::Harness">TAP::Harness</a></li>

        <li>A slower version along the lines you talk is <a target="_blank" href=
        "http://search.cpan.org/perldoc?Sub::Assert">Sub::Assert</a></li>
      </ul>
      <hr>

      <p>Ok! This is what i was looking for -- <a target="_blank" href=
      "http://www.wgz.org/chromatic/perl/Test-Tutorial.pdf">PDF Warning: Test-Tutorial.pdf</a> .
      The <code>Test::Harness</code> is used for writing Perl module tests.</p>
    </blockquote>

    <p><b><a target="_blank" href=
    "https://stackoverflow.com/users/42082/ape-inago">Ape-inago</a></b> ,Jun 21, 2009 at 13:51</p>

    <blockquote>
      <pre>
$var_to_check =~ /sometest/ or die "bad variable!";
</pre>

      <p>I tend to throw things like this in my code, and later use a find and replace to get rid
      of them (in production code).</p>

      <p>Also, ' <a target="_blank" href="http://perldoc.perl.org/functions/eval.html">eval</a> '
      can be used to run a section of code and capture errors and can be used to create exception
      handling functionality. If you are asserting that a value is not 0, perhaps you want to throw
      an exception and handle that case in a special way?</p>
    </blockquote>

    <p><b>&gt;</b> ,</p>

    <blockquote>
      <pre>
if ( $next_sunrise_time &gt; 24*60*60 ) { warn( "assertion failed" ); } # Assert that the sun must rise in the next 24 hours.
</pre>

      <p>You can do this if you do not have access to Perl 5.9 which is required for <a target=
      "_blank" href="http://search.cpan.org/perldoc?Carp%3a%3aAssert">Carp::Assert</a> .</p>
    </blockquote><!--TAGS: .  -->
</blockquote>
</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20171129X_how_can_i_have_variable_assertions_in_perl"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20171117X_bruce_gray_your_perl_5_brain_on_perl_6" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20171113X_perls_worst_best_practices"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[Nov 17, 2017] <a target="_blank" href="https://www.youtube.com/watch?v=EUXclZ_d0DQ">Bruce
  Gray - Your Perl 5 Brain, on Perl 6 </a><a target="_blank" href="https://www.youtube.com/watch?v=EUXclZ_d0DQ"> </a> by Bruce Gray</h4>

  <blockquote>
    <h6>Nov 17, 2017 | <a target="_blank" href=
    "https://www.youtube.com/watch?v=EUXclZ_d0DQ">www.youtube.com</a></h6>

    <p>Published on Jun 21, 2017</p>

    <p>In which I detail the Perl 6 elements that have most changed my Perl 5 coding, and share the
    Perl 5 techniques I have adopted.</p>

    <p>I eat, sleep, live, and breathe Perl!</p>

    <p>Consultant and Contract Programmer Frequent PerlMongers speaker Dedicated Shakespeare
    theater-goer Armchair Mathematician Author of Blue_Tiger, a tool for modernizing Perl.</p>

    <p>36 years coding 22 years Perl 16 years Married 15 YAPCs 7 Hackathons 3 PerlWhirls Perl
    interests: Refactoring, Perl Idioms / Micropatterns, RosettaCode, and Perl 6.</p>
    <!--TAGS: .  -->
</blockquote>
</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20171117X_bruce_gray_your_perl_5_brain_on_perl_6"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20171113X_perls_worst_best_practices" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20120126X_appendix_b_perl_best_practices_oreilly_media"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[Nov 13, 2017] <a target="_blank" href="https://www.youtube.com/watch?v=Y1I1KcKvz9Q">&#8206;Perl&#39;s Worst Best Practices&#8206;</a> by Daina Pettit </h4>

<h5><a target="_blank" href="https://www.youtube.com/watch?v=Y1I1KcKvz9Q">YouTube 2016</a></h5>

<h5>Some good critique</h5>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20171113X_perls_worst_best_practices"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20120126X_appendix_b_perl_best_practices_oreilly_media" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20111027X_how_can_i_convert_perl_code_into_html_with_syntax_highlighting"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[Jan 26, 2012] <a target="_blank" href="http://oreilly.com/perl/excerpts/perl-best-practices/appendix-b.html">Appendix B Perl Best 
Practices - O&#39;Reilly Media</a></h4>

<h5>This is very questionable recommendations. Should be taken with  a grain  of salt</h5>
<blockquote>
   <b>Table of Contents</b><dl>
      <dt><a href="#perlbp-APP-B-SECT-1">B.1. Chapter 2, [Code Layout]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-2">B.2. Chapter 3, [Naming Conventions]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-3">B.3. Chapter 4, [Values and Expressions]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-4">B.4. Chapter 5, [Variables]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-5">B.5. Chapter 6, [Control Structures]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-6">B.6. Chapter 7, [Documentation]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-7">B.7. Chapter 8, [Built-in Functions]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-8">B.8. Chapter 9, [Subroutines]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-9">B.9. Chapter 10, [I/O]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-10">B.10. Chapter 11, [References]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-11">B.11. Chapter 12, [Regular Expressions]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-12">B.12. Chapter 13, [Error Handling]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-13">B.13. Chapter 14, [Command-Line Processing]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-14">B.14. Chapter 15, [Objects]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-15">B.15. Chapter 16, [Class Hierarchies]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-16">B.16. Chapter 17, [Modules]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-17">B.17. Chapter 18, [Testing and Debugging]</a> </dt>
      <dt><a href="#perlbp-APP-B-SECT-18">B.18. Chapter 19, [Miscellanea]</a> </dt>
   </dl>

   <p>This appendix lists the complete set of 256 guidelines presented in this book. The section heading under which each guideline 
   appears is also provided in square brackets.</p>
   <b>Chapter 2, [Code Layout]</b>
   <ul type="disc">

      <li>Brace and parenthesize in K&amp;R style. [<em>Bracketing</em>]</li>

      <li>Separate your control keywords from the following opening bracket. [<em>Keywords</em>]</li>

      <li>Don&#39;t use unnecessary parentheses for builtins and &quot;honorary&quot; builtins. [<em>Builtins</em>]</li>

      <li>Separate complex keys or indices from their surrounding brackets. [<em>Keys and Indices</em>]</li>

      <li>Use whitespace to help binary operators stand out from their operands. [<em>Operators</em>]</li>

      <li>Place a semicolon after every statement. [<em>Semicolons</em>]</li>

      <li>Place a comma after every value in a multiline list. [<em>Commas</em>]</li>

      <li>Use 78-column lines. [<em>Line Lengths</em>]</li>

      <li>Use four-column indentation levels. [<em>Indentation</em>]</li>

      <li>Indent with spaces, not tabs. [<em>Tabs</em>]</li>

      <li>Code in paragraphs. [<em>Chunking</em>]</li>

      <li>Align corresponding items vertically. [<em>Vertical Alignment</em>]</li>

      <li>Break long expressions before an operator. [<em>Breaking Long Lines</em>]</li>

      <li>Factor out long expressions in the middle of statements. [<em>Non-Terminal Expressions</em>]</li>

      <li>Always break a long expression at the operator of the lowest possible precedence. [<em>Breaking by Precedence</em>]</li>

      <li>Break long assignments before the assignment operator. [<em>Assignments</em>]</li>

      <li>Enforce your chosen layout style mechanically. [<em>Automated Layout</em>]</li>
   </ul>
   <b>Chapter 3, [Naming Conventions]</b>
   <ul type="disc">

      <li>Use grammatical templates when forming identifiers. [<em>Identifiers</em>]</li>

      <li>Name booleans after their associated test. [<em>Booleans</em>]</li>

      <li>Mark variables that store references with a _ref suffix. [<em>Reference Variables</em>]</li>

      <li>Name arrays in the plural and hashes in the singular. [<em>Arrays and Hashes</em>]</li>

      <li>Use underscores to separate words in multiword identifiers. [<em>Underscores</em>]</li>

      <li>Distinguish different program components by case. [<em>Capitalization</em>]</li>

      <li>Abbr idents by prefx. [<em>Abbreviations</em>]</li>

      <li>Abbreviate only when the meaning remains unambiguous. [<em>Ambiguous Abbreviations</em>]</li>

      <li>Avoid using inherently ambiguous words in names. [<em>Ambiguous Names</em>]</li>

      <li>Prefix &quot;for internal use only&quot; subroutines with an underscore. [<em>Utility Subroutines</em>]</li>
   </ul>
   <b>Chapter 4, [Values and Expressions]</b>
   <ul type="disc">

      <li>Use interpolating string delimiters only for strings that actually interpolate. [<em>String Delimiters</em>]</li>

      <li>Don&#39;t use &quot;&quot; or &#39;&#39; for an empty string. [<em>Empty Strings</em>]</li>

      <li>Don&#39;t write one-character strings in visually ambiguous ways. [<em>Single-Character Strings</em>]</li>

      <li>Use named character escapes instead of numeric escapes. [<em>Escaped Characters</em>]</li>

      <li>Use named constants, but don&#39;t use constant. [<em>Constants</em>]</li>

      <li>Don&#39;t pad decimal numbers with leading zeros. [<em>Leading Zeros</em>]</li>

      <li>Use underscores to improve the readability of long numbers. [<em>Long Numbers</em>]</li>

      <li>Lay out multiline strings over multiple lines. [<em>Multiline Strings</em>]</li>

      <li>Use a heredoc when a multiline string exceeds two lines. [<em>Here Documents</em>]</li>

      <li>Use a &quot;theredoc&quot; when a heredoc would compromise your indentation. [<em>Heredoc Indentation</em>]</li>

      <li>Make every heredoc terminator a single uppercase identifier with a standard prefix. [<em>Heredoc Terminators</em>]</li>

      <li>When introducing a heredoc, quote the terminator. [<em>Heredoc Quoters</em>]</li>

      <li>Don&#39;t use barewords. [<em>Barewords</em>]</li>

      <li>Reserve =&gt; for pairs. [<em>Fat Commas</em>]</li>

      <li>Don&#39;t use commas to sequence statements. [<em>Thin Commas</em>]</li>

      <li>Don&#39;t mix high- and low-precedence booleans. [<em>Low-Precedence Operators</em>]</li>

      <li>Parenthesize every rrreraw list. [<em>Lists</em>]</li>

      <li>Use table-lookup to test for membership in lists of strings; use any() for membership of lists of anything else. [<em>List 
      Membership</em>]</li>
   </ul>
   <b>Chapter 5, [Variables]</b>
   <ul type="disc">

      <li>Avoid using non-lexical variables. [<em>Lexical Variables</em>]</li>

      <li>Don&#39;t use package variables in your own development. [<em>Package Variables</em>]</li>

      <li>If you&#39;re forced to modify a package variable, localize it. [<em>Localization</em>]</li>

      <li>Initialize any variable you localize. [<em>Initialization</em>]</li>

      <li>use English for the less familiar punctuation variables. [<em>Punctuation Variables</em>]</li>

      <li>If you&#39;re forced to modify a punctuation variable, localize it. [<em>Localizing Punctuation Variables</em>]</li>

      <li>Don&#39;t use the regex match variables. [<em>Match Variables</em>]</li>

      <li>Beware of any modification via $_. [<em>Dollar-Underscore</em>]</li>

      <li>Use negative indices when counting from the end of an array. [<em>Array Indices</em>]</li>

      <li>Take advantage of hash and array slicing. [<em>Slicing</em>]</li>

      <li>Use a tabular layout for slices. [<em>Slice Layout</em>]</li>

      <li>Factor large key or index lists out of their slices. [<em>Slice Factoring</em>]</li>
   </ul>
   <b>Chapter 6, [Control Structures]</b>
   <ul type="disc">

      <li>Use block if, not postfix if. [<em>If Blocks</em>]</li>

      <li>Reserve postfix if for flow-of-control statements. [<em>Postfix Selectors</em>]</li>

      <li>Don&#39;t use postfix unless, for, while, or until. [<em>Other Postfix Modifiers</em>]</li>

      <li>Don&#39;t use unless or until at all. [<em>Negative Control Statements</em>]</li>

      <li>Avoid C-style for statements. [<em>C-Style Loops</em>]</li>

      <li>Avoid subscripting arrays or hashes within loops. [<em>Unnecessary Subscripting</em>]</li>

      <li>Never subscript more than once in a loop. [<em>Necessary Subscripting</em>]</li>

      <li>Use named lexicals as explicit for loop iterators. [<em>Iterator Variables</em>]</li>

      <li>Always declare a for loop iterator variable with my. [<em>Non-Lexical Loop Iterators</em>]</li>

      <li>Use map instead of for when generating new lists from old. [<em>List Generation</em>]</li>

      <li>Use grep and first instead of for when searching for values in a list. [<em>List Selections</em>]</li>

      <li>Use for instead of map when transforming a list in place. [<em>List Transformation</em>]</li>

      <li>Use a subroutine call to factor out complex list transformations. [<em>Complex Mappings</em>]</li>

      <li>Never modify $_ in a list function. [<em>List Processing Side Effects</em>]</li>

      <li>Avoid cascading an if. [<em>Multipart Selections</em>]</li>

      <li>Use table look-up in preference to cascaded equality tests. [<em>Value Switches</em>]</li>

      <li>When producing a value, use tabular ternaries. [<em>Tabular Ternaries</em>]</li>

      <li>Reject as many iterations as possible, as early as possible. [<em>Linear Coding</em>]</li>

      <li>Don&#39;t contort loop structures just to consolidate control. [<em>Distributed Control</em>]</li>

      <li>Use for and redo instead of an irregularly counted while. [<em>Redoing</em>]</li>

      <li>Label every loop that is exited explicitly, and use the label with every next, last, or redo. [<em>Loop Labels</em>]</li>
   </ul>
   <b>Chapter 7, [Documentation]</b>
   <ul type="disc">

      <li>Distinguish user documentation from technical documentation. [<em>Types of Documentation</em>]</li>

      <li>Create standard POD templates for modules and applications. [<em>Boilerplates</em>]</li>

      <li>Extend and customize your standard POD templates. [<em>Extended Boilerplates</em>]</li>

      <li>Put user documentation in source files. [<em>Location</em>]</li>

      <li>Keep all user documentation in a single place within your source file. [<em>Contiguity</em>]</li>

      <li>Place POD as close as possible to the end of the file. [<em>Position</em>]</li>

      <li>Subdivide your technical documentation appropriately. [<em>Technical Documentation</em>]</li>

      <li>Use block templates for major comments. [<em>Comments</em>]</li>

      <li>Use full-line comments to explain the algorithm. [<em>Algorithmic Documentation</em>]</li>

      <li>Use end-of-line comments to point out subtleties and oddities. [<em>Elucidating Documentation</em>]</li>

      <li>Comment anything that has puzzled or tricked you. [<em>Defensive Documentation</em>]</li>

      <li>Consider whether it&#39;s better to rewrite than to comment. [<em>Indicative Documentation</em>]</li>

      <li>Use &quot;invisible&quot; POD sections for longer technical discussions. [<em>Discursive Documentation</em>]</li>

      <li>Check the spelling, syntax, and sanity of your documentation. [<em>Proofreading</em>]</li>
   </ul>
   <b>Chapter 8, [Built-in Functions]</b>
   <ul type="disc">

      <li>Don&#39;t recompute sort keys inside a sort. [<em>Sorting</em>]</li>

      <li>Use reverse to reverse a list. [<em>Reversing Lists</em>]</li>

      <li>Use scalar reverse to reverse a scalar. [<em>Reversing Scalars</em>]</li>

      <li>Use unpack to extract fixed-width fields. [<em>Fixed-Width Data</em>]</li>

      <li>Use split to extract simple variable-width fields. [<em>Separated Data</em>]</li>

      <li>Use Text::CSV_XS to extract complex variable-width fields. [<em>Variable-Width Data</em>]</li>

      <li>Avoid string eval. [<em>String Evaluations</em>]</li>

      <li>Consider building your sorting routines with Sort::Maker. [<em>Automating Sorts</em>]</li>

      <li>Use 4-arg substr instead of lvalue substr. [<em>Substrings</em>]</li>

      <li>Make appropriate use of lvalue values. [<em>Hash Values</em>]</li>

      <li>Use glob, not &lt;...&gt;. [<em>Globbing</em>]</li>

      <li>Avoid a raw select for non-integer sleeps. [<em>Sleeping</em>]</li>

      <li>Always use a block with a map and grep. [<em>Mapping and Grepping</em>]</li>

      <li>Use the &quot;non-builtin builtins&quot;. [<em>Utilities</em>]</li>
   </ul>
   <b>Chapter 9, [Subroutines]</b>
   <ul type="disc">

      <li>Call subroutines with parentheses but without a leading &amp;. [<em>Call Syntax</em>]</li>

      <li>Don&#39;t give subroutines the same names as built-in functions. [<em>Homonyms</em>]</li>

      <li>Always unpack @_ first. [<em>Argument Lists</em>]</li>

      <li>Use a hash of named arguments for any subroutine that has more than three parameters. [<em>Named 
      Arguments</em>]</li>

      <li>Use definedness or existence to test for missing arguments. [<em>Missing Arguments</em>]</li>

      <li>Resolve any default argument values as soon as @_ is unpacked. [<em>Default Argument Values</em>]</li>

      <li>Always return scalar in scalar returns. [<em>Scalar Return Values</em>]</li>

      <li>Make list-returning subroutines return the &quot;obvious&quot; value in scalar context. [<em>Contextual Return Values</em>]</li>

      <li>When there is no &quot;obvious&quot; scalar context return value, consider Contextual::Return instead. [<em>Multi-Contextual Return 
      Values</em>]</li>

      <li>Don&#39;t use subroutine prototypes. [<em>Prototypes</em>]</li>

      <li>Always return via an explicit return. [<em>Implicit Returns</em>]</li>

      <li>Use a bare return to return failure. [<em>Returning Failure</em>]</li>
   </ul>
   <b>Chapter 10, [I/O]</b>
   <ul type="disc">

      <li>Don&#39;t use bareword filehandles. [<em>Filehandles</em>]</li>

      <li>Use indirect filehandles. [<em>Indirect Filehandles</em>]</li>

      <li>If you have to use a package filehandle, localize it first. [<em>Localizing Filehandles</em>]</li>

      <li>Use either the IO::File module or the three-argument form of open. [<em>Opening Cleanly</em>]</li>

      <li>Never open, close, or print to a file without checking the outcome. [<em>Error Checking</em>]</li>

      <li>Close filehandles explicitly, and as soon as possible. [<em>Cleanup</em>]</li>

      <li>Use while (&lt;&gt;), not for (&lt;&gt;). [<em>Input Loops</em>]</li>

      <li>Prefer line-based I/O to slurping. [<em>Line-Based Input</em>]</li>

      <li>Slurp a filehandle with a do block for purity. [<em>Simple Slurping</em>]</li>

      <li>Slurp a stream with Perl6::Slurp for power and simplicity. [<em>Power Slurping</em>]</li>

      <li>Avoid using *STDIN, unless you really mean it. [<em>Standard Input</em>]</li>

      <li>Always put filehandles in braces within any print statement. [<em>Printing to Filehandles</em>]</li>

      <li>Always prompt for interactive input. [<em>Simple Prompting</em>]</li>

      <li>Don&#39;t reinvent the standard test for interactivity. [<em>Interactivity</em>]</li>

      <li>Use the IO::Prompt module for prompting. [<em>Power Prompting</em>]</li>

      <li>Always convey the progress of long non-interactive operations within interactive applications. [<em>Progress Indicators</em>]</li>

      <li>Consider using the Smart::Comments module to automate your progress indicators. [<em>Automatic Progress Indicators</em>]</li>

      <li>Avoid a raw select when setting autoflushes. [<em>Autoflushing</em>]</li>
   </ul>
   <b>Chapter 11, [References]</b>
   <ul type="disc">

      <li>Wherever possible, dereference with arrows. [<em>Dereferencing</em>]</li>

      <li>Where prefix dereferencing<a name="IDX-APP-B-0122"></a> is unavoidable, put braces around the reference. [<em>Braced References</em>]</li>

      <li>Never use symbolic references<a name="IDX-APP-B-0123"></a>. [<em>Symbolic References</em>]</li>

      <li>Use weaken to prevent circular data structures from leaking memory. [<em>Cyclic References</em>]</li>
   </ul>
   <b>Chapter 12, [Regular Expressions]</b>
   <ul type="disc">

      <li>Always use the /x flag. [<em>Extended Formatting</em>]</li>

      <li>Always use the /m flag. [<em>Line Boundaries</em>]</li>

      <li>Use \A and \z as string boundary anchors. [<em>String Boundaries</em>]</li>

      <li>Use \z, not \Z, to indicate &quot;end of string&quot;. [<em>End of String</em>]</li>

      <li>Always use the /s flag. [<em>Matching Anything</em>]</li>

      <li>Consider mandating the Regexp::Autoflags module. [<em>Lazy Flags</em>]</li>

      <li>Use m{...} in preference to /.../ in multiline regexes. [<em>Brace Delimiters</em>]</li>

      <li>Don&#39;t use any delimiters other than /.../ or m{...}. [<em>Other Delimiters</em>]</li>

      <li>Prefer singular character classes to escaped metacharacters<a name="IDX-APP-B-0124"></a>. [<em>Metacharacters</em>]</li>

      <li>Prefer named characters to escaped metacharacters. [<em>Named Characters</em>]</li>

      <li>Prefer properties to enumerated character classes. [<em>Properties</em>]</li>

      <li>Consider matching arbitrary whitespace, rather than specific whitespace characters. [<em>Whitespace</em>]</li>

      <li>Be specific when matching &quot;as much as possible&quot;. [<em>Unconstrained Repetitions</em>]</li>

      <li>Use capturing parentheses only when you intend to capture<a name="IDX-APP-B-0125"></a>. [<em>Capturing Parentheses</em>]</li>

      <li>Use the numeric capture variables<a name="IDX-APP-B-0126"></a> only when you&#39;re sure that the preceding match succeeded. [<em>Captured 
      Values</em>]</li>

      <li>Always give captured substrings proper names. [<em>Capture Variables</em>]</li>

      <li>Tokenize input using the /gc flag. [<em>Piecewise Matching</em>]</li>

      <li>Build regular expressions from tables. [<em>Tabular Regexes</em>]</li>

      <li>Build complex regular expressions from simpler pieces. [<em>Constructing Regexes</em>]</li>

      <li>Consider using Regexp::Common instead of writing your own regexes. [<em>Canned Regexes</em>]</li>

      <li>Always use character classes instead of single-character alternations. [<em>Alternations</em>]</li>

      <li>Factor out common affixes from alternations. [<em>Factoring Alternations</em>]</li>

      <li>Prevent useless backtracking. [<em>Backtracking</em>]</li>

      <li>Prefer fixed-string eq comparisons to fixed-pattern regex matches. [<em>String Comparisons</em>]</li>
   </ul>
   <b>Chapter 13, [Error Handling]</b>
   <ul type="disc">

      <li>Throw exceptions instead of returning special values or setting flags. [<em>Exceptions</em>]</li>

      <li>Make failed builtins throw exceptions too. [<em>Builtin Failures</em>]</li>

      <li>Make failures fatal in all contexts. [<em>Contextual Failure</em>]</li>

      <li>Be careful when testing for failure of the system builtin. [<em>Systemic Failure</em>]</li>

      <li>Throw exceptions on all failures, including recoverable ones. [<em>Recoverable Failure</em>]</li>

      <li>Have exceptions report from the caller&#39;s location, not from the place where they were thrown. [<em>Reporting Failure</em>]</li>

      <li>Compose error messages in the recipient&#39;s dialect. [<em>Error Messages</em>]</li>

      <li>Document every error message in the recipient&#39;s dialect. [<em>Documenting Errors</em>]</li>

      <li>Use exception objects whenever failure data needs to be conveyed to a handler. [<em>OO Exceptions</em>]</li>

      <li>Use exception objects when error messages may change. [<em>Volatile Error Messages</em>]</li>

      <li>Use exception objects when two or more exceptions are related. [<em>Exception Hierarchies</em>]</li>

      <li>Catch exception objects in most-derived-first order. [<em>Processing Exceptions</em>]</li>

      <li>Build exception classes automatically. [<em>Exception Classes</em>]</li>

      <li>Unpack the exception variable in extended exception handlers. [<em>Unpacking Exceptions</em>]</li>
   </ul>
   <b>Chapter 14, [Command-Line Processing]</b>
   <ul type="disc">

      <li>Enforce a single consistent command-line structure. [<em>Command-Line Structure</em>]</li>

      <li>Adhere to a standard set of conventions in your command-line syntax. [<em>Command-Line Conventions</em>]</li>

      <li>Standardize your meta-options. [<em>Meta-options</em>]</li>

      <li>Allow the same filename to be specified for both input and output. [<em>In-situ Arguments</em>]</li>

      <li>Standardize on a single approach to command-line processing. [<em>Command-Line Processing</em>]</li>

      <li>Ensure that your interface, run-time messages, and documentation remain consistent. [<em>Interface Consistency</em>]</li>

      <li>Factor out common command-line interface components into a shared module. [<em>Interapplication Consistency</em>]</li>
   </ul>
   <b>Chapter 15, [Objects]</b>
   <ul type="disc">

      <li>Make object orientation a choice, not a default. [<em>Using OO</em>]</li>

      <li>Choose object orientation using appropriate criteria. [<em>Criteria</em>]</li>

      <li>Don&#39;t use pseudohashes. [<em>Pseudohashes</em>]</li>

      <li>Don&#39;t use restricted hashes. [<em>Restricted Hashes</em>]</li>

      <li>Always use fully encapsulated objects. [<em>Encapsulation</em>]</li>

      <li>Give every constructor the same standard name. [<em>Constructors</em>]</li>

      <li>Don&#39;t let a constructor clone objects. [<em>Cloning</em>]</li>

      <li>Always provide a destructor for every inside-out class. [<em>Destructors</em>]</li>

      <li>When creating methods, follow the general guidelines for subroutines. [<em>Methods</em>]</li>

      <li>Provide separate read and write accessors. [<em>Accessors</em>]</li>

      <li>Don&#39;t use lvalue accessors. [<em>Lvalue Accessors</em>]</li>

      <li>Don&#39;t use the indirect<a name="IDX-APP-B-0159"></a> object syntax. [<em>Indirect Objects</em>]</li>

      <li>Provide an optimal interface, rather than a minimal one. [<em>Class Interfaces</em>]</li>

      <li>Overload only the isomorphic operators of algebraic classes. [<em>Operator Overloading</em>]</li>

      <li>Always consider overloading the boolean, numeric, and string coercions of objects. [<em>Coercions</em>]</li>
   </ul>
   <b>Chapter 16, [Class Hierarchies]</b>
   <ul type="disc">

      <li>Don&#39;t manipulate the list of base classes directly. [<em>Inheritance</em>]</li>

      <li>Use distributed encapsulated objects. [<em>Objects</em>]</li>

      <li>Never use the one-argument form of bless. [<em>Blessing Objects</em>]</li>

      <li>Pass constructor arguments as labeled values, using a hash reference. [<em>Constructor Arguments</em>]</li>

      <li>Distinguish arguments for base classes by class name as well. [<em>Base Class Initialization</em>]</li>

      <li>Separate your construction, initialization, and destruction processes. [<em>Construction and Destruction</em>]</li>

      <li>Build the standard class infrastructure automatically. [<em>Automating Class Hierarchies</em>]</li>

      <li>Use Class::Std to automate the deallocation of attribute data. [<em>Attribute Demolition</em>]</li>

      <li>Have attributes initialized and verified automatically. [<em>Attribute Building</em>]</li>

      <li>Specify coercions as :STRINGIFY, :NUMERIFY, and :BOOLIFY methods. [<em>Coercions</em>]</li>

      <li>Use :CUMULATIVE methods instead of SUPER:: calls. [<em>Cumulative Methods</em>]</li>

      <li>Don&#39;t use AUTOLOAD(). [<em>Autoloading</em>]</li>
   </ul>
   <b>Chapter 17, [Modules]</b>
   <ul type="disc">

      <li>Design the module&#39;s interface first. [<em>Interfaces</em>]</li>

      <li>Place original code inline. Place duplicated code in a subroutine. Place duplicated subroutines in a module. [<em>Refactoring</em>]</li>

      <li>Use three-part version numbers. [<em>Version Numbers</em>]</li>

      <li>Enforce your version requirements programmatically. [<em>Version Requirements</em>]</li>

      <li>Export judiciously and, where possible, only by request. [<em>Exporting</em>]</li>

      <li>Consider exporting declaratively. [<em>Declarative Exporting</em>]</li>

      <li>Never make variables part of a module&#39;s interface. [<em>Interface Variables</em>]</li>

      <li>Build new module frameworks automatically. [<em>Creating Modules</em>]</li>

      <li>Use core modules wherever possible. [<em>The Standard Library</em>]</li>

      <li>Use CPAN modules where feasible. [<em>CPAN</em>]</li>
   </ul>

   <p><b>Chapter 18, [Testing and Debugging]</b></p>
   <ul type="disc">

      <li>Write the test cases first. [<em>Test Cases</em>]</li>

      <li>Standardize your tests with Test::Simple or Test::More. [<em>Modular Testing</em>]</li>

      <li>Standardize your test suites with Test::Harness. [<em>Test Suites</em>]</li>

      <li>Write test cases that fail. [<em>Failure</em>]</li>

      <li>Test both the likely and the unlikely. [<em>What to Test</em>]</li>

      <li>Add new test cases before you start debugging. [<em>Debugging and Testing</em>]</li>

      <li>Always use strict. [<em>Strictures</em>]</li>

      <li>Always turn on warnings explicitly. [<em>Warnings</em>]</li>

      <li>Never assume that a warning-free compilation implies correctness. [<em>Correctness</em>]</li>

      <li>Turn off strictures or warnings explicitly, selectively, and in the smallest possible scope. [<em>Overriding Strictures</em>]</li>

      <li>Learn at least a subset of the <em>perl</em> debugger. [<em>The Debugger</em>]</li>

      <li>Use serialized warnings when debugging &quot;manually&quot;. [<em>Manual Debugging</em>]</li>

      <li>Consider using &quot;smart comments&quot; when debugging, rather than warn statements. [<em>Semi-Automatic Debugging</em>]</li>
   </ul>

   <p><b>Chapter 19, [Miscellanea]</b></p>
   <ul type="disc">

      <li>Use a revision control system. [<em>Revision Control</em>]</li>

      <li>Integrate non-Perl code into your applications via the Inline:: modules. [<em>Other Languages</em>]</li>

      <li>Keep your configuration language uncomplicated. [<em>Configuration Files</em>]</li>

      <li>Don&#39;t use formats. [<em>Formats</em>]</li>

      <li>Don&#39;t tie variables or filehandles. [<em>Ties</em>]</li>

      <li>Don&#39;t be clever. [<em>Cleverness</em>]</li>

      <li>If you must rely on cleverness, encapsulate it. [<em>Encapsulated Cleverness</em>]</li>

      <li>Don&#39;t optimize code-benchmark it. [<em>Benchmarking</em>]</li>

      <li>Don&#39;t optimize data structures-measure them. [<em>Memory</em>]</li>

      <li>Look for opportunities to use caches. [<em>Caching</em>]</li>

      <li>Automate your subroutine caching. [<em>Memoization</em>]</li>

      <li>Benchmark any caching strategy you use. [<em>Caching for Optimization</em>]</li>

      <li>Don&#39;t optimize applications-profile them. [<em>Profiling</em>]</li>

      <li>Be careful to preserve semantics when refactoring syntax. [<em>Enbugging</em>]</li>
   </ul>
</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20120126X_appendix_b_perl_best_practices_oreilly_media"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20111027X_how_can_i_convert_perl_code_into_html_with_syntax_highlighting" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20111027X_togotutor"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[Oct 27, 2011] <a target="_blank" href="http://www.perlmonks.org/?node_id=174891">How can I convert Perl code into HTML with syntax 
highlighting?</a></h4>

<h5><a target="_blank" href="http://www.vim.org/">vim</a> can do it. </h5>
<blockquote>

   <h6>Jun 16, 2002 | <a target="_blank" href="http://www.perlmonks.org">www.perlmonks.org</a></h6>

   <p><b>ChemBoy</b></p>
   <blockquote>

      <p>Use <a target="_blank" href="http://perltidy.sourceforge.net">Perltidy</a>&#39;s HTML formatting options (<tt>-html</tt> etc.) 
      As <a target="_blank" href="http://perltidy.sourceforge.net/perltidy.html#html%20options">shown</a> in the Perltidy manual, you 
      can generate a whole page or a simple preformatted section, with embedded or linked style sheets, and set different colors for 
      all manner of different constructs. <a href="?node_id=174905">Answer: How can I convert Perl code into HTML with syntax highlighting?</a><br>
      </p>
   </blockquote>

   <p><b><a href="?node_id=133637">yodabjorn</a> </b></p>
   <blockquote>

      <p><a target="_blank" href="http://www.vim.org/">vim</a> can do it. </p>
   </blockquote>
   <blockquote>
      <tt><font size="-1">:runtime! syntax/2html.vim </font></tt><font size="-1">

      <p><a href="?abspart=1;displaytype=displaycode;node_id=174891;part=1">[download]</a></p>
      </font>

      <p>To view the help page: </p>
      <tt><font size="-1">:help 2html </font></tt><font size="-1">

      <p><a href="?abspart=1;displaytype=displaycode;node_id=174891;part=2">[download]</a></p>
   </blockquote>
   </font>
   <blockquote>
      Details are in the vim help under &quot;convert-to-HTML&quot;.
   </blockquote>

   <p><b><a href="?node_id=43868">epoptai</a> </b></p>
   <blockquote>

      <p><a target="_blank" href="http://www.palfrader.org/code2html/">Code2HTML</a> is a free Perl script that syntax highlights 15 
      different programming languages. </p>
   </blockquote>

   <p><b><a href="?node_id=832495">choroba</a></b> </p>
   <blockquote>
      <a target="_blank" href="http://www.gnu.org/software/emacs/">Emacs</a> can do it. I often use the
      <a target="_blank" href="http://www.emacswiki.org/emacs/HtmlFontify">htmlfontify</a> library.
   </blockquote>
</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20111027X_how_can_i_convert_perl_code_into_html_with_syntax_highlighting"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20111027X_togotutor" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20111027X_perl_style_guide_juerds_site"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4><a href="?node_id=854709">togotutor</a> </h4>
<blockquote>
   There is a robust online Source Code to HTML option at
   <a target="_blank" href="http://www.togotutor.com/code-to-html/perl-to-html.php">ToGoTutor - Code2Html</a> . The website has tools 
   for Perl, Java and other languages.&lt;</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20111027X_togotutor"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20111027X_perl_style_guide_juerds_site" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20050714X_perl_com_ten_essential_development_practices"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4><a target="_blank" href="http://juerd.nl/site.plp/perlstyle">Perl style guide - Juerd&#39;s site </a></h4>
<blockquote>

   <p>This is how I like my code, in no specific order. :)</p>
   <ul>

      <li>4 space indents </li>

      <li>No tabs in code (includes indents) </li>

      <li>Always <tt>Class-&gt;method</tt>, never <tt>method Class</tt> (this includes &quot;new&quot;!) </li>

      <li>Cuddled else: <tt>} else {</tt> </li>

      <li>Opening curly on the same line as the keyword it belongs to </li>

      <li>Closing vertically aligned with that keyword </li>

      <li>Space after comma or semi-colon, but not before </li>

      <li>No extra spaces around or inside parens: <tt>foo, (bar, baz), quux</tt> </li>

      <li>Extra spaces in arrayref constructor: <tt>[ foo, bar ]</tt> </li>

      <li>Extra spaces in hashref constructor: <tt>{ foo =&gt; bar }</tt> </li>

      <li>Extra spaces in code delimiting curlies: <tt>sort { $a &lt;=&gt; $b } @foo</tt> </li>

      <li>No <tt>$a</tt> or <tt>$b</tt> except when sorting </li>

      <li>No parens unless needed for clarity </li>

      <li>Space between special keyword and its arguments: <tt>if (...) { ... }</tt> </li>

      <li>No space between keyword and its arguments if the &quot;looks like a function, therefore it is a function&quot; rule applies: <nobr>
      <tt>print((split)[22])</tt></nobr>, not <nobr><tt>print ((split)[22])</tt></nobr>. (And of course not <nobr><tt>print (split)[22]</tt></nobr>)
      </li>

      <li>No subroutine prototypes if they&#39;re ignored anyway </li>

      <li>No subroutine prototypes just to hint the number of arguments </li>

      <li>Prototypes enforce context, so use them only if that makes sense </li>

      <li>No globals when access from another package is not needed </li>

      <li><tt>use strict</tt> and <tt>-w</tt>. Loading of normal modules comes after loading strict. </li>

      <li>Lots of modules, but not to replace few-liners or simple regexes </li>

      <li>Comments on code lines have two spaces before and one after the # symbol </li>

      <li>No double spaces except for vertical alignment and comments </li>

      <li>Only <tt>&amp;&amp; || !</tt> where parens would be needed with <tt>and or not</tt> </li>

      <li>No double empty lines </li>

      <li>Empty line between logical code chunks </li>

      <li>Explicit returns from subs </li>

      <li>Guards (<tt>return if ...</tt>) are nicer than large else-blocks </li>

      <li>No space between array/hash and index/key: <tt>$foo[0], $foo{bar}</tt> </li>

      <li>No quotes for simple literal hash keys </li>

      <li>Space around index/key if it is complex: <tt>$foo{ $bar{baz}{bar} }</tt> </li>

      <li>Long lines: indent according to parens, but always 4 spaces (or <tt>[]</tt>, <tt>{}</tt>, etc) </li>

      <li>Long lines: continuing lines are indented </li>

      <li>Long lines: Lines end with operator, unless it&#39;s <tt>|| &amp;&amp; and or</tt> </li>

      <li>No &quot;outdent&quot;s </li>

      <li>No half indents </li>

      <li>No double indents </li>

      <li><tt>grep EXPR</tt> and <tt>map EXPR</tt> when BLOCK is not needed </li>

      <li>Logical order in comparisons: <tt>$foo == 4</tt>, but never <tt>4 == $foo</tt> </li>

      <li>English identifiers </li>

      <li>Not the English.pm module </li>

      <li>Multi-word identifiers have no separation, or are separated by underscores </li>

      <li>Lowercase identifiers, but uppercase for constants </li>

      <li>Whatever tool is useful: no OO when it does not make sense </li>

      <li>It&#39;s okay to import symbols </li>

      <li>No here-documents, but multi-line q/qq. Even repeated prints are better :) (Okay, here-docs can be used when they&#39;re far away 
      from code that contains any logic. Code MUST NOT break when (un)indented.) </li>

      <li>Always check return values where they are important </li>

      <li>No spaces around: <tt>-&gt; **</tt> </li>

      <li>Spaces around: <tt>=~ !~ * / % + - . &lt;&lt; &gt;&gt;</tt> comparison_ops <tt>&amp; | ^ &amp;&amp; || ?:</tt> assignment_ops <tt>=&gt; and or xor</tt>
      </li>

      <li>Spaces or no spaces, depending on complexity: <tt>.. ... x</tt> </li>

      <li>No space after, unless complex: <tt>~ u+ u-</tt> </li>

      <li>Long lines: break between method calls, <tt>-&gt;</tt> comes first on a line, space after it </li>

      <li><tt>=&gt;</tt> where it makes sense </li>

      <li><tt>qw</tt> where useful </li>

      <li><tt>qw</tt> when importing, but <tt>&#39;&#39;</tt> when specifying pragma behaviour </li>

      <li><tt>()</tt> for empty list, not <tt>qw()</tt> </li>

      <li><tt>-&gt;</tt> to dereference, where possible </li>

      <li>No abbreviations (acronyms are okay, and so are VERY common abbreviations) NEVER &quot;ary&quot; </li>

      <li>Data type not represented in variable name: <tt>%foo</tt> and <tt>@foo</tt>, but not <tt>%foo_hash</tt> or <tt>@foo_array</tt>
      </li>

      <li>Sometimes: data type of referent in reference variable names: <tt>$bla_hash</tt> is okay </li>

      <li>Sometimes: data type &#39;reference&#39; in reference variable names: <tt>$hashref</tt> is okay </li>

      <li>No one-letter variable names, unless $i or alike </li>

      <li><tt>$i</tt> is a(n index) counter </li>

      <li>Dummy variables can be called foo, bar, baz, quux or just dummy </li>

      <li>Taint mode *only* for setuid programs </li>

      <li>No sub main(), unless it needs to be called more often than once </li>

      <li>Subs before main code! </li>

      <li>Declare variables on first use, not before (unless required) </li>

      <li>\cM &gt; \x0d &gt; \015. \r only where it makes sense as carriage return. </li>

      <li>Complex regexes get <tt>/x</tt> </li>

      <li>No space between <tt>++</tt>/<tt>--</tt> and the variable </li>

      <li>List assignment for parameters/arguments, not lots of shifts </li>

      <li>Only shift $self from @_ if @_ is used elsewhere in the sub </li>

      <li>Direct @_ access is okay in very short subs </li>

      <li>No eval STRING if not needed </li>

      <li>Constructor &quot;new&quot; does not clone. Only handles a *class* as <tt>$_[0]</tt> </li>

      <li>Constructor that clones is called &quot;clone&quot; </li>

      <li>Constructor can be something else than &quot;new&quot;, but &quot;new&quot; is an alias </li>

      <li>No setting of <tt>$|</tt> when it is not needed </li>

      <li>Lexical filehandles </li>

      <li>No v-strings </li>

      <li>Single quotes when double-quote features not used </li>

      <li>In DBI: value interpolation using placeholders only </li>

      <li><tt>use base &#39;BaseClass&#39;</tt> instead of <tt>use BaseClass</tt> and setting <tt>@ISA</tt> </li>

      <li>Comments where code is unclear </li>

      <li>Comments usually explain the WHY, not the HOW </li>

      <li>POD at the bottom, not top, not interleaved </li>

      <li>Sane variable scopes </li>

      <li>No local, except for perlvar vars </li>

      <li>No C-style loop for skipless iteration </li>

      <li>No looping over indexes if only the element is used </li>

      <li>80 characters width. It&#39;s okay to give up some whitespace </li>

      <li>Unbalanced custom delimiters are not metacharacters and not alphanumeric </li>

      <li>RHS of complex <tt>s///e</tt> is delimited by <tt>{}</tt> </li>

      <li>Favourite custom delimiter is <tt>[]</tt> </li>

      <li>Semi-colon only left out for implicit return or in single-statement block </li>

      <li>No <tt>$&amp;</tt>, <tt>$`</tt> or <tt>$&#39;</tt> </li>

      <li>Localization of globals if they&#39;re to be changed (<tt>local $_</tt> often avoids weird bugs) </li>

      <li>Semi-colon not on its own line </li>

      <li>(in|de)crement in void context is post(in|de)crement </li>

      <li>No <tt>map</tt> or <tt>grep</tt> in void context </li>

      <li><tt>?</tt> and <tt>:</tt> begin lines in complex expressions </li>

      <li>True and false are always implied. No <tt>$foo == 0</tt> when testing for truth. </li>

      <li>Only constructors return $self. Accessor methods never do this. </li>

      <li>Stacking methods is okay, but a non-constructor method should never return $self. </li>

      <li>Accessor methods should behave like variables (<a title="Off-site link: http://search.cpan.org/perldoc?Attribute::Property" target="_blank" href="http://juerd.nl/elsewhere.plp?href=http://search.cpan.org/perldoc?Attribute::Property">Attribute::Property</a>!)
      </li>

      <li>Other methods should behave like subroutines </li>

      <li><tt>our $VERSION</tt>, not <tt>use vars qw($VERSION);</tt> </li>

      <li>Module version numbers are ^\d+\.\d\d\z </li>

      <li>Error checking is done using <tt>or</tt>. This means <tt>open or do { ... }</tt> instead of <tt>unless (open) { ... }</tt> 
      when handling the error is more than a simple statement. </li>

      <li>The result of the modulus operator (<tt>%</tt>) has no useful boolean meaning (it is reversed), so explicit <tt>== 0</tt> 
      should be used.</li>
   </ul>
</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20111027X_perl_style_guide_juerds_site"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20050714X_perl_com_ten_essential_development_practices" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#NEWS_TOC"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[July 14, 2005] <a target="_blank" href="http://www.perl.com/lpt/a/2005/07/14/bestpractices.html">Perl.com- Ten Essential Development 
Practices</a> by Damian Conway</h4>

<h5>Beware, Damian Conway is an OO fundamentalist... </h5>
<blockquote>

   <p>The following ten tips come from <em><a target="_blank" href="http://www.oreilly.com/catalog/perlbp/">Perl Best Practices</a></em>, 
   a new book of Perl coding and development guidelines by Damian Conway.</p>

   <h5>1. Design the Module&#39;s Interface First</h5>

   <p>The most important aspect of any module is not how it implements the facilities it provides, but the way in which it provides 
   those facilities in the first place. If the module&#39;s API is too awkward, or too complex, or too extensive, or too fragmented, or 
   even just poorly named, developers will avoid using it. They&#39;ll write their own code instead. In that way, a poorly designed module 
   can actually reduce the overall maintainability of a system.</p>

   <p>Designing module interfaces requires both experience and creativity. Perhaps the easiest way to work out how an interface should 
   work is to &quot;play test&quot; it: to write examples of code that will use the module before implementing the module itself. These examples 
   will not be wasted when the design is complete. You can usually recycle them into demos, documentation examples, or the core of a 
   test suite.</p>

   <p>The key, however, is to write that code as if the module were already available, and write it the way you&#39;d most like the module 
   to work.</p>

   <p>Once you have some idea of the interface you want to create, convert your &quot;play tests&quot; into actual tests (see Tip #2). Then it&#39;s 
   just a Simple Matter Of Programming to make the module work the way that the code examples and the tests want it to.</p>

   <p>Of course, it may not be possible for the module to work the way you&#39;d most like, in which case attempting to implement it that 
   way will help you determine what aspects of your API are not practical, and allow you to work out what might be an acceptable alternative.</p>

   <h5>2. Write the Test Cases Before the Code</h5>

   <p>Probably the single best practice in all of software development is writing your test suite first.</p>

   <p>A test suite is an executable, self-verifying specification of the behavior of a piece of software. If you have a test suite, 
   you can--at any point in the development process--verify that the code works as expected. If you have a test suite, you can--after 
   any changes during the maintenance cycle--verify that the code still works as expected.</p>

   <p>Write the tests first. Write them as soon as you know what your interface will be (see #1). Write them before you start coding 
   your application or module. Unless you have tests, you have no unequivocal specification of what the software should do, and no way 
   of knowing whether it does it.</p>

   <p>Writing tests always seems like a chore, and an unproductive chore at that: you don&#39;t have anything to test yet, so why write 
   tests? Yet most developers will--almost automatically--write driver software to test their new module in an ad hoc way:</p>

   <pre><tt>&gt; cat try_inflections.pl # Test my shiny new English inflections module... use Lingua::EN::Inflect qw( inflect ); # Try some plurals (both standard and unusual inflections)... my %plural_of = ( &#39;house&#39; =&gt; &#39;houses&#39;, &#39;mouse&#39; =&gt; &#39;mice&#39;, &#39;box&#39; =&gt; &#39;boxes&#39;, &#39;ox&#39; =&gt; &#39;oxen&#39;, &#39;goose&#39; =&gt; &#39;geese&#39;, &#39;mongoose&#39; =&gt; &#39;mongooses&#39;, &#39;law&#39; =&gt; &#39;laws&#39;, &#39;mother-in-law&#39; =&gt; &#39;mothers-in-law&#39;, ); # For each of them, print both the expected result and the actual inflection... for my $word ( keys %plural_of ) { my $expected = $plural_of{$word}; my $computed = inflect( &quot;PL_N($word)&quot; ); print &quot;For $word:\n&quot;, &quot;\tExpected: $expected\n&quot;, &quot;\tComputed: $computed\n&quot;; }</tt></pre>

   <p>A driver like that is actually harder to write than a test suite, because you have to worry about formatting the output in a way 
   that is easy to read. It&#39;s also much harder to use the driver than it would be to use a test suite, because every time you run it 
   you have to wade though that formatted output and verify &quot;by eye&quot; that everything is as it should be. That&#39;s also error-prone; eyes 
   are not optimized for picking out small differences in the middle of large amounts of nearly identical text.</p>

   <p>Instead of hacking together a driver program, it&#39;s easier to write a test program using the standard
   <a target="_blank" href="http://search.cpan.org/perldoc?Test::Simple">Test::Simple</a> module. Instead of <tt>print</tt> statements 
   showing what&#39;s being tested, you just write calls to the <tt>ok()</tt> subroutine, specifying as its first argument the condition 
   under which things are okay, and as its second argument a description of what you&#39;re actually testing:</p>

   <pre><tt>&gt; cat inflections.t use Lingua::EN::Inflect qw( inflect); use Test::Simple qw( no_plan); my %plural_of = ( &#39;mouse&#39; =&gt; &#39;mice&#39;, &#39;house&#39; =&gt; &#39;houses&#39;, &#39;ox&#39; =&gt; &#39;oxen&#39;, &#39;box&#39; =&gt; &#39;boxes&#39;, &#39;goose&#39; =&gt; &#39;geese&#39;, &#39;mongoose&#39; =&gt; &#39;mongooses&#39;, &#39;law&#39; =&gt; &#39;laws&#39;, &#39;mother-in-law&#39; =&gt; &#39;mothers-in-law&#39;, ); for my $word ( keys %plural_of ) { my $expected = $plural_of{$word}; my $computed = inflect( &quot;PL_N($word)&quot; ); ok( $computed eq $expected, &quot;$word -&gt; $expected&quot; ); }</tt></pre>

   <p>Note that this code loads <tt>Test::Simple</tt> with the argument <tt>qw( no_plan )</tt>. Normally that argument would be <tt>
   tests =&gt; count</tt>, indicating how many tests to expect, but here the tests are generated from the <tt>%plural_of</tt> table at 
   run time, so the final count will depend on how many entries are in that table. Specifying a fixed number of tests when loading the 
   module is useful if you happen know that number at compile time, because then the module can also &quot;meta-test:&quot; verify that you carried 
   out all the tests you expected to.</p>

   <p>The <tt>Test::Simple</tt> program is slightly more concise and readable than the original driver code, and the output is much 
   more compact and informative:</p>

   <pre><tt>&gt; perl inflections.t ok 1 - house -&gt; houses ok 2 - law -&gt; laws not ok 3 - mongoose -&gt; mongooses # Failed test (inflections.t at line 21) ok 4 - goose -&gt; geese ok 5 - ox -&gt; oxen not ok 6 - mother-in-law -&gt; mothers-in-law # Failed test (inflections.t at line 21) ok 7 - mouse -&gt; mice ok 8 - box -&gt; boxes 1..8 # Looks like you failed 2 tests of 8. </tt></pre>

   <p>More importantly, this version requires far less effort to verify the correctness of each test. You just scan down the left margin 
   looking for a <tt>not</tt> and a comment line.</p>

   <p>You might prefer to use the <a target="_blank" href="http://search.cpan.org/perldoc?Test::More">Test::More</a> module instead 
   of <tt>Test::Simple</tt>. Then you can specify the actual and expected values separately, by using the <tt>is()</tt> subroutine, 
   rather than <tt>ok()</tt>:</p>

   <pre><tt>use Lingua::EN::Inflect qw( inflect ); use Test::More qw( no_plan ); # Now using more advanced testing tools my %plural_of = ( &#39;mouse&#39; =&gt; &#39;mice&#39;, &#39;house&#39; =&gt; &#39;houses&#39;, &#39;ox&#39; =&gt; &#39;oxen&#39;, &#39;box&#39; =&gt; &#39;boxes&#39;, &#39;goose&#39; =&gt; &#39;geese&#39;, &#39;mongoose&#39; =&gt; &#39;mongooses&#39;, &#39;law&#39; =&gt; &#39;laws&#39;, &#39;mother-in-law&#39; =&gt; &#39;mothers-in-law&#39;, ); for my $word ( keys %plural_of ) { my $expected = $plural_of{$word}; my $computed = inflect( &quot;PL_N($word)&quot; ); # Test expected and computed inflections for string equality... is( $computed, $expected, &quot;$word -&gt; $expected&quot; ); }</tt></pre>

   <p>Apart from no longer having to type the <tt>eq</tt> yourself, this version also produces more detailed error messages:</p>

   <pre><tt>&gt; perl inflections.t ok 1 - house -&gt; houses ok 2 - law -&gt; laws not ok 3 - mongoose -&gt; mongooses # Failed test (inflections.t at line 20) # got: &#39;mongeese&#39; # expected: &#39;mongooses&#39; ok 4 - goose -&gt; geese ok 5 - ox -&gt; oxen not ok 6 - mother-in-law -&gt; mothers-in-law # Failed test (inflections.t at line 20) # got: &#39;mothers-in-laws&#39; # expected: &#39;mothers-in-law&#39; ok 7 - mouse -&gt; mice ok 8 - box -&gt; boxes 1..8 # Looks like you failed 2 tests of 8.</tt></pre>

   <p>The <a target="_blank" href="http://search.cpan.org/perldoc?Test::Tutorial">Test::Tutorial</a> documentation that comes with Perl 
   5.8 provides a gentle introduction to both <tt>Test::Simple</tt> and <tt>Test::More</tt>.</p>

   <h5>3. Create Standard POD Templates for Modules and Applications</h5>

   <p>One of the main reasons documentation can often seem so unpleasant is the &quot;blank page effect.&quot; Many programmers simply don&#39;t know 
   how to get started or what to say.</p>

   <p>Perhaps the easiest way to make writing documentation less forbidding (and hence, more likely to actually occur) is to circumvent 
   that initial empty screen by providing a template that developers can cut and paste into their code.</p>

   <p>For a module, that documentation template might look something like this:</p>

   <pre><tt>=head1 NAME &lt;Module::Name&gt; - &lt;One-line description of module&#39;s purpose&gt; =head1 VERSION The initial template usually just has: This documentation refers to &lt;Module::Name&gt; version 0.0.1. =head1 SYNOPSIS use &lt;Module::Name&gt;; # Brief but working code example(s) here showing the most common usage(s) # This section will be as far as many users bother reading, so make it as # educational and exemplary as possible. =head1 DESCRIPTION A full description of the module and its features. May include numerous subsections (i.e., =head2, =head3, etc.). =head1 SUBROUTINES/METHODS A separate section listing the public components of the module&#39;s interface. These normally consist of either subroutines that may be exported, or methods that may be called on objects belonging to the classes that the module provides. Name the section accordingly. In an object-oriented module, this section should begin with a sentence (of the form &quot;An object of this class represents ...&quot;) to give the reader a high-level context to help them understand the methods that are subsequently described. =head1 DIAGNOSTICS A list of every error and warning message that the module can generate (even the ones that will &quot;never happen&quot;), with a full explanation of each problem, one or more likely causes, and any suggested remedies. =head1 CONFIGURATION AND ENVIRONMENT A full explanation of any configuration system(s) used by the module, including the names and locations of any configuration files, and the meaning of any environment variables or properties that can be set. These descriptions must also include details of any configuration language used. =head1 DEPENDENCIES A list of all of the other modules that this module relies upon, including any restrictions on versions, and an indication of whether these required modules are part of the standard Perl distribution, part of the module&#39;s distribution, or must be installed separately. =head1 INCOMPATIBILITIES A list of any modules that this module cannot be used in conjunction with. This may be due to name conflicts in the interface, or competition for system or program resources, or due to internal limitations of Perl (for example, many modules that use source code filters are mutually incompatible). =head1 BUGS AND LIMITATIONS A list of known problems with the module, together with some indication of whether they are likely to be fixed in an upcoming release. Also, a list of restrictions on the features the module does provide: data types that cannot be handled, performance issues and the circumstances in which they may arise, practical limitations on the size of data sets, special cases that are not (yet) handled, etc. The initial template usually just has: There are no known bugs in this module. Please report problems to &lt;Maintainer name(s)&gt; (&lt;contact address&gt;) Patches are welcome. =head1 AUTHOR &lt;Author name(s)&gt; (&lt;contact address&gt;) =head1 LICENSE AND COPYRIGHT Copyright (c) &lt;year&gt; &lt;copyright holder&gt; (&lt;contact address&gt;). All rights reserved. followed by whatever license you wish to release it under. For Perl code that is often just: This module is free software; you can redistribute it and/or modify it under the same terms as Perl itself. See L&lt;perlartistic&gt;. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.</tt></pre>

   <p>Of course, the specific details that your templates provide may vary from those shown here, according to your other coding practices. 
   The most likely variation will be in the license and copyright, but you may also have specific in-house conventions regarding version 
   numbering, the grammar of diagnostic messages, or the attribution of authorship.</p>

   <h5>4. Use a Revision Control System</h5>

   <p>Maintaining control over the creation and modification of your source code is utterly essential for robust team-based development. 
   And not just over source code: you should be revision controlling your documentation, and data files, and document templates, and 
   makefiles, and style sheets, and change logs, and any other resources your system requires.</p>

   <p>Just as you wouldn&#39;t use an editor without an Undo command or a word processor that can&#39;t merge documents, so too you shouldn&#39;t 
   use a file system you can&#39;t rewind, or a development environment that can&#39;t integrate the work of many contributors.</p>

   <p>Programmers make mistakes, and occasionally those mistakes will be catastrophic. They will reformat the disk containing the most 
   recent version of the code. Or they&#39;ll mistype an editor macro and write zeros all through the source of a critical core module. 
   Or two developers will unwittingly edit the same file at the same time and half their changes will be lost. Revision control systems 
   can prevent those kinds of problems.</p>

   <p>Moreover, occasionally the very best debugging technique is to just give up, stop trying to get yesterday&#39;s modifications to work 
   correctly, roll the code back to a known stable state, and start over again. Less drastically, comparing the current condition of 
   your code with the most recent stable version from your repository (even just a line-by-line <tt>diff</tt>) can often help you isolate 
   your recent &quot;improvements&quot; and work out which of them is the problem.</p>

   <p>Revision control systems such as RCS, CVS, Subversion, Monotone, <tt>darcs</tt>, Perforce, GNU arch, or BitKeeper can protect 
   against calamities, and ensure that you always have a working fallback position if maintenance goes horribly wrong. The various systems 
   have different strengths and limitations, many of which stem from fundamentally different views on what exactly revision control 
   is. It&#39;s a good idea to audition the various revision control systems, and find the one that works best for you. <em>Pragmatic Version 
   Control Using Subversion</em>, by Mike Mason (Pragmatic Bookshelf, 2005) and
   <a target="_blank" href="http://www.oreilly.com/catalog/cvs/"><em>Essential CVS</em></a>, by Jennifer Vesperman (O&#39;Reilly, 2003) 
   are useful starting points.</p>

   <h5>5. Create Consistent Command-Line Interfaces</h5>

   <p>Command-line interfaces have a strong tendency to grow over time, accreting new options as you add features to the application. 
   Unfortunately, the evolution of such interfaces is rarely designed, managed, or controlled, so the set of flags, options, and arguments 
   that a given application accepts are likely to be ad hoc and unique.</p>

   <p>This also means they&#39;re likely to be inconsistent with the unique ad hoc sets of flags, options, and arguments that other related 
   applications provide. The result is inevitably a suite of programs, each of which is driven in a distinct and idiosyncratic way. 
   For example:</p>

   <pre><tt>&gt; orchestrate source.txt -to interim.orc &gt; remonstrate +interim.rem -interim.orc &gt; fenestrate --src=interim.rem --dest=final.wdw Invalid input format &gt; fenestrate --help Unknown option: --help. Type &#39;fenestrate -hmo&#39; for help</tt></pre>

   <p>Here, the <tt>orchestrate</tt> utility expects its input file as its first argument, while the <tt>-to</tt> flag specifies its 
   output file. The related <tt>remonstrate</tt> tool uses <tt>-infile</tt> and <tt>+outfile</tt> options instead, with the output file 
   coming first. The <tt>fenestrate</tt> program seems to require GNU-style &quot;long options:&quot; <tt>--src=infile</tt> and <tt>--dest=outfile</tt>, 
   except, apparently, for its oddly named help flag. All in all, it&#39;s a mess.</p>

   <p>When you&#39;re providing a suite of programs, all of them should appear to work the same way, using the same flags and options for 
   the same features across all applications. This enables your users to take advantage of existing knowledge--instead of continually 
   asking you.</p>

   <p>Those three programs should work like this:</p>

   <pre><tt>&gt; orchestrate -i source.txt -o dest.orc &gt; remonstrate -i source.orc -o dest.rem &gt; fenestrate -i source.rem -o dest.wdw Input file (&#39;source.rem&#39;) not a valid Remora file (type &quot;fenestrate --help&quot; for help) &gt; fenestrate --help fenestrate - convert Remora .rem files to Windows .wdw format Usage: fenestrate [-i &lt;infile&gt;] [-o &lt;outfile&gt;] [-cstq] [-h|-v] Options: -i &lt;infile&gt; Specify input source [default: STDIN] -o &lt;outfile&gt; Specify output destination [default: STDOUT] -c Attempt to produce a more compact representation -h Use horizontal (landscape) layout -v Use vertical (portrait) layout -s Be strict regarding input -t Be extra tolerant regarding input -q Run silent --version Print version information --usage Print the usage line of this summary --help Print this summary --man Print the complete manpage</tt></pre>

   <p>Here, every application that takes input and output files uses the same two flags to do so. A user who wants to use the <tt>substrate</tt> 
   utility (to convert that final .wdw file to a subroutine) is likely to be able to guess correctly the required syntax:</p>

   <pre><tt>&gt; substrate -i dest.wdw -o dest.sub</tt></pre>

   <p>Anyone who can&#39;t guess that probably can guess that:</p>

   <pre><tt>&gt; substrate --help</tt></pre>

   <p>is likely to render aid and comfort.</p>

   <p>A large part of making interfaces consistent is being consistent in specifying the individual components of those interfaces. 
   Some conventions that may help to design consistent and predictable interfaces include:</p>
   <ul>

      <li>Require a flag preceding every piece of command-line data, except filenames.<p>Users don&#39;t want to have to remember that your 
      application requires &quot;input file, output file, block size, operation, fallback strategy,&quot; and requires them in that precise order:</p>

      <pre><tt>&gt; lustrate sample_data proc_data 1000 normalize log</tt></pre>

      <p>They want to be able to say explicitly what they mean, in any order that suits them:</p>

      <pre><tt>&gt; lustrate sample_data proc_data -op=normalize -b1000 --fallback=log</tt></pre>
      </li>

      <li>Provide a flag for each filename, too, especially when a program can be given files for different purposes.<p>Users might 
      also not want to remember the order of the two positional filenames, so let them label those arguments as well, and specify them 
      in whatever order they prefer:</p>

      <pre><tt>&gt; lustrate -i sample_data -op normalize -b1000 --fallback log -o proc_data</tt></pre>
      </li>

      <li>Use a single <tt>-</tt> prefix for short-form flags, up to three letters (<tt>-v</tt>, <tt>-i</tt>, <tt>-rw</tt>, <tt>-in</tt>,
      <tt>-out</tt>).<p>Experienced users appreciate short-form flags as a way of reducing typing and limiting command-line clutter. 
      Don&#39;t make them type two dashes in these shortcuts.</p>
      </li>

      <li>Use a double <tt>--</tt> prefix for longer flags (<tt>--verbose</tt>, <tt>--interactive</tt>, <tt>--readwrite</tt>, <tt>--input</tt>,
      <tt>--output</tt>).<p>Flags that are complete words improve the readability of a command line (in a shell script, for example). 
      The double dash also helps to distinguish between the longer flag name and any nearby file names.</p>
      </li>

      <li>If a flag expects an associated value, allow an optional <tt>=</tt> between the flag and the value.<p>Some people prefer to 
      visually associate a value with its preceding flag:</p>

      <pre><tt>&gt; lustrate -i=sample_data -op=normalize -b=1000 --fallback=log -o=proc_data</tt></pre>

      <p>Others don&#39;t:</p>

      <pre><tt>&gt; lustrate -i sample_data -op normalize -b1000 --fallback log -o proc_data</tt></pre>

      <p>Still others want a bit each way:</p>

      <pre><tt>&gt; lustrate -i sample_data -o proc_data -op=normalize -b=1000 --fallback=log</tt></pre>

      <p>Let the user choose.</p>
      </li>

      <li>Allow single-letter options to be &quot;bundled&quot; after a single dash.<p>It&#39;s irritating to have to type repeated dashes for a series 
      of flags:</p>

      <pre><tt>&gt; lustrate -i sample_data -v -l -x</tt></pre>

      <p>Allow experienced users to also write:</p>

      <pre><tt>&gt; lustrate -i sample_data -vlx</tt></pre>
      </li>

      <li>Provide a multi-letter version of every single-letter flag.<p>Short-form flags may be nice for experienced users, but they 
      can be troublesome for new users: hard to remember and even harder to recognize. Don&#39;t force people to do either. Give them a 
      verbose alternative to every concise flag; full words that are easier to remember, and also more self-documenting in shell scripts.</p>
      </li>

      <li>Always allow <tt>-</tt> as a special filename.<p>A widely used convention is that a dash (<tt>-</tt>) where an input file 
      is expected means &quot;read from standard input,&quot; and a dash where an output file is expected means &quot;write to standard output.&quot;</p>
      </li>

      <li>Always allow <tt>--</tt> as a file list marker.<p>Another widely used convention is that the appearance of a double dash (<tt>--</tt>) 
      on the command line marks the end of any flagged options, and indicates that the remaining arguments are a list of filenames, 
      even if some of them look like flags.</p>
      </li>
   </ul>

   <h5>6. Agree Upon a Coherent Layout Style and Automate It with <tt>perltidy</tt></h5>

   <p>Formatting. Indentation. Style. Code layout. Whatever you choose to call it, it&#39;s one of the most contentious aspects of programming 
   discipline. More and bloodier wars have been fought over code layout than over just about any other aspect of coding.</p>

   <p>What is the best practice here? Should you use classic Kernighan and Ritchie style? Or go with BSD code formatting? Or adopt the 
   layout scheme specified by the GNU project? Or conform to the Slashcode coding guidelines?</p>

   <p>Of course not! Everyone knows that <em>&lt;insert your personal coding style here&gt;</em> is the One True Layout Style, the only sane 
   choice, as ordained by <em>&lt;insert your favorite Programming Deity here&gt;</em> since Time Immemorial! Any other choice is manifestly 
   absurd, willfully heretical, and self-evidently a Work of Darkness!</p>

   <p>That&#39;s precisely the problem. When deciding on a layout style, it&#39;s hard to decide where rational choices end and rationalized 
   habits begin.</p>

   <p>Adopting a coherently designed approach to code layout, and then applying that approach consistently across all your coding, is 
   fundamental to best-practice programming. Good layout can improve the readability of a program, help detect errors within it, and 
   make the structure of your code much easier to comprehend. Layout matters.</p>

   <p>However, most coding styles--including the four mentioned earlier--confer those benefits almost equally well. While it&#39;s true 
   that having a consistent code layout scheme matters very much indeed, the particular code layout scheme you ultimately decide upon 
   does not matter at all! All that matters is that you adopt a single, coherent style; one that works for your entire programming team, 
   and, having agreed upon that style, that you then apply it consistently across all your development.</p>

   <p>In the long term, it&#39;s best to train yourself and your team to code in a consistent, rational, and readable style. However, the 
   time and commitment necessary to accomplish that isn&#39;t always available. In such cases, a reasonable compromise is to prescribe a 
   standard code-formatting tool that must be applied to all code before it&#39;s committed, reviewed, or otherwise displayed in public.</p>

   <p>There is now an excellent code formatter available for Perl: <a target="_blank" href="http://perltidy.sourceforge.net/"><tt>perltidy</tt></a>. 
   It provides an extensive range of user-configurable options for indenting, block delimiter positioning, column-like alignment, and 
   comment positioning.</p>

   <p>Using <tt>perltidy</tt>, you can convert code like this:</p>

   <pre><tt>if($sigil eq &#39;$&#39;){ if($subsigil eq &#39;?&#39;){ $sym_table{substr($var_name,2)}=delete $sym_table{locate_orig_var($var)}; $internal_count++;$has_internal{$var_name}++ } else { ${$var_ref} = q{$sym_table{$var_name}}; $external_count++; $has_external{$var_name}++; }} elsif ($sigil eq &#39;@&#39;&amp;&amp;$subsigil eq &#39;?&#39;) { @{$sym_table{$var_name}} = grep {defined $_} @{$sym_table{$var_name}}; } elsif ($sigil eq &#39;%&#39; &amp;&amp; $subsigil eq &#39;?&#39;) { delete $sym_table{$var_name}{$EMPTY_STR}; } else { ${$var_ref} = q{$sym_table{$var_name}} }</tt></pre>

   <p>into something readable:</p>

   <pre><tt>if ( $sigil eq &#39;$&#39; ) { if ( $subsigil eq &#39;?&#39; ) { $sym_table{ substr( $var_name, 2 ) } = delete $sym_table{ locate_orig_var($var) }; $internal_count++; $has_internal{$var_name}++; } else { ${$var_ref} = q{$sym_table{$var_name}}; $external_count++; $has_external{$var_name}++; } } elsif ( $sigil eq &#39;@&#39; &amp;&amp; $subsigil eq &#39;?&#39; ) { @{ $sym_table{$var_name} } = grep {defined $_} @{ $sym_table{$var_name} }; } elsif ( $sigil eq &#39;%&#39; &amp;&amp; $subsigil eq &#39;?&#39; ) { delete $sym_table{$var_name}{$EMPTY_STR}; } else { ${$var_ref} = q{$sym_table{$var_name}}; }</tt></pre>

   <p>Mandating that everyone use a common tool to format their code can also be a simple way of sidestepping the endless objections, 
   acrimony, and dogma that always surround any discussion on code layout. If <tt>perltidy</tt> does all the work for them, then it 
   will cost developers almost no effort to adopt the new guidelines. They can simply set up an editor macro that will &quot;straighten&quot; 
   their code whenever they need to.</p>

   <h5>7. Code in Commented Paragraphs</h5>

   <p>A paragraph is a collection of statements that accomplish a single task: in literature, it&#39;s a series of sentences conveying a 
   single idea; in programming, a series of instructions implementing a single step of an algorithm.</p>

   <p>Break each piece of code into sequences that achieve a single task, placing a single empty line between each sequence. To further 
   improve the maintainability of the code, place a one-line comment at the start of each such paragraph, describing what the sequence 
   of statements does. Like so:</p>

   <pre><tt># Process an array that has been recognized... sub addarray_internal { my ($var_name, $needs_quotemeta) = @_; # Cache the original... $raw .= $var_name; # Build meta-quoting code, if requested... my $quotemeta = $needs_quotemeta ? q{map {quotemeta $_} } : $EMPTY_STR; # Expand elements of variable, conjoin with ORs... my $perl5pat = qq{(??{join q{|}, $quotemeta \@{$var_name}})}; # Insert debugging code if requested... my $type = $quotemeta ? &#39;literal&#39; : &#39;pattern&#39;; debug_now(&quot;Adding $var_name (as $type)&quot;); add_debug_mesg(&quot;Trying $var_name (as $type)&quot;); return $perl5pat; }</tt></pre>

   <p>Paragraphs are useful because humans can focus on only a few pieces of information at once. Paragraphs are one way of aggregating 
   small amounts of related information, so that the resulting &quot;chunk&quot; can fit into a single slot of the reader&#39;s limited short-term 
   memory. Paragraphs enable the physical structure of a piece of writing to reflect and emphasize its logical structure.</p>

   <p>Adding comments at the start of each paragraph further enhances the chunking by explicitly summarizing the purpose of each chunk 
   (note: the purpose, not the behavior). Paragraph comments need to explain why the code is there and what it achieves, not merely 
   paraphrase the precise computational steps it&#39;s performing.</p>

   <p>Note, however, that the contents of paragraphs are only of secondary importance here. It is the vertical gaps separating each 
   paragraph that are critical. Without them, the readability of the code declines dramatically, even if the comments are retained:</p>

   <pre><tt>sub addarray_internal { my ($var_name, $needs_quotemeta) = @_; # Cache the original... $raw .= $var_name; # Build meta-quoting code, if required... my $quotemeta = $needs_quotemeta ? q{map {quotemeta $_} } : $EMPTY_STR; # Expand elements of variable, conjoin with ORs... my $perl5pat = qq{(??{join q{|}, $quotemeta \@{$var_name}})}; # Insert debugging code if requested... my $type = $quotemeta ? &#39;literal&#39; : &#39;pattern&#39;; debug_now(&quot;Adding $var_name (as $type)&quot;); add_debug_mesg(&quot;Trying $var_name (as $type)&quot;); return $perl5pat; }</tt></pre>

   <h5>8. Throw Exceptions Instead of Returning Special Values or Setting Flags</h5>

   <p>Returning a special error value on failure, or setting a special error flag, is a very common error-handling technique. Collectively, 
   they&#39;re the basis for virtually all error notification from Perl&#39;s own built-in functions. For example, the built-ins <tt>eval</tt>,
   <tt>exec</tt>, <tt>flock</tt>, <tt>open</tt>, <tt>print</tt>, <tt>stat</tt>, and <tt>system</tt> all return special values on error. 
   Unfortunately, they don&#39;t all use the same special value. Some of them also set a flag on failure. Sadly, it&#39;s not always the same 
   flag. See the <a target="_blank" href="http://perldoc.perl.org/perlfunc.html"><tt>perlfunc</tt></a> manpage for the gory details.</p>

   <p>Apart from the obvious consistency problems, error notification via flags and return values has another serious flaw: developers 
   can silently ignore flags and return values, and ignoring them requires absolutely no effort on the part of the programmer. In fact, 
   in a void context, ignoring return values is Perl&#39;s default behavior. Ignoring an error flag that has suddenly appeared in a special 
   variable is just as easy: you simply don&#39;t bother to check the variable.</p>

   <p>Moreover, because ignoring a return value is the void-context default, there&#39;s no syntactic marker for it. There&#39;s no way to look 
   at a program and immediately see where a return value is deliberately being ignored, which means there&#39;s also no way to be sure that 
   it&#39;s not being ignored accidentally.</p>

   <p>The bottom line: regardless of the programmer&#39;s (lack of) intention, an error indicator is being ignored. That&#39;s not good programming.</p>

   <p>Ignoring error indicators frequently causes programs to propagate errors in entirely the wrong direction. For example:</p>

   <pre><tt># Find and open a file by name, returning the filehandle # or undef on failure... sub locate_and_open { my ($filename) = @_; # Check acceptable directories in order... for my $dir (@DATA_DIRS) { my $path = &quot;$dir/$filename&quot;; # If file exists in an acceptable directory, open and return it... if (-r $path) { open my $fh, &#39;&lt;&#39;, $path; return $fh; } } # Fail if all possible locations tried without success... return; } # Load file contents up to the first &lt;DATA/&gt; marker... sub load_header_from { my ($fh) = @_; # Use DATA tag as end-of-&quot;line&quot;... local $/ = &#39;&lt;DATA/&gt;&#39;; # Read to end-of-&quot;line&quot;... return &lt;$fh&gt;; } # and later... for my $filename (@source_files) { my $fh = locate_and_open($filename); my $head = load_header_from($fh); print $head; }</tt></pre>

   <p>The <tt>locate_and_open()</tt> subroutine simply assumes that the call to <tt>open</tt> works, immediately returning the filehandle 
   (<tt>$fh</tt>), whatever the actual outcome of the <tt>open</tt>. Presumably, the expectation is that whoever calls <tt>locate_and_open()</tt> 
   will check whether the return value is a valid filehandle.</p>

   <p>Except, of course, &quot;whoever&quot; doesn&#39;t check. Instead of testing for failure, the main <tt>for</tt> loop takes the failure value 
   and immediately propagates it &quot;across&quot; the block, to the rest of the statements in the loop. That causes the call to <tt>loader_header_from()</tt> 
   to propagate the error value &quot;downwards.&quot; It&#39;s in that subroutine that the attempt to treat the failure value as a filehandle eventually 
   kills the program:</p>

   <pre><tt>readline() on unopened filehandle at demo.pl line 28.</tt></pre>

   <p>Code like that--where an error is reported in an entirely different part of the program from where it actually occurred--is particularly 
   onerous to debug.</p>

   <p>Of course, you could argue that the fault lies squarely with whoever wrote the loop, for using <tt>locate_and_open()</tt> without 
   checking its return value. In the narrowest sense, that&#39;s entirely correct--but the deeper fault lies with whoever actually wrote
   <tt>locate_and_open()</tt> in the first place, or at least, whoever assumed that the caller would always check its return value.</p>

   <p>Humans simply aren&#39;t like that. Rocks almost never fall out of the sky, so humans soon conclude that they never do, and stop looking 
   up for them. Fires rarely break out in their homes, so humans soon forget that they might, and stop testing their smoke detectors 
   every month. In the same way, programmers inevitably abbreviate &quot;almost never fails&quot; to &quot;never fails,&quot; and then simply stop checking.</p>

   <p>That&#39;s why so very few people bother to verify their <tt>print</tt> statements:</p>

   <pre><tt>if (!print &#39;Enter your name: &#39;) { print {*STDLOG} warning =&gt; &#39;Terminal went missing!&#39; }</tt></pre>

   <p>It&#39;s human nature to &quot;trust but not verify.&quot;</p>

   <p>Human nature is why returning an error indicator is not best practice. Errors are (supposed to be) unusual occurrences, so error 
   markers will almost never be returned. Those tedious and ungainly checks for them will almost never do anything useful, so eventually 
   they&#39;ll be quietly omitted. After all, leaving the tests off almost always works just fine. It&#39;s so much easier not to bother. Especially 
   when not bothering is the default!</p>

   <p>Don&#39;t return special error values when something goes wrong; throw an exception instead. The great advantage of exceptions is 
   that they reverse the usual default behaviors, bringing untrapped errors to immediate and urgent attention. On the other hand, ignoring 
   an exception requires a deliberate and conspicuous effort: you have to provide an explicit <tt>eval</tt> block to neutralize it.</p>

   <p>The <tt>locate_and_open()</tt> subroutine would be much cleaner and more robust if the errors within it threw exceptions:</p>

   <pre><tt># Find and open a file by name, returning the filehandle # or throwing an exception on failure... sub locate_and_open { my ($filename) = @_; # Check acceptable directories in order... for my $dir (@DATA_DIRS) { my $path = &quot;$dir/$filename&quot;; # If file exists in acceptable directory, open and return it... if (-r $path) { open my $fh, &#39;&lt;&#39;, $path or croak( &quot;Located $filename at $path, but could not open&quot;); return $fh; } } # Fail if all possible locations tried without success... croak( &quot;Could not locate $filename&quot; ); } # and later... for my $filename (@source_files) { my $fh = locate_and_open($filename); my $head = load_header_from($fh); print $head; }</tt></pre>

   <p>Notice that the main <tt>for</tt> loop didn&#39;t change at all. The developer using <tt>locate_and_open()</tt> still assumes that 
   nothing can go wrong. Now there&#39;s some justification for that expectation, because if anything does go wrong, the thrown exception 
   will automatically terminate the loop.</p>

   <p>Exceptions are a better choice even if you are the careful type who religiously checks every return value for failure:</p>

   <pre><tt>SOURCE_FILE: for my $filename (@source_files) { my $fh = locate_and_open($filename); next SOURCE_FILE if !defined $fh; my $head = load_header_from($fh); next SOURCE_FILE if !defined $head; print $head; }</tt></pre>

   <p>Constantly checking return values for failure clutters your code with validation statements, often greatly decreasing its readability. 
   In contrast, exceptions allow an algorithm to be implemented without having to intersperse any error-handling infrastructure at all. 
   You can factor the error-handling out of the code and either relegate it to after the surrounding <tt>eval</tt>, or else dispense 
   with it entirely:</p>

   <pre><tt>for my $filename (@directory_path) { # Just ignore any source files that don&#39;t load... eval { my $fh = locate_and_open($filename); my $head = load_header_from($fh); print $head; } }</tt></pre>

   <h5>9. Add New Test Cases Before you Start Debugging</h5>

   <p>The first step in any debugging process is to isolate the incorrect behavior of the system, by producing the shortest demonstration 
   of it that you reasonably can. If you&#39;re lucky, this may even have been done for you:</p>

   <pre><tt>To: <a href="/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="387c7b77766f7961785b48595616574a5f">[email&#160;protected]</a> From: <a href="/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="e4978597878c85a494819688898b8a8f97ca8b9683">[email&#160;protected]</a> Subject: Bug in inflect module Zdravstvuite, I have been using your Lingua::EN::Inflect module to normalize terms in a data-mining application I am developing, but there seems to be a bug in it, as the following example demonstrates: use Lingua::EN::Inflect qw( PL_N ); print PL_N(&#39;man&#39;), &quot;\n&quot;; # Prints &quot;men&quot;, as expected print PL_N(&#39;woman&#39;), &quot;\n&quot;; # Incorrectly prints &quot;womans&quot;</tt></pre>

   <p>Once you have distilled a short working example of the bug, convert it to a series of tests, such as:</p>

   <pre><tt>use Lingua::EN::Inflect qw( PL_N ); use Test::More qw( no_plan ); is(PL_N(&#39;man&#39;) , &#39;men&#39;, &#39;man -&gt; men&#39; ); is(PL_N(&#39;woman&#39;), &#39;women&#39;, &#39;woman -&gt; women&#39; );</tt></pre>

   <p>Don&#39;t try to fix the problem straight away, though. Instead, immediately add those tests to your test suite. If that testing has 
   been well set up, that can often be as simple as adding a couple of entries to a table:</p>

   <pre><tt>my %plural_of = ( &#39;mouse&#39; =&gt; &#39;mice&#39;, &#39;house&#39; =&gt; &#39;houses&#39;, &#39;ox&#39; =&gt; &#39;oxen&#39;, &#39;box&#39; =&gt; &#39;boxes&#39;, &#39;goose&#39; =&gt; &#39;geese&#39;, &#39;mongoose&#39; =&gt; &#39;mongooses&#39;, &#39;law&#39; =&gt; &#39;laws&#39;, &#39;mother-in-law&#39; =&gt; &#39;mothers-in-law&#39;, # Sascha&#39;s bug, reported 27 August 2004... &#39;man&#39; =&gt; &#39;men&#39;, &#39;woman&#39; =&gt; &#39;women&#39;, );</tt></pre>

   <p>The point is: if the original test suite didn&#39;t report this bug, then that test suite was broken. It simply didn&#39;t do its job 
   (finding bugs) adequately. Fix the test suite first by adding tests that cause it to fail:</p>

   <pre><tt>&gt; perl inflections.t ok 1 - house -&gt; houses ok 2 - law -&gt; laws ok 3 - man -&gt; men ok 4 - mongoose -&gt; mongooses ok 5 - goose -&gt; geese ok 6 - ox -&gt; oxen not ok 7 - woman -&gt; women # Failed test (inflections.t at line 20) # got: &#39;womans&#39; # expected: &#39;women&#39; ok 8 - mother-in-law -&gt; mothers-in-law ok 9 - mouse -&gt; mice ok 10 - box -&gt; boxes 1..10 # Looks like you failed 1 tests of 10.</tt></pre>

   <p>Once the test suite is detecting the problem correctly, then you&#39;ll be able to tell when you&#39;ve correctly fixed the actual bug, 
   because the tests will once again fall silent.</p>

   <p>This approach to debugging is most effective when the test suite covers the full range of manifestations of the problem. When 
   adding test cases for a bug, don&#39;t just add a single test for the simplest case. Make sure you include the obvious variations as 
   well:</p>

   <pre><tt>my %plural_of = ( &#39;mouse&#39; =&gt; &#39;mice&#39;, &#39;house&#39; =&gt; &#39;houses&#39;, &#39;ox&#39; =&gt; &#39;oxen&#39;, &#39;box&#39; =&gt; &#39;boxes&#39;, &#39;goose&#39; =&gt; &#39;geese&#39;, &#39;mongoose&#39; =&gt; &#39;mongooses&#39;, &#39;law&#39; =&gt; &#39;laws&#39;, &#39;mother-in-law&#39; =&gt; &#39;mothers-in-law&#39;, # Sascha&#39;s bug, reported 27 August 2004... &#39;man&#39; =&gt; &#39;men&#39;, &#39;woman&#39; =&gt; &#39;women&#39;, &#39;human&#39; =&gt; &#39;humans&#39;, &#39;man-at-arms&#39; =&gt; &#39;men-at-arms&#39;, &#39;lan&#39; =&gt; &#39;lans&#39;, &#39;mane&#39; =&gt; &#39;manes&#39;, &#39;moan&#39; =&gt; &#39;moans&#39;, );</tt></pre>

   <p>The more thoroughly you test the bug, the more completely you will fix it.</p>

   <h5>10. Don&#39;t Optimize Code--Benchmark It</h5>

   <p>If you need a function to remove duplicate elements of an array, it&#39;s natural to think that a &quot;one-liner&quot; like this:</p>

   <pre><tt>sub uniq { return keys %{ { map {$_=&gt;1} @_ } } }</tt></pre>

   <p>will be more efficient than two statements:</p>

   <pre><tt>sub uniq { my %seen; return grep {!$seen{$_}++} @_; }</tt></pre>

   <p>Unless you are deeply familiar with the internals of the Perl interpreter (in which case you already have far more serious personal 
   issues to deal with), intuitions about the relative performance of two constructs are exactly that: unconscious guesses.</p>

   <p>The only way to know for sure which of two--or more--alternatives will perform better is to actually time each of them. The standard
   <a target="_blank" href="http://search.cpan.org/perldoc?Benchmark">Benchmark</a> module makes that easy:</p>

   <pre><tt># A short list of not-quite-unique values... our @data = qw( do re me fa so la ti do ); # Various candidates... sub unique_via_anon { return keys %{ { map {$_=&gt;1} @_ } }; } sub unique_via_grep { my %seen; return grep { !$seen{$_}++ } @_; } sub unique_via_slice { my %uniq; @uniq{@_} = (); return keys %uniq; } # Compare the current set of data in @data sub compare { my ($title) = @_; print &quot;\n[$title]\n&quot;; # Create a comparison table of the various timings, making sure that # each test runs at least 10 CPU seconds... use Benchmark qw( cmpthese ); cmpthese -10, { anon =&gt; &#39;my @uniq = unique_via_anon(@data)&#39;, grep =&gt; &#39;my @uniq = unique_via_grep(@data)&#39;, slice =&gt; &#39;my @uniq = unique_via_slice(@data)&#39;, }; return; } compare(&#39;8 items, 10% repetition&#39;); # Two copies of the original data... @data = (@data) x 2; compare(&#39;16 items, 56% repetition&#39;); # One hundred copies of the original data... @data = (@data) x 50; compare(&#39;800 items, 99% repetition&#39;);</tt></pre>

   <p>The <tt>cmpthese()</tt> subroutine takes a number, followed by a reference to a hash of tests. The number specifies either the 
   exact number of times to run each test (if the number is positive), or the absolute number of CPU seconds to run the test for (if 
   the number is negative). Typical values are around 10,000 repetitions or ten CPU seconds, but the module will warn you if the test 
   is too short to produce an accurate benchmark.</p>

   <p>The keys of the test hash are the names of your tests, and the corresponding values specify the code to be tested. Those values 
   can be either strings (which are <tt>eval</tt>&#39;d to produce executable code) or subroutine references (which are called directly).</p>

   <p>The benchmarking code shown above would print out something like the following:</p>

   <pre><tt>[8 items, 10% repetitions] Rate anon grep slice anon 28234/s -- -24% -47% grep 37294/s 32% -- -30% slice 53013/s 88% 42% -- [16 items, 50% repetitions] Rate anon grep slice anon 21283/s -- -28% -51% grep 29500/s 39% -- -32% slice 43535/s 105% 48% -- [800 items, 99% repetitions] Rate anon grep slice anon 536/s -- -65% -89% grep 1516/s 183% -- -69% slice 4855/s 806% 220% --</tt></pre>

   <p>Each of the tables printed has a separate row for each named test. The first column lists the absolute speed of each candidate 
   in repetitions per second, while the remaining columns allow you to compare the relative performance of any two tests. For example, 
   in the final test tracing across the <tt>grep</tt> row to the <tt>anon</tt> column reveals that the <tt>grep</tt>ped solution was 
   1.83 times (183 percent) faster than using an anonymous hash. Tracing further across the same row also indicates that <tt>grep</tt>ping 
   was 69 percent slower (-69 percent faster) than slicing.</p>

   <p>Overall, the indication from the three tests is that the slicing-based solution is consistently the fastest for this particular 
   set of data on this particular machine. It also appears that as the data set increases in size, slicing also scales much better than 
   either of the other two approaches.</p>

   <p>However, those two conclusions are effectively drawn from only three data points (namely, the three benchmarking runs). To get 
   a more definitive comparison of the three methods, you&#39;d also need to test other possibilities, such as a long list of non-repeating 
   items, or a short list with nothing but repetitions.</p>

   <p>Better still, test on the real data that you&#39;ll actually be &quot;unique-ing.&quot;</p>

   <p>For example, if that data is a sorted list of a quarter of a million words, with only minimal repetitions, and which has to remain 
   sorted, then test exactly that:</p>

   <pre><tt>our @data = slurp &#39;/usr/share/biglongwordlist.txt&#39;; use Benchmark qw( cmpthese ); cmpthese 10, { # Note: the non-grepped solutions need a post-uniqification re-sort anon =&gt; &#39;my @uniq = sort(unique_via_anon(@data))&#39;, grep =&gt; &#39;my @uniq = unique_via_grep(@data)&#39;, slice =&gt; &#39;my @uniq = sort(unique_via_slice(@data))&#39;, };</tt></pre>

   <p>Not surprisingly, this benchmark indicates that the <tt>grep</tt>ped solution is markedly superior on a large sorted data set:</p>

   <pre><tt>s/iter anon slice grep anon 4.28 -- -3% -46% slice 4.15 3% -- -44% grep 2.30 86% 80% --</tt></pre>

   <p>Perhaps more interestingly, the <tt>grep</tt>ped solution still benchmarks as being marginally faster when the two hash-based 
   approaches aren&#39;t re-sorted. This suggests that the better scalability of the sliced solution as seen in the earlier benchmark is 
   a localized phenomenon, and is eventually undermined by the growing costs of allocation, hashing, and bucket-overflows as the sliced 
   hash grows very large.</p>

   <p>Above all, that last example demonstrates that benchmarks only benchmark the cases you actually benchmark, and that you can only 
   draw useful conclusions about performance from benchmarking real data.</p>
</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20050714X_perl_com_ten_essential_development_practices"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20031017X_part_13_perl_style" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20031017X_the_road_to_better_programming_introduction_and_chapter_1"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4>[Oct 17,2003] <a target="_blank" href="http://mailman.linuxchix.org/pipermail/courses/2003-October/001349.html">Part 13 Perl Style</a> 
by Dan Richter </h4>
<blockquote>

   <p><b>Part 13: Perl Style </b></p>
   <ul>

      <li>1) Introduction </li>

      <li>2) The Many Faces of Perl </li>

      <li>3) The Special File Handle &quot;ARGV&quot; </li>

      <li>4) Exercise </li>

      <li>5) [Non-]Answer to Previous Exercise </li>

      <li>6) Past Information </li>

      <li>7) Credits </li>

      <li>8) Licensing </li>

      <li>1) Introduction </li>
   </ul>

   <p>An important concept in Perl is that &quot;There Is More Than One Way To Do It&quot; (TIMTOWTDY, pronounced &quot;Tim Towtdy&quot;). We have seen 
   this to some degree already. A simple example is &quot;tr///&quot; and &quot;y///&quot;, which mean exactly the same thing. As another example, &quot;unless&quot; 
   is the same as &quot;if not&quot;. Likewise, &quot;s///&quot; can usually do the job of &quot;tr///&quot;. </p>

   <p>This is one of the ways in which Perl is similar to human languages, which have many different words to express similar concepts 
   (as a quick look through a thesaurus will demonstrate). Perl borrows other ideas from human languages as well. One simple example 
   is the way &quot;if&quot; can be placed before or after the command to be conditionally executed. In addition, the implied use of &quot;$_&quot; is foreign 
   to English speakers, but implied subjects are common in other languages such as Latin and Chinese. </p>

   <p>Larry Wall, who invented Perl, is fascinated with human languages. He even studied them in graduate school - and for an interesting 
   reason: &quot;At the time, [my wife and I] were actually planning to be missionaries (more specifically, Bible translators), but we had 
   to drop that idea for health reasons.&quot; But Wall doesn&#39;t regret the change of plans: he figures that Perl is actually more useful 
   to the missionaries than he personally could have been. </p>

   <p>Perl&#39;s similarity to a human language goes hand-in-hand with its ability to do many different types of jobs. As Jon Udell puts 
   it, &quot;[Larry] Wall [inventor of Perl] believes that people think about things in different ways, that natural languages accommodate 
   many mindsets, and that programming languages should too.&quot; </p>

   <p>We&#39;re now going to see a bit of Perl&#39;s human-language-like versatility. </p>

   <h5>2) The Many Faces of Perl</h5>

   <p>As we have seen, Perl commands can be written as functions or statements. </p>

   <pre>
	<font color="#0000FF">   die(&quot;Error!&quot;) unless defined($foo);  
	# Look like functions.
   die &quot;Error!&quot;  unless defined $foo;    # Look
	like statements.</font>
	</pre>
   This changes the appearance only: the functionality is exactly the same.

   <p>More interestingly, the following code looks like it comes from a shell script, but it is actually valid Perl code: </p>

   <pre>
<font color="#0000FF">   (-d $dir) || mkdir $dir;    #
	Create directory if it doesn&#39;t exist.
   $size=`wc -c $dir/*`;       # Get
	sizes of files.
   print &lt;&lt;EOF;
      Looks like
      everything&#39;s OK!
   EOF</font>
	</pre>
   Yes, Perl borrows &quot;-d&quot;, &quot;mkdir&quot; back-ticks and &quot;here-documents&quot; (the &quot;&lt;&lt;EOF&quot; part) from your favourite shell.

   <p>Of course, the similarity is limited. Functions like &quot;mkdir&quot; are Perl built-ins; you cannot execute an arbitrary shell command 
   directly from Perl. (That&#39;s why we had to use back-ticks to execute &quot;wc&quot;.) In addition, you can&#39;t use &quot;&lt;&quot;, &quot;&gt;&quot; and &quot;|&quot; to pipe I/O 
   the way you would in a shell script (except when using &quot;open&quot;, as we saw last week). </p>

   <p>But I still think that it&#39;s impressive that a Perl program can be written like C or like a shell script. </p>

   <h5>3) The Special File Handle &quot;ARGV&quot;</h5>
   Okay, I admit that this section isn&#39;t too related to Perl style, but I had to put it somewhere!

   <p>&quot;ARGV&quot; is a special file handle which sequentially opens (for reading) all the files specified on the command-line (i.e., every 
   command-line argument is interpreted as a file name and is opened). The files are read in the order in which they were specified 
   on the command-line and as though the contents formed one big file (i.e., when you reach the end of a file, Perl automatically closes 
   it and opens the next). The variable &quot;$ARGV&quot; always contains the name of the file that is currently being read by &quot;&lt;ARGV&gt;&quot;. </p>

   <p>In practice, you never see &quot;&lt;ARGV&gt;&quot;. Instead, you see it written as &quot;&lt;&gt;&quot;, because &quot;ARGV&quot; is the default file handle for reading 
   if none is specified. </p>

   <p>For example, consider the following code: </p>

   <pre>
  <font color="#0000FF">while ( &lt;&gt; ) {    # while (
	defined ( $_ = &lt;ARGV&gt; ) ) {
     print;         
	#  print $_;
   }</font></pre>
   If you store this code in a file named &quot;test.pl&quot; and run:

   <pre>
  <font color="#0000FF">perl test.pl file1 file2 file3</font></pre>
   the program will output the contents of &quot;file1&quot;, then the contents of &quot;file2&quot;, then the contents of &quot;file3&quot;.

   <p>A dash (&quot;-&quot;) given as a command-line argument refers to standard input. Also, if no command-line arguments are given at all, standard 
   input is assumed. </p>
</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20031017X_part_13_perl_style"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20031017X_the_road_to_better_programming_introduction_and_chapter_1" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20031017X_the_road_to_better_programming"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4 align="left"><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road1.html">The road to better programming 
Introduction and chapter 1</a> by Teodor Zlatanov</h4>
<blockquote>

   <h6>November 1, 2001</h6>

   <p><a href="#author1">Zlatanov</a> (<a href="/cdn-cgi/l/email-protection#b3c7c9c9f3dad4dfdcc69dd0dcde"><span class="__cf_email__" data-cfemail="e4909e9ea48d83888b91ca878b89">[email&#160;protected]</span></a>)</p>

   <p>Programmer, Gold Software Systems</p>
   <blockquote>

      <p>The success or failure of any software programming group depends largely on its ability to work together as a team. From manager 
      to members, to well-conceived, yet dynamic guidelines, the team as a whole is defined by the unison of its parts. Shattering the 
      myth of the faultless programmer, Teodor dismantles the uninspired software group and then builds it up again into a synchronized, 
      energized ensemble.<br xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
      <br xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
      Welcome to a series of articles on developerWorks comprising a complete guide to better programming in Perl. In this first installment, 
      Teodor introduces his book and looks at coding guidelines from a fresh perspective.</p>
   </blockquote>

   <p>This is the book for the beginner to intermediate Perl programmer. But even an advanced Perl programmer can find the majority 
   of the chapters exciting and relevant, from the tips of Part I to the project management tools presented in Part II to the <tt>Parse::RecDescentsource</tt> 
   code analysis scripts in Part III.</p>

   <p>The words <i xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">program</i> and
   <i xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">script</i> are used interchangeably. In Perl, the two mean pretty much the 
   same thing. A program can, indeed, be made up of many scripts, and a script can contain many programs, but for simplicity&#39;s sake, 
   we will use the two terms with the understanding that one script file contains only one program.</p>

   <p><b><a name="g1">Goals of the book</a></b></p>

   <p><b xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">Part I</b> is full of tips to improve your Perl skills, ranging from 
   best programming practices to code debugging. It does not teach you Perl programming. There are many books with that purpose, and 
   they would be hard to surpass in clarity and completeness.</p>

   <p><b xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">Part II</b> will teach you how a small Perl software team can be better 
   managed with the standard tools of software project management. Often, Perl programmers embody the &quot;herd of cats&quot; view of software 
   teams. Part II will apply project management tools to a small (2- to 6-person) Perl development team, and will examine how managing 
   such a team successfully is different from the classic project management approach.</p>

   <p><b xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">Part III</b> will develop tools to analyze source code (Perl and C examples 
   will be developed) and to help you manage your team better. Analysis of source code is superficial at best today, ranging from the 
   obvious and irrelevant &quot;lines of code&quot; metrics to function points (see
   <a href="#resources" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" trackclick="no">Resources</a> later in this article), 
   which do not help in understanding the programmer&#39;s mindset. Understanding the programmer&#39;s mindset will be the goal of Part III. 
   Tools will be developed that help track metrics such as comment legibility and consistency, repetitiveness of code, and code legibility. 
   These metrics will be introduced as a part of a software project, not its goal.</p>

   <p>There is no perfection in programming, only its pursuit. Good programmers learn something new every day and continually improve 
   their skills and technique. Rigidity and inflexibility are forever the enemy of ingenuity and creativity.</p>

   <p><b><a name="1">In pursuit of perfection</a></b></p>

   <p>The most common mistake a programmer can make is not in the list of bugs for his program. It is not a function of the programmer&#39;s 
   age or language of choice. It is, simply, the assumption that his abilities are complete and there is
   <i xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">no room for improvement.</i></p>

   <p>Arguably, such is human nature; but I would argue that human nature is always on the prowl for knowledge and improvement. Only 
   hubris and the fear of being proven wrong hold us back. Resisting them both will not only make a better programmer, but a better 
   person as well.</p>

   <p>The social interactions and the quality of the people, I believe, are what create successful software teams more than any other 
   factors. Constant improvement in a programmer&#39;s skills and the ability to take criticism are fundamental requirements for members 
   of a software team. These requirements should precede all others.</p>

   <p>Think back to the last time you changed your style. Was it the new algorithm you learned, or commenting style, or simply a different 
   way of naming your variables? Whatever it was, it was only a step along the way, not the final change that made your code complete 
   and perfect. </p>

   <p>A programmer shouldn&#39;t be required to follow precise code guidelines to the letter; nor should he improvise those guidelines to 
   get the job done. Consider an orchestra -- neither static, soulless performers nor wildly improvisational virtuosos (though the latter 
   is more acclaimed). A static performer simply follows the notes without putting effort and soul into the music; the virtuoso must 
   restrain herself from errantly exploring new pieces of the melody or marching to the beat of her own drum.</p>

   <p><b><a name="2">Striking a concordant tone</a></b></p>

   <p>Code guidelines are like the written directions a musician follows -- when to come on, when to come off, how fast to play, what 
   beat, etc. The notes themselves, to extend the analogy somewhat precariously, are the goals of the project -- sometimes lone high 
   notes, and sometimes a harmony of instruments.</p>

   <p>In an orchestra, there is a conductor that directs but does not tell every musician how to play, and everyone has a part in the 
   performance. The conductor creates harmony. Because music has been around for many more centuries than the art of programming, perhaps 
   these are lessons well worth learning. The software project manager is neither a gorilla nor a walled-off convict. She is a part 
   of the team just like everyone else.</p>

   <p>The guidelines presented in this series are not to be blindly extracted into an official coding policy. The coding standards in 
   your project are uniquely yours, and they reflect your very own orchestral composition. Don&#39;t force programmers to do things exactly 
   right, thereby creating an atmosphere of distrust and fear. You can forget about code reviews, or admission of responsibility for 
   the smallest bugs.</p>

   <p>Instead, present the guidelines and watch how people react. If no one adopts the comment format you like, perhaps it&#39;s a bad format. 
   If people write without cleverness, perhaps you have been too clever in the guidelines. If the debugger you thought everyone must 
   run is sitting in a dusty room, still packed, then rethink the need for Whizzo Debugger 3.4. Maybe everyone is happy with Acme Debugger 
   1.0 for a reason.</p>

   <p>Of course, programmers can be stubborn for no reason at all, only out of reluctance to change. It&#39;s hard to convince people that 
   20 years of experience do not entitle them to an organized religion. On the other hand, freshly minted college graduates often lack 
   self-confidence. Recognize and adapt to those characteristics, and to all the others of your team. Present ideas to the stubbornly 
   experienced in such a way that they feel they have helped with it. Build up the college graduates with guidance and support until 
   they can fly on their own.</p>

   <p><b><a name="3">All this, just for a few coding guidelines?</a></b></p>

   <p>Coding guidelines are fundamental to a software team, just as direction and harmony are to music. They create consistency and 
   cohesiveness. New team members will feel welcome and gel more quickly. Ye olde team members will accept newcomers more readily. The 
   loss of a team member will not cripple the project just because someone can&#39;t understand someone else&#39;s code.</p>

   <p>Keep in mind that speed is not the only measure of improvement in a program&#39;s code. Consider ease of testing, documentation, and 
   maintenance just as important to any software project, especially for the long term. A language as flexible as Perl facilitates good 
   coding in every stage of the software project. Although this book focuses on Perl, many of the principles are valid for other languages 
   such as C, C++, Java, and Python.</p>

   <p>Finally, be an innovator. Regardless of your position in the team -- manager or member -- always look for new ideas and put them 
   into action. Perfection may be impossible, but it&#39;s a worthy goal. Innovators are the true strength of a team and without them the 
   melody grows stale very quickly. Stay in touch with your peers; continually learn new things from them. A medium such as Usenet (see
   <a href="#resources" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" trackclick="no">Resources</a>) is a great place for an 
   exchange of ideas. Teach and learn, to and from each other. Remember, there&#39;s always room for improvement. Above all, have fun, and 
   let the music begin.</p>

   <p><b><a name="resources">Resources</a> </b></p>
   <ul>

      <li>Read
      <a target="_blank" href="http://www-106.ibm.com/developerworks/views/linux/articles.jsp?sort_order=desc&expand=&sort_by=Date&show_abstract=true&view_by=Search&search_by=better+programming%3A" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" trackclick="no">
      more chapters</a> of <i xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">The road to better programming.</i></li>

      <li>Read
      <a target="_blank" href="https://www.amazon.com/exec/obidos/ISBN=0070328137/104-5461564-8341548" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" trackclick="no">
      <i>Applied Software Measurement: Assuring Productivity and Quality, 2nd edition</i></a> (McGraw-Hill), by Capers Jones.</li>

      <li>Read the
      <a target="_blank" href="http://ourworld.compuserve.com/homepages/softcomp/fpfaq.htm" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" trackclick="no">
      Function Points FAQ</a>, and check out the
      <a target="_blank" href="http://www.ifpug.org/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" trackclick="no">International 
      Function Points User Group</a>.</li>

      <li>Read
      <a target="_blank" href="http://www.faqs.org/usenet/index.html" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" trackclick="no">
      all you ever wanted to know about Usenet</a>.</li>

      <li>Read Teodor&#39;s other Perl articles in the <i xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">developerWorks</i> &quot;Cultured 
      Perl&quot; series:
      <ul xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-plset/index.html" trackclick="no">A programmer&#39;s 
         Linux-oriented setup</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-perl3/index.html" trackclick="no">Application 
         configuration with Perl</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-punix.html" trackclick="no">Automating UNIX 
         system administration with Perl</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-pl-deb.html" trackclick="no">Debugging Perl 
         with ease</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-japh.html" trackclick="no">The elegance 
         of JAPH</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-genperl/" trackclick="no">Genetic algorithms 
         applied with Perl </a></li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-p101/" trackclick="no">One-liners 101</a>
         </li>

         <li>
         <a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-perl-parsing/index.html" trackclick="no">Parsing 
         with Perl modules</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-p560.html" trackclick="no">Perl 5.6 for 
         C and Java programmers</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-pexcel/" trackclick="no">Reading and writing 
         Excel files with Perl</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-perl5.html" trackclick="no">Review of Programming 
         Perl, Third Edition</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-perl1/" trackclick="no">Small observations 
         about the big picture</a> </li>

         <li><a target="_blank" href="http://www-106.ibm.com/developerworks/linux/library/l-perl-speak.html" trackclick="no">Writing 
         Perl programs that speak English</a></li>
      </ul>
      </li>

      <li>Browse
      <a target="_blank" href="http://www-106.ibm.com/developerworks/linux/?article=lr" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" trackclick="no">
      more Linux resources</a> on <i xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">developerWorks</i>.</li>

      <li>Browse
      <a target="_blank" href="http://www-124.ibm.com/developerworks/opensource/?article=osr" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" trackclick="no">
      more Open source resources</a> on <i xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">developerWorks</i>.</li>
   </ul>

   <p>About the author</p>

   <p><b><i>Teodor Zlatanov graduated with an M.S. in computer engineering from Boston University in 1999. He has worked as a programmer 
   since 1992, using Perl, Java, C, and C++. His interests are in open source work on text parsing</i></b></p>
</blockquote>

</em></b></i><center><table border="0" width="100"><tr>
<td align="center"><a href="#n20031017X_the_road_to_better_programming_introduction_and_chapter_1"><img border="0" src="/Images/up.png" width="16" height="16"></a></td>
<td align="center"><a name="n20031017X_the_road_to_better_programming" href="#NEWS_TOC"><img border="0" src="/Images/home.gif" width="16" height="18"></a></td>
<td align="center"><a href="#n20200930X_postfix_if_conditions_are_useful_mainly_to_specify_exist_condition_in_the_loop_rarely"><img border="0" src="/Images/down.png" width="16" height="16"></a></td>
</tr></table></center>

<h4><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road1.html">The road to better programming</a></h4>
<blockquote>
   <ul>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road1.html">The road to better programming- Introduction 
      and chapter 1</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road2.html">The road to better programming- Chapter 
      2</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road3.html">The road to better programming- Chapter 
      3</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road4.html">The road to better programming- Chapter 
      4</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road5.html">The road to better programming- Chapter 
      5</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road6.html">The road to better programming- Chapter 
      6. Developing cfperl, from the beginning</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road7.html">The road to better programming- Chapter 
      7. Top-level control flow and configuration</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road8.html">The road to better programming- Chapter 
      8. The top-level and compound-class parsers</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road9.html">The road to better programming- Chapter 
      9. The classes and default parsers</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road10.html">The road to better programming- Chapter 
      10. User management with cfperl</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road11.html">The road to better programming- Chapter 
      11. Crontab management with cfperl</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road12.html">The road to better programming- Chapter 
      12. File editing with the perledit- section</a></li>
   </ul>
</blockquote>

</em></b></i><h2><a name="Recommended_Links">Recommended Links</a></h2>
<table border="0" width="100%" height="220">
<tr>
<td width="100%" align="center">	

<h3>Google matched content</h3>
<script data-cfasync="false" src="/cdn-cgi/scripts/5c5dd728/cloudflare-static/email-decode.min.js"></script><script async src="//pagead2.googlesyndication.com/pagead/js/adsbygoogle.js"></script>
<ins class="adsbygoogle"
     style="display:block"
     data-ad-format="autorelaxed"
     data-ad-client="ca-pub-4031247137266443"
     data-ad-slot="1626659014"></ins>
<script>
     (adsbygoogle = window.adsbygoogle || []).push({});
</script>
</td>
</table>
<h3>Softpanorama Recommended</h3>


<h3><a name="Top_articles">Top articles</a></h3><h3><a name="Sites">Sites</a></h3>

<h4>Internal</h4>
<ul>

   <li><b><a target="_blank" href="../../SE/programming_style.shtml">Programming Style</a></b></li>
</ul>

<h4>External</h4>
<ul>

   <li><a target="_blank" href="http://www.vromans.org/johan/software/sw_pbpref.html">Perl Best Practices Quick Reference Guide</a></li>

   <li><a target="_blank" href="http://oreilly.com/perl/excerpts/perl-best-practices/appendix-b.html">Appendix B Perl Best Practices 
   - O&#39;Reilly Media</a></li>

   <li><a target="_blank" href="http://perldoc.perl.org/perlstyle.html">perlstyle - perldoc.perl.org</a></li>

   <li><a target="_blank" href="http://www.perl.com/doc/FMTEYEWTK/style/slide-index.html">Perl Style</a> Tom Christiansen  guide 
   (1998)</li>

   <li><a target="_blank" href="http://www.kuleuven.net/perlcourse/perlingo.html">Perl Style Guide - version 1.12</a></li>

   <li><a target="_blank" href="http://www.perl.com/lpt/a/2005/07/14/bestpractices.html">Perl.com- Ten Essential Development Practices</a>
   </li>

   <li><a target="_blank" href="http://webdeveloper.com/cgi-perl/cgi_perl_style.html">Perl Style - WebDeveloper.com </a>Perl Style By 
   Tom Christiansen This article first appeared November 21st, 1998.</li>

   <li>A
   <a target="new" s_oid="http://www.oreilly.com/catalog/perlbp/chapter/ch09.pdf" s_oidt="0" href="http://www.oreilly.com/catalog/perlbp/chapter/ch09.pdf">
   sample chapter</a> on subroutines from Damian Conway book. </li>

   <li><a target="_blank" href="http://mailman.linuxchix.org/pipermail/courses/2003-October/001349.html">[Courses] [Perl] Part 13 Perl 
   Style</a>  Dan Richter Oct 17,2003 </li>

   <li><a target="_blank" href="http://ali.as/devel/code.html">Perl Style Guide for Large Scale Object Oriented Development</a></li>

   <li>

   <p>The road to better programming-</p>
   <ul>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road1.html">The road to better programming- Introduction 
      and chapter 1</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road2.html">The road to better programming- Chapter 
      2</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road3.html">The road to better programming- Chapter 
      3</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road4.html">The road to better programming- Chapter 
      4</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road5.html">The road to better programming- Chapter 
      5</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road6.html">The road to better programming- Chapter 
      6. Developing cfperl, from the beginning</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road7.html">The road to better programming- Chapter 
      7. Top-level control flow and configuration</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road8.html">The road to better programming- Chapter 
      8. The top-level and compound-class parsers</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road9.html">The road to better programming- Chapter 
      9. The classes and default parsers</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road10.html">The road to better programming- Chapter 
      10. User management with cfperl</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road11.html">The road to better programming- Chapter 
      11. Crontab management with cfperl</a></li>

      <li><a target="_blank" href="http://www.ibm.com/developerworks/linux/library/l-road12.html">The road to better programming- Chapter 
      12. File editing with the perledit- section</a></li>
   </ul>
   </li>
</ul>
<hr>
<hr noshade color="#FF0000" size="5">

<h2><a name="Etc">Etc</a></h2>

<p><b>Society</b></p>
<blockquote>

<p><b><a href="/Skeptics/groupthink.shtml"><font size="2">Groupthink</font></a><font size="2"> :
<a href="/Skeptics/Political_skeptic/Two_party_system_as_poliarchy/index.shtml">Two Party System 
as Polyarchy</a> : 
<a href="/Skeptics/Financial_skeptic/Casino_capitalism/Corruption_of_regulators/index.shtml">
Corruption of Regulators</a> :
<a href="/Social/Bureaucracy/index.shtml">Bureaucracies</a> :
<a href="/Social/Toxic_managers/Micromanagers/understanding_micromanagers.shtml">Understanding Micromanagers 
and Control Freaks</a> : <a href="/Social/Toxic_managers/index.shtml">Toxic Managers</a> :&nbsp;&nbsp;
<a href="/Skeptics/Pseudoscience/harvard_mafia.shtml">Harvard Mafia</a> :
<a href="/Social/Toxic_managers/Communication/diplomatic_communication.shtml">Diplomatic Communication</a> 
: <a href="/Social/Toxic_managers/surviving_a_bad_performance_review.shtml">Surviving a Bad Performance 
Review</a> : <a href="/Skeptics/Financial_skeptic/index.shtml">Insufficient Retirement Funds as 
Immanent Problem of Neoliberal Regime</a> : <a href="/Skeptics/index.shtml">PseudoScience</a> :
<a href="/Skeptics/Political_skeptic/index.shtml">Who Rules America</a> :
<a href="/Skeptics/Political_skeptic/Neoliberalism/index.shtml">Neoliberalism
</a>&nbsp;: <a href="/Skeptics/Political_skeptic/Elite_theory/iron_law_of_oligarchy.shtml">The Iron 
Law of Oligarchy</a> : </font><a href="/Skeptics/Political_skeptic/libertarianism.shtml">
<font size="2">Libertarian Philosophy</font></a></b></p>
</blockquote>

<p><b>Quotes</b></p>
<blockquote>

<p><b><font size="2" face="Arial"> 
<a href="/Skeptics/Quotes/war_and_peace_quotes.shtml">War and Peace</a> </font>
<font face="Arial"><font size="2">: <a href="/Skeptics/Quotes/financial_quotes.shtml">Skeptical 
Finance</a> : <a href="/Skeptics/Quotes/famous_galbraith_quotes.shtml">John 
Kenneth Galbraith</a> :<a href="/Skeptics/Quotes/talleyrand_quotes.shtml">Talleyrand</a> :
<a href="/Skeptics/Quotes/oscar_wilde_quotes.shtml">Oscar Wilde</a> :
<a href="/Skeptics/Quotes/bismarck_quotes.shtml">Otto Von Bismarck</a> :
<a href="/Skeptics/Quotes/keynes_quotes.shtml">Keynes</a> :
<a href="/Skeptics/Quotes/george_carlin.shtml">George Carlin</a> :
<a href="/Skeptics/skeptical_quotes.shtml">Skeptics</a> :
<a href="/Skeptics/Quotes/propaganda.shtml">Propaganda</a>&nbsp; : <a href="/SE/quotes.shtml">SE 
quotes</a> : <a href="/Lang/quotes.shtml">Language Design and Programming Quotes</a> :
<a href="/Bulletin/quotes.shtml">Random IT-related quotes</a> :&nbsp;
<a href="/Skeptics/Quotes/somerset_maugham.shtml">Somerset Maugham</a> :
<a href="/Skeptics/Quotes/marcus_aurelius.shtml">Marcus Aurelius</a> :
<a href="/Skeptics/Quotes/kurt_vonnegut_quotes.shtml">Kurt Vonnegut</a> :
<a href="/Skeptics/Quotes/eric_hoffer.shtml">Eric Hoffer</a> :
<a href="/Skeptics/Quotes/churchill_quotes.shtml">Winston Churchill</a> :
<a href="/Skeptics/Quotes/napoleon_quotes.shtml">Napoleon Bonaparte</a> :
<a href="/Skeptics/Quotes/ambrose_bierce.shtml">Ambrose Bierce</a> :&nbsp;
<a href="/Skeptics/Quotes/bernard_shaw.shtml">Bernard Shaw</a> : </font>
<a href="/Skeptics/Quotes/mark_twain_quotes.shtml"><font size="2">Mark Twain Quotes</font></a></font></b></p>
</blockquote>

<p><b>Bulletin:</b></p>
<blockquote>

<p><b><font face="Arial"><a href="http://softpanorama.biz/Bulletin/Sp2013_v25/bulletin25_12.shtml">
<font size="2">Vol 25, No.12 (December, 2013) Rational Fools vs. Efficient Crooks The efficient 
markets hypothesis</font></a><font size="2"> :
<a href="http://softpanorama.biz/Skeptics/Political_skeptic/Bulletin/political_skeptic2013.shtml">
Political Skeptic Bulletin, 2013</a> :
<a href="http://softpanorama.biz/Skeptics/Financial_skeptic/Unemployment/Bulletin/unempoyment2010.shtml">
Unemployment Bulletin, 2010</a> :
<a href="http://softpanorama.biz/Bulletin/Sp2011_v23/bulletin23_10.shtml">&nbsp;Vol 23, No.10 
(October, 2011) An observation about corporate security departments</a> :
<a href="http://softpanorama.biz/Skeptics/Political_skeptic/Fifth_column/Color_revolutions/Euromaydan/Bulletin/euromaydan14_06.shtml">
Slightly Skeptical Euromaydan Chronicles, June 2014</a> :
<a href="http://softpanorama.biz/Skeptics/Financial_skeptic/Casino_capitalism/12_Apostols_of_deregulation/Greenspan/Bulletin/greenspan_bulletin2008.shtml">
Greenspan legacy bulletin, 2008</a> :
<a href="/Bulletin/Sp2013_v25/bulletin25_10.shtml">Vol 25, No.10 (October, 2013) Cryptolocker Trojan 
(Win32/Crilock.A)</a> :
<a href="/Bulletin/Sp2013_v25/bulletin25_08.shtml">Vol 25, No.08 (August, 2013) Cloud providers 
as intelligence collection hubs</a> : 
<a href="http://softpanorama.biz/Skeptics/Financial_skeptic/Humor/Bulletin/financial_humor2010.shtml">
Financial Humor Bulletin, 2010</a> :
<a href="http://softpanorama.biz/Skeptics/Financial_skeptic/Inequality/Bulletin/inequality2009.shtml">
Inequality Bulletin, 2009</a> :
<a href="http://softpanorama.biz/Skeptics/Financial_skeptic/Humor/Bulletin/financial_humor2008.shtml">
Financial Humor Bulletin, 2008</a> :
<a href="http://softpanorama.biz/Copyright/Bulletin/copyleft_problems2004.shtml">Copyleft Problems 
Bulletin, 2004</a> :
<a href="http://softpanorama.biz/Skeptics/Financial_skeptic/Humor/Bulletin/financial_humor2011.shtml">
Financial Humor Bulletin, 2011</a> :
<a href="http://softpanorama.biz/Skeptics/Financial_skeptic/Energy/Bulletin/energy_bulletin2010.shtml">
Energy Bulletin, 2010</a> : <a href="http://softpanorama.biz/Malware/Bulletin/malware2010.shtml">
Malware Protection Bulletin, 2010</a> : <a href="/Bulletin/Sp2014_v26/bulletin26_01.shtml">Vol 26, 
No.1 (January, 2013) Object-Oriented Cult</a> :
<a href="http://softpanorama.biz/Skeptics/Political_skeptic/Bulletin/political_skeptic2011.shtml">
Political Skeptic Bulletin, 2011</a> :
<a href="/Bulletin/Sp2011_v23/bulletin23_11.shtml">Vol 23, No.11 (November, 2011) Softpanorama classification 
of sysadmin horror stories</a> : <a href="/Bulletin/Sp2013_v25/bulletin25_05.shtml">Vol 25, No.05 
(May, 2013) Corporate bullshit as a communication method</a>&nbsp; : </font><a href="/Bulletin/Sp2013_v25/bulletin25_06.shtml">
<font size="2">Vol 25, No.06 (June, 2013) A Note on the Relationship of Brooks Law and Conway Law</font></a></font></b></p>
</blockquote>

<p align="left"><b>History:</b></p>
<blockquote>

<p><b><font face="Arial"><a href="/History/index.shtml"><font size="2">Fifty glorious years (1950-2000): 
the triumph of the US computer engineering</font></a><font size="2"> :
<a href="/People/Knuth/index.shtml">Donald Knuth</a> : <a href="/People/Knuth/taocp.shtml">TAoCP 
and its Influence of Computer Science</a> : <a href="/People/Stallman/index.shtml">Richard Stallman</a> 
: <a href="/People/Torvalds/index.shtml">Linus Torvalds</a>&nbsp; :
<a href="/People/Wall/index.shtml">Larry Wall </a>&nbsp;:
<a href="/People/Ousterhout/index.shtml">John K. Ousterhout</a> : <a href="/History/ctss.shtml">
CTSS</a> : <a href="/History/multix.shtml">Multix OS</a> <a href="/History/Unix/index.shtml">Unix 
History</a> : <a href="/People/Shell_giants/introduction.shtml">Unix shell history</a> :
<a href="/Editors/Vimorama/history.shtml">VI editor</a> :
<a href="/Scripting/Piporama/history.shtml">History of pipes concept</a> :
<a href="/Solaris/solaris_history.shtml">Solaris</a> : <a href="/History/dos_history.shtml">MS DOS</a> 
:&nbsp; <a href="/History/lang_history.shtml">Programming Languages History</a> :
<a href="/Lang/pl1.shtml">PL/1</a> : <a href="/Lang/simula67.shtml">Simula 67</a> :
<a href="/Lang/Cilorama/history.shtml">C</a> :
<a href="/People/Stallman/history_of_gcc_development.shtml">History of GCC development</a> :&nbsp;
<a href="/People/Scripting_giants/scripting_languages_as_vhll.shtml">Scripting Languages</a> :
<a href="/Scripting/Perlbook/Ch01/perl_history.shtml">Perl history&nbsp; </a>&nbsp;:
<a href="/History/os_history.shtml">OS History</a> : <a href="/Mail/history.shtml">Mail</a> :
<a href="/DNS/history.shtml">DNS</a> : <a href="/Net/Application_layer/SSH/ssh_history.shtml">SSH</a> 
: <a href="/History/cpu_history.shtml">CPU Instruction Sets</a> :
<a href="/Hardware/Sun/history_of_sparc.shtml">SPARC systems 1987-2006</a> :
<a href="/OFM/Paradigm/Ch03/norton_commander.shtml">Norton Commander</a> :
<a href="/Windows/Norton_utilities/history.shtml">Norton Utilities</a> :
<a href="/Windows/Ghosting/ghost_history.shtml">Norton Ghost</a> :
<a href="/Office/Frontpage/history.shtml">Frontpage history</a> :
<a href="/Malware/Malware_defense_history/index.shtml">Malware Defense History</a> :
<a href="/Utilities/Screen/history.shtml">GNU Screen</a> : </font>
<a href="/OSS/oss_early_history.shtml"><font size="2">OSS early history</font></a></font></b></p>
</blockquote>

<p><b>Classic books:</b></p>
<blockquote>

<p><b><font face="Arial"><a href="/Bookshelf/Classic/peter_principle.shtml"><font size="2">The Peter 
Principle</font></a><font size="2"> : <a href="/Bookshelf/Classic/parkinson_law.shtml">Parkinson 
Law</a> : <a href="/Bookshelf/Classic/nineteen_eighty_four.shtml">1984</a> :
<a href="/Bookshelf/Classic/tmmm.shtml">The Mythical Man-Month</a> :&nbsp;
<a href="/Bookshelf/Classic/polya_htsi.shtml">How to Solve It by George Polya</a> :
<a href="/Bookshelf/Classic/taocp.shtml">The Art of Computer Programming</a> :
<a href="/Bookshelf/Classic/teops.shtml">The Elements of Programming Style</a> :
<a href="/Bookshelf/Classic/unix_haters_handhook.shtml">The Unix Hater’s Handbook</a> :
<a href="/Bookshelf/Classic/jargon_file.shtml">The Jargon file</a> :
<a href="/Bookshelf/Classic/true_believer.shtml">The True Believer</a> :
<a href="/Bookshelf/Classic/programming_pearls.shtml">Programming Pearls</a> :
<a href="/Bookshelf/Classic/good_soldier_svejk.shtml">The Good Soldier Svejk</a> : </font>
<a href="/Bookshelf/Classic/power_elite.shtml"><font size="2">The Power Elite</font></a></font></b></p>
</blockquote>

<p><b>Most popular humor pages:</b></p>
<blockquote>

<p><font face="Arial"><b><a href="/Bulletin/Humor/Slackerism/it_slacker_manifest.shtml">
<font size="2">Manifest of the Softpanorama IT Slacker Society</font></a><font size="2"> :
<a href="/Bulletin/Humor/Slackerism/ten_commandments_of_software_slackerism.shtml">Ten Commandments 
of the IT Slackers Society</a> : <a href="/Bulletin/Humor/index.shtml">Computer Humor Collection</a> 
: <a href="/Bulletin/Humor/bsd_logo_story.shtml">BSD Logo Story</a> :
<a href="/Bulletin/Humor/cuckoo_egg.shtml">The Cuckoo&#39;s Egg </a>:
<a href="/Bulletin/Humor/slang.shtml">IT Slang</a> : <a href="/Lang/Cpp_rama/humor.shtml">C++ Humor</a> 
: <a href="/Bulletin/Humor/Archive/humor059.shtml">ARE YOU A BBS ADDICT?</a> :
<a href="/Bulletin/Humor/Archive/humor092.shtml">The Perl Purity Test</a> :
<a href="/Bulletin/Humor/Archive/humor065.shtml">Object oriented programmers of all nations</a> 
: <a href="/Skeptics/Financial_skeptic/Humor/financial_humor.shtml">Financial Humor</a> :
<a href="/Skeptics/Financial_skeptic/Humor/Bulletin/financial_humor2008.shtml">Financial Humor Bulletin, 
2008</a> : <a href="/Skeptics/Financial_skeptic/Humor/Bulletin/financial_humor2010.shtml">Financial 
Humor Bulletin, 2010</a> : <a href="/Editors/humor.shtml">The Most Comprehensive Collection of Editor-related 
Humor</a> : <a href="/Lang/programming_languages_humor.shtml">Programming Language Humor</a> :
<a href="/Skeptics/Financial_skeptic/Casino_capitalism/Systemic_instability_of_financial_sector/TBTF/Goldman_Sachs/humor.shtml">
Goldman Sachs related humor</a> :
<a href="/Skeptics/Financial_skeptic/Casino_capitalism/Twelve_apostles_of_deregulation/Greenspan/greenspan_humor.shtml">
Greenspan humor</a> : <a href="/Lang/Cilorama/humor.shtml">C Humor</a> :
<a href="/Scripting/humor.shtml">Scripting Humor</a> :
<a href="/Bulletin/Humor/real_programmers_humor.shtml">Real Programmers Humor</a> :
<a href="/WWW/humor.shtml">Web Humor</a> : <a href="/Copyright/humor.shtml">GPL-related Humor</a> 
: <a href="/OFM/ofm_humor.shtml">OFM Humor</a> :
<a href="/Skeptics/Political_skeptic/humor.shtml">Politically Incorrect Humor</a> :
<a href="/Security/IDS/humor.shtml">IDS Humor</a> : <a href="/Bulletin/Humor/linux_sucks.shtml">
&quot;Linux Sucks&quot; Humor </a>: <a href="/Links/Russian/Culture/Music/russian_musical_humor.shtml">Russian 
Musical Humor</a> : <a href="/Bulletin/Humor/best_russian_programmer_humor.shtml">Best Russian Programmer 
Humor</a> : <a href="/Bulletin/Humor/Archive/humor070.shtml">Microsoft plans to buy Catholic Church</a> 
: <a href="/People/Stallman/rms_related_humor.shtml">Richard Stallman Related Humor</a> :
<a href="/Admin/humor.shtml">Admin Humor</a> : <a href="/People/Wall/perl_related_humor.shtml">Perl-related 
Humor</a> : <a href="/People/Torvalds/linus_torvalds_related_humor.shtml">Linus Torvalds Related 
humor</a> : <a href="/Skeptics/humor.shtml">PseudoScience Related Humor</a> :
<a href="/Net/net_humor.shtml">Networking Humor</a> :
<a href="/Scripting/Shellorama/humor.shtml">Shell Humor</a> :
<a href="/Skeptics/Financial_skeptic/Humor/Bulletin/financial_humor2011.shtml">Financial Humor Bulletin, 
2011</a> : <a href="/Skeptics/Financial_skeptic/Humor/Bulletin/financial_humor2012.shtml">Financial 
Humor Bulletin, 2012</a> :
<a href="/Skeptics/Financial_skeptic/Humor/Bulletin/financial_humor2013.shtml">Financial Humor Bulletin, 
2013</a> : <a href="/Lang/Javarama/humor.shtml">Java Humor</a> : <a href="/SE/humor.shtml">Software 
Engineering Humor</a> : <a href="/Solaris/humor.shtml">Sun Solaris Related Humor</a> :
<a href="/Education/humor.shtml">Education Humor</a> : <a href="/Admin/Tivoli/ibm_humor.shtml">IBM 
Humor</a> : <a href="/Lang/Asmorama/humor.shtml">Assembler-related Humor</a> :
<a href="/Editors/Vimorama/vim_humor.shtml">VIM Humor</a> : <a href="/Malware/humor.shtml">Computer 
Viruses Humor</a> : <a href="/Bulletin/Humor/Archive/humor034.shtml">Bright tomorrow is rescheduled 
to a day after tomorrow</a> : <a href="/Bulletin/Humor/classic_computer_humor.shtml">Classic Computer 
Humor</a> </font></b></font></p>
</blockquote>

<p align="left"><b><a href="/Bulletin/Humor/last_but_not_least.shtml">The Last but not Least</a> </b> <em>Technology is dominated by 
two types of people: those who understand what they do not manage and those who manage what they do not understand ~Archibald Putt. 
Ph.D</em></p>

<hr size="5" noshade color="#FF0000"><font face="Verdana" size="1">

<p><i><b>Copyright © 1996-2021 by </b></i><b><i>Softpanorama Society</i></b>. <a target="_blank" href>www.softpanorama.org</a> 
was initially created as a service to the (now defunct) UN Sustainable Development Networking Programme (<a target="_blank" href="http://www.un.org/Depts/dhl/sflib/">SDNP</a>) 
without any remuneration. This document is an industrial compilation designed and <b>created exclusively 
for educational use</b> and is distributed under the <a href="/license.shtml">Softpanorama Content License</a>. 
Original materials copyright belong 
to respective owners. <i><b>Quotes are made<font color="#FF0000"> for educational purposes only</font> 
in compliance with the fair use doctrine. </b></i> 
</p>

<p><em><b>FAIR USE NOTICE</b> </em>This site contains 
		copyrighted material the use of which has not always been specifically 
		authorized by the copyright owner. We are making such material available 
		to advance understanding of computer science, IT technology, economic, scientific, and social  
		issues. We believe this constitutes a 'fair use' of any such 
		copyrighted material as provided by section 107 of the US Copyright Law according to which 
such material can be distributed without profit exclusively for research and educational purposes.</p>

<p><b>This is a Spartan WHYFF (We Help You For Free) 
site written by people for whom English is not a native language.</b> Grammar and spelling errors should 
be expected. <b>The site contain some broken links as it develops like a living tree...</b></p>

<table border="0" width="100%">
<tr>
<td>

<p align="center"><font face="Verdana" size="1">
<input type="image" src="https://www.paypalobjects.com/en_US/i/btn/btn_donateCC_LG.gif" border="0" name="submit" alt="PayPal - The safer, easier way to pay online!">
<img alt="" border="0" src="https://www.paypalobjects.com/en_US/i/scr/pixel.gif" width="1" height="1">
</font></td>
<td><font face="Verdana" size="1">You can use PayPal to to buy a cup of coffee for authors 
of this site </font></td>
</tr>
</table>

<p><b>Disclaimer:</b> </p>

<p><i>The statements, views and opinions presented on this web page are those of the author (or 
referenced source) and are 
not endorsed by, nor do they necessarily reflect, the opinions of the Softpanorama society.</i> <i>We do not warrant the correctness 
of the information provided or its fitness for any purpose. </i>The site uses AdSense so you need to be aware of Google privacy policy. You you do not want to be 
tracked by Google please disable Javascript for this site. <em>This site is perfectly usable without 
Javascript.</em> 
</font></p>


<p><i><font color="#FF0000">Last updated: <!--webbot bot="Timestamp" s-type="EDITED" s-format="%B %d, %Y" startspan -->September 30, 2020<!--webbot bot="Timestamp" i-checksum="37949" endspan --></font></i></p>

</body>

</html>