Wednesday, July 30, 2014

iOS define local and global NSString constant

situee.blogspot.com

1. Local Constant

There are two type of constants.

(1)  static NSString * const kMyConstant = @"constant string";

(2)  #define kMyConstant @"constant string"


"static const" is a better way, although "#define" is much shorter,

Here are some reasons:

  • "#define" are not type-safe. 
  • We can test the value of the "static const" constant with debugger;
  • Memory. Preprocessor create a new string each time the macro appear, while "static const" reuse the same string.

2. Global Constant

----------------  .h file  --------------

extern NSString *const kMyConstant;

----------------  .m file  --------------
// define in a .m file OUTSIDE @implementation like...

NSString *const kMyConstant = @"my constant";

The extern declaration says that there exists an NSString * const by the name kMyConstant whose storage is created in some other place.[1]

Remove the static -- that specifies that kMyConstant is only visible in files linked with this one.[1]

C pointer and constant

Review C pointer and constant:
situee.blogspot.com

1. Pointer to Constant Value

const char *ptr = 'a';  OR    char const *ptr = 'a'

"const" precedes the *, that means the pointer treat *ptr  as constant

This is NOT valid     :  
*ptr = 'b';

Another example;

int nValue = 5;
const int *pnPtr = &nValue;

Thus, the following is okay:
    nValue = 6; // nValue is non-const
But the following is not:
    *pnPtr = 6; // pnPtr treats its value as const


2. Constant Pointer

char * const ptr = 'a';

This means the pointer ptr is a constant pointer

This is NOT valid:
char char_B = 'b';
ptr = &char_B;


3. Constant Pointer to Constant Value

const char * const ptr = 'a';  OR    char const * const ptr = 'a';


Reference:


http://www.learncpp.com/cpp-tutorial/610-pointers-and-const/
http://www.cprogramming.com/reference/pointers/const_pointers.html
http://www.noxeos.com/2011/07/29/c-const-static-keywords/
http://www.codeguru.com/cpp/cpp/cpp_mfc/general/article.php/c6967/Constant-Pointers-and-Pointers-to-Constants.htm

Sunday, July 20, 2014

Android - SuppressLint cannot be resolved to a type

After upgrading my Android SDK 20 and ADT-23.0.2, some projects got error "SuppressLint cannot be resolved to a type"



I fixed this by

1. In project property -> Android, change to another API level and change back.
2. change the order in project property ->Java build path -> Order and Export.
3. also please try refresh and clean the project.

Hope it can help and save some hours.

Tuesday, July 08, 2014

OpenDrive cloud storage direct link

I am trying OpenDrive storage service.
You will get 5G storage, and 1G traffic for a basic account.
It is great that you can share file by direct link.
I think it is very useful for a developer.
We can put XML, JSON, image files on the OpenDrive for testing.

Go to their website and have a look:
www.opendrive.com




Wednesday, January 22, 2014

Setup DNSmasq MAC DNS server


$ brew install dnsmasq

