HarmonyOS Sample Network Management

HarmonyOS Sample Network Management

[[425392]]

For more information, please visit:

Hongmeng Technology Community jointly built by 51CTO and Huawei

https://harmonyos..com

1. Introduction

This example demonstrates how to use the network management module related interfaces and demonstrates the following functions:

Function 1: Use the default network, open a connection, and send an HTTP request.

Function 2: Statistics the upstream/downstream traffic of the specified UID.

Function 3: Use Socket to realize communication between different devices.

This function requires that WIFI is turned on and the communicating devices are connected to the same WIFI to form a local area network.

In terms of operation, start the server first, then start the client, and then send a message from the client to see if the server receives the message.

Function 4: Use HTTP cache to create a cache for the next request, reducing data traffic and loading time.

Note that the following permissions are required:

ohos.permission.GET_NETWORK_INFO Get network connection information.

ohos.permission.SET_NETWORK_INFO Modify the network connection status.

ohos.permission.INTERNET Allows an application to open network sockets and make network connections.

For details, see the official document Network Management Development Overview

2. Build the environment

Install DevEco Studio. For details, see DevEco Studio Download.

Set up the DevEco Studio development environment. The DevEco Studio development environment needs to rely on the network environment. You need to connect to the network to ensure the normal use of the tool. You can configure the development environment according to the following two situations:

If you can access the Internet directly, you only need to download the HarmonyOS SDK.

If the network cannot directly access the Internet and needs to access it through a proxy server, please refer to Configuring the Development Environment.

After downloading the source code, use DevEco Studio to open the project and run the simulator.

To run on a real device, you need to change the buddleName in config.json to your own. If you have not done so, please configure it on AppGallery Connect. For details, see Debugging with the Simulator.

3. Code structure

3.1 Code Structure


3.2 Related Documents

Core Classes:

HttpURLConnection.java //URLConnection that supports HTTP specific functions

URLConnection.java //URL connection

URL.java //Pointer to a "resource" on the World Wide Web

NetStatusCallback.java //Callback class for network status, triggers onAvailable function when available network appears

DataFlowStatistics.java //This class provides an interface for querying the overall flow statistics and flow statistics of a specified cellular network, application, and network card.

DatagramSocket.java //This class represents a socket used to send and receive datagram packets.

DatagramPacket.java //Datagram packet

WifiDevice.java //This class provides Wi-Fi management interface

NetManager.java //Provides interfaces to manage and use data networks.

NetHandle.java // Data network

InetAddress.java //Network IP address

HttpResponseCache.java //This class caches HTTP and HTTPS responses for reuse

Custom class:

ThreadPoolUtil.java //Thread pool tool class

MainAbilitySlice.java //Main page

NetRequestSlice.java //Network request & traffic statistics function page

SocketClientSlice.java //Socket client

SocketServerSlice.java //Socket server

HttpCacheSlice.java //HTTP cache function page

Page Layout:

http_cache_slice.xml //HTTP cache sample page

net_request.slice.xml //HTTP request page

socket_client_slice.xml //Socket communication client page

socket_server_slice.xml //Socket communication server page

main_ability_slice.xml //Main page

4. Example explanation

4.1. Interface layout



4.2. Background code

4.2.1 NetRequestSlice.java Network request & traffic statistics function

a. Initialize the network management object NetManager

  1. // Initialize the network management object
  2. netManager = NetManager.getInstance( null );

b. Get a new thread from the thread pool to process the connection request

