目录
此内容是否有帮助?

# Android-Advanced

# 1. Managing User Identity

SDK instances would use random UUID as the default distinct ID of each user by default, which would be used as the identity identification ID of users under an unlogged-in state. It should be noted that the distinct ID would change after the user reinstalled the App or use the APP with a new device.

# 1.1. Identify

::: Tips

Generally speaking, you do not need to customize a distinct ID. Please ensure that you understand User Identification Rulesbefore setting a distinct ID.

If you need to change the distinct ID, please call the api immediately after SDK is initialized. To avoid the generation of useless accounts, please do not call such a process multiple times.

:::

If your App has its own distinct ID management system for each user, you can call identify to set the distinct ID:

// set distinct ID as Thinker
instance.identify("Thinker");

If you need to get the current distinct ID, please call getDistinctId:

//Return distinct ID
String distinctId = instance.getDistinctId();

# 1.2 Login

When the users log in, login could be called to set the account ID of the user. TE would use the account ID as the identity identification ID, and the account ID that has been set would be saved before logout is called. The previous account ID would be replaced if login has been called multiple times.

// The login unique identifier of the user, corresponding to the #account_id in data tracking. #Account_id now is TE
instance.login("TE");

Login events wouldn't be uploaded in this method.

# 1.3 Removing Account ID

After the user logs out, logout could be called to remove the account ID. The distinct ID would be used as the identity identification ID before the next time login is called.

instance.logout();

It is recommended that you call logout upon explicit logout event. For example, call logout when the user commits the behavior of canceling an account; do not call such a process when the App is closed.

Logout events wouldn't be uploaded in this method.

# Sending Events

After SDK is initialized, you can track user behaviour data. In general, ordinary events could meet business requirements. You can also use the First/Updatable Event based on your own business requirements.

# 2.1 Ordinary Events

You can call track to upload events. It is suggested that you set event properties based on the data tracking plan drafted previously. Here is an example of a user buying an item:

try {
    JSONObject properties = new JSONObject();
    properties."product_name", "product name");
    instance.track("product_buy",properties);
} catch (JSONException e) {
    e.printStackTrace();   
}

# 2.2 First Events

The First Events refers to events that would only be recorded once for the ID of a certain device or other dimensions. For example, under certain scenarios, you may want to record the activation event on a certain device. In this case, you can perform data tracking with the first event.

JSONObject properties = new JSONObject();
try {
    properties.put("key", "value");
} catch (JSONException e) {
    e.printStackTrace();
}
instance.track(new TDFirstEvent("device_activation", properties));

If you want to judge whether an event is the First Event from other dimensions, you can define a first_check_id for the First Event:

//set the user ID as the first_check_id of the first event to track the first initialization event of the user.
TDFirstEvent firstEvent = new TDFirstEvent("account_activation", properties);
firstEvent.setFirstCheckId("TE");
instance.track(firstEvent);

Note: Since the server has to check whether the event is the First Event, the First Event will be put in storage one hour later by default.

# 2.3 Updatable Events

You can meet the requirements for event data modification under specific scenarios through Updatable Events. The ID of Updatable Events should be specified and uploaded when the objects of Updatable Events are created. TE would determine the data to be updated according to the event name and event ID.

 //The event property status is 3 after reporting, with the price being 100
JSONObject properties = new JSONObject();
try {
    properties.put("status", 3);
    properties.put("price", 100);
} catch (JSONException e) {
    e.printStackTrace();
}
mInstance.track(new TDUpdatableEvent("UPDATABLE_EVENT", properties, "test_event_id"));

//The event property status is 5 after reporting, with the price remaining the same
JSONObject properties_new = new JSONObject();
try {
    properties_new.put("status", 5);
} catch (JSONException e) {
    e.printStackTrace();
}
mInstance.track(new TDUpdatableEvent("UPDATABLE_EVENT", properties_new, "test_event_id"));

# 2.4 Overwritable Events

Despite the similarity with Updatable Events, Overwritable Events would replace all historical data with the latest data. Looking from the perspective of effect, such a process is equivalent to the behavior of deleting the previous data while putting the latest data in storage. TE would determine the data to be updated according to the event name and event ID.

// Instance: Assume the event name is OVERWRITE_EVENT when reporting an overwritable event
//The event property status is 3 after reporting, with the price being 100
JSONObject properties = new JSONObject();
try {
    properties.put("status", 3);
    properties.put("price", 100);
} catch (JSONException e) {
    e.printStackTrace();
}
mInstance.track(new TDOverWritableEvent("OVERWRITE_EVENT", properties, "test_event_id"));

//The event property status is 5 after reporting, with the price deleted
JSONObject properties_new = new JSONObject();
try {
    properties_new.put("status", 5);
} catch (JSONException e) {
    e.printStackTrace();
}
mInstance.track(new TDOverWritableEvent("OVERWRITE_EVENT", properties_new, "test_event_id"));

# 2.5 Super Properties

