2014-03-28

← Older revision

Revision as of 13:43, 28 March 2014

(3 intermediate revisions by one user not shown)

Line 2:

Line 2:

 

 

 

=== Plan for cross-platform up front ===

 

=== Plan for cross-platform up front ===

 

+

 

+

See also:

 

+

* [[Multiplatform Programming Guide]]

 

+

* [[Introduction to platform-sensitive development]]

 

 

 

There are several advantages to doing your development in a cross-platform way. Probably the most important advantage is that it can result in better, more reliable software. This is because cross-platform development requires that you spend more time planning your software before you start writing code. It also forces you to develop good programming habits, to test your software more, and to develop a more disciplined approach toward what you're doing. These things (planning, testing, discipline) tend to spill over into other aspects of your development, resulting in still more improvements in design and reliability.

 

There are several advantages to doing your development in a cross-platform way. Probably the most important advantage is that it can result in better, more reliable software. This is because cross-platform development requires that you spend more time planning your software before you start writing code. It also forces you to develop good programming habits, to test your software more, and to develop a more disciplined approach toward what you're doing. These things (planning, testing, discipline) tend to spill over into other aspects of your development, resulting in still more improvements in design and reliability.

Line 11:

Line 15:

 

The best way to make an application cross-platform is to design it to be cross-platform. If you plan on adding cross-platform support to your finished application later, you may find that you'll have to rewrite parts of it or, worse, find yourself in a Windows-only cul-de-sac that makes adding    cross-platform support difficult or impossible.

 

The best way to make an application cross-platform is to design it to be cross-platform. If you plan on adding cross-platform support to your finished application later, you may find that you'll have to rewrite parts of it or, worse, find yourself in a Windows-only cul-de-sac that makes adding    cross-platform support difficult or impossible.

 

 



If you've already developed an application for Windows with Delphi, it's possible in some situations to convert it to cross-platform app without starting over. If you want to do a one-way conversion of your Delphi code and forms to Lazarus, you can use some of the conversion commands on the Lazarus Tools menu. You can also convert your Delphi app to Lazarus, but retain Delphi compatibility (see this [
http://wiki.lazarus.freepascal.org/XDev_Toolkit
link] for more information).

+