The ohos.permission.GET_NETWORK_INFO permission is required to obtain the default data network.

  1. //Use the thread pool to get a new thread to process
  2. ThreadPoolUtil.submit(() -> {
  3. HiLog.debug(LABEL_LOG, "%{public}s" , "ThreadPoolUtil submit" );
  4. //Get the default data network. When both wifi and traffic are turned off, netId==0, otherwise netId=390/391. You must have the ohos.permission.GET_NETWORK_INFO permission
  5. NetHandle netHandle = netManager.getDefaultNet();
  6. //Receive callback for status changes of the default data network
  7. netManager.addDefaultNetStatusCallback(callback);
  8.  
  9. //netManager.setAppNet(netHandle);
  10.  
  11. // A URLConnection that supports HTTP specific features.
  12. HttpURLConnection connection = null ;
  13. //Output stream
  14. try (ByteArrayOutputStream outputStream = new ByteArrayOutputStream()) {
  15. //Requested URL
  16. String urlString = inputText.getText();
  17.  
  18. URL url = new URL(urlString);
  19. //Use netHandle to open the URL connection without using a proxy
  20. URLConnection urlConnection = netHandle.openConnection(url, java.net.Proxy.NO_PROXY);
  21. HiLog.debug(LABEL_LOG, "%{public}s" , "netHandle openConnection" );
  22. //Strong conversion type
  23. if (urlConnection instanceof HttpURLConnection) {
  24. connection = (HttpURLConnection) urlConnection;
  25. }
  26. // Request type
  27. connection .setRequestMethod( "GET" );
  28. //connect
  29. connection.connect ( ) ;
  30. //Traffic statistics
  31. trafficDataStatistics( false );
  32.  
  33. try (InputStream inputStream = urlConnection.getInputStream()) {
  34. byte[] cache = new byte[2 * 1024];
  35. int len ​​= inputStream.read ( cache);
  36. while (len != -1) {
  37. //
  38. outputStream.write(cache, 0, len);
  39. len = inputStream.read (cache);
  40. }
  41. } catch (IOException e) {
  42. HiLog.error(LABEL_LOG, "%{public}s" , "netRequest inner IOException" );
  43. }
  44. //Return result
  45. String result = new String(outputStream.toByteArray());
  46.  
  47. //UI display
  48. getUITaskDispatcher().asyncDispatch(() -> outText.setText(result));
  49. //Statistics completed
  50. trafficDataStatistics( true );
  51.  
  52. } catch (IOException e) {
  53. HiLog.error(LABEL_LOG, "%{public}s" , "netRequest IOException" );
  54. }
  55. });

c. Perform data traffic statistics according to application ID

Get it once before sending the request and once after the request is completed.

There is a typo in the gitee code. tx represents uplink traffic and rx represents downlink traffic. For details, see the official document explaining traffic statistics.

  1. /**
  2. * Perform data traffic statistics based on application ID
  3. *
  4. * @param isStart
  5. */
  6. private void trafficDataStatistics(boolean isStart) {
  7. int uid = 0;
  8. try {
  9. //Get the application UID based on the given package name and user ID.
  10. uid = getBundleManager().getUidByBundleName(getBundleName(), 0);
  11. } catch (RemoteException e) {
  12. HiLog.error(LABEL_LOG, "%{public}s" , "trafficDataStatistics RemoteException" );
  13. }
  14. if (isStart) {
  15. //Get the downstream traffic of the specified UID.
  16. rx = DataFlowStatistics.getUidRxBytes(uid);
  17. //Get the upstream traffic of the specified UID.
  18. tx = DataFlowStatistics.getUidTxBytes(uid);
  19. } else {
  20. rx = DataFlowStatistics.getUidRxBytes(uid) - rx;
  21. tx = DataFlowStatistics.getUidTxBytes(uid) - tx;
  22.  
  23. //Set UI display
  24. getUITaskDispatcher().asyncDispatch(() -> statisticsText.setText(
  25. "TrafficDataStatistics:" + System.lineSeparator()
  26. + "Receive traffic:" + rx + System.lineSeparator()
  27. + "Sent traffic:" + tx));
  28. }
  29. }
  30. }

4.2.2 SocketClientSlice.java/SocketServerSlice.java Socket Client/Server

To realize Socket communication, a client and a server are required. The server listens to the specified port on the specified network card, and the client sends data to the specified IP and port to realize communication.