Super properties refer to properties that would be uploaded by each event. Super properties could be divided into static super properties and dynamic super propertiesbased on the update frequency. You can select different methods for super property setting according to business requirements; we recommend that you set super properties first before sending events. In the same event, when the keys of super properties, self-defined event properties, and preset properties are the same, we would assign value according to the following priority: self-defined properties>dynamic super properties>static super properties>preset properties.

# 2.5.1 Static Super Properties

Static super properties are properties that all events might have and would change with a low frequency, for example, the user membership class. After setting static super properties through setSuperProperties, SDK would use the preset super properties as the event properties when tracking events.

//set super properties
try {
    JSONObject superProperties = new JSONObject();
    superProperties.put("vip_level",2);
    instance.setSuperProperties(superProperties);
} catch (JSONException e) {
    e.printStackTrace();
}

Static super properties would be saved in local storage, and should not be called every time the App is closed. If such properties already exist, the reset properties would replace the original properties. If such properties do not exist, properties would be newly created. In addition to property setting, we also provide other APIs to set and manage static super properties and meet general business requirements.

//clear a certain super property
instance.unsetSuperProperty("Channel");
//clear all certain super properties
instance.clearSuperProperties();
//obtain all certain super properties
instance.getSuperProperties();

# 2.5.2 Dynamic super properties

Dynamic super properties that all events might have and would change with a high frequency, for example, the quantity of the gold coins the user possesses. After setting dynamic super properties through setDynamicSuperPropertiesTracker, SDK would obtain the properties in getDynamicSuperProperties during event tracking automatically, and add such properties to the event triggered.

int coin = 0;
instance.setDynamicSuperPropertiesTracker(
    new ThinkingAnalyticsSDK.DynamicSuperPropertiesTracker() {
        @Override
        public JSONObject getDynamicSuperProperties() {
            Coin++;//frequency update of gold coin quantity 
            try {
                dynamicSuperProperties.put("coin",coni);
            } catch (JSONException e) {
                e.printStackTrace();
            }
            return dynamicSuperProperties;
        }
    });

# 2.6 Timing Events

If you need to record the duration of a certain event, you can call timeEvent . Configure the name of the event you want to record. When you upload the event, #duration would be added to your event property automatically to record the duration of the event (unit: second). It should be noted that only one task can be timed with the same event name.

//The following instance has recorded the time the user spent on a certain product page
try {
    //The user enters the product page and starts the timing
    instance.timeEvent("stay_shop");
    /**do someting
    .......
    **/
    //the timing would end when the user leaves the product page. "stay_shop" event would carry#duration, a property representing event duration. 
    instance.track("stay_shop");
} catch (JSONException e) {
    e.printStackTrace();
}

# 3. User Properties

User property setting APIs supported by TE include: user_set,user_setOnce,user_add,user_unset,user_delete,user_append,user_uniqAppend.

# 3.1 User_Set

You can call user_set to set general user properties. The original properties would be replaced if the properties uploaded via the API are used. The type of newly-created user properties must conform to that of the uploaded properties. User name setting is taken as the example here:

try {
    //the username now is TA
    JSONObject properties = new JSONObject();
    properties.put("username","TA");
    instance.user_set(properties);
    //the userName now is TE
    JSONObject newProperties = new JSONObject();
    newProperties.put("username","TE");
    instance.user_set(newProperties);
} catch (JSONException e) {
    e.printStackTrace();
}

# 3.2 User_SetOnce

If the user property you want to upload only needs to be set once, you can call user_setOnce to set the property. If such property had been set before, this message would be neglected. Let's take the setting of the first payment time as an example:

try {
    //first_payment_time is 2018-01-01 01:23:45.678
    JSONObject properties = new JSONObject();
    properties.put("first_payment_time","2018-01-01 01:23:45.678");
    instance.user_setOnce(properties);
    
    //first_payment_time is still 2018-01-01 01:23:45.678
    JSONObject newProperties = new JSONObject();
    newProperties.put("first_payment_time","2018-12-31 01:23:45.678");
    instance.user_setOnce(newProperties); 
       
} catch (JSONException e) {
    e.printStackTrace();
}

# 3.3 User_Add

When you want to upload numeric property for cumulative operation, you can call user_add.

If the property has not been set, it would be given a value of 0 before computing. A negative value could be uploaded, which is equivalent to subtraction operation. Let's take the accumulative payment amount as an example:

try {
    //in this case, the total_revenue is 30
    JSONObject properties = new JSONObject();additive operation
    properties.put("total_revenue",30);
    instance.user_add(properties);
    
    //in this case, the total_revenue is 678
    JSONObject newProperties = new JSONObject();
    newProperties.put("total_revenue",648);
    instance.user_add(newProperties);
} catch (JSONException e) {
    e.printStackTrace();
}

The property key is a string, and the value is only allowed to be a numeric value.

# 3.4 User_Unset

When you need to clear the user properties of users, you can call user_unset to clear specific properties. user_unset would not create properties that have not been created in the cluster.