$ cp /usr/local/opt/dnsmasq/dnsmasq.conf.example /usr/local/etc/dnsmasq.conf
$ sudo cp -fv /usr/local/opt/dnsmasq/*.plist /Library/LaunchDaemons
$ sudo launchctl load /Library/LaunchDaemons/homebrew.mxcl.dnsmasq.plist

Go to System Network preference,
Set the only one DNS as 127.0.0.1

/usr/local/etc/dnsmasq.conf

#upstream dns servers
resolv-file=/etc/resolv.dnsmasq.conf

#for the host domain to the ip, like the "hosts" file
address=/dev/127.0.0.1
address=/double-click.net/127.0.0.1

# set listen address as your IP
listen-address=127.0.0.1, 192.168.11.12

/etc/resolv.dnsmasq.conf  

Put upstream server in /etc/resolv.dnsmasq.conf , such as 8.8.8.8
nameserver upstream_dns_server
nameserver 8.8.8.8


Kill the process, it will restart and reload new config.

$ sudo kill $(ps aux | grep '[d]nsmasq' | awk '{print $2}')


Reference:

Install DNSmasq locally on Mac OS X via Homebrew
http://blog.philippklaus.de/2012/02/install-dnsmasq-locally-on-mac-os-x-via-homebrew/

Setup a wildcard TLD using Dnsmasq on OS X
http://bhamrick.com/2013/04/18/setup-a-wildcard-tld-using-dnsmasq-on-os-x/

https://gist.github.com/g3d/2709563

http://unixhelp.ed.ac.uk/CGI/man-cgi?ps

Appendix:

 ps aux  : to see every process on the system using bSD syntax;

$ ps aux 
USER       PID  %CPU %MEM  VSZ RSS     TTY   STAT START   TIME COMMAND
timothy  29217  0.0  0.0 11916 4560 pts/21   S+   08:15   0:00 pine 
root     29505  0.0  0.0 38196 2728 ?        Ss   Mar07   0:00 sshd: can [priv]  
can      29529  0.0  0.0 38332 1904 ?        S    Mar07   0:00 sshd: can@notty 
USER = user owning the process
PID = process ID of the process
%CPU = It is the CPU time used divided by the time the process has been running.
%MEM = ratio of the process’s resident set size to the physical memory on the machine
VSZ = virtual memory usage of entire process
RSS = resident set size, the non-swapped physical memory that a task has used
TTY = controlling tty (terminal)
STAT = multi-character process state
START = starting time or date of the process
TIME = cumulative CPU time
COMMAND = command with all its arguments
ps -aux VS ps aux

Note that "ps -aux" is distinct from "ps aux". The POSIX and UNIX standards require that "ps -aux" print all processes owned by a user named "x", as well as printing all processes that would be selected by the -a option. If the user named "x" does not exist, this ps may interpret the command as "ps aux" instead and print a warning. This behavior is intended to aid in transitioning old scripts and habits. It is fragile, subject to change, and thus should not be relied upon.

Thursday, October 17, 2013

Fix Android cannot show Logcat, "Invalid Argument"

Fix problem: Android cannot show Logcat, "Invalid Argument"
  • Unplug and plug your device
  • Switch ON/OFF the developer option – USB debugging
  • Clicking at the device on DDMS-Device tab
  • Use command "adb logcat -c" followed by unplug/plug device.
  • On DDMS-Device tab, click the down triangle, choose “Reset adb”
  • Huawei device: *#*#2846579#*#*  Go ProjectMenu / Background Setting / Log setting
  • Use app to show logcat, such as CatLog (play.google.com/store/apps/details?id=com.nolanlawson.logcat
  • Run “echo 1 > /sys/kernel/logger/log_main/enable” in “adb shell”
Reference :
[1] http://stackoverflow.com/questions/2250112/why-doesnt-logcat-show-anything-in-my-android/2945333#2945333
[2] http://stackoverflow.com/questions/3823934/android-the-ddms-shows-the-message-logcat-read-invalid-argument

Sunday, September 22, 2013

Use Launch Daemon to Start Tomcat Automatically on Mac

Start tomcat automatically on Mac.

1. create launchd_wrapper.sh at tomcat/bin
2. create /Library/LaunchDaemons/org.apache.tomcat.plist
3. restart your mac

Note: Replace "/Users/username/tomcat" with your tomcat path in launchd_wrapper.sh and plist file.

try to launch tomcat with the plist
"sudo launchctl load -w /Library/LaunchDaemons/org.apache.tomcat.plist"

If you got error: "launchctl: Dubious ownership on file (skipping)" [5]
Please try change owner and permission of the plist file
a. change the owner to root with "sudo chown root <filename>",
b. change the permissions with "sudo chmod 644 <filename>"
   (4 for read access, 2 for write access, 1 for execute access, added up. The first number is for the owner, the second for the group, the third for everyone.) 


-----  http://situee.blogspot.com ---------

Tuesday, August 20, 2013

Batch rename images to sequence number with script

To get some test images.
Get images from images.baidu.com, as follows,

















create a shell script (.sh) file with following content

n=0
for file in *.jpg; do
    file_name="$n.jpg"
    n=$(( $n+1 ))
    mv $file $file_name
done


Images are renamed to sequence number


Monday, August 05, 2013

iOS handle Remote Push Notification

There are three cases to handle remote push notification.
  1. Your app was just launched
    handle callback didFinishLaunchingWithOptions application launched by clicking "View" when app is not running
  2. Your app was just brought from background to foreground
    by clicking "View" when app is running in background.
    handle callback didReceiveRemoteNotification
  3. Your app was already running in the foreground
    handle callback didReceiveRemoteNotification
Both (2) and (3) are handled by didReceiveRemoteNotification, use the following code to distinguish forground and background.

Thursday, August 01, 2013

Using Ti.Google.Analytics in Titanium Mobile

In previous post "Titanium Javascript Execution Context", we know that different javascript execution context == different scopes. When implementing the Google Analytics, we will meet this problem.

The Ti.Google.Analytics project has been converted to use CommonJS. [1]

var gaModule = require('Ti.Google.Analytics'); var analytics = new gaModule('UA-XXXXXX-X'); // Call the next function if you want to reset the analytics to a new first time visit. // This is useful for development only and should not go into a production app. //analytics.reset(); // The analytics object functions must be called on app.js otherwise it will loose it's context Ti.App.addEventListener('analytics_trackPageview', function(e){
analytics.trackPageview('/iPad' + e.pageUrl); }); Ti.App.addEventListener('analytics_trackEvent', function(e){ analytics.trackEvent(e.category, e.action, e.label, e.value); }); // Starts a new session as long as analytics.enabled = true // Function takes an integer which is the dispatch interval in seconds analytics.start(10,true); // You don't need to call stop on application close, but this is just to show you can callstop at any time (Basically sets enabled = false) Titanium.App.addEventListener('close', function(e){ analytics.stop(); });
By using commonJS, all of our windows are in same context, we can simply call analytics.trackPageview directly inside windows.

However, for windows created by window.url property, the global variable "analytics" is not available.  For compatible reason, in window created by "url" property, we can still fire a global event to track.  Like this,

    Ti.App.fireEvent("analytics_trackPageview", {pageUrl:'myhomepage'});