a. The Socket server starts listening and waits for receiving data

  1. //Listening port
  2. private static final int PORT = 8888;
  3.  
  4. /**
  5. * Start socket service monitoring
  6. *
  7. * @param component
  8. */
  9. private void startServer(Component component) {
  10. HiLog.debug(LABEL_LOG, "startServer" );
  11.      
  12. //Thread pool gets new thread processing
  13. ThreadPoolUtil.submit(() -> {
  14. //Start listening on the specified port
  15. try (DatagramSocket socket = new DatagramSocket(PORT)) {
  16. //Datagram packet
  17. DatagramPacket packet = new DatagramPacket(new byte[255], 255);
  18. //Infinite loop receiving data
  19. while ( true ) {
  20. //Receive data
  21. socket.receive(packet);
  22.                  
  23. // Synchronize the received data to be displayed through a dedicated thread
  24. getUITaskDispatcher().syncDispatch(() -> outText.setText(
  25. "Receive a message from :" + packet.getAddress().getHostAddress()
  26. + System.lineSeparator() + " on port " + packet.getPort()
  27. + System.lineSeparator() + "message:" + new String(
  28. packet.getData()). substring (0, packet.getLength())
  29. ));
  30. packet.setLength(255);
  31.  
  32. // Delay to allow time for data processing
  33. Thread.sleep(1000);
  34. }
  35. } catch (IOException | InterruptedException e) {
  36. e.printStackTrace();
  37. HiLog.error(LABEL_LOG, "%{public}s" , "StartServer IOException | InterruptedException" + e);
  38. }
  39. });
  40. }
  41.  
  42.  
  43. /**
  44. * Get the server IP address and display it to the client to send a message
  45. *
  46. * @return  
  47. */
  48. private String getLocationIpAddress() {
  49. HiLog.debug(LABEL_LOG, "getLocationIpAddress" );
  50. // Provides interfaces to manage Wi-Fi.
  51. WifiDevice wifiDevice = WifiDevice.getInstance(this);
  52. HiLog.debug(LABEL_LOG, "wifiDevice:" + wifiDevice);
  53. //WifiLinkedInfo provides information about Wi-Fi connections.
  54. Optional<WifiLinkedInfo> linkedInfo = wifiDevice.getLinkedInfo();
  55. HiLog.debug(LABEL_LOG, "linkedInfo:" + linkedInfo);
  56. //Get IP address
  57. int ip = linkedInfo.get().getIpAddress();
  58. HiLog.debug(LABEL_LOG, "ip:" + ip);
  59. return (ip & 0xFF) + "." + ((ip >> 8) & 0xFF) + "." + ((ip >> 16) & 0xFF) + "." + (ip >> 24 & 0xFF);
  60. }

b.Socket client sends data and waits for receiving data

Initialize NetManager object->new a DatagramSocket->get the current data network NetHandle->get the server IP address object InetAddress

->Bind DatagramSocket to NetHandle ->Create a DatagramPacket ->Send data

  1. //Communication port
  2. private static final int PORT = 8888;
  3.  
  4.  
  5. /**
  6. * Send network request
  7. * @param component
  8. */
  9. private void netRequest(Component component) {
  10. HiLog.debug(LABEL_LOG, "netRequest" );
  11. //Start a new thread
  12. ThreadPoolUtil.submit(() -> {
  13. // Initialize the network management object
  14. NetManager netManager = NetManager.getInstance( null );
  15.  
  16. // Check if the default data network is activated.
  17. if (!netManager.hasDefaultNet()) {
  18. return ;
  19. }
  20. //new socket
  21. try (DatagramSocket socket = new DatagramSocket()) {
  22. //Get the current data network
  23. NetHandle netHandle = netManager.getDefaultNet();
  24.  
  25. //Get the server's IP address
  26. InetAddress address = netHandle.getByName(inputText.getText());
  27. //Bind the socket to the current network
  28. netHandle.bindSocket(socket);
  29. byte[] buffer = "I'm from Client" .getBytes();
  30. //Data packet
  31. DatagramPacket request = new DatagramPacket(buffer, buffer.length, address, PORT);
  32. //Send data
  33. socket.send(request);
  34. HiLog.debug(LABEL_LOG, "send socket" );
  35. } catch (IOException e) {
  36. e.printStackTrace();
  37. HiLog.error(LABEL_LOG, "%{public}s" , "netRequest IOException" +e);
  38. }
  39. });
  40. }