If you've already developed an application for Windows with Delphi, it's possible in some situations to convert it to cross-platform app without starting over. If you want to do a one-way conversion of your Delphi code and forms to Lazarus, you can use some of the conversion commands on the Lazarus Tools menu. You can also convert your Delphi app to Lazarus, but retain Delphi compatibility (see this [
[XDev Toolkit|
link] for more information).

 

 

 

=== Separate code into UI and non-UI units ===

 

=== Separate code into UI and non-UI units ===

 

 



Probably one of the most important programming practices you can employ with any application, whether cross-platform or not, is to divide your code into user
-
interface (UI) units and non-user interface (non-UI) units. In general, this means that your non-UI units should not implement any of the program's user interface and your UI units should not contain any of the computational or database parts of the program. In practice, a good way to enforce this separation is to require that the units you develop only use units that are appropriate to their side of this "divide". For example
, your
non-UI units should not use LCL units such as Forms, Dialogs, StdCtrls, Button, Graphics, Controls, LMessages, and so forth, or any of your own units that use these units.
And your
UI units should probably not be using units such as Math, Dbf, MySQLDB4, etc. Follow the example of the Free Pascal and Lazarus units themselves, where general-purpose units like SysUtils, Classes
and
Variants don't use any LCL units and LCL units like Forms and Controls don't supply any general-purpose routines.

+

See also:

 

+

* [[SQLdb Tutorial4]] Info about using data modules with SQLDB components. Data modules can also be used for non-database non-GUI controls.

 

+

 

 

+

Probably one of the most important programming practices you can employ with any application, whether cross-platform or not, is to divide your code into user interface (UI) units and non-user interface (non-UI) units. In general, this means that your non-UI units should not implement any of the program's user interface and your UI units should not contain any of the computational or database parts of the program. In practice, a good way to enforce this separation is to require that the units you develop only use units that are appropriate to their side of this "divide".

 

+

For example
:

 

+

* Your
non-UI units should not use LCL units such as Forms, Dialogs, StdCtrls, Button, Graphics, Controls, LMessages, and so forth, or any of your own units that use these units.  

 

+

* Your
UI units should probably not be using units such as Math, Dbf, MySQLDB4, etc.  

 

+

 

 

+

Follow the example of the Free Pascal and Lazarus units themselves, where general-purpose units like SysUtils, Classes
,
Variants
and [[LazUtils]]
don't use any LCL units and LCL units like Forms and Controls don't supply any general-purpose routines.

 

 

 

There are several advantages to dividing up your code like this. For example, it means that you will be able to use your non-UI units in any type of program, whether a GUI app, library, console app, even a Web app, without making any changes to the code. And it also means that if necessary you could develop a completely different user interface for your program or switch to a different UI library or control without changing the underlying non-UI code. It's also critical that this partitioning be done in large projects so that multiple developers can work on different parts of the program without stepping on each other's toes.

 

There are several advantages to dividing up your code like this. For example, it means that you will be able to use your non-UI units in any type of program, whether a GUI app, library, console app, even a Web app, without making any changes to the code. And it also means that if necessary you could develop a completely different user interface for your program or switch to a different UI library or control without changing the underlying non-UI code. It's also critical that this partitioning be done in large projects so that multiple developers can work on different parts of the program without stepping on each other's toes.

Line 35:

Line 47:

 

   ./mybatch.sh

 

   ./mybatch.sh

 

 



There are many way that batch files can be used in development. For example, you might have one set of compiler options that you use during development and another set that you use to create the executable that you distribute. Here's an example of a command you can put in a batch file that runs the Free Pascal compiler
:

+

There are many way that batch files can be used in development. For example, you might have one set of compiler options that you use during development and another set that you use to create the executable that you distribute. Here's an example of a command you can put in a batch file that runs the Free Pascal compiler
.

 

 



With
Windows:

+

On
Windows:



 

+

<syntaxhighlight lang="dos">



 
c:\lazarus\pp\bin\i386-win32\fpc.exe -Sda -Cirot -gl %1

+

c:\lazarus\pp\bin\i386-win32\fpc.exe -Sda -Cirot -gl %1



 

+

</syntaxhighlight>



With Unix:

+

 

 



 
/usr/local/bin/fpc -Sda -Cirot -gl $@

+

On Unix/Linux:

 

+

<syntaxhighlight lang="bash">

 

+

/usr/local/bin/fpc -Sda -Cirot -gl $@

 

+

</syntaxhighlight>

 

 

 

The above command compiles the unit or non-GUI program specified on the command line with Delphi compatibility, assertions, I/O checking, range checking, overflow checking, stack checking, and line number tracing turned on. Note that you can add additional pre- and post-compile commands to a batch file. For example, you could run the strip.exe utility to shrink the size of your executable or copy the executable to a different folder for testing.

 

The above command compiles the unit or non-GUI program specified on the command line with Delphi compatibility, assertions, I/O checking, range checking, overflow checking, stack checking, and line number tracing turned on. Note that you can add additional pre- and post-compile commands to a batch file. For example, you could run the strip.exe utility to shrink the size of your executable or copy the executable to a different folder for testing.

Line 49:

Line 63:

 

Here's a similar command that you can put in a batch file to compile a GUI program that you specify on the command line:

 

Here's a similar command that you can put in a batch file to compile a GUI program that you specify on the command line:

 

 



With Windows (all on one line):

+

{{Note|The examples below directly call the FPC compiler. It may be easier to use [[lazbuild]] which will let you compile Lazarus projects for you without any need to set include paths etc}}

 

 



 
c:\lazarus\pp\bin\i386-win32\fpc.exe -dLCL -WG -Sda -Cirot -gl

+

On Windows (all on one line):



 
-Fuc:\lazarus\lcl\units\i386-win32;c:\lazarus\lcl\units\i386-win32\win32 %1

+

<syntaxhighlight lang="dos">

 

+

c:\lazarus\pp\bin\i386-win32\fpc.exe -dLCL -WG -Sda -Cirot -gl -Fuc:\lazarus\lcl\units\i386-win32;c:\lazarus\lcl\units\i386-win32\win32 %1

 

+

</syntaxhighlight>

 

 



With
OS X
(all on one line)
:

+

On
OS X:

 

+

<syntaxhighlight lang="bash">

 

+

/usr/local/bin/fpc -dLCL -WG -Sda -Cirot -gl \

 

+

-Fu/usr/local/share/lazarus/lcl/units/powerpc-darwin \

 

+

-Fu/usr/local/share/lazarus/lcl/units/powerpc-darwin/gtk \

 

+

-Fl/usr/X11R6/lib/ -Fl/sw/lib/ $@

 

+

</syntaxhighlight>

 

 



  /usr/local/bin/fpc -dLCL -WG -Sda -Cirot -gl -Fu/usr/local/share/lazarus/lcl/units/powerpc-darwin

 



  -Fu/usr/local/share/lazarus/lcl/units/powerpc-darwin/gtk -Fl/usr/X11R6/lib/ -Fl/sw/lib/ $@

 

 

 

 

=== How to avoid using a debugger ===

 

=== How to avoid using a debugger ===

 

+

See also:

 

+

* [[Debugger Setup]]

 

+

* [[GDB Debugger Tips]]

 

 

 

Using a debugger is rarely much fun. It often means that you've given up trying to figure out what's wrong with your program and have turned to the debugger as a last resort. Here are some simple techniques that you can employ that will help minimize your use of a debugger:

 

Using a debugger is rarely much fun. It often means that you've given up trying to figure out what's wrong with your program and have turned to the debugger as a last resort. Here are some simple techniques that you can employ that will help minimize your use of a debugger:

 

 



* Develop with all run-time checks turned on. You can turn off some or all checks
when you create an executable
for distribution, although
the
code
-
size and performance improvements of turning checks off are pretty negligible. Plus, leaving checks turned on can help you locate bugs that turn up
later
.

+

* Set up a debug and a release/default build mode. Use the debug build mode to enable run-time checks, asseertions, line tracing. You can use the default/release mode to disable them.



+

*
* Develop with all run-time checks turned on. You can turn off some or all checks
in your release/default build used
for distribution
to your users
, although code size and performance improvements of turning checks off are pretty negligible. Plus, leaving checks turned on can help you locate bugs that turn up
in production (e.g. hitting rarely used code paths)
.



* Use assertions. Like run-time checks, assertions can be turned off later, although leaving them on can help you diagnose bugs that your users encounter.

+

*
* Use assertions. Like run-time checks, assertions can be turned off later, although leaving them on can help you diagnose bugs that your users encounter.  



*
Compile with
line number tracing
turned
on.
You can turn this off when you release your program to shrink the size of the executable.

+

*
* Set
line number tracing on. If your code handles exceptions, you can determine the line number and source file name of where the exception occurred using line number tracing and the FPC BackTraceStrFunc function, as in
the
example code
below.



*
If your code handles exceptions, you can
still
determine the line number and source file name of where the exception occurred using line number tracing and the FPC BackTraceStrFunc function, as in
this
example code
:

+

 

 

 

<syntaxhighlight>try

 

<syntaxhighlight>try

Line 80:

Line 102:

 

The {$IFDEF} and {$ENDIF} lines ensure that you can compile this code with Delphi too.

 

The {$IFDEF} and {$ENDIF} lines ensure that you can compile this code with Delphi too.

 

 



* Initialize variables. Although both Delphi and FPC initialize global variables and object fields, variables local to routines are not initialized. It's a good idea to initialize all variables just to be clear about what their initial values are.

+

* Initialize variables. Although both Delphi and FPC initialize global variables and object fields, variables local to routines are not
guaranteed to be
initialized. It's a good idea to initialize all variables just to be clear about what their initial values are.

 

* Turn on compiler warnings and pay attention to them.

 

* Turn on compiler warnings and pay attention to them.

 

* Use "with" sparingly. Although the Pascal with statement can save you a lot of typing, it can also hide incorrect field references if you use nested with statements or if a field with the same name as a field in your with's target is added someday to an ancestor class.

 

* Use "with" sparingly. Although the Pascal with statement can save you a lot of typing, it can also hide incorrect field references if you use nested with statements or if a field with the same name as a field in your with's target is added someday to an ancestor class.



* Run your code through multiple compilers. There will always be differences in how different compilers process your code. Sometimes these differences are dictated by the underlying operating system or hardware. Try to compile your program (or at least the non-UI units) with two or three compilers. For example, if your code compiles and runs correctly with Delphi and on at least two of FPC's platforms, you'll be that much closer not only to a stable cross-platform
app
but also to having a program that you know will work.

+

* Run your code through multiple compilers
(or compilers on different operating systems, e.g. Windows and Linux)
. There will always be differences in how different compilers process your code. Sometimes these differences are dictated by the underlying operating system or hardware. Try to compile your program (or at least the non-UI units) with two or three compilers. For example, if your code compiles and runs correctly with Delphi and on at least two of FPC's platforms, you'll be that much closer not only to a stable cross-platform
application
but also to having a program that you know will work.

 

 

 

[[Category:Tutorials]]

 

[[Category:Tutorials]]

 

+

[[Category:Lazarus]]

 

+

[[Category:FPC]]

 

+

[[Category:Development]]

 

+

[[Category:Debugging]]

Show more