// reset properties of a single user
instance.user_unset("key1");
// reset properties of several users
instance.user_unset("key1", "key2", "key3");

# 3.5 User_Delete

You can call user_delete to delete a user. After deleting the user, you would no longer be able to inquire about its user property, but could still query the events data triggered by the user.

instance.user_delete();

# 3.6 User_Append

You can call user_append to add user properties of array type.

try {
    // list is the value of user property user_list, JSONArray type
    JSONArray list = new JSONArray("[\"apple\", \"ball\"]");
    JSONObject properties = new JSONObject();
    properties.put("user_list", list);
    // call user_append to add elements for user property user_list. In this case, mon-existing elements would be newly created
    mInstance.user_append(properties);
} catch (JSONException e) {
    e.printStackTrace();
}

# 3.7 User_UniqAppend

You can call user_uniqAppend to add user properties of array type. You can delete duplicated user property by calling user_uniqAppend interface. If you call user_append API, duplicated user property will be merged.

try {
    // list is the value of user property user_list, JSONArray type
    //in this case, the property value of user_list is ["apple","ball"]
    JSONArray list = new JSONArray("[\"apple\", \"ball\"]");
    JSONObject properties = new JSONObject();
    properties.put("user_list", list);
    mInstance.user_append(properties);
    
    
    //in this case, the property value of user_list is ["apple","apple","ball","cube"]
    JSONArray list1 = new JSONArray("[\"apple\", \"cube\"]");
    JSONObject properties1 = new JSONObject();
    properties1.put("user_list", list1);
    mInstance.user_append(properties1);
    
    //in this case, the property value of user_list is ["apple","ball","cube"]
    mInstance.user_uniqAppend(properties1);
    
} catch (JSONException e) {
    e.printStackTrace();
}

# 4. Encryption

The SDK supports data encryption using AES+RSA. The data encryption function requires the cooperation of the client and the server. For specific usage methods, please consult customer success manager.

TDConfig config = TDConfig.getInstance(mContext,TA_APP_ID,TA_SERVER_URL);
//Enable the encryption function
config.enableEncrypt(true);
TDSecreteKey secreteKey = new TDSecreteKey();
secreteKey.publicKey = "publicKey";
secreteKey.version = 1;
secreteKey.symmetricEncryption = "AES";
secreteKey.asymmetricEncryption = "RSA";
//set public key information
config.setSecretKey(secreteKey);

# 5. Enable H5 Connection

If you need to connect with the JavaScript SDK that tracks the data of H5, please call the following API when initializing WebView. For detailed information, please refer to H5 and APP SDK Connection

// connect with H5 
instance.setJsBridge(webView);

# 6. Other

# 6.1 Device ID

You can call getDeviceId to obtain the device ID:

String deviceID = instance.getDeviceId();//the value of device ID is Android ID

# 6.2 Default Timezone

SDK would use the local time as the event time by default. You can also assign a timezone by setting the default timezone API. In this way, the time of all events could be aligned according to the timezone set by you:

// get TDConfig instance
TDConfig config = TDConfig.getInstance(this, TE_APP_ID, TE_SERVER_URL);
// set UTC as the default timezone
config.setDefaultTimeZone(TimeZone.getTimeZone("UTC"));
// initialize SDK
instance = ThinkingAnalyticsSDK.sharedInstance(config);

The local timezone information of the device would be lost if a specific timezone is used to align event time. If you need to save the local timezone information of the device, please add relevant properties for the event.

# 6.3 Time Calibration

SDK would use local time as the event time by default. If the user modifies the device time manually, analysis would be affected. At this time, time calibration could be performed to ensure the accuracy of event time. We provide two time calibration methods: timestamp and NTP.

  • You can use the current timestamp obtained from the server to calibrate the time of SDK. Thereafter, all calling operations not assigned with a specific time would use the calibrated time as the occurrence time, including event data and user property setting.
// 1585633785954 is the current unix time stamp, with the unit being millisecond; the corresponding Beijing time is 2020-03-31 13:49:45
ThinkingAnalyticsSDK.calibrateTime(1585633785954);
  • You can also set the address of NTP server, after which SDK would try to obtain the current time from the uploaded NTP server address and calibrate the SDK time. If you failed to obtain the current return results within the default timeout interval (3s), local time would be used to track data.
//use the NTP service of Apple Inc for time calibration 
ThinkingAnalyticsSDK.calibrateTimeWithNtp("time.apple.com");
  1. Time calibration may fail due to unstable NTP server. It is suggested that you use a time stamp for time calibration as the priority

  2. You should select your NTP server address carefully to ensure that the device of the user could obtain server time rapidly under sound network conditions

# 6.4 Flush

You can call the flush API to report data to TE immediately .

ta.flush();

# 6.5 Region Code

You can obtain such information through getLocalRegion to obtain the region code of the user device.

ThinkingAnalyticsSDK.getLocalRegion();