4.2.3 HttpCacheSlice.java HTTP caching function

When an application repeatedly opens the same web page, it can prioritize reading content from the cache file, thereby reducing data traffic, lowering device power consumption, and improving application performance.

Q: How to set the priority to read content from the cache file?

A: No additional settings are required, it is automatic~~

When setting up a cache, you need to consider the cache location and cache size.

a. Initialize cache install

  1. /**
  2. * Enable caching
  3. * Automatically cache data after turning it on, no manual processing required
  4. */
  5. private void initCache(Component component) {
  6. //Default cache directory
  7. File httpCacheDir = new File(this.getCacheDir(), "http" );
  8. // Cache size
  9. long httpCacheSize = 10 * 1024 * 1024;
  10. try {
  11. //Configure the cache directory and maximum cache space
  12. HttpResponseCache.install(httpCacheDir, httpCacheSize);
  13. HiLog.debug(LABEL_LOG, "%{public}s" , "initCache,cache installed[" + httpCacheDir + "]" );
  14. } catch (IOException e) {
  15. HiLog.error(LABEL_LOG, "%{public}s" , "initCache IOException" );
  16. }
  17. }

b. Save the cache and write the cache to the file system flush

  1. /**
  2. * Write the cache to the file system to prevent cache loss
  3. */
  4. private void flushCache(Component component) {
  5. HiLog.debug(LABEL_LOG, "%{public}s" , "flushCache" );
  6. try {
  7. //Get the cache object
  8. HttpResponseCache cache = HttpResponseCache.getInstalled();
  9. HiLog.debug(LABEL_LOG, "%{public}s" , "cache:" +cache);
  10. if (cache != null ) {
  11. try {
  12. //Write the cache to the file system. If cache is closed, an error will be reported. //java.lang.IllegalStateException: cache is closed
  13. cache.flush();
  14. getUITaskDispatcher().syncDispatch(() -> {
  15. //Image loading time, test the difference between caching and not caching
  16. duration.setText( "cache flush success" );
  17. });
  18. HiLog.debug(LABEL_LOG, "%{public}s" , "cache flush" );
  19. } catch (IOException e) {
  20. HiLog.error(LABEL_LOG, "%{public}s" , "onStop IOException" );
  21. }
  22. }
  23. } catch (IllegalStateException e) {
  24. e.printStackTrace();
  25. }
  26. }

c. Disable the cache and delete the data in it delete

  1. /**
  2. * Disable the cache and delete the data in it
  3. */
  4. private void deleteCache(Component component) {
  5. HiLog.debug(LABEL_LOG, "%{public}s" , "deleteCache" );
  6. //Get the cache object
  7. HttpResponseCache cache = HttpResponseCache.getInstalled();
  8. HiLog.debug(LABEL_LOG, "%{public}s" , "cache:" +cache);
  9. if (cache != null ) {
  10. try {
  11. // Disable the cache and delete the data in it.
  12. cache.delete ();
  13. image.setPixelMap( null );
  14. HiLog.debug(LABEL_LOG, "%{public}s" , "cache delete" );
  15. } catch (IOException e) {
  16. HiLog.error(LABEL_LOG, "%{public}s" , "onStop IOException" );
  17. }
  18.  
  19. }
  20. }
  21.  
  22. /**
  23. * Test request
  24. *
  25. * @param component
  26. */
  27. private void startRequest(Component component) {
  28. HiLog.debug(LABEL_LOG, "%{public}s" , "startRequest" );
  29. ThreadPoolUtil.submit(() -> {
  30. try {
  31. long startTime=System.currentTimeMillis();
  32. HiLog.debug(LABEL_LOG, "%{public}s" , "startTime:" +startTime);
  33. //Request URL
  34. URL url = new URL(inputText.getText());
  35. URLConnection urlConnection = url.openConnection();
  36. HiLog.debug(LABEL_LOG, "%{public}s" , "openConnection" );
  37. // Determine the connection type
  38. if (urlConnection instanceof HttpURLConnection) {
  39. HiLog.debug(LABEL_LOG, "%{public}s" , "urlConnection" );
  40. HttpURLConnection connection = (HttpURLConnection) urlConnection;
  41. HiLog.debug(LABEL_LOG, "%{public}s" , "connect:" + connection );
  42. //connect
  43. connection.connect ( ) ;
  44. HiLog.debug(LABEL_LOG, "%{public}s" , "connected" );
  45. //Connection result
  46. if ( connection .getResponseCode() == HttpURLConnection.HTTP_OK) {
  47. HiLog.debug(LABEL_LOG, "%{public}s" , "HTTP_OK" );
  48. //Describes image data source options, including for example the image format represented as image/png.
  49. ImageSource.SourceOptions srcOpts = new ImageSource.SourceOptions();
  50. ImageSource imageSource = ImageSource. create ( connection .getInputStream(), srcOpts);
  51. // Provide the image as a matrix of pixels.
  52. PixelMap pixelMap = imageSource.createPixelmap( null );
  53. HiLog.debug(LABEL_LOG, "%{public}s" , "pixelMap:" +pixelMap);
  54. //Specialized thread synchronization settings picture display
  55. getUITaskDispatcher().syncDispatch(() -> {
  56. image.setPixelMap(pixelMap);
  57. //Image loading time, test the difference between caching and not caching
  58. duration.setText(String.valueOf(System.currentTimeMillis()-startTime)+ " ms" );
  59. });
  60. HiLog.debug(LABEL_LOG, "%{public}s" , "setPixelMap" );
  61. HiLog.debug(LABEL_LOG, "%{public}s" , "endTime:" + (System.currentTimeMillis()-startTime));
  62. }
  63. HiLog.debug(LABEL_LOG, "%{public}s" , "finish" );
  64. //Close the connection
  65. connection .disconnect();
  66. }
  67. } catch (Exception e) {
  68. HiLog.error(LABEL_LOG, "%{public}s" , "initCache Exception" +e);
  69. getUITaskDispatcher().syncDispatch(() -> {
  70. //Image loading time, test the difference between caching and not caching
  71. duration.setText( "cache is closed, please open cache" );
  72. });
  73. }
  74. });
  75. }

5. Summary

(1) Two ways to open a network connection

  1. URLConnection urlConnection = url.openConnection();
  1. //Use netHandle to open the URL connection without using a proxy
  2. URLConnection urlConnection = netHandle.openConnection(url, java.net.Proxy.NO_PROXY);

(2) When caching is not enabled, the time to send a request is between 400-2000ms. When caching is enabled, you need to click twice to send a request, and the time remains at around 90-200ms.

(3) After disabling and clearing the cache delete/close, no request can be sent until the cache is enabled again. The official documentation for this operation states "Close the cache at the end."

6. Complete code

Download attachments directly

For more information, please visit:

Hongmeng Technology Community jointly built by 51CTO and Huawei

https://harmonyos..com

<<:  Pairing private networks with 5G to boost smart city development

>>:  5G is changing society, but is millimeter wave still missing?

Recommend

What is 5G network slicing and how can it benefit various industries?

Network slicing divides the network into multiple...

Expert: China ranks second in the world in terms of the number of IPv6 addresses

[[230257]] The global Internet Protocol (IP) addr...

10 things to know about LoRaWAN and NB-IoT

The Internet of Things has a variety of connectio...

10 common mistakes in integrated wiring, see how many you have made

Integrated wiring is a modular and highly flexibl...

Network packet loss troubleshooting solution

1. Location and processing of network packet loss...

What impact will 5G have on us?

1. First of all, it is clear that fiber optic